module Hasura.RQL.DML.Internal where import qualified Database.PG.Query as Q import qualified Hasura.SQL.DML as S import Hasura.Prelude import Hasura.RQL.GBoolExp import Hasura.RQL.Types import Hasura.SQL.Error import Hasura.SQL.Types import Hasura.SQL.Value import Control.Lens import Data.Aeson.Types import qualified Data.HashMap.Strict as M import qualified Data.HashSet as HS import qualified Data.Sequence as DS import qualified Data.Text as T newtype DMLP1 a = DMLP1 {unDMLP1 :: StateT (DS.Seq Q.PrepArg) P1 a} deriving ( Functor, Applicative , Monad , MonadState (DS.Seq Q.PrepArg) , MonadError QErr ) liftDMLP1 :: (QErrM m, UserInfoM m, CacheRM m, HasSQLGenCtx m) => DMLP1 a -> m (a, DS.Seq Q.PrepArg) liftDMLP1 = liftP1 . flip runStateT DS.empty . unDMLP1 instance CacheRM DMLP1 where askSchemaCache = DMLP1 $ lift askSchemaCache instance UserInfoM DMLP1 where askUserInfo = DMLP1 $ lift askUserInfo instance HasSQLGenCtx DMLP1 where askSQLGenCtx = DMLP1 $ lift askSQLGenCtx mkAdminRolePermInfo :: TableInfo PGColumnInfo -> RolePermInfo mkAdminRolePermInfo ti = RolePermInfo (Just i) (Just s) (Just u) (Just d) where pgCols = map pgiColumn $ getCols $ _tiFieldInfoMap ti tn = _tiName ti i = InsPermInfo (HS.fromList pgCols) tn annBoolExpTrue M.empty [] s = SelPermInfo (HS.fromList pgCols) tn annBoolExpTrue Nothing True [] u = UpdPermInfo (HS.fromList pgCols) tn annBoolExpTrue M.empty [] d = DelPermInfo tn annBoolExpTrue [] askPermInfo' :: (UserInfoM m) => PermAccessor c -> TableInfo PGColumnInfo -> m (Maybe c) askPermInfo' pa tableInfo = do roleName <- askCurRole let mrpi = getRolePermInfo roleName return $ mrpi >>= (^. permAccToLens pa) where rpim = _tiRolePermInfoMap tableInfo getRolePermInfo roleName | roleName == adminRole = Just $ mkAdminRolePermInfo tableInfo | otherwise = M.lookup roleName rpim askPermInfo :: (UserInfoM m, QErrM m) => PermAccessor c -> TableInfo PGColumnInfo -> m c askPermInfo pa tableInfo = do roleName <- askCurRole mPermInfo <- askPermInfo' pa tableInfo case mPermInfo of Just c -> return c Nothing -> throw400 PermissionDenied $ mconcat [ pt <> " on " <>> _tiName tableInfo , " for role " <>> roleName , " is not allowed. " ] where pt = permTypeToCode $ permAccToType pa isTabUpdatable :: RoleName -> TableInfo PGColumnInfo -> Bool isTabUpdatable role ti | role == adminRole = True | otherwise = isJust $ M.lookup role rpim >>= _permUpd where rpim = _tiRolePermInfoMap ti askInsPermInfo :: (UserInfoM m, QErrM m) => TableInfo PGColumnInfo -> m InsPermInfo askInsPermInfo = askPermInfo PAInsert askSelPermInfo :: (UserInfoM m, QErrM m) => TableInfo PGColumnInfo -> m SelPermInfo askSelPermInfo = askPermInfo PASelect askUpdPermInfo :: (UserInfoM m, QErrM m) => TableInfo PGColumnInfo -> m UpdPermInfo askUpdPermInfo = askPermInfo PAUpdate askDelPermInfo :: (UserInfoM m, QErrM m) => TableInfo PGColumnInfo -> m DelPermInfo askDelPermInfo = askPermInfo PADelete verifyAsrns :: (MonadError QErr m) => [a -> m ()] -> [a] -> m () verifyAsrns preds xs = indexedForM_ xs $ \a -> mapM_ ($ a) preds checkSelOnCol :: (UserInfoM m, QErrM m) => SelPermInfo -> PGCol -> m () checkSelOnCol selPermInfo = checkPermOnCol PTSelect (spiCols selPermInfo) checkPermOnCol :: (UserInfoM m, QErrM m) => PermType -> HS.HashSet PGCol -> PGCol -> m () checkPermOnCol pt allowedCols pgCol = do roleName <- askCurRole unless (HS.member pgCol allowedCols) $ throw400 PermissionDenied $ permErrMsg roleName where permErrMsg roleName | roleName == adminRole = "no such column exists : " <>> pgCol | otherwise = mconcat [ "role " <>> roleName , " does not have permission to " , permTypeToCode pt <> " column " <>> pgCol ] binRHSBuilder :: PGColumnType -> Value -> DMLP1 S.SQLExp binRHSBuilder colType val = do preparedArgs <- get scalarValue <- parsePGScalarValue colType val put (preparedArgs DS.|> toBinaryValue scalarValue) return $ toPrepParam (DS.length preparedArgs + 1) (pstType scalarValue) fetchRelTabInfo :: (QErrM m, CacheRM m) => QualifiedTable -> m (TableInfo PGColumnInfo) fetchRelTabInfo refTabName = -- Internal error modifyErrAndSet500 ("foreign " <> ) $ askTabInfo refTabName type SessVarBldr m = PGType PGScalarType -> SessVar -> m S.SQLExp fetchRelDet :: (UserInfoM m, QErrM m, CacheRM m) => RelName -> QualifiedTable -> m (FieldInfoMap PGColumnInfo, SelPermInfo) fetchRelDet relName refTabName = do roleName <- askCurRole -- Internal error refTabInfo <- fetchRelTabInfo refTabName -- Get the correct constraint that applies to the given relationship refSelPerm <- modifyErr (relPermErr refTabName roleName) $ askSelPermInfo refTabInfo return (_tiFieldInfoMap refTabInfo, refSelPerm) where relPermErr rTable roleName _ = mconcat [ "role " <>> roleName , " does not have permission to read relationship " <>> relName , "; no permission on" , " table " <>> rTable ] checkOnColExp :: (UserInfoM m, QErrM m, CacheRM m) => SelPermInfo -> SessVarBldr m -> AnnBoolExpFldSQL -> m AnnBoolExpFldSQL checkOnColExp spi sessVarBldr annFld = case annFld of AVCol colInfo _ -> do let cn = pgiColumn colInfo checkSelOnCol spi cn return annFld AVRel relInfo nesAnn -> do relSPI <- snd <$> fetchRelDet (riName relInfo) (riRTable relInfo) modAnn <- checkSelPerm relSPI sessVarBldr nesAnn resolvedFltr <- convAnnBoolExpPartialSQL sessVarBldr $ spiFilter relSPI return $ AVRel relInfo $ andAnnBoolExps modAnn resolvedFltr convAnnBoolExpPartialSQL :: (Applicative f) => SessVarBldr f -> AnnBoolExpPartialSQL -> f AnnBoolExpSQL convAnnBoolExpPartialSQL f = traverseAnnBoolExp (convPartialSQLExp f) convPartialSQLExp :: (Applicative f) => SessVarBldr f -> PartialSQLExp -> f S.SQLExp convPartialSQLExp f = \case PSESQLExp sqlExp -> pure sqlExp PSESessVar colTy sessVar -> f colTy sessVar sessVarFromCurrentSetting :: (Applicative f) => PGType PGScalarType -> SessVar -> f S.SQLExp sessVarFromCurrentSetting pgType sessVar = pure $ sessVarFromCurrentSetting' pgType sessVar sessVarFromCurrentSetting' :: PGType PGScalarType -> SessVar -> S.SQLExp sessVarFromCurrentSetting' ty sessVar = flip S.SETyAnn (S.mkTypeAnn ty) $ case ty of PGTypeScalar baseTy -> withConstructorFn baseTy sessVarVal PGTypeArray _ -> sessVarVal where curSess = S.SEUnsafe "current_setting('hasura.user')::json" sessVarVal = S.SEOpApp (S.SQLOp "->>") [curSess, S.SELit $ T.toLower sessVar] checkSelPerm :: (UserInfoM m, QErrM m, CacheRM m) => SelPermInfo -> SessVarBldr m -> AnnBoolExpSQL -> m AnnBoolExpSQL checkSelPerm spi sessVarBldr = traverse (checkOnColExp spi sessVarBldr) convBoolExp :: (UserInfoM m, QErrM m, CacheRM m) => FieldInfoMap PGColumnInfo -> SelPermInfo -> BoolExp -> SessVarBldr m -> (PGColumnType -> Value -> m S.SQLExp) -> m AnnBoolExpSQL convBoolExp cim spi be sessVarBldr prepValBldr = do abe <- annBoolExp rhsParser cim $ unBoolExp be checkSelPerm spi sessVarBldr abe where rhsParser pgType val = case pgType of PGTypeScalar ty -> prepValBldr ty val PGTypeArray ofTy -> do -- for arrays, we don't use the prepared builder vals <- runAesonParser parseJSON val WithScalarType scalarType scalarValues <- parsePGScalarValues ofTy vals return $ S.SETyAnn (S.SEArray $ map (toTxtValue . WithScalarType scalarType) scalarValues) (S.mkTypeAnn $ PGTypeArray scalarType) dmlTxErrorHandler :: Q.PGTxErr -> QErr dmlTxErrorHandler = mkTxErrorHandler $ \case PGIntegrityConstraintViolation _ -> True PGDataException _ -> True PGSyntaxErrorOrAccessRuleViolation (Just (PGErrorSpecific code)) -> code `elem` [ PGUndefinedObject , PGInvalidColumnReference ] _ -> False toJSONableExp :: Bool -> PGColumnType -> S.SQLExp -> S.SQLExp toJSONableExp strfyNum colTy expn | isScalarColumnWhere isGeoType colTy = S.SEFnApp "ST_AsGeoJSON" [ expn , S.SEUnsafe "15" -- max decimal digits , S.SEUnsafe "4" -- to print out crs ] Nothing `S.SETyAnn` S.jsonTypeAnn | isScalarColumnWhere isBigNum colTy && strfyNum = expn `S.SETyAnn` S.textTypeAnn | otherwise = expn -- validate headers validateHeaders :: (UserInfoM m, QErrM m) => [T.Text] -> m () validateHeaders depHeaders = do headers <- getVarNames . userVars <$> askUserInfo forM_ depHeaders $ \hdr -> unless (hdr `elem` map T.toLower headers) $ throw400 NotFound $ hdr <<> " header is expected but not found" -- validate limit and offset int values onlyPositiveInt :: MonadError QErr m => Int -> m () onlyPositiveInt i = when (i < 0) $ throw400 NotSupported "unexpected negative value"