mirror of
https://github.com/hasura/graphql-engine.git
synced 2025-01-05 14:27:59 +03:00
021e769235
PR-URL: https://github.com/hasura/graphql-engine-mono/pull/8312 GitOrigin-RevId: fc7c7a14115c0b228abc2f48d3cfac51d2852277
187 lines
6.7 KiB
Haskell
187 lines
6.7 KiB
Haskell
-- | Validate logical models against postgres-like flavors.
|
|
module Hasura.Backends.Postgres.Instances.LogicalModels
|
|
( validateLogicalModel,
|
|
logicalModelToPreparedStatement,
|
|
)
|
|
where
|
|
|
|
import Data.Aeson (toJSON)
|
|
import Data.Environment qualified as Env
|
|
import Data.HashMap.Strict qualified as HashMap
|
|
import Data.Map.Strict (Map)
|
|
import Data.Map.Strict qualified as Map
|
|
import Data.Set (Set)
|
|
import Data.Set qualified as Set
|
|
import Data.Text qualified as Text
|
|
import Data.Text.Extended (commaSeparated, toTxt)
|
|
import Database.PG.Query qualified as PG
|
|
import Hasura.Backends.Postgres.Connection qualified as PG
|
|
import Hasura.Backends.Postgres.Connection.Connect (withPostgresDB)
|
|
import Hasura.Backends.Postgres.Instances.Types ()
|
|
import Hasura.Backends.Postgres.SQL.Types (PGScalarType, pgScalarTypeToText)
|
|
import Hasura.Base.Error
|
|
import Hasura.CustomReturnType
|
|
import Hasura.LogicalModel.Metadata
|
|
( InterpolatedItem (..),
|
|
InterpolatedQuery (..),
|
|
LogicalModelArgumentName,
|
|
LogicalModelMetadata (..),
|
|
)
|
|
import Hasura.LogicalModel.Types (NullableScalarType (nstType), getLogicalModelName)
|
|
import Hasura.Prelude
|
|
import Hasura.SQL.Backend
|
|
|
|
-- | Prepare a logical model query against a postgres-like database to validate it.
|
|
validateLogicalModel ::
|
|
forall m pgKind.
|
|
(MonadIO m, MonadError QErr m) =>
|
|
Env.Environment ->
|
|
PG.PostgresConnConfiguration ->
|
|
LogicalModelMetadata ('Postgres pgKind) ->
|
|
m ()
|
|
validateLogicalModel env connConf model = do
|
|
preparedQuery <- logicalModelToPreparedStatement model
|
|
|
|
-- We don't need to deallocate the prepared statement because 'withPostgresDB'
|
|
-- opens a new connection, runs a statement, and then closes the connection.
|
|
-- Since a prepared statement only lasts for the duration of the session, once
|
|
-- the session closes, the prepared statement is deallocated as well.
|
|
runRaw (PG.fromText $ preparedQuery)
|
|
where
|
|
runRaw :: PG.Query -> m ()
|
|
runRaw stmt =
|
|
liftEither
|
|
=<< liftIO
|
|
( withPostgresDB
|
|
env
|
|
connConf
|
|
( PG.rawQE
|
|
( \e ->
|
|
(err400 ValidationFailed "Failed to validate query")
|
|
{ qeInternal = Just $ ExtraInternal $ toJSON e
|
|
}
|
|
)
|
|
stmt
|
|
[]
|
|
False
|
|
)
|
|
)
|
|
|
|
---------------------------------------
|
|
|
|
-- | The environment and fresh-name generator used by 'renameIQ'.
|
|
data RenamingState = RenamingState
|
|
{ rsNextFree :: Int,
|
|
rsBoundVars :: Map LogicalModelArgumentName Int
|
|
}
|
|
|
|
-- | 'Rename' an 'InterpolatedQuery' expression with 'LogicalModelArgumentName' variables
|
|
-- into one which uses ordinal arguments instead of named arguments, suitable
|
|
-- for a prepared query.
|
|
renameIQ ::
|
|
InterpolatedQuery LogicalModelArgumentName ->
|
|
( InterpolatedQuery Int,
|
|
Map Int LogicalModelArgumentName
|
|
)
|
|
renameIQ = runRenaming . fmap InterpolatedQuery . mapM renameII . getInterpolatedQuery
|
|
where
|
|
runRenaming :: forall a. State RenamingState a -> (a, Map Int LogicalModelArgumentName)
|
|
runRenaming action =
|
|
let (res, st) = runState action (RenamingState 1 mempty)
|
|
in (res, inverseMap $ rsBoundVars st)
|
|
|
|
drawFree :: State RenamingState Int
|
|
drawFree = do
|
|
i <- gets rsNextFree
|
|
modify (\s -> s {rsNextFree = i + 1})
|
|
return i
|
|
|
|
-- Rename a variable, assigning a fresh argument index when encounting new
|
|
-- variables and reusing the previously assigned indices when encountering a
|
|
-- previously treated variable accordingly.
|
|
renameII ::
|
|
InterpolatedItem LogicalModelArgumentName ->
|
|
State RenamingState (InterpolatedItem Int)
|
|
renameII = traverse \v -> do
|
|
env <- gets rsBoundVars
|
|
(Map.lookup v env)
|
|
`onNothing` ( do
|
|
i <- drawFree
|
|
modify \s -> s {rsBoundVars = Map.insert v i (rsBoundVars s)}
|
|
return i
|
|
)
|
|
|
|
-- When renaming from the named representation to the ordinal representation
|
|
-- it is convenient for the variable renaming environment to be keyed by the
|
|
-- names.
|
|
-- When subsequently rendering the prepared statement definition however, it
|
|
-- is more convenient to inspect the environment by index.
|
|
-- Therefore we invert the map as part of renaming.
|
|
inverseMap :: Ord b => Map a b -> Map b a
|
|
inverseMap = Map.fromList . map swap . Map.toList
|
|
where
|
|
swap (a, b) = (b, a)
|
|
|
|
-- | Pretty print an interpolated query with numbered parameters.
|
|
renderIQ :: InterpolatedQuery Int -> Text
|
|
renderIQ (InterpolatedQuery items) = foldMap printItem items
|
|
where
|
|
printItem :: InterpolatedItem Int -> Text
|
|
printItem (IIText t) = t
|
|
printItem (IIVariable i) = "$" <> tshow i
|
|
|
|
-----------------------------------------
|
|
|
|
-- | Convert a logical model to a prepared statement to be validate.
|
|
--
|
|
-- Used by 'validateLogicalModel'. Exported for testing.
|
|
logicalModelToPreparedStatement ::
|
|
forall m pgKind.
|
|
MonadError QErr m =>
|
|
LogicalModelMetadata ('Postgres pgKind) ->
|
|
m Text
|
|
logicalModelToPreparedStatement model = do
|
|
let name = getLogicalModelName $ _lmmRootFieldName model
|
|
let (preparedIQ, argumentMapping) = renameIQ $ _lmmCode model
|
|
logimoCode :: Text
|
|
logimoCode = renderIQ preparedIQ
|
|
prepname = "_logimo_vali_" <> toTxt name
|
|
|
|
occurringArguments, declaredArguments, undeclaredArguments :: Set LogicalModelArgumentName
|
|
occurringArguments = Set.fromList (Map.elems argumentMapping)
|
|
declaredArguments = Set.fromList $ HashMap.keys (_lmmArguments model)
|
|
undeclaredArguments = occurringArguments `Set.difference` declaredArguments
|
|
|
|
argumentTypes :: Map Int PGScalarType
|
|
argumentTypes = nstType <$> Map.fromList (HashMap.toList $ _lmmArguments model) `Map.compose` argumentMapping
|
|
|
|
argumentSignature
|
|
| argumentTypes /= mempty = "(" <> commaSeparated (pgScalarTypeToText <$> Map.elems argumentTypes) <> ")"
|
|
| otherwise = ""
|
|
|
|
returnedColumnNames :: Text
|
|
returnedColumnNames =
|
|
commaSeparated $ HashMap.keys (crtColumns (_lmmReturns model))
|
|
|
|
wrapInCTE :: Text -> Text
|
|
wrapInCTE query =
|
|
Text.intercalate
|
|
"\n"
|
|
[ "WITH " <> ctename <> " AS (",
|
|
query,
|
|
")",
|
|
"SELECT " <> returnedColumnNames,
|
|
"FROM " <> ctename
|
|
]
|
|
where
|
|
ctename = "_cte" <> prepname
|
|
|
|
preparedQuery = "PREPARE " <> prepname <> argumentSignature <> " AS " <> wrapInCTE logimoCode
|
|
|
|
when (Set.empty /= undeclaredArguments) $
|
|
throwError $
|
|
err400 ValidationFailed $
|
|
"Undeclared arguments: " <> commaSeparated (map tshow $ Set.toList undeclaredArguments)
|
|
|
|
return preparedQuery
|