2019-10-18 11:29:47 +03:00
|
|
|
{- |
|
|
|
|
Description: Add/Drop computed fields in metadata
|
|
|
|
-}
|
|
|
|
module Hasura.RQL.DDL.ComputedField
|
|
|
|
( AddComputedField(..)
|
|
|
|
, ComputedFieldDefinition(..)
|
|
|
|
, runAddComputedField
|
|
|
|
, DropComputedField
|
|
|
|
, runDropComputedField
|
2020-12-08 17:22:31 +03:00
|
|
|
, dropComputedFieldInMetadata
|
2019-10-18 11:29:47 +03:00
|
|
|
) where
|
|
|
|
|
|
|
|
import Hasura.Prelude
|
|
|
|
|
2021-02-14 09:07:52 +03:00
|
|
|
import qualified Data.HashMap.Strict.InsOrd as OMap
|
2020-08-27 19:36:39 +03:00
|
|
|
|
|
|
|
import Data.Aeson
|
2020-10-27 16:53:49 +03:00
|
|
|
import Data.Text.Extended
|
2021-03-15 16:02:58 +03:00
|
|
|
|
|
|
|
import qualified Hasura.SQL.AnyBackend as AB
|
2020-08-27 19:36:39 +03:00
|
|
|
|
2019-10-18 11:29:47 +03:00
|
|
|
import Hasura.EncJSON
|
2021-02-14 09:07:52 +03:00
|
|
|
import Hasura.Incremental (Cacheable)
|
2019-10-18 11:29:47 +03:00
|
|
|
import Hasura.RQL.DDL.Deps
|
2020-12-08 17:22:31 +03:00
|
|
|
import Hasura.RQL.DDL.Permission
|
2019-10-18 11:29:47 +03:00
|
|
|
import Hasura.RQL.Types
|
|
|
|
|
|
|
|
|
2021-02-14 09:07:52 +03:00
|
|
|
data AddComputedField b
|
2019-10-18 11:29:47 +03:00
|
|
|
= AddComputedField
|
2020-12-28 15:56:00 +03:00
|
|
|
{ _afcSource :: !SourceName
|
2021-02-14 09:07:52 +03:00
|
|
|
, _afcTable :: !(TableName b)
|
2019-10-18 11:29:47 +03:00
|
|
|
, _afcName :: !ComputedFieldName
|
2021-02-14 09:07:52 +03:00
|
|
|
, _afcDefinition :: !(ComputedFieldDefinition b)
|
2019-10-18 11:29:47 +03:00
|
|
|
, _afcComment :: !(Maybe Text)
|
2021-02-14 09:07:52 +03:00
|
|
|
} deriving (Generic)
|
|
|
|
deriving instance (Backend b) => Show (AddComputedField b)
|
|
|
|
deriving instance (Backend b) => Eq (AddComputedField b)
|
|
|
|
instance (Backend b) => NFData (AddComputedField b)
|
|
|
|
instance (Backend b) => Cacheable (AddComputedField b)
|
|
|
|
instance (Backend b) => ToJSON (AddComputedField b) where
|
|
|
|
toJSON = genericToJSON hasuraJSON
|
|
|
|
|
|
|
|
instance (Backend b) => FromJSON (AddComputedField b) where
|
2020-12-28 15:56:00 +03:00
|
|
|
parseJSON = withObject "Object" $ \o ->
|
|
|
|
AddComputedField
|
|
|
|
<$> o .:? "source" .!= defaultSource
|
|
|
|
<*> o .: "table"
|
|
|
|
<*> o .: "name"
|
|
|
|
<*> o .: "definition"
|
2021-01-14 10:02:46 +03:00
|
|
|
<*> o .:? "comment"
|
2019-10-18 11:29:47 +03:00
|
|
|
|
2021-02-14 09:07:52 +03:00
|
|
|
runAddComputedField :: (MonadError QErr m, CacheRWM m, MetadataM m) => AddComputedField 'Postgres -> m EncJSON
|
2019-10-18 11:29:47 +03:00
|
|
|
runAddComputedField q = do
|
2020-12-28 15:56:00 +03:00
|
|
|
withPathK "table" $ askTabInfo source table
|
2021-03-15 16:02:58 +03:00
|
|
|
let metadataObj = MOSourceObjId source
|
|
|
|
$ AB.mkAnyBackend
|
|
|
|
$ SMOTableObj table
|
|
|
|
$ MTOComputedField computedFieldName
|
2020-12-08 17:22:31 +03:00
|
|
|
metadata = ComputedFieldMetadata computedFieldName (_afcDefinition q) (_afcComment q)
|
|
|
|
buildSchemaCacheFor metadataObj
|
|
|
|
$ MetadataModifier
|
2020-12-28 15:56:00 +03:00
|
|
|
$ tableMetadataSetter source table.tmComputedFields
|
2020-12-08 17:22:31 +03:00
|
|
|
%~ OMap.insert computedFieldName metadata
|
2019-11-20 21:21:30 +03:00
|
|
|
pure successMsg
|
2020-12-08 17:22:31 +03:00
|
|
|
where
|
2020-12-28 15:56:00 +03:00
|
|
|
source = _afcSource q
|
2020-12-08 17:22:31 +03:00
|
|
|
table = _afcTable q
|
|
|
|
computedFieldName = _afcName q
|
2019-10-18 11:29:47 +03:00
|
|
|
|
2021-02-14 09:07:52 +03:00
|
|
|
data DropComputedField b
|
2019-10-18 11:29:47 +03:00
|
|
|
= DropComputedField
|
2020-12-28 15:56:00 +03:00
|
|
|
{ _dccSource :: !SourceName
|
2021-02-14 09:07:52 +03:00
|
|
|
, _dccTable :: !(TableName b)
|
2019-10-18 11:29:47 +03:00
|
|
|
, _dccName :: !ComputedFieldName
|
|
|
|
, _dccCascade :: !Bool
|
2021-02-14 09:07:52 +03:00
|
|
|
} deriving (Generic)
|
|
|
|
deriving instance (Backend b) => Show (DropComputedField b)
|
|
|
|
deriving instance (Backend b) => Eq (DropComputedField b)
|
|
|
|
instance (Backend b) => ToJSON (DropComputedField b) where
|
|
|
|
toJSON = genericToJSON hasuraJSON
|
2019-10-18 11:29:47 +03:00
|
|
|
|
2021-02-14 09:07:52 +03:00
|
|
|
instance (Backend b) => FromJSON (DropComputedField b) where
|
2019-10-18 11:29:47 +03:00
|
|
|
parseJSON = withObject "Object" $ \o ->
|
|
|
|
DropComputedField
|
2020-12-28 15:56:00 +03:00
|
|
|
<$> o .:? "source" .!= defaultSource
|
|
|
|
<*> o .: "table"
|
2019-10-18 11:29:47 +03:00
|
|
|
<*> o .: "name"
|
|
|
|
<*> o .:? "cascade" .!= False
|
|
|
|
|
|
|
|
runDropComputedField
|
2021-03-15 16:02:58 +03:00
|
|
|
:: forall b m
|
|
|
|
. (QErrM m, CacheRWM m, MetadataM m, BackendMetadata b)
|
2021-02-14 09:07:52 +03:00
|
|
|
=> DropComputedField b -> m EncJSON
|
2020-12-28 15:56:00 +03:00
|
|
|
runDropComputedField (DropComputedField source table computedField cascade) = do
|
2019-10-18 11:29:47 +03:00
|
|
|
-- Validation
|
2020-12-28 15:56:00 +03:00
|
|
|
fields <- withPathK "table" $ _tciFieldInfoMap <$> askTableCoreInfo source table
|
2019-10-18 11:29:47 +03:00
|
|
|
void $ withPathK "name" $ askComputedFieldInfo fields computedField
|
|
|
|
|
|
|
|
-- Dependencies check
|
|
|
|
sc <- askSchemaCache
|
2021-03-15 16:02:58 +03:00
|
|
|
let deps = getDependentObjs sc
|
|
|
|
$ SOSourceObj source
|
|
|
|
$ AB.mkAnyBackend
|
|
|
|
$ SOITableObj table
|
|
|
|
$ TOComputedField computedField
|
2019-10-18 11:29:47 +03:00
|
|
|
when (not cascade && not (null deps)) $ reportDeps deps
|
|
|
|
|
2019-11-20 21:21:30 +03:00
|
|
|
withNewInconsistentObjsCheck do
|
2020-12-08 17:22:31 +03:00
|
|
|
metadataModifiers <- mapM purgeComputedFieldDependency deps
|
|
|
|
buildSchemaCache $ MetadataModifier $
|
2020-12-28 15:56:00 +03:00
|
|
|
tableMetadataSetter source table
|
2021-03-15 16:02:58 +03:00
|
|
|
%~ dropComputedFieldInMetadata computedField . foldl' (.) id metadataModifiers
|
2019-10-18 11:29:47 +03:00
|
|
|
pure successMsg
|
|
|
|
where
|
|
|
|
purgeComputedFieldDependency = \case
|
2021-02-14 09:07:52 +03:00
|
|
|
-- TODO: do a better check of ensuring that the dependency is as expected.
|
|
|
|
-- i.e, the only allowed dependent objects on a computed fields are permissions
|
|
|
|
-- on the same table
|
2021-03-15 16:02:58 +03:00
|
|
|
SOSourceObj _ exists
|
|
|
|
| Just (SOITableObj _ (TOPerm roleName permType))
|
|
|
|
<- AB.unpackAnyBackend @b exists ->
|
|
|
|
pure $ dropPermissionInMetadata roleName permType
|
2019-10-18 11:29:47 +03:00
|
|
|
d -> throw500 $ "unexpected dependency for computed field "
|
|
|
|
<> computedField <<> "; " <> reportSchemaObj d
|
|
|
|
|
2020-12-08 17:22:31 +03:00
|
|
|
dropComputedFieldInMetadata
|
2021-02-14 09:07:52 +03:00
|
|
|
:: ComputedFieldName -> TableMetadata b -> TableMetadata b
|
2020-12-08 17:22:31 +03:00
|
|
|
dropComputedFieldInMetadata name =
|
|
|
|
tmComputedFields %~ OMap.delete name
|