mirror of
https://github.com/hasura/graphql-engine.git
synced 2025-01-05 22:34:22 +03:00
d66abcefab
PR-URL: https://github.com/hasura/graphql-engine-mono/pull/5032 GitOrigin-RevId: 206a673f4bd59700436f6160feeb7df415f637ad
853 lines
31 KiB
Haskell
853 lines
31 KiB
Haskell
-- | Translate from the DML to the MySQL dialect.
|
|
module Hasura.Backends.MySQL.FromIr
|
|
( fromSelectRows,
|
|
mkSQLSelect,
|
|
fromRootField,
|
|
FromIr,
|
|
Error (..),
|
|
runFromIr,
|
|
)
|
|
where
|
|
|
|
import Control.Monad.Validate
|
|
import Data.HashMap.Strict qualified as HM
|
|
import Data.HashSet.InsOrd qualified as OSet
|
|
import Data.Map.Strict (Map)
|
|
import Data.Map.Strict qualified as M
|
|
import Data.Proxy
|
|
import Data.Text qualified as T
|
|
import Database.MySQL.Base.Types qualified as MySQL
|
|
import Hasura.Backends.MySQL.Instances.Types ()
|
|
import Hasura.Backends.MySQL.Types
|
|
import Hasura.Prelude hiding (GT)
|
|
import Hasura.RQL.IR qualified as IR
|
|
import Hasura.RQL.Types.Column qualified as IR
|
|
import Hasura.RQL.Types.Common qualified as IR
|
|
import Hasura.RQL.Types.Relationships.Local qualified as IR
|
|
import Hasura.SQL.Backend
|
|
|
|
data FieldSource
|
|
= ExpressionFieldSource (Aliased Expression)
|
|
| JoinFieldSource (Aliased Join)
|
|
| AggregateFieldSource [Aliased Aggregate]
|
|
deriving (Eq, Show)
|
|
|
|
-- | Most of these errors should be checked for legitimacy.
|
|
data Error
|
|
= UnsupportedOpExpG (IR.OpExpG 'MySQL Expression)
|
|
| IdentifierNotSupported
|
|
| FunctionNotSupported
|
|
| NodesUnsupportedForNow
|
|
| ConnectionsNotSupported
|
|
deriving (Show, Eq)
|
|
|
|
-- | The base monad used throughout this module for all conversion
|
|
-- functions.
|
|
--
|
|
-- It's a Validate, so it'll continue going when it encounters errors
|
|
-- to accumulate as many as possible.
|
|
--
|
|
-- It also contains a mapping from entity prefixes to counters. So if
|
|
-- my prefix is "table" then there'll be a counter that lets me
|
|
-- generate table1, table2, etc. Same for any other prefix needed
|
|
-- (e.g. names for joins).
|
|
--
|
|
-- A ReaderT is used around this in most of the module too, for
|
|
-- setting the current entity that a given field name refers to. See
|
|
-- @fromColumn@.
|
|
newtype FromIr a = FromIr
|
|
{ unFromIr :: StateT (Map Text Int) (Validate (NonEmpty Error)) a
|
|
}
|
|
deriving (Functor, Applicative, Monad, MonadValidate (NonEmpty Error))
|
|
|
|
--------------------------------------------------------------------------------
|
|
-- Runners
|
|
|
|
runFromIr :: FromIr a -> Validate (NonEmpty Error) a
|
|
runFromIr fromIr = evalStateT (unFromIr fromIr) mempty
|
|
|
|
data NameTemplate
|
|
= ArrayRelationTemplate Text
|
|
| ArrayAggregateTemplate Text
|
|
| ObjectRelationTemplate Text
|
|
| TableTemplate Text
|
|
| ForOrderAlias Text
|
|
| IndexTemplate
|
|
|
|
generateEntityAlias :: NameTemplate -> FromIr Text
|
|
generateEntityAlias template = do
|
|
FromIr (modify' (M.insertWith (+) prefix start))
|
|
i <- FromIr get
|
|
pure (prefix <> tshow (fromMaybe start (M.lookup prefix i)))
|
|
where
|
|
start = 1
|
|
prefix = T.take 20 rendered
|
|
rendered =
|
|
case template of
|
|
ArrayRelationTemplate sample -> "ar_" <> sample
|
|
ArrayAggregateTemplate sample -> "aa_" <> sample
|
|
ObjectRelationTemplate sample -> "or_" <> sample
|
|
TableTemplate sample -> "t_" <> sample
|
|
ForOrderAlias sample -> "order_" <> sample
|
|
IndexTemplate -> "idx_"
|
|
|
|
-- | This is really the start where you query the base table,
|
|
-- everything else is joins attached to it.
|
|
fromQualifiedTable :: TableName -> FromIr From
|
|
fromQualifiedTable schemadTableName@(TableName {name}) = do
|
|
alias <- generateEntityAlias (TableTemplate name)
|
|
pure
|
|
( FromQualifiedTable
|
|
( Aliased
|
|
{ aliasedThing =
|
|
schemadTableName,
|
|
aliasedAlias = alias
|
|
}
|
|
)
|
|
)
|
|
|
|
fromAlias :: From -> EntityAlias
|
|
fromAlias (FromQualifiedTable Aliased {aliasedAlias}) = EntityAlias aliasedAlias
|
|
fromAlias (FromSelect Aliased {aliasedAlias}) = EntityAlias aliasedAlias
|
|
|
|
trueExpression :: Expression
|
|
trueExpression = ValueExpression (BitValue True)
|
|
|
|
existsFieldName :: Text
|
|
existsFieldName = "exists_placeholder"
|
|
|
|
fromGExists :: IR.GExists 'MySQL Expression -> ReaderT EntityAlias FromIr Select
|
|
fromGExists IR.GExists {_geTable, _geWhere} = do
|
|
selectFrom <- lift (fromQualifiedTable _geTable)
|
|
whereExpression <-
|
|
local (const (fromAlias selectFrom)) (fromGBoolExp _geWhere)
|
|
pure
|
|
Select
|
|
{ selectOrderBy = Nothing,
|
|
selectProjections =
|
|
OSet.fromList
|
|
[ ExpressionProjection
|
|
( Aliased
|
|
{ aliasedThing = trueExpression,
|
|
aliasedAlias = existsFieldName
|
|
}
|
|
)
|
|
],
|
|
selectFrom = selectFrom,
|
|
selectGroupBy = [],
|
|
selectJoins = mempty,
|
|
selectWhere = Where [whereExpression],
|
|
selectSqlTop = NoTop,
|
|
selectSqlOffset = Nothing,
|
|
selectFinalWantedFields = Nothing
|
|
}
|
|
|
|
fromGBoolExp :: IR.GBoolExp 'MySQL Expression -> ReaderT EntityAlias FromIr Expression
|
|
fromGBoolExp = do
|
|
\case
|
|
IR.BoolAnd expressions ->
|
|
fmap AndExpression (traverse fromGBoolExp expressions)
|
|
IR.BoolOr expressions ->
|
|
fmap OrExpression (traverse fromGBoolExp expressions)
|
|
IR.BoolNot expression ->
|
|
fmap NotExpression (fromGBoolExp expression)
|
|
IR.BoolExists gExists ->
|
|
fmap ExistsExpression (fromGExists gExists)
|
|
IR.BoolField expression ->
|
|
pure expression
|
|
|
|
fromAnnBoolExp ::
|
|
IR.GBoolExp 'MySQL (IR.AnnBoolExpFld 'MySQL Expression) ->
|
|
ReaderT EntityAlias FromIr Expression
|
|
fromAnnBoolExp boolExp = do
|
|
fields <- traverse fromAnnBoolExpFld boolExp
|
|
fromGBoolExp fields
|
|
|
|
-- | For boolean operators, various comparison operators used need
|
|
-- special handling to ensure that SQL Server won't outright reject
|
|
-- the comparison. See also 'shouldCastToVarcharMax'.
|
|
fromColumnInfoForBoolExp :: IR.ColumnInfo 'MySQL -> ReaderT EntityAlias FromIr Expression
|
|
fromColumnInfoForBoolExp IR.ColumnInfo {ciColumn = column, ciType = _ciType} = do
|
|
fieldName <- columnNameToFieldName column <$> ask
|
|
pure (ColumnExpression fieldName)
|
|
|
|
fromAnnBoolExpFld ::
|
|
IR.AnnBoolExpFld 'MySQL Expression ->
|
|
ReaderT EntityAlias FromIr Expression
|
|
fromAnnBoolExpFld =
|
|
\case
|
|
IR.AVColumn columnInfo opExpGs -> do
|
|
expression <- fromColumnInfoForBoolExp columnInfo
|
|
expressions <- traverse (lift . fromOpExpG expression) opExpGs
|
|
pure (AndExpression expressions)
|
|
IR.AVRelationship IR.RelInfo {riMapping = mapping, riRTable = table} annBoolExp -> do
|
|
selectFrom <- lift (fromQualifiedTable table)
|
|
foreignKeyConditions <- fromMapping selectFrom mapping
|
|
|
|
whereExpression <-
|
|
local (const (fromAlias selectFrom)) (fromAnnBoolExp annBoolExp)
|
|
pure
|
|
( ExistsExpression
|
|
Select
|
|
{ selectOrderBy = Nothing,
|
|
selectProjections =
|
|
OSet.fromList
|
|
[ ExpressionProjection
|
|
( Aliased
|
|
{ aliasedThing = trueExpression,
|
|
aliasedAlias = existsFieldName
|
|
}
|
|
)
|
|
],
|
|
selectFrom = selectFrom,
|
|
selectGroupBy = [],
|
|
selectJoins = mempty,
|
|
selectWhere = Where (foreignKeyConditions <> [whereExpression]),
|
|
selectSqlTop = NoTop,
|
|
selectSqlOffset = Nothing,
|
|
selectFinalWantedFields = Nothing
|
|
}
|
|
)
|
|
|
|
-- | The context given by the reader is of the previous/parent
|
|
-- "remote" table. The WHERE that we're generating goes in the child,
|
|
-- "local" query. The @From@ passed in as argument is the local table.
|
|
--
|
|
-- We should hope to see e.g. "post.category = category.id" for a
|
|
-- local table of post and a remote table of category.
|
|
--
|
|
-- The left/right columns in @HashMap Column Column@ corresponds
|
|
-- to the left/right of @select ... join ...@. Therefore left=remote,
|
|
-- right=local in this context.
|
|
fromMapping ::
|
|
From ->
|
|
HashMap Column Column ->
|
|
ReaderT EntityAlias FromIr [Expression]
|
|
fromMapping localFrom = traverse columnsToEqs . HM.toList
|
|
where
|
|
columnsToEqs (remoteColumn, localColumn) = do
|
|
localFieldName <- local (const (fromAlias localFrom)) (fromColumn localColumn)
|
|
remoteFieldName <- fromColumn remoteColumn
|
|
pure
|
|
( OpExpression
|
|
EQ'
|
|
(ColumnExpression localFieldName)
|
|
(ColumnExpression remoteFieldName)
|
|
)
|
|
|
|
fromColumn :: Column -> ReaderT EntityAlias FromIr FieldName
|
|
fromColumn column = columnNameToFieldName column <$> ask
|
|
|
|
columnNameToFieldName :: Column -> EntityAlias -> FieldName
|
|
columnNameToFieldName (Column fieldName) EntityAlias {entityAliasText = fieldNameEntity} =
|
|
FieldName {fName = fieldName, fNameEntity = fieldNameEntity}
|
|
|
|
fromOpExpG :: Expression -> IR.OpExpG 'MySQL Expression -> FromIr Expression
|
|
fromOpExpG expression op =
|
|
case op of
|
|
IR.AEQ True val -> do
|
|
pure $ OpExpression EQ' expression val
|
|
_ -> refute (pure (UnsupportedOpExpG op))
|
|
|
|
data Args = Args
|
|
{ argsWhere :: Where,
|
|
argsOrderBy :: Maybe (NonEmpty OrderBy),
|
|
argsJoins :: [Join],
|
|
argsTop :: Top,
|
|
argsOffset :: Maybe Int,
|
|
argsDistinct :: Proxy (Maybe (NonEmpty FieldName)),
|
|
argsExistingJoins :: Map TableName EntityAlias
|
|
}
|
|
deriving (Show)
|
|
|
|
data UnfurledJoin = UnfurledJoin
|
|
{ unfurledJoin :: Join,
|
|
-- | Recorded if we joined onto an object relation.
|
|
unfurledObjectTableAlias :: Maybe (TableName, EntityAlias)
|
|
}
|
|
deriving (Show)
|
|
|
|
fromColumnInfo :: IR.ColumnInfo 'MySQL -> ReaderT EntityAlias FromIr FieldName
|
|
fromColumnInfo IR.ColumnInfo {ciColumn = column} =
|
|
columnNameToFieldName column <$> ask
|
|
|
|
tableNameText :: TableName -> Text
|
|
tableNameText (TableName {name}) = name
|
|
|
|
aggFieldName :: Text
|
|
aggFieldName = "agg"
|
|
|
|
-- | Unfurl the nested set of object relations (tell'd in the writer)
|
|
-- that are terminated by field name (IR.AOCColumn and
|
|
-- IR.AOCArrayAggregation).
|
|
unfurlAnnOrderByElement ::
|
|
IR.AnnotatedOrderByElement 'MySQL Expression ->
|
|
WriterT (Seq UnfurledJoin) (ReaderT EntityAlias FromIr) (FieldName, Maybe ScalarType)
|
|
unfurlAnnOrderByElement =
|
|
\case
|
|
IR.AOCColumn columnInfo -> do
|
|
fieldName <- lift (fromColumnInfo columnInfo)
|
|
pure
|
|
( fieldName,
|
|
case IR.ciType columnInfo of
|
|
IR.ColumnScalar t -> Just t
|
|
_ -> Nothing
|
|
)
|
|
IR.AOCObjectRelation IR.RelInfo {riRTable = table} annBoolExp annOrderByElementG -> do
|
|
selectFrom <- lift (lift (fromQualifiedTable table))
|
|
joinAliasEntity <-
|
|
lift (lift (generateEntityAlias (ForOrderAlias (tableNameText table))))
|
|
whereExpression <-
|
|
lift (local (const (fromAlias selectFrom)) (fromAnnBoolExp annBoolExp))
|
|
tell
|
|
( pure
|
|
UnfurledJoin
|
|
{ unfurledJoin =
|
|
Join
|
|
{ joinSelect =
|
|
Select
|
|
{ selectProjections = OSet.fromList [StarProjection],
|
|
selectSqlTop = NoTop,
|
|
selectSqlOffset = Nothing,
|
|
selectFrom = selectFrom,
|
|
selectJoins = [],
|
|
selectWhere =
|
|
Where [whereExpression],
|
|
selectOrderBy = Nothing,
|
|
selectFinalWantedFields = Nothing,
|
|
selectGroupBy = []
|
|
},
|
|
joinRightTable = fromAlias selectFrom,
|
|
joinType = OnlessJoin,
|
|
joinFieldName = name table,
|
|
joinTop = NoTop,
|
|
joinOffset = Nothing
|
|
},
|
|
unfurledObjectTableAlias = Just (table, EntityAlias joinAliasEntity)
|
|
}
|
|
)
|
|
local
|
|
(const (EntityAlias joinAliasEntity))
|
|
(unfurlAnnOrderByElement annOrderByElementG)
|
|
IR.AOCArrayAggregation IR.RelInfo {riMapping = mapping, riRTable = tableName} annBoolExp annAggregateOrderBy -> do
|
|
selectFrom <- lift (lift (fromQualifiedTable tableName))
|
|
let alias = aggFieldName
|
|
joinAliasEntity <-
|
|
lift (lift (generateEntityAlias (ForOrderAlias (tableNameText tableName))))
|
|
foreignKeyConditions <- lift (fromMapping selectFrom mapping)
|
|
whereExpression <-
|
|
lift (local (const (fromAlias selectFrom)) (fromAnnBoolExp annBoolExp))
|
|
aggregate <-
|
|
lift
|
|
( local
|
|
(const (fromAlias selectFrom))
|
|
( case annAggregateOrderBy of
|
|
IR.AAOCount -> pure (CountAggregate StarCountable)
|
|
IR.AAOOp text columnInfo -> do
|
|
fieldName <- fromColumnInfo columnInfo
|
|
pure (OpAggregate text (pure (ColumnExpression fieldName)))
|
|
)
|
|
)
|
|
tell
|
|
( pure
|
|
( UnfurledJoin
|
|
{ unfurledJoin =
|
|
Join
|
|
{ joinSelect =
|
|
Select
|
|
{ selectProjections =
|
|
OSet.fromList
|
|
[ AggregateProjection
|
|
Aliased
|
|
{ aliasedThing = aggregate,
|
|
aliasedAlias = alias
|
|
}
|
|
],
|
|
selectSqlTop = NoTop,
|
|
selectGroupBy = [],
|
|
selectFrom = selectFrom,
|
|
selectJoins = [],
|
|
selectWhere =
|
|
Where
|
|
(foreignKeyConditions <> [whereExpression]),
|
|
selectOrderBy = Nothing,
|
|
selectSqlOffset = Nothing,
|
|
selectFinalWantedFields = Nothing
|
|
},
|
|
joinFieldName = "",
|
|
joinRightTable = EntityAlias "",
|
|
joinType = OnlessJoin,
|
|
joinTop = NoTop,
|
|
joinOffset = Nothing
|
|
},
|
|
unfurledObjectTableAlias = Nothing
|
|
}
|
|
)
|
|
)
|
|
pure
|
|
( FieldName {fNameEntity = joinAliasEntity, fName = alias},
|
|
Nothing
|
|
)
|
|
|
|
-- | Produce a valid ORDER BY construct, telling about any joins
|
|
-- needed on the side.
|
|
fromAnnOrderByItemG ::
|
|
IR.AnnotatedOrderByItemG 'MySQL Expression ->
|
|
WriterT (Seq UnfurledJoin) (ReaderT EntityAlias FromIr) OrderBy
|
|
fromAnnOrderByItemG IR.OrderByItemG {obiType, obiColumn = obiColumn, obiNulls} = do
|
|
(orderByFieldName, orderByType) <- unfurlAnnOrderByElement obiColumn
|
|
let orderByNullsOrder = fromMaybe NullsAnyOrder obiNulls
|
|
orderByOrder = fromMaybe Asc obiType
|
|
pure OrderBy {..}
|
|
|
|
fromSelectArgsG :: IR.SelectArgsG 'MySQL Expression -> ReaderT EntityAlias FromIr Args
|
|
fromSelectArgsG selectArgsG = do
|
|
let argsOffset = fromIntegral <$> moffset
|
|
argsWhere <-
|
|
maybe (pure mempty) (fmap (Where . pure) . fromAnnBoolExp) mannBoolExp
|
|
argsTop <-
|
|
maybe (pure mempty) (pure . Top) mlimit
|
|
let argsDistinct = Proxy
|
|
(argsOrderBy, joins) <-
|
|
runWriterT (traverse fromAnnOrderByItemG (maybe [] toList orders))
|
|
-- Any object-relation joins that we generated, we record their
|
|
-- generated names into a mapping.
|
|
let argsExistingJoins =
|
|
M.fromList (mapMaybe unfurledObjectTableAlias (toList joins))
|
|
pure
|
|
Args
|
|
{ argsJoins = toList (fmap unfurledJoin joins),
|
|
argsOrderBy = nonEmpty argsOrderBy,
|
|
..
|
|
}
|
|
where
|
|
IR.SelectArgs
|
|
{ _saWhere = mannBoolExp,
|
|
_saLimit = mlimit,
|
|
_saOffset = moffset,
|
|
_saOrderBy = orders
|
|
} = selectArgsG
|
|
|
|
-- | Here is where we project a field as a column expression. If
|
|
-- number stringification is on, then we wrap it in a
|
|
-- 'ToStringExpression' so that it's casted when being projected.
|
|
fromAnnColumnField ::
|
|
IR.AnnColumnField 'MySQL Expression ->
|
|
ReaderT EntityAlias FromIr Expression
|
|
fromAnnColumnField annColumnField = do
|
|
fieldName <- fromColumn column
|
|
if typ == IR.ColumnScalar MySQL.Geometry
|
|
then pure $ MethodExpression (ColumnExpression fieldName) "STAsText" []
|
|
else pure (ColumnExpression fieldName)
|
|
where
|
|
IR.AnnColumnField
|
|
{ _acfColumn = column,
|
|
_acfType = typ,
|
|
_acfAsText = _asText :: Bool,
|
|
_acfArguments = _ :: Maybe Void
|
|
} = annColumnField
|
|
|
|
fromRelName :: IR.RelName -> FromIr Text
|
|
fromRelName relName =
|
|
pure (IR.relNameToTxt relName)
|
|
|
|
-- fromAggregateField :: IR.AggregateField 'MySQL -> ReaderT EntityAlias FromIr Aggregate
|
|
-- fromAggregateField aggregateField =
|
|
-- case aggregateField of
|
|
-- IR.AFExp text -> pure (TextAggregate text)
|
|
-- IR.AFCount countType -> CountAggregate <$> case countType of
|
|
-- StarCountable -> pure StarCountable
|
|
-- NonNullFieldCountable names -> NonNullFieldCountable <$> traverse fromColumn names
|
|
-- DistinctCountable names -> DistinctCountable <$> traverse fromColumn names
|
|
-- IR.AFOp _ -> error "fromAggregatefield: not implemented"
|
|
|
|
fromTableAggregateFieldG ::
|
|
(IR.FieldName, IR.TableAggregateFieldG 'MySQL Void Expression) -> ReaderT EntityAlias FromIr FieldSource
|
|
fromTableAggregateFieldG (IR.FieldName _name, _field) = error "fromTableAggregateFieldG: not implemented yet"
|
|
|
|
fieldSourceProjections :: FieldSource -> [Projection]
|
|
fieldSourceProjections =
|
|
\case
|
|
ExpressionFieldSource aliasedExpression ->
|
|
pure (ExpressionProjection aliasedExpression)
|
|
JoinFieldSource Aliased {aliasedThing = Join {..}} ->
|
|
map
|
|
( \(_left, right@(FieldName {fName})) ->
|
|
ExpressionProjection
|
|
Aliased
|
|
{ aliasedAlias = fName,
|
|
aliasedThing = ColumnExpression right
|
|
}
|
|
)
|
|
fields
|
|
where
|
|
fields =
|
|
case joinType of
|
|
ArrayJoin fs -> fs
|
|
ObjectJoin fs -> fs
|
|
ArrayAggregateJoin fs -> fs
|
|
OnlessJoin -> mempty
|
|
AggregateFieldSource aggregates -> fmap AggregateProjection aggregates
|
|
|
|
fieldSourceJoin :: FieldSource -> Maybe Join
|
|
fieldSourceJoin =
|
|
\case
|
|
JoinFieldSource aliasedJoin -> pure (aliasedThing aliasedJoin)
|
|
ExpressionFieldSource {} -> Nothing
|
|
AggregateFieldSource {} -> Nothing
|
|
|
|
fromSelectAggregate ::
|
|
Maybe (EntityAlias, HashMap Column Column) ->
|
|
IR.AnnSelectG 'MySQL (IR.TableAggregateFieldG 'MySQL Void) Expression ->
|
|
FromIr Select
|
|
fromSelectAggregate mparentRelationship annSelectG = do
|
|
selectFrom <-
|
|
case from of
|
|
IR.FromTable qualifiedObject -> fromQualifiedTable qualifiedObject
|
|
IR.FromIdentifier {} -> refute $ pure IdentifierNotSupported
|
|
IR.FromFunction {} -> refute $ pure FunctionNotSupported
|
|
_mforeignKeyConditions <- fmap (Where . fromMaybe []) $
|
|
for mparentRelationship $
|
|
\(entityAlias, mapping) ->
|
|
runReaderT (fromMapping selectFrom mapping) entityAlias
|
|
fieldSources <-
|
|
runReaderT (traverse fromTableAggregateFieldG fields) (fromAlias selectFrom)
|
|
filterExpression <-
|
|
runReaderT (fromAnnBoolExp permFilter) (fromAlias selectFrom)
|
|
Args
|
|
{ argsOrderBy,
|
|
argsWhere,
|
|
argsJoins,
|
|
argsTop,
|
|
argsDistinct = Proxy,
|
|
argsOffset
|
|
} <-
|
|
runReaderT (fromSelectArgsG args) (fromAlias selectFrom)
|
|
let selectProjections =
|
|
concatMap (toList . fieldSourceProjections) fieldSources
|
|
pure
|
|
Select
|
|
{ selectProjections = OSet.fromList selectProjections,
|
|
selectFrom = selectFrom,
|
|
selectJoins = argsJoins <> mapMaybe fieldSourceJoin fieldSources,
|
|
selectWhere = argsWhere <> Where [filterExpression],
|
|
selectOrderBy = argsOrderBy,
|
|
selectSqlOffset = argsOffset,
|
|
selectSqlTop = permissionBasedTop <> argsTop,
|
|
selectFinalWantedFields = Nothing,
|
|
selectGroupBy = []
|
|
}
|
|
where
|
|
permissionBasedTop =
|
|
maybe NoTop Top mPermLimit
|
|
IR.AnnSelectG
|
|
{ _asnFields = fields,
|
|
_asnFrom = from,
|
|
_asnPerm = perm,
|
|
_asnArgs = args,
|
|
_asnStrfyNum = _num,
|
|
_asnNamingConvention = _tCase
|
|
} = annSelectG
|
|
IR.TablePerm {_tpLimit = mPermLimit, _tpFilter = permFilter} = perm
|
|
|
|
-- _fromTableAggFieldG ::
|
|
-- (Int, (IR.FieldName, IR.TableAggregateFieldG 'MySQL Void Expression)) ->
|
|
-- Maybe (ReaderT EntityAlias FromIr (Int, (IR.FieldName, [Projection])))
|
|
-- _fromTableAggFieldG = \case
|
|
-- (index, (fieldName, IR.TAFAgg (aggregateFields :: [(IR.FieldName, IR.AggregateField 'MySQL)]))) -> Just do
|
|
-- aggregates <-
|
|
-- for aggregateFields \(fieldName', aggregateField) ->
|
|
-- fromAggregateField aggregateField <&> \aliasedThing ->
|
|
-- Aliased {aliasedAlias = IR.getFieldNameTxt fieldName', ..}
|
|
-- pure (index, (fieldName, fieldSourceProjections $ AggregateFieldSource aggregates))
|
|
-- _ -> Nothing
|
|
|
|
-- _fromTableNodesFieldG ::
|
|
-- Map TableName EntityAlias ->
|
|
-- StringifyNumbers ->
|
|
-- (Int, (IR.FieldName, IR.TableAggregateFieldG 'MySQL Void Expression)) ->
|
|
-- Maybe (ReaderT EntityAlias FromIr (Int, (IR.FieldName, [Projection])))
|
|
-- _fromTableNodesFieldG argsExistingJoins stringifyNumbers = \case
|
|
-- (index, (fieldName, IR.TAFNodes () (annFieldsG :: [(IR.FieldName, IR.AnnFieldG 'MySQL Void Expression)]))) -> Just do
|
|
-- fieldSources' <- fromAnnFieldsG argsExistingJoins stringifyNumbers `traverse` annFieldsG
|
|
-- let nodesProjections' :: [Projection] = concatMap fieldSourceProjections fieldSources'
|
|
-- pure (index, (fieldName, nodesProjections'))
|
|
-- _ -> Nothing
|
|
|
|
-- -- | Get FieldSource from a TAFExp type table aggregate field
|
|
-- _fromTableExpFieldG ::
|
|
-- (Int, (IR.FieldName, IR.TableAggregateFieldG 'MySQL Void Expression)) ->
|
|
-- Maybe (ReaderT EntityAlias FromIr (Int, [Projection]))
|
|
-- _fromTableExpFieldG = \case
|
|
-- (index, (IR.FieldName name, IR.TAFExp text)) -> Just $
|
|
-- pure
|
|
-- (index, fieldSourceProjections $
|
|
-- ExpressionFieldSource
|
|
-- Aliased
|
|
-- { aliasedThing = ValueExpression (TextValue text)
|
|
-- , aliasedAlias = name
|
|
-- })
|
|
-- _ -> Nothing
|
|
|
|
fromArrayAggregateSelectG ::
|
|
IR.AnnRelationSelectG 'MySQL (IR.AnnAggregateSelectG 'MySQL Void Expression) ->
|
|
ReaderT EntityAlias FromIr Join
|
|
fromArrayAggregateSelectG annRelationSelectG = do
|
|
fieldName <- lift (fromRelName _aarRelationshipName)
|
|
joinSelect' <- do
|
|
lhsEntityAlias <- ask
|
|
-- With this, the foreign key relations are injected automatically
|
|
-- at the right place by fromSelectAggregate.
|
|
lift (fromSelectAggregate (pure (lhsEntityAlias, mapping)) annSelectG)
|
|
alias <- lift (generateEntityAlias (ArrayAggregateTemplate fieldName))
|
|
joinOn <- fromMappingFieldNames (EntityAlias alias) mapping
|
|
pure
|
|
Join
|
|
{ joinSelect = joinSelect' {selectSqlTop = NoTop, selectSqlOffset = Nothing},
|
|
joinFieldName = "",
|
|
joinRightTable = EntityAlias "",
|
|
joinType = ArrayAggregateJoin joinOn,
|
|
joinTop = selectSqlTop joinSelect',
|
|
joinOffset = selectSqlOffset joinSelect'
|
|
}
|
|
where
|
|
IR.AnnRelationSelectG
|
|
{ _aarRelationshipName,
|
|
_aarColumnMapping = mapping :: HashMap Column Column,
|
|
_aarAnnSelect = annSelectG
|
|
} = annRelationSelectG
|
|
|
|
fromArraySelectG :: IR.ArraySelectG 'MySQL Void Expression -> ReaderT EntityAlias FromIr Join
|
|
fromArraySelectG =
|
|
\case
|
|
IR.ASSimple arrayRelationSelectG ->
|
|
fromArrayRelationSelectG arrayRelationSelectG
|
|
IR.ASAggregate arrayAggregateSelectG ->
|
|
fromArrayAggregateSelectG arrayAggregateSelectG
|
|
|
|
fromObjectRelationSelectG ::
|
|
IR.ObjectRelationSelectG 'MySQL Void Expression ->
|
|
ReaderT EntityAlias FromIr Join
|
|
fromObjectRelationSelectG annRelationSelectG = do
|
|
from <- lift $ fromQualifiedTable tableFrom
|
|
let entityAlias :: EntityAlias = fromAlias from
|
|
fieldSources <-
|
|
local
|
|
(const entityAlias)
|
|
(traverse fromAnnFieldsG fields)
|
|
let selectProjections =
|
|
concatMap (toList . fieldSourceProjections) fieldSources
|
|
filterExpression <- local (const entityAlias) (fromAnnBoolExp tableFilter)
|
|
joinOn <- fromMappingFieldNames entityAlias mapping
|
|
let joinFieldProjections =
|
|
map
|
|
( \(fieldName', _) ->
|
|
FieldNameProjection
|
|
Aliased
|
|
{ aliasedThing = fieldName',
|
|
aliasedAlias = fName fieldName'
|
|
}
|
|
)
|
|
joinOn
|
|
joinFieldName <- lift (fromRelName _aarRelationshipName)
|
|
pure
|
|
Join
|
|
{ joinSelect =
|
|
Select
|
|
{ selectOrderBy = Nothing,
|
|
selectProjections =
|
|
OSet.fromList joinFieldProjections
|
|
<> OSet.fromList selectProjections, -- Ordering is right-biased.
|
|
selectGroupBy = [],
|
|
selectFrom = from,
|
|
selectJoins = mapMaybe fieldSourceJoin fieldSources,
|
|
selectWhere = Where [filterExpression],
|
|
selectSqlTop = NoTop,
|
|
selectSqlOffset = Nothing,
|
|
selectFinalWantedFields = pure (fieldTextNames fields)
|
|
},
|
|
joinFieldName,
|
|
joinRightTable = EntityAlias "",
|
|
joinType = ObjectJoin joinOn,
|
|
joinTop = NoTop,
|
|
joinOffset = Nothing
|
|
}
|
|
where
|
|
IR.AnnObjectSelectG
|
|
{ _aosFields = fields :: IR.AnnFieldsG 'MySQL Void Expression,
|
|
_aosTableFrom = tableFrom :: TableName,
|
|
_aosTableFilter = tableFilter :: IR.AnnBoolExp 'MySQL Expression
|
|
} = annObjectSelectG
|
|
IR.AnnRelationSelectG
|
|
{ _aarRelationshipName,
|
|
_aarColumnMapping = mapping :: HashMap Column Column,
|
|
_aarAnnSelect = annObjectSelectG :: IR.AnnObjectSelectG 'MySQL Void Expression
|
|
} = annRelationSelectG
|
|
|
|
isEmptyExpression :: Expression -> Bool
|
|
isEmptyExpression (AndExpression []) = True
|
|
isEmptyExpression (OrExpression []) = True
|
|
isEmptyExpression _ = False
|
|
|
|
fromSelectRows :: IR.AnnSelectG 'MySQL (IR.AnnFieldG 'MySQL Void) Expression -> FromIr Select
|
|
fromSelectRows annSelectG = do
|
|
selectFrom <-
|
|
case from of
|
|
IR.FromTable qualifiedObject -> fromQualifiedTable qualifiedObject
|
|
IR.FromIdentifier {} -> refute $ pure IdentifierNotSupported
|
|
IR.FromFunction {} -> refute $ pure FunctionNotSupported
|
|
Args
|
|
{ argsOrderBy,
|
|
argsWhere,
|
|
argsJoins,
|
|
argsDistinct = Proxy,
|
|
argsOffset,
|
|
argsTop
|
|
} <-
|
|
runReaderT (fromSelectArgsG args) (fromAlias selectFrom)
|
|
fieldSources <-
|
|
runReaderT
|
|
(traverse fromAnnFieldsG fields)
|
|
(fromAlias selectFrom)
|
|
filterExpression <-
|
|
runReaderT (fromAnnBoolExp permFilter) (fromAlias selectFrom)
|
|
let selectProjections =
|
|
concatMap (toList . fieldSourceProjections) fieldSources
|
|
pure
|
|
Select
|
|
{ selectOrderBy = argsOrderBy,
|
|
selectGroupBy = [],
|
|
selectProjections = OSet.fromList selectProjections,
|
|
selectFrom = selectFrom,
|
|
selectJoins = argsJoins <> mapMaybe fieldSourceJoin fieldSources,
|
|
selectWhere = argsWhere <> Where ([filterExpression | not (isEmptyExpression filterExpression)]),
|
|
selectSqlOffset = argsOffset,
|
|
selectSqlTop = permissionBasedTop <> argsTop,
|
|
selectFinalWantedFields = pure (fieldTextNames fields)
|
|
}
|
|
where
|
|
permissionBasedTop =
|
|
maybe NoTop Top mPermLimit
|
|
IR.AnnSelectG
|
|
{ _asnFields = fields,
|
|
_asnFrom = from,
|
|
_asnPerm = perm,
|
|
_asnArgs = args,
|
|
_asnNamingConvention = _tCase
|
|
} = annSelectG
|
|
IR.TablePerm {_tpLimit = mPermLimit, _tpFilter = permFilter} = perm
|
|
|
|
fromArrayRelationSelectG :: IR.ArrayRelationSelectG 'MySQL Void Expression -> ReaderT EntityAlias FromIr Join
|
|
fromArrayRelationSelectG annRelationSelectG = do
|
|
joinFieldName <- lift (fromRelName _aarRelationshipName)
|
|
sel <- lift (fromSelectRows annSelectG)
|
|
joinOn <- fromMappingFieldNames (fromAlias (selectFrom sel)) mapping
|
|
let joinFieldProjections =
|
|
map
|
|
( \(fieldName', _) ->
|
|
FieldNameProjection
|
|
Aliased
|
|
{ aliasedThing = fieldName',
|
|
aliasedAlias = fName fieldName'
|
|
}
|
|
)
|
|
joinOn
|
|
pure
|
|
Join
|
|
{ joinSelect =
|
|
sel
|
|
{ selectProjections =
|
|
OSet.fromList joinFieldProjections <> selectProjections sel,
|
|
-- Above: Ordering is right-biased.
|
|
selectSqlTop = NoTop,
|
|
selectSqlOffset = Nothing
|
|
},
|
|
joinRightTable = fromAlias (selectFrom sel),
|
|
joinType = ArrayJoin joinOn,
|
|
-- Above: Needed by DataLoader to determine the type of
|
|
-- Haskell-native join to perform.
|
|
joinFieldName,
|
|
joinTop = selectSqlTop sel,
|
|
joinOffset = selectSqlOffset sel
|
|
}
|
|
where
|
|
IR.AnnRelationSelectG
|
|
{ _aarRelationshipName,
|
|
_aarColumnMapping = mapping :: HashMap Column Column,
|
|
_aarAnnSelect = annSelectG
|
|
} = annRelationSelectG
|
|
|
|
-- | The main sources of fields, either constants, fields or via joins.
|
|
fromAnnFieldsG ::
|
|
(IR.FieldName, IR.AnnFieldG 'MySQL Void Expression) ->
|
|
ReaderT EntityAlias FromIr FieldSource
|
|
fromAnnFieldsG (IR.FieldName name, field) =
|
|
case field of
|
|
IR.AFColumn annColumnField -> do
|
|
expression <- fromAnnColumnField annColumnField
|
|
pure
|
|
( ExpressionFieldSource
|
|
Aliased {aliasedThing = expression, aliasedAlias = name}
|
|
)
|
|
IR.AFExpression text ->
|
|
pure
|
|
( ExpressionFieldSource
|
|
Aliased
|
|
{ aliasedThing = ValueExpression (TextValue text),
|
|
aliasedAlias = name
|
|
}
|
|
)
|
|
IR.AFObjectRelation objectRelationSelectG ->
|
|
fmap
|
|
( \aliasedThing ->
|
|
JoinFieldSource (Aliased {aliasedThing, aliasedAlias = name})
|
|
)
|
|
(fromObjectRelationSelectG objectRelationSelectG)
|
|
IR.AFArrayRelation arraySelectG ->
|
|
fmap
|
|
( \aliasedThing ->
|
|
JoinFieldSource (Aliased {aliasedThing, aliasedAlias = name})
|
|
)
|
|
(fromArraySelectG arraySelectG)
|
|
|
|
mkSQLSelect ::
|
|
IR.JsonAggSelect ->
|
|
IR.AnnSelectG 'MySQL (IR.AnnFieldG 'MySQL Void) Expression ->
|
|
FromIr Select
|
|
mkSQLSelect jsonAggSelect annSimpleSel = do
|
|
case jsonAggSelect of
|
|
IR.JASMultipleRows -> fromSelectRows annSimpleSel
|
|
IR.JASSingleObject ->
|
|
fromSelectRows annSimpleSel <&> \sel ->
|
|
sel
|
|
{ selectSqlTop = Top 1
|
|
}
|
|
|
|
-- | Convert from the IR database query into a select.
|
|
fromRootField :: IR.QueryDB 'MySQL Void Expression -> FromIr Select
|
|
fromRootField =
|
|
\case
|
|
(IR.QDBSingleRow s) -> mkSQLSelect IR.JASSingleObject s
|
|
(IR.QDBMultipleRows s) -> mkSQLSelect IR.JASMultipleRows s
|
|
(IR.QDBAggregation s) -> fromSelectAggregate Nothing s
|
|
|
|
fromMappingFieldNames ::
|
|
EntityAlias ->
|
|
HashMap Column Column ->
|
|
ReaderT EntityAlias FromIr [(FieldName, FieldName)]
|
|
fromMappingFieldNames localFrom =
|
|
traverse
|
|
( \(remoteColumn, localColumn) -> do
|
|
localFieldName <- local (const localFrom) (fromColumn localColumn)
|
|
remoteFieldName <- fromColumn remoteColumn
|
|
pure
|
|
( (,)
|
|
(localFieldName)
|
|
(remoteFieldName)
|
|
)
|
|
)
|
|
. HM.toList
|
|
|
|
fieldTextNames :: IR.AnnFieldsG 'MySQL Void Expression -> [Text]
|
|
fieldTextNames = fmap (\(IR.FieldName name, _) -> name)
|