mirror of
https://github.com/hasura/graphql-engine.git
synced 2024-12-17 12:31:52 +03:00
204cd3514b
1. Reuses postgres connections during startup which reduces the overhead of opening and closing connections. 2. Faster schema cache building. This is done by fetching all the required data in a single sql statement.
188 lines
6.5 KiB
Haskell
188 lines
6.5 KiB
Haskell
module Hasura.RQL.DDL.Schema.Function where
|
|
|
|
import Hasura.EncJSON
|
|
import Hasura.GraphQL.Utils (isValidName, showNames)
|
|
import Hasura.Prelude
|
|
import Hasura.RQL.Types
|
|
import Hasura.SQL.Types
|
|
|
|
import Data.Aeson
|
|
import Data.Aeson.Casing
|
|
import Data.Aeson.TH
|
|
import Language.Haskell.TH.Syntax (Lift)
|
|
|
|
import qualified Hasura.GraphQL.Schema as GS
|
|
import qualified Language.GraphQL.Draft.Syntax as G
|
|
|
|
import qualified Data.HashMap.Strict as M
|
|
import qualified Data.Sequence as Seq
|
|
import qualified Data.Text as T
|
|
import qualified Database.PG.Query as Q
|
|
|
|
|
|
data PGTypType
|
|
= PTBASE
|
|
| PTCOMPOSITE
|
|
| PTDOMAIN
|
|
| PTENUM
|
|
| PTRANGE
|
|
| PTPSUEDO
|
|
deriving (Show, Eq)
|
|
$(deriveJSON defaultOptions{constructorTagModifier = drop 2} ''PGTypType)
|
|
|
|
data RawFuncInfo
|
|
= RawFuncInfo
|
|
{ rfiHasVariadic :: !Bool
|
|
, rfiFunctionType :: !FunctionType
|
|
, rfiReturnTypeSchema :: !SchemaName
|
|
, rfiReturnTypeName :: !T.Text
|
|
, rfiReturnTypeType :: !PGTypType
|
|
, rfiReturnsSet :: !Bool
|
|
, rfiInputArgTypes :: ![PGColType]
|
|
, rfiInputArgNames :: ![T.Text]
|
|
, rfiReturnsTable :: !Bool
|
|
} deriving (Show, Eq)
|
|
$(deriveJSON (aesonDrop 3 snakeCase) ''RawFuncInfo)
|
|
|
|
mkFunctionArgs :: [PGColType] -> [T.Text] -> [FunctionArg]
|
|
mkFunctionArgs tys argNames =
|
|
bool withNames withNoNames $ null argNames
|
|
where
|
|
withNoNames = flip map tys $ \ty -> FunctionArg Nothing ty
|
|
withNames = zipWith mkArg argNames tys
|
|
|
|
mkArg "" ty = FunctionArg Nothing ty
|
|
mkArg n ty = flip FunctionArg ty $ Just $ FunctionArgName n
|
|
|
|
validateFuncArgs :: MonadError QErr m => [FunctionArg] -> m ()
|
|
validateFuncArgs args =
|
|
unless (null invalidArgs) $ throw400 NotSupported $
|
|
"arguments: " <> showNames invalidArgs
|
|
<> " are not in compliance with GraphQL spec"
|
|
where
|
|
funcArgsText = mapMaybe (fmap getFuncArgNameTxt . faName) args
|
|
invalidArgs = filter (not . isValidName) $ map G.Name funcArgsText
|
|
|
|
mkFunctionInfo
|
|
:: QErrM m => QualifiedFunction -> RawFuncInfo -> m FunctionInfo
|
|
mkFunctionInfo qf rawFuncInfo = do
|
|
-- throw error if function has variadic arguments
|
|
when hasVariadic $ throw400 NotSupported "function with \"VARIADIC\" parameters are not supported"
|
|
-- throw error if return type is not composite type
|
|
when (retTyTyp /= PTCOMPOSITE) $ throw400 NotSupported "function does not return a \"COMPOSITE\" type"
|
|
-- throw error if function do not returns SETOF
|
|
unless retSet $ throw400 NotSupported "function does not return a SETOF"
|
|
-- throw error if return type is not a valid table
|
|
unless returnsTab $ throw400 NotSupported "function does not return a SETOF table"
|
|
-- throw error if function type is VOLATILE
|
|
when (funTy == FTVOLATILE) $ throw400 NotSupported "function of type \"VOLATILE\" is not supported now"
|
|
|
|
let funcArgs = mkFunctionArgs inpArgTyps inpArgNames
|
|
validateFuncArgs funcArgs
|
|
|
|
let funcArgsSeq = Seq.fromList funcArgs
|
|
dep = SchemaDependency (SOTable retTable) "table"
|
|
retTable = QualifiedObject retSn (TableName retN)
|
|
return $ FunctionInfo qf False funTy funcArgsSeq retTable [dep]
|
|
where
|
|
RawFuncInfo hasVariadic funTy retSn retN retTyTyp
|
|
retSet inpArgTyps inpArgNames returnsTab
|
|
= rawFuncInfo
|
|
|
|
saveFunctionToCatalog :: QualifiedFunction -> Bool -> Q.TxE QErr ()
|
|
saveFunctionToCatalog (QualifiedObject sn fn) isSystemDefined =
|
|
Q.unitQE defaultTxErrorHandler [Q.sql|
|
|
INSERT INTO "hdb_catalog"."hdb_function" VALUES ($1, $2, $3)
|
|
|] (sn, fn, isSystemDefined) False
|
|
|
|
delFunctionFromCatalog :: QualifiedFunction -> Q.TxE QErr ()
|
|
delFunctionFromCatalog (QualifiedObject sn fn) =
|
|
Q.unitQE defaultTxErrorHandler [Q.sql|
|
|
DELETE FROM hdb_catalog.hdb_function
|
|
WHERE function_schema = $1
|
|
AND function_name = $2
|
|
|] (sn, fn) False
|
|
|
|
newtype TrackFunction
|
|
= TrackFunction
|
|
{ tfName :: QualifiedFunction}
|
|
deriving (Show, Eq, FromJSON, ToJSON, Lift)
|
|
|
|
trackFunctionP1
|
|
:: (CacheRM m, UserInfoM m, QErrM m) => TrackFunction -> m ()
|
|
trackFunctionP1 (TrackFunction qf) = do
|
|
adminOnly
|
|
rawSchemaCache <- askSchemaCache
|
|
when (M.member qf $ scFunctions rawSchemaCache) $
|
|
throw400 AlreadyTracked $ "function already tracked : " <>> qf
|
|
|
|
trackFunctionP2Setup :: (QErrM m, CacheRWM m, MonadTx m)
|
|
=> QualifiedFunction -> RawFuncInfo -> m ()
|
|
trackFunctionP2Setup qf rawfi = do
|
|
fi <- mkFunctionInfo qf rawfi
|
|
let retTable = fiReturnType fi
|
|
err = err400 NotExists $ "table " <> retTable <<> " is not tracked"
|
|
sc <- askSchemaCache
|
|
void $ liftMaybe err $ M.lookup retTable $ scTables sc
|
|
addFunctionToCache fi
|
|
|
|
trackFunctionP2 :: (QErrM m, CacheRWM m, MonadTx m)
|
|
=> QualifiedFunction -> m EncJSON
|
|
trackFunctionP2 qf = do
|
|
sc <- askSchemaCache
|
|
let defGCtx = scDefaultRemoteGCtx sc
|
|
funcNameGQL = GS.qualObjectToName qf
|
|
-- check function name is in compliance with GraphQL spec
|
|
unless (isValidName funcNameGQL) $ throw400 NotSupported $
|
|
"function name " <> qf <<> " is not in compliance with GraphQL spec"
|
|
-- check for conflicts in remote schema
|
|
GS.checkConflictingNode defGCtx funcNameGQL
|
|
|
|
-- fetch function info
|
|
functionInfos <- liftTx fetchFuncDets
|
|
rawfi <- case functionInfos of
|
|
[] ->
|
|
throw400 NotExists $ "no such function exists in postgres : " <>> qf
|
|
[rawfi] -> return rawfi
|
|
_ ->
|
|
throw400 NotSupported $
|
|
"function " <> qf <<> " is overloaded. Overloaded functions are not supported"
|
|
trackFunctionP2Setup qf rawfi
|
|
liftTx $ saveFunctionToCatalog qf False
|
|
return successMsg
|
|
where
|
|
QualifiedObject sn fn = qf
|
|
fetchFuncDets = map (Q.getAltJ . runIdentity) <$>
|
|
Q.listQE defaultTxErrorHandler [Q.sql|
|
|
SELECT function_info
|
|
FROM hdb_catalog.hdb_function_info_agg
|
|
WHERE function_schema = $1
|
|
AND function_name = $2
|
|
|] (sn, fn) True
|
|
|
|
runTrackFunc
|
|
:: ( QErrM m, CacheRWM m, MonadTx m
|
|
, UserInfoM m
|
|
)
|
|
=> TrackFunction -> m EncJSON
|
|
runTrackFunc q = do
|
|
trackFunctionP1 q
|
|
trackFunctionP2 $ tfName q
|
|
|
|
newtype UnTrackFunction
|
|
= UnTrackFunction
|
|
{ utfName :: QualifiedFunction }
|
|
deriving (Show, Eq, FromJSON, ToJSON, Lift)
|
|
|
|
runUntrackFunc
|
|
:: ( QErrM m, CacheRWM m, MonadTx m
|
|
, UserInfoM m
|
|
)
|
|
=> UnTrackFunction -> m EncJSON
|
|
runUntrackFunc (UnTrackFunction qf) = do
|
|
adminOnly
|
|
void $ askFunctionInfo qf
|
|
liftTx $ delFunctionFromCatalog qf
|
|
delFunctionFromCache qf
|
|
return successMsg
|