{-# LANGUAGE DeriveLift #-} {-# LANGUAGE FlexibleContexts #-} {-# LANGUAGE GeneralizedNewtypeDeriving #-} {-# LANGUAGE MultiWayIf #-} {-# LANGUAGE OverloadedStrings #-} {-# LANGUAGE QuasiQuotes #-} {-# LANGUAGE TemplateHaskell #-} {-# LANGUAGE TypeApplications #-} {-# LANGUAGE TypeFamilies #-} module Hasura.RQL.DDL.Schema.Table where import Hasura.Prelude import Hasura.RQL.DDL.Deps import Hasura.RQL.DDL.Permission import Hasura.RQL.DDL.Permission.Internal import Hasura.RQL.DDL.QueryTemplate import Hasura.RQL.DDL.Relationship import Hasura.RQL.DDL.Schema.Diff import Hasura.RQL.DDL.Utils import Hasura.RQL.Types import Hasura.SQL.Types import qualified Database.PG.Query as Q import Data.Aeson import Data.Aeson.Casing import Data.Aeson.TH import Instances.TH.Lift () import Language.Haskell.TH.Syntax (Lift) import qualified Data.HashMap.Strict as M import qualified Data.Text as T import qualified Data.Text.Encoding as TE import qualified Database.PostgreSQL.LibPQ as PQ delTableFromCatalog :: QualifiedTable -> Q.Tx () delTableFromCatalog (QualifiedTable sn tn) = Q.unitQ [Q.sql| DELETE FROM "hdb_catalog"."hdb_table" WHERE table_schema = $1 AND table_name = $2 |] (sn, tn) False saveTableToCatalog :: QualifiedTable -> Q.Tx () saveTableToCatalog (QualifiedTable sn tn) = Q.unitQ [Q.sql| INSERT INTO "hdb_catalog"."hdb_table" VALUES ($1, $2) |] (sn, tn) False -- Build the TableInfo with all its columns getTableInfo :: QualifiedTable -> Q.TxE QErr TableInfo getTableInfo qt@(QualifiedTable sn tn) = do tableExists <- Q.catchE defaultTxErrorHandler $ Q.listQ [Q.sql| SELECT true from information_schema.tables WHERE table_schema = $1 AND table_name = $2; |] (sn, tn) False -- if no columns are found, there exists no such view/table unless (tableExists == [Identity True]) $ throw400 NotExists $ "no such table/view exists in postgres : " <>> qt -- Fetch the column details colData <- Q.catchE defaultTxErrorHandler $ Q.listQ [Q.sql| SELECT column_name, to_json(udt_name) FROM information_schema.columns WHERE table_schema = $1 AND table_name = $2 |] (sn, tn) False return $ mkTableInfo qt $ map (fmap Q.getAltJ) colData newtype TrackTable = TrackTable { tName :: QualifiedTable } deriving (Show, Eq, FromJSON, ToJSON, Lift) trackExistingTableOrViewP1 :: TrackTable -> P1 () trackExistingTableOrViewP1 (TrackTable vn) = do adminOnly rawSchemaCache <- getSchemaCache <$> lift ask when (M.member vn $ scTables rawSchemaCache) $ throw400 AlreadyTracked $ "view/table already tracked : " <>> vn trackExistingTableOrViewP2Setup :: (P2C m) => QualifiedTable -> m () trackExistingTableOrViewP2Setup tn = do ti <- liftTx $ getTableInfo tn addTableToCache ti trackExistingTableOrViewP2 :: (P2C m) => QualifiedTable -> m RespBody trackExistingTableOrViewP2 vn = do trackExistingTableOrViewP2Setup vn liftTx $ Q.catchE defaultTxErrorHandler $ saveTableToCatalog vn return successMsg instance HDBQuery TrackTable where type Phase1Res TrackTable = () phaseOne = trackExistingTableOrViewP1 phaseTwo (TrackTable tn) _ = trackExistingTableOrViewP2 tn schemaCachePolicy = SCPReload purgeDep :: (CacheRWM m, MonadError QErr m, MonadTx m) => SchemaObjId -> m () purgeDep schemaObjId = case schemaObjId of (SOTableObj tn (TOPerm rn pt)) -> do liftTx $ dropPermFromCatalog tn rn pt withPermType pt delPermFromCache rn tn (SOTableObj qt (TORel rn)) -> do liftTx $ delRelFromCatalog qt rn delFldFromCache (fromRel rn) qt (SOQTemplate qtn) -> do liftTx $ delQTemplateFromCatalog qtn delQTemplateFromCache qtn _ -> throw500 $ "unexpected dependent object : " <> reportSchemaObj schemaObjId processTableChanges :: (P2C m) => TableInfo -> TableDiff -> m () processTableChanges ti tableDiff = do when (isJust mNewName) $ throw400 NotSupported $ "table renames are not yet supported : " <>> tn -- for all the dropped columns forM_ droppedCols $ \droppedCol -> -- Drop the column from the cache delFldFromCache (fromPGCol droppedCol) tn -- In the newly added columns check that there is no conflict with relationships forM_ addedCols $ \colInfo@(PGColInfo colName _) -> case M.lookup (fromPGCol colName) $ tiFieldInfoMap ti of Just (FIRelationship _) -> throw400 AlreadyExists $ "cannot add column " <> colName <<> " in table " <> tn <<> " as a relationship with the name already exists" _ -> addFldToCache (fromPGCol colName) (FIColumn colInfo) tn sc <- askSchemaCache -- for rest of the columns forM_ alteredCols $ \(PGColInfo oColName oColTy, nci@(PGColInfo nColName nColTy)) -> if | oColName /= nColName -> throw400 NotSupported $ "column renames are not yet supported : " <> tn <<> "." <>> oColName | oColTy /= nColTy -> do let colId = SOTableObj tn $ TOCol oColName depObjs = getDependentObjsWith (== "on_type") sc colId if null depObjs then updateFldInCache oColName $ FIColumn nci else throw400 DependencyError $ "cannot change type of column " <> oColName <<> " in table " <> tn <<> " because of the following dependencies : " <> reportSchemaObjs depObjs | otherwise -> return () where updateFldInCache cn ci = do delFldFromCache (fromPGCol cn) tn addFldToCache (fromPGCol cn) ci tn tn = tiName ti TableDiff mNewName droppedCols addedCols alteredCols _ = tableDiff processSchemaChanges :: (P2C m) => SchemaDiff -> m () processSchemaChanges schemaDiff = do -- Purge the dropped tables forM_ droppedTables $ \qtn@(QualifiedTable sn tn) -> do liftTx $ Q.catchE defaultTxErrorHandler $ do Q.unitQ [Q.sql| DELETE FROM "hdb_catalog"."hdb_relationship" WHERE table_schema = $1 AND table_name = $2 |] (sn, tn) False Q.unitQ [Q.sql| DELETE FROM "hdb_catalog"."hdb_permission" WHERE table_schema = $1 AND table_name = $2 |] (sn, tn) False delTableFromCatalog qtn delTableFromCache qtn -- Get schema cache sc <- askSchemaCache forM_ alteredTables $ \(oldQtn, tableDiff) -> do ti <- case M.lookup oldQtn $ scTables sc of Just ti -> return ti Nothing -> throw500 $ "old table metadata not found in cache : " <>> oldQtn processTableChanges ti tableDiff where SchemaDiff droppedTables alteredTables = schemaDiff data UntrackTable = UntrackTable { utTable :: !QualifiedTable , utCascade :: !(Maybe Bool) } deriving (Show, Eq, Lift) $(deriveJSON (aesonDrop 2 snakeCase){omitNothingFields=True} ''UntrackTable) unTrackExistingTableOrViewP1 :: UntrackTable -> P1 (UntrackTable, TableInfo) unTrackExistingTableOrViewP1 ut@(UntrackTable vn _) = do adminOnly rawSchemaCache <- getSchemaCache <$> lift ask case M.lookup vn (scTables rawSchemaCache) of Just ti -> return (ut, ti) Nothing -> throw400 AlreadyUntracked $ "view/table already untracked : " <>> vn unTrackExistingTableOrViewP2 :: (P2C m) => UntrackTable -> TableInfo -> m RespBody unTrackExistingTableOrViewP2 (UntrackTable vn cascade) tableInfo = do sc <- askSchemaCache -- Get Foreign key constraints to this table fKeyTables <- liftTx getFKeyTables let fKeyDepIds = mkFKeyObjIds $ filterTables fKeyTables $ scTables sc -- Report back with an error if any fkey object ids are present when (fKeyDepIds /= []) $ reportDepsExt fKeyDepIds [] -- Get relational and query template dependants let allRels = getAllRelations $ scTables sc directRelDep = (vn, getRels $ tiFieldInfoMap tableInfo) relDeps = directRelDep : foldl go [] allRels relDepIds = concatMap mkObjIdFromRel relDeps queryTDepIds = getDependentObjsOfQTemplateCache (SOTable vn) (scQTemplates sc) allDepIds = relDepIds <> queryTDepIds -- Report bach with an error if cascade is not set when (allDepIds /= [] && not (or cascade)) $ reportDepsExt allDepIds [] -- Purge all the dependants from state mapM_ purgeDep allDepIds -- update the schema cache with the changes processSchemaChanges $ SchemaDiff [vn] [] return successMsg where QualifiedTable sn tn = vn getFKeyTables = Q.catchE defaultTxErrorHandler $ Q.listQ [Q.sql| SELECT constraint_name, table_schema, table_name FROM hdb_catalog.hdb_foreign_key_constraint WHERE ref_table_table_schema = $1 AND ref_table =$2 |] (sn, tn) False filterTables tables tc = flip filter tables $ \(_, s, t) -> isJust $ M.lookup (QualifiedTable s t) tc mkFKeyObjIds tables = flip map tables $ \(cn, s, t) -> SOTableObj (QualifiedTable s t) (TOCons cn) getAllRelations tc = map getRelInfo $ M.toList tc getRelInfo (qt, ti) = (qt, getRels $ tiFieldInfoMap ti) go l (qt, ris) = if any isDep ris then (qt, filter isDep ris):l else l isDep relInfo = vn == riRTable relInfo mkObjIdFromRel (qt, ris) = flip map ris $ \ri -> SOTableObj qt (TORel $ riName ri) instance HDBQuery UntrackTable where type Phase1Res UntrackTable = (UntrackTable, TableInfo) phaseOne = unTrackExistingTableOrViewP1 phaseTwo _ = uncurry unTrackExistingTableOrViewP2 schemaCachePolicy = SCPReload buildSchemaCache :: Q.TxE QErr SchemaCache buildSchemaCache = flip execStateT emptySchemaCache $ do tables <- lift $ Q.catchE defaultTxErrorHandler fetchTables forM_ tables $ \(sn, tn) -> modifyErr (\e -> "table " <> tn <<> "; " <> e) $ trackExistingTableOrViewP2Setup $ QualifiedTable sn tn -- Fetch all the relationships relationships <- lift $ Q.catchE defaultTxErrorHandler fetchRelationships forM_ relationships $ \(sn, tn, rn, rt, Q.AltJ rDef) -> modifyErr (\e -> "table " <> tn <<> "; rel " <> rn <<> "; " <> e) $ case rt of ObjRel -> do using <- decodeValue rDef objRelP2Setup (QualifiedTable sn tn) $ RelDef rn using Nothing ArrRel -> do using <- decodeValue rDef arrRelP2Setup (QualifiedTable sn tn) $ RelDef rn using Nothing -- Fetch all the permissions permissions <- lift $ Q.catchE defaultTxErrorHandler fetchPermissions forM_ permissions $ \(sn, tn, rn, pt, Q.AltJ pDef) -> modifyErr (\e -> "table " <> tn <<> "; role " <> rn <<> "; " <> e) $ case pt of PTInsert -> permHelper sn tn rn pDef PAInsert PTSelect -> permHelper sn tn rn pDef PASelect PTUpdate -> permHelper sn tn rn pDef PAUpdate PTDelete -> permHelper sn tn rn pDef PADelete -- Fetch all the query templates qtemplates <- lift $ Q.catchE defaultTxErrorHandler fetchQTemplates forM_ qtemplates $ \(qtn, Q.AltJ qtDefVal) -> do qtDef <- decodeValue qtDefVal qCtx <- mkAdminQCtx <$> get qti <- liftP1 qCtx $ createQueryTemplateP1 $ CreateQueryTemplate qtn qtDef Nothing addQTemplateToCache qti where permHelper sn tn rn pDef pa = do qCtx <- mkAdminQCtx <$> get perm <- decodeValue pDef let qt = QualifiedTable sn tn permDef = PermDef rn perm Nothing createPerm = WithTable qt permDef p1Res <- liftP1 qCtx $ phaseOne createPerm addPermP2Setup qt permDef p1Res addPermToCache qt rn pa p1Res -- p2F qt rn p1Res fetchTables = Q.listQ [Q.sql| SELECT table_schema, table_name from hdb_catalog.hdb_table |] () False fetchRelationships = Q.listQ [Q.sql| SELECT table_schema, table_name, rel_name, rel_type, rel_def::json FROM hdb_catalog.hdb_relationship |] () False fetchPermissions = Q.listQ [Q.sql| SELECT table_schema, table_name, role_name, perm_type, perm_def::json FROM hdb_catalog.hdb_permission |] () False fetchQTemplates = Q.listQ [Q.sql| SELECT template_name, template_defn :: json FROM hdb_catalog.hdb_query_template |] () False data RunSQL = RunSQL { rSql :: T.Text , rCascade :: !(Maybe Bool) } deriving (Show, Eq, Lift) $(deriveJSON (aesonDrop 1 snakeCase){omitNothingFields=True} ''RunSQL) data RunSQLRes = RunSQLRes { rrResultType :: !T.Text , rrResult :: !Value } deriving (Show, Eq) $(deriveJSON (aesonDrop 2 snakeCase){omitNothingFields=True} ''RunSQLRes) runSqlP2 :: (P2C m) => RunSQL -> m RespBody runSqlP2 (RunSQL t cascade) = do -- Drop hdb_views so no interference is caused to the sql query liftTx $ Q.catchE defaultTxErrorHandler $ Q.unitQ clearHdbViews () False -- Get the metadata before the sql query, everything, need to filter this oldMetaU <- liftTx $ Q.catchE defaultTxErrorHandler fetchTableMeta -- Run the SQL res <- liftTx $ Q.multiQE rawSqlErrHandler $ Q.fromBuilder $ TE.encodeUtf8Builder t -- Get the metadata after the sql query newMeta <- liftTx $ Q.catchE defaultTxErrorHandler fetchTableMeta sc <- askSchemaCache let existingTables = M.keys $ scTables sc oldMeta = flip filter oldMetaU $ \tm -> tmTable tm `elem` existingTables schemaDiff = getSchemaDiff oldMeta newMeta indirectDeps <- getSchemaChangeDeps schemaDiff -- Report back with an error if cascade is not set when (indirectDeps /= [] && not (or cascade)) $ reportDepsExt indirectDeps [] -- Purge all the indirect dependents from state mapM_ purgeDep indirectDeps -- update the schema cache with the changes processSchemaChanges schemaDiff postSc <- askSchemaCache -- recreate the insert permission infra forM_ (M.elems $ scTables postSc) $ \ti -> do let tn = tiName ti forM_ (M.elems $ tiRolePermInfoMap ti) $ \rpi -> maybe (return ()) (liftTx . buildInsInfra tn) $ _permIns rpi return $ encode (res :: RunSQLRes) where rawSqlErrHandler :: Q.PGTxErr -> QErr rawSqlErrHandler txe = let e = err400 PostgresError "query execution failed" in e {qeInternal = Just $ toJSON txe} instance HDBQuery RunSQL where type Phase1Res RunSQL = () phaseOne _ = adminOnly phaseTwo q _ = runSqlP2 q schemaCachePolicy = SCPReload -- Should be used only after checking the status resToCSV :: PQ.Result -> ExceptT T.Text IO [[T.Text]] resToCSV r = do nr <- liftIO $ PQ.ntuples r nc <- liftIO $ PQ.nfields r hdr <- forM [0..pred nc] $ \ic -> do colNameBS <- liftIO $ PQ.fname r ic maybe (return "unknown") decodeBS colNameBS rows <- forM [0..pred nr] $ \ir -> forM [0..pred nc] $ \ic -> do cellValBS <- liftIO $ PQ.getvalue r ir ic maybe (return "NULL") decodeBS cellValBS return $ hdr:rows where decodeBS = either (throwError . T.pack . show) return . TE.decodeUtf8' instance Q.FromRes RunSQLRes where fromRes (Q.ResultOkEmpty _) = return $ RunSQLRes "CommandOk" Null fromRes (Q.ResultOkData res) = do csvRows <- resToCSV res return $ RunSQLRes "TuplesOk" $ toJSON csvRows