module Hasura.RQL.DDL.Permission ( CreatePerm , runCreatePerm , PermDef(..) , InsPerm(..) , InsPermDef , CreateInsPerm , buildInsPermInfo , SelPerm(..) , SelPermDef , CreateSelPerm , buildSelPermInfo , UpdPerm(..) , UpdPermDef , CreateUpdPerm , buildUpdPermInfo , DelPerm(..) , DelPermDef , CreateDelPerm , buildDelPermInfo , IsPerm(..) , DropPerm , runDropPerm , dropPermissionInMetadata , SetPermComment(..) , runSetPermComment , fetchPermDef ) where import Hasura.Prelude import qualified Data.HashMap.Strict as HM import qualified Data.HashMap.Strict.InsOrd as OMap import qualified Data.HashSet as HS import qualified Database.PG.Query as Q import Data.Aeson import Data.Aeson.Casing import Data.Aeson.TH import Data.Text.Extended import Hasura.Backends.Postgres.SQL.Types import Hasura.EncJSON import Hasura.RQL.DDL.Permission.Internal import Hasura.RQL.DML.Internal hiding (askPermInfo) import Hasura.RQL.Types import Hasura.SQL.Types import Hasura.Session {- Note [Backend only permissions] ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ As of writing this note, Hasura permission system is meant to be used by the frontend. After introducing "Actions", the webhook handlers now can make GraphQL mutations to the server with some backend logic. These mutations shouldn't be exposed to frontend for any user since they'll bypass the business logic. For example:- We've a table named "user" and it has a "email" column. We need to validate the email address. So we define an action "create_user" and it expects the same inputs as "insert_user" mutation (generated by Hasura). Now, a role has permission for both actions and insert operation on the table. If the insert permission is not marked as "backend_only: true" then it visible to the frontend client along with "creat_user". Backend only permissions adds an additional privilege to Hasura generated operations. Those are accessable only if the request is made with `x-hasura-admin-secret` (if authorization is configured), `x-hasura-use-backend-only-permissions` (value must be set to "true"), `x-hasura-role` to identify the role and other required session variables. backend_only `x-hasura-admin-secret` `x-hasura-use-backend-only-permissions` Result ------------ --------------------- ------------------------------------- ------ FALSE ANY ANY Mutation is always visible TRUE FALSE ANY Mutation is always hidden TRUE TRUE (OR NOT-SET) FALSE Mutation is hidden TRUE TRUE (OR NOT-SET) TRUE Mutation is shown -} type CreateInsPerm b = CreatePerm (InsPerm b) procSetObj :: (QErrM m) => QualifiedTable -> FieldInfoMap (FieldInfo 'Postgres) -> Maybe (ColumnValues Value) -> m (PreSetColsPartial 'Postgres, [Text], [SchemaDependency]) procSetObj tn fieldInfoMap mObj = do (setColTups, deps) <- withPathK "set" $ fmap unzip $ forM (HM.toList setObj) $ \(pgCol, val) -> do ty <- askPGType fieldInfoMap pgCol $ "column " <> pgCol <<> " not found in table " <>> tn sqlExp <- valueParser (CollectableTypeScalar ty) val let dep = mkColDep (getDepReason sqlExp) tn pgCol return ((pgCol, sqlExp), dep) return (HM.fromList setColTups, depHeaders, deps) where setObj = fromMaybe mempty mObj depHeaders = getDepHeadersFromVal $ Object $ mapKeys getPGColTxt setObj getDepReason = bool DRSessionVariable DROnType . isStaticValue class (ToJSON a) => IsPerm a where permAccessor :: PermAccessor 'Postgres (PermInfo a) buildPermInfo :: (QErrM m, TableCoreInfoRM 'Postgres m) => QualifiedTable -> FieldInfoMap (FieldInfo 'Postgres) -> PermDef a -> m (WithDeps (PermInfo a)) getPermAcc1 :: PermDef a -> PermAccessor 'Postgres (PermInfo a) getPermAcc1 _ = permAccessor getPermAcc2 :: DropPerm a -> PermAccessor 'Postgres (PermInfo a) getPermAcc2 _ = permAccessor addPermToMetadata :: PermDef a -> TableMetadata -> TableMetadata runCreatePerm :: (UserInfoM m, CacheRWM m, IsPerm a, MonadError QErr m, MetadataM m) => CreatePerm a -> m EncJSON runCreatePerm (WithTable tn pd) = do let pt = permAccToType $ getPermAcc1 pd role = _pdRole pd metadataObject = MOTableObj tn $ MTOPerm role pt buildSchemaCacheFor metadataObject $ MetadataModifier $ metaTables.ix tn %~ addPermToMetadata pd pure successMsg runDropPerm :: (IsPerm a, UserInfoM m, CacheRWM m, MonadTx m, MetadataM m) => DropPerm a -> m EncJSON runDropPerm dp@(DropPerm table role) = do tabInfo <- askTabInfo table let permType = permAccToType $ getPermAcc2 dp askPermInfo tabInfo role $ getPermAcc2 dp withNewInconsistentObjsCheck $ buildSchemaCache $ MetadataModifier $ metaTables.ix table %~ dropPermissionInMetadata role permType return successMsg dropPermissionInMetadata :: RoleName -> PermType -> TableMetadata -> TableMetadata dropPermissionInMetadata rn = \case PTInsert -> tmInsertPermissions %~ OMap.delete rn PTSelect -> tmSelectPermissions %~ OMap.delete rn PTDelete -> tmDeletePermissions %~ OMap.delete rn PTUpdate -> tmUpdatePermissions %~ OMap.delete rn buildInsPermInfo :: (QErrM m, TableCoreInfoRM 'Postgres m) => QualifiedTable -> FieldInfoMap (FieldInfo 'Postgres) -> PermDef (InsPerm 'Postgres) -> m (WithDeps (InsPermInfo 'Postgres)) buildInsPermInfo tn fieldInfoMap (PermDef _rn (InsPerm checkCond set mCols mBackendOnly) _) = withPathK "permission" $ do (be, beDeps) <- withPathK "check" $ procBoolExp tn fieldInfoMap checkCond (setColsSQL, setHdrs, setColDeps) <- procSetObj tn fieldInfoMap set void $ withPathK "columns" $ indexedForM insCols $ \col -> askPGType fieldInfoMap col "" let fltrHeaders = getDependentHeaders checkCond reqHdrs = fltrHeaders `union` setHdrs insColDeps = map (mkColDep DRUntyped tn) insCols deps = mkParentDep tn : beDeps ++ setColDeps ++ insColDeps insColsWithoutPresets = insCols \\ HM.keys setColsSQL return (InsPermInfo (HS.fromList insColsWithoutPresets) be setColsSQL backendOnly reqHdrs, deps) where backendOnly = Just True == mBackendOnly allCols = map pgiColumn $ getCols fieldInfoMap insCols = maybe allCols (convColSpec fieldInfoMap) mCols -- TODO this is a dirty hack, hardcoding permissions to postgres. When -- implementing support for other backends, the type family 'PermInfo' probably -- needs to be refactored. type instance PermInfo (InsPerm b) = InsPermInfo b instance IsPerm (InsPerm 'Postgres) where permAccessor = PAInsert buildPermInfo = buildInsPermInfo addPermToMetadata permDef = tmInsertPermissions %~ OMap.insert (_pdRole permDef) permDef buildSelPermInfo :: (QErrM m, TableCoreInfoRM 'Postgres m) => QualifiedTable -> FieldInfoMap (FieldInfo 'Postgres) -> SelPerm 'Postgres -> m (WithDeps (SelPermInfo 'Postgres)) buildSelPermInfo tn fieldInfoMap sp = withPathK "permission" $ do let pgCols = convColSpec fieldInfoMap $ spColumns sp (be, beDeps) <- withPathK "filter" $ procBoolExp tn fieldInfoMap $ spFilter sp -- check if the columns exist void $ withPathK "columns" $ indexedForM pgCols $ \pgCol -> askPGType fieldInfoMap pgCol autoInferredErr -- validate computed fields scalarComputedFields <- withPathK "computed_fields" $ indexedForM computedFields $ \fieldName -> do computedFieldInfo <- askComputedFieldInfo fieldInfoMap fieldName case _cfiReturnType computedFieldInfo of CFRScalar _ -> pure fieldName CFRSetofTable returnTable -> throw400 NotSupported $ "select permissions on computed field " <> fieldName <<> " are auto-derived from the permissions on its returning table " <> returnTable <<> " and cannot be specified manually" let deps = mkParentDep tn : beDeps ++ map (mkColDep DRUntyped tn) pgCols ++ map (mkComputedFieldDep DRUntyped tn) scalarComputedFields depHeaders = getDependentHeaders $ spFilter sp mLimit = spLimit sp withPathK "limit" $ mapM_ onlyPositiveInt mLimit return ( SelPermInfo (HS.fromList pgCols) (HS.fromList computedFields) be mLimit allowAgg depHeaders , deps ) where allowAgg = spAllowAggregations sp computedFields = spComputedFields sp autoInferredErr = "permissions for relationships are automatically inferred" type CreateSelPerm b = CreatePerm (SelPerm b) -- TODO see TODO for PermInfo above. type instance PermInfo (SelPerm b) = SelPermInfo b instance IsPerm (SelPerm 'Postgres) where permAccessor = PASelect buildPermInfo tn fieldInfoMap (PermDef _ a _) = buildSelPermInfo tn fieldInfoMap a addPermToMetadata permDef = tmSelectPermissions %~ OMap.insert (_pdRole permDef) permDef type CreateUpdPerm b = CreatePerm (UpdPerm b) buildUpdPermInfo :: (QErrM m, TableCoreInfoRM 'Postgres m) => QualifiedTable -> FieldInfoMap (FieldInfo 'Postgres) -> UpdPerm 'Postgres -> m (WithDeps (UpdPermInfo 'Postgres)) buildUpdPermInfo tn fieldInfoMap (UpdPerm colSpec set fltr check) = do (be, beDeps) <- withPathK "filter" $ procBoolExp tn fieldInfoMap fltr checkExpr <- traverse (withPathK "check" . procBoolExp tn fieldInfoMap) check (setColsSQL, setHeaders, setColDeps) <- procSetObj tn fieldInfoMap set -- check if the columns exist void $ withPathK "columns" $ indexedForM updCols $ \updCol -> askPGType fieldInfoMap updCol relInUpdErr let updColDeps = map (mkColDep DRUntyped tn) updCols deps = mkParentDep tn : beDeps ++ maybe [] snd checkExpr ++ updColDeps ++ setColDeps depHeaders = getDependentHeaders fltr reqHeaders = depHeaders `union` setHeaders updColsWithoutPreSets = updCols \\ HM.keys setColsSQL return (UpdPermInfo (HS.fromList updColsWithoutPreSets) tn be (fst <$> checkExpr) setColsSQL reqHeaders, deps) where updCols = convColSpec fieldInfoMap colSpec relInUpdErr = "relationships can't be used in update" -- TODO see TODO for PermInfo above type instance PermInfo (UpdPerm b) = UpdPermInfo b instance IsPerm (UpdPerm 'Postgres) where permAccessor = PAUpdate buildPermInfo tn fieldInfoMap (PermDef _ a _) = buildUpdPermInfo tn fieldInfoMap a addPermToMetadata permDef = tmUpdatePermissions %~ OMap.insert (_pdRole permDef) permDef type CreateDelPerm b = CreatePerm (DelPerm b) buildDelPermInfo :: (QErrM m, TableCoreInfoRM 'Postgres m) => QualifiedTable -> FieldInfoMap (FieldInfo 'Postgres) -> DelPerm 'Postgres -> m (WithDeps (DelPermInfo 'Postgres)) buildDelPermInfo tn fieldInfoMap (DelPerm fltr) = do (be, beDeps) <- withPathK "filter" $ procBoolExp tn fieldInfoMap fltr let deps = mkParentDep tn : beDeps depHeaders = getDependentHeaders fltr return (DelPermInfo tn be depHeaders, deps) -- TODO see TODO for PermInfo above type instance PermInfo (DelPerm b) = DelPermInfo b instance IsPerm (DelPerm 'Postgres) where permAccessor = PADelete buildPermInfo tn fieldInfoMap (PermDef _ a _) = buildDelPermInfo tn fieldInfoMap a addPermToMetadata permDef = tmDeletePermissions %~ OMap.insert (_pdRole permDef) permDef data SetPermComment = SetPermComment { apTable :: !QualifiedTable , apRole :: !RoleName , apPermission :: !PermType , apComment :: !(Maybe Text) } deriving (Show, Eq) $(deriveJSON (aesonDrop 2 snakeCase) ''SetPermComment) setPermCommentP1 :: (UserInfoM m, QErrM m, CacheRM m) => SetPermComment -> m () setPermCommentP1 (SetPermComment qt rn pt _) = do tabInfo <- askTabInfo qt action tabInfo where action tabInfo = case pt of PTInsert -> assertPermDefined rn PAInsert tabInfo PTSelect -> assertPermDefined rn PASelect tabInfo PTUpdate -> assertPermDefined rn PAUpdate tabInfo PTDelete -> assertPermDefined rn PADelete tabInfo setPermCommentP2 :: (QErrM m, MonadTx m) => SetPermComment -> m EncJSON setPermCommentP2 apc = do liftTx $ setPermCommentTx apc return successMsg runSetPermComment :: (QErrM m, CacheRM m, MonadTx m, UserInfoM m) => SetPermComment -> m EncJSON runSetPermComment defn = do setPermCommentP1 defn setPermCommentP2 defn setPermCommentTx :: SetPermComment -> Q.TxE QErr () setPermCommentTx (SetPermComment (QualifiedObject sn tn) rn pt comment) = Q.unitQE defaultTxErrorHandler [Q.sql| UPDATE hdb_catalog.hdb_permission SET comment = $1 WHERE table_schema = $2 AND table_name = $3 AND role_name = $4 AND perm_type = $5 |] (comment, sn, tn, rn, permTypeToCode pt) True fetchPermDef :: QualifiedTable -> RoleName -> PermType -> Q.TxE QErr (Value, Maybe Text) fetchPermDef (QualifiedObject sn tn) rn pt = first Q.getAltJ . Q.getRow <$> Q.withQE defaultTxErrorHandler [Q.sql| SELECT perm_def::json, comment FROM hdb_catalog.hdb_permission WHERE table_schema = $1 AND table_name = $2 AND role_name = $3 AND perm_type = $4 |] (sn, tn, rn, permTypeToCode pt) True