mirror of
https://github.com/hasura/graphql-engine.git
synced 2025-01-05 14:27:59 +03:00
b124c943a9
PR-URL: https://github.com/hasura/graphql-engine-mono/pull/9373 Co-authored-by: Gil Mizrahi <8547573+soupi@users.noreply.github.com> Co-authored-by: Daniel Harvey <4729125+danieljharvey@users.noreply.github.com> GitOrigin-RevId: b998da0e2725f687a00c2c6c993435e44df4f70f
211 lines
7.6 KiB
Haskell
211 lines
7.6 KiB
Haskell
-- | The modules in the @Hasura.Backends.MSSQL.FromIr@ namespace translates the
|
|
-- RQL IR into TSQL, the SQL dialect of MSSQL, as defined in abstract syntax in
|
|
-- "Hasura.Backends.MSSQL.Types".
|
|
--
|
|
-- The translation happens in the @FromIr@ monad, which manages identifier
|
|
-- scoping and error collection.
|
|
--
|
|
-- The actual rendering of this AST into TSQL text happens in
|
|
-- "Hasura.Backends.MSSQL.ToQuery".
|
|
module Hasura.Backends.MSSQL.FromIr
|
|
( -- * The central Monad
|
|
FromIr,
|
|
runFromIrErrorOnCTEs,
|
|
runFromIrUseCTEs,
|
|
runFromIrUseCTEsT,
|
|
Error (..),
|
|
tellBefore,
|
|
tellAfter,
|
|
tellCTE,
|
|
|
|
-- * Name generation
|
|
NameTemplate (..),
|
|
generateAlias,
|
|
)
|
|
where
|
|
|
|
import Control.Monad.Validate
|
|
import Control.Monad.Validate qualified as V
|
|
import Control.Monad.Writer.Strict
|
|
import Data.Map.Strict (Map)
|
|
import Data.Map.Strict qualified as M
|
|
import Data.Text qualified as T
|
|
import Data.Text.Extended qualified as T
|
|
import Hasura.Backends.MSSQL.Instances.Types ()
|
|
import Hasura.Backends.MSSQL.Types.Internal as TSQL
|
|
import Hasura.Base.Error (QErr, throw500)
|
|
import Hasura.NativeQuery.Metadata (InterpolatedQuery, NativeQueryName (getNativeQueryName))
|
|
import Hasura.Prelude
|
|
import Hasura.RQL.IR qualified as IR
|
|
import Hasura.RQL.Types.BackendType
|
|
|
|
-- | Allow the query process to emit extra setup / teardown steps
|
|
data IRWriter = IRWriter
|
|
{ irwBefore :: [TempTableDDL],
|
|
irwAfter :: [TempTableDDL],
|
|
irwCTEs :: Maybe With
|
|
}
|
|
|
|
-- | Unique name counter
|
|
data IRState = IRState
|
|
{ irsCounter :: Int,
|
|
irsMap :: Map Text Int
|
|
}
|
|
|
|
instance Semigroup IRWriter where
|
|
(IRWriter a b c) <> (IRWriter a' b' c') = IRWriter (a <> a') (b' <> b) (c <> c')
|
|
|
|
instance Monoid IRWriter where
|
|
mempty = IRWriter mempty mempty Nothing
|
|
|
|
-- | add a step to be run before the main query
|
|
tellBefore :: TempTableDDL -> FromIr ()
|
|
tellBefore step =
|
|
tell (IRWriter {irwBefore = [step], irwAfter = mempty, irwCTEs = Nothing})
|
|
|
|
-- | add a step to be run after the main query
|
|
tellAfter :: TempTableDDL -> FromIr ()
|
|
tellAfter step =
|
|
tell (IRWriter {irwBefore = mempty, irwAfter = [step], irwCTEs = Nothing})
|
|
|
|
tellCTE :: NativeQueryName -> InterpolatedQuery Expression -> FromIr Text
|
|
tellCTE name cte = do
|
|
counter <- irsCounter <$> get
|
|
modify' \s -> s {irsCounter = (counter + 1)}
|
|
let alias = T.toTxt (getNativeQueryName name) <> tshow counter
|
|
tell
|
|
IRWriter
|
|
{ irwBefore = mempty,
|
|
irwAfter = mempty,
|
|
irwCTEs = Just (With $ pure $ CTEUnsafeRawSQL <$> Aliased cte alias)
|
|
}
|
|
pure alias
|
|
|
|
-- | The central Monad used throughout for all conversion functions.
|
|
--
|
|
-- It has the following features:
|
|
--
|
|
-- * It's a 'MonadValidate', so it'll continue going when it encounters 'Error's
|
|
-- to accumulate as many as possible.
|
|
--
|
|
-- * It has a facility for generating fresh, unique aliases, which lets the
|
|
-- translation output retain a resemblance with source names without the
|
|
-- translation process needing to be bothered about potential name shadowing.
|
|
-- See 'generateAlias'.
|
|
--
|
|
-- * It has a writer part for reporting native queries that need to be wrapped in a CTE
|
|
--
|
|
-- The Inner part 'FromIrInner' containing the state and validate are extracted to a different
|
|
-- type so we can peel the writer for queries and report errors in the process if needed.
|
|
newtype FromIr a = FromIr
|
|
{ unFromIr :: WriterT IRWriter FromIrInner a
|
|
}
|
|
deriving
|
|
( Functor,
|
|
Applicative,
|
|
Monad,
|
|
MonadValidate (NonEmpty Error),
|
|
MonadWriter IRWriter,
|
|
MonadState IRState
|
|
)
|
|
|
|
-- | We extract the state and validate parts of FromIr so we can peel off
|
|
-- the writer part of 'FromIr' for queries and report errors in the process if needed.
|
|
type FromIrInner = StateT IRState (Validate (NonEmpty Error))
|
|
|
|
-- | Run a 'FromIr' action, throwing errors that have been collected using the
|
|
-- supplied action, and attach CTEs created from native queries to the select query.
|
|
runFromIrUseCTEs :: (MonadError QErr m) => FromIr Select -> m (QueryWithDDL Select)
|
|
runFromIrUseCTEs fromir = runIdentity <$> runFromIr attachCTEs (Identity fromir)
|
|
|
|
-- | Run a 'FromIr' action, throwing errors that have been collected using the
|
|
-- supplied action, and attach CTEs created from native queries to the select query.
|
|
runFromIrUseCTEsT :: (Traversable t, MonadError QErr m) => t (FromIr Select) -> m (t (QueryWithDDL Select))
|
|
runFromIrUseCTEsT = runFromIr attachCTEs
|
|
|
|
-- | Run a 'FromIr' action, throwing errors that have been collected using the
|
|
-- supplied action, and discard CTEs created from native queries to the select query.
|
|
--
|
|
-- If CTEs were reported, we throw an error, since we don't support native queries
|
|
-- in this context yet.
|
|
runFromIrErrorOnCTEs :: (MonadError QErr m) => FromIr a -> m (QueryWithDDL a)
|
|
runFromIrErrorOnCTEs fromir = runIdentity <$> runFromIr errorOnCTEs (Identity fromir)
|
|
|
|
-- | Run a 'FromIr' action, throwing errors that have been collected using the supplied action.
|
|
runFromIr :: (Traversable t, MonadError QErr m) => ((a, IRWriter) -> FromIrInner (QueryWithDDL a)) -> t (FromIr a) -> m (t (QueryWithDDL a))
|
|
runFromIr toResult =
|
|
flip onLeft (throw500 . tshow)
|
|
. V.runValidate
|
|
. flip evalStateT (IRState 0 mempty)
|
|
. (traverse toResult =<<)
|
|
. traverse (runWriterT . unFromIr)
|
|
|
|
-- | attach CTEs created from native queries to the select query.
|
|
attachCTEs :: (MonadValidate (NonEmpty Error) m) => (Select, IRWriter) -> m (QueryWithDDL Select)
|
|
attachCTEs (select, IRWriter before after ctes) =
|
|
pure
|
|
$ QueryWithDDL
|
|
{ qwdBeforeSteps = before,
|
|
qwdQuery = select {selectWith = ctes <> selectWith select},
|
|
qwdAfterSteps = after
|
|
}
|
|
|
|
-- | If CTEs were reported, we throw an error, since we don't support native queries
|
|
-- in this context yet.
|
|
errorOnCTEs :: (MonadValidate (NonEmpty Error) m) => (a, IRWriter) -> m (QueryWithDDL a)
|
|
errorOnCTEs (result, IRWriter {irwBefore, irwAfter, irwCTEs}) =
|
|
case irwCTEs of
|
|
Nothing ->
|
|
pure
|
|
$ QueryWithDDL
|
|
{ qwdBeforeSteps = irwBefore,
|
|
qwdQuery = result,
|
|
qwdAfterSteps = irwAfter
|
|
}
|
|
Just _ -> refute $ pure NativeQueriesNotSupported
|
|
|
|
-- | Errors that may happen during translation.
|
|
data Error
|
|
= UnsupportedOpExpG (IR.OpExpG 'MSSQL Expression)
|
|
| FunctionNotSupported
|
|
| NativeQueriesNotSupported
|
|
deriving (Show, Eq)
|
|
|
|
-- | Hints about the type of entity that 'generateAlias' is producing an alias
|
|
-- for.
|
|
data NameTemplate
|
|
= ArrayRelationTemplate Text
|
|
| ArrayAggregateTemplate Text
|
|
| ObjectRelationTemplate Text
|
|
| TableTemplate Text
|
|
| ForOrderAlias Text
|
|
|
|
-- | Generate a fresh alias for a given entity to remove ambiguity and naming
|
|
-- conflicts between scopes at the TSQL level.
|
|
--
|
|
-- Names are generated in the form @type_name_occurrence@, where:
|
|
--
|
|
-- * @type@ hints at the type of entity,
|
|
-- * @name@ refers to the source name being aliased, and
|
|
-- * @occurrence@ is an integer counter that distinguishes each occurrence of @type_name@.
|
|
--
|
|
-- Example outputs:
|
|
--
|
|
-- > do
|
|
-- > "ar_articles_1" <- generateAlias (ArrayRelationTemplate "articles")
|
|
-- > "ar_articles_2" <- generateAlias (ArrayRelationTemplate "articles")
|
|
-- > "t_users_1" <- generateAlias (TableTemplate "users")
|
|
generateAlias :: NameTemplate -> FromIr Text
|
|
generateAlias template = do
|
|
FromIr (modify' (\s -> s {irsMap = M.insertWith (+) rendered 1 (irsMap s)}))
|
|
occurrence <- M.findWithDefault 1 rendered . irsMap <$> FromIr get
|
|
pure (rendered <> tshow occurrence)
|
|
where
|
|
rendered = T.take 20
|
|
$ case template of
|
|
ArrayRelationTemplate sample -> "ar_" <> sample
|
|
ArrayAggregateTemplate sample -> "aa_" <> sample
|
|
ObjectRelationTemplate sample -> "or_" <> sample
|
|
TableTemplate sample -> "t_" <> sample
|
|
ForOrderAlias sample -> "order_" <> sample
|