mirror of
https://github.com/hasura/graphql-engine.git
synced 2025-01-05 14:27:59 +03:00
439 lines
15 KiB
Haskell
439 lines
15 KiB
Haskell
{-# 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
|