2019-09-30 22:50:57 +03:00
|
|
|
|
{-# LANGUAGE UndecidableInstances #-}
|
|
|
|
|
|
2019-08-28 15:19:21 +03:00
|
|
|
|
-- | Construction of multiplexed live query plans; see "Hasura.GraphQL.Execute.LiveQuery" for
|
|
|
|
|
-- details.
|
|
|
|
|
module Hasura.GraphQL.Execute.LiveQuery.Plan
|
|
|
|
|
( MultiplexedQuery
|
|
|
|
|
, mkMultiplexedQuery
|
2019-09-30 22:50:57 +03:00
|
|
|
|
, resolveMultiplexedValue
|
|
|
|
|
|
|
|
|
|
, CohortId
|
|
|
|
|
, newCohortId
|
|
|
|
|
, CohortVariables
|
|
|
|
|
, executeMultiplexedQuery
|
2019-08-28 15:19:21 +03:00
|
|
|
|
|
|
|
|
|
, LiveQueryPlan(..)
|
|
|
|
|
, ParameterizedLiveQueryPlan(..)
|
|
|
|
|
, ReusableLiveQueryPlan
|
|
|
|
|
, ValidatedQueryVariables
|
|
|
|
|
, buildLiveQueryPlan
|
|
|
|
|
, reuseLiveQueryPlan
|
2019-09-30 22:50:57 +03:00
|
|
|
|
|
|
|
|
|
, LiveQueryPlanExplanation
|
|
|
|
|
, explainLiveQueryPlan
|
2019-08-28 15:19:21 +03:00
|
|
|
|
) where
|
|
|
|
|
|
|
|
|
|
import Hasura.Prelude
|
|
|
|
|
|
|
|
|
|
import qualified Data.Aeson.Casing as J
|
|
|
|
|
import qualified Data.Aeson.Extended as J
|
|
|
|
|
import qualified Data.Aeson.TH as J
|
|
|
|
|
import qualified Data.HashMap.Strict as Map
|
|
|
|
|
import qualified Data.Text as T
|
2019-09-30 22:50:57 +03:00
|
|
|
|
import qualified Data.UUID.V4 as UUID
|
2019-08-28 15:19:21 +03:00
|
|
|
|
import qualified Database.PG.Query as Q
|
|
|
|
|
import qualified Language.GraphQL.Draft.Syntax as G
|
|
|
|
|
|
2019-09-30 22:50:57 +03:00
|
|
|
|
-- remove these when array encoding is merged
|
|
|
|
|
import qualified Database.PG.Query.PTI as PTI
|
|
|
|
|
import qualified PostgreSQL.Binary.Encoding as PE
|
|
|
|
|
|
2019-08-28 15:19:21 +03:00
|
|
|
|
import Control.Lens
|
|
|
|
|
import Data.Has
|
2019-09-30 22:50:57 +03:00
|
|
|
|
import Data.UUID (UUID)
|
2019-08-28 15:19:21 +03:00
|
|
|
|
|
|
|
|
|
import qualified Hasura.GraphQL.Resolve as GR
|
|
|
|
|
import qualified Hasura.GraphQL.Transport.HTTP.Protocol as GH
|
|
|
|
|
import qualified Hasura.GraphQL.Validate as GV
|
|
|
|
|
import qualified Hasura.SQL.DML as S
|
|
|
|
|
|
|
|
|
|
import Hasura.Db
|
2019-09-30 22:50:57 +03:00
|
|
|
|
import Hasura.EncJSON
|
2019-08-28 15:19:21 +03:00
|
|
|
|
import Hasura.RQL.Types
|
|
|
|
|
import Hasura.SQL.Error
|
|
|
|
|
import Hasura.SQL.Types
|
|
|
|
|
import Hasura.SQL.Value
|
|
|
|
|
|
|
|
|
|
-- -------------------------------------------------------------------------------------------------
|
|
|
|
|
-- Multiplexed queries
|
|
|
|
|
|
|
|
|
|
newtype MultiplexedQuery = MultiplexedQuery { unMultiplexedQuery :: Q.Query }
|
|
|
|
|
deriving (Show, Eq, Hashable, J.ToJSON)
|
|
|
|
|
|
|
|
|
|
mkMultiplexedQuery :: Q.Query -> MultiplexedQuery
|
|
|
|
|
mkMultiplexedQuery baseQuery =
|
|
|
|
|
MultiplexedQuery . Q.fromText $ foldMap Q.getQueryText [queryPrefix, baseQuery, querySuffix]
|
|
|
|
|
where
|
|
|
|
|
queryPrefix =
|
|
|
|
|
[Q.sql|
|
|
|
|
|
select
|
|
|
|
|
_subs.result_id, _fld_resp.root as result
|
|
|
|
|
from
|
|
|
|
|
unnest(
|
|
|
|
|
$1::uuid[], $2::json[]
|
|
|
|
|
) _subs (result_id, result_vars)
|
|
|
|
|
left outer join lateral
|
|
|
|
|
(
|
|
|
|
|
|]
|
|
|
|
|
|
|
|
|
|
querySuffix =
|
|
|
|
|
[Q.sql|
|
|
|
|
|
) _fld_resp ON ('true')
|
|
|
|
|
|]
|
|
|
|
|
|
2019-09-30 22:50:57 +03:00
|
|
|
|
-- | Resolves an 'GR.UnresolvedVal' by converting 'GR.UVPG' values to SQL expressions that refer to
|
|
|
|
|
-- the @result_vars@ input object, collecting variable values along the way.
|
|
|
|
|
resolveMultiplexedValue
|
|
|
|
|
:: (MonadState (GV.ReusableVariableValues, Seq (WithScalarType PGScalarValue)) m)
|
|
|
|
|
=> GR.UnresolvedVal -> m S.SQLExp
|
|
|
|
|
resolveMultiplexedValue = \case
|
|
|
|
|
GR.UVPG annPGVal -> do
|
2019-09-14 09:01:06 +03:00
|
|
|
|
let GR.AnnPGVal varM _ colVal = annPGVal
|
2019-09-30 22:50:57 +03:00
|
|
|
|
varJsonPath <- case varM of
|
|
|
|
|
Just varName -> do
|
|
|
|
|
modifying _1 $ Map.insert varName colVal
|
2019-10-03 22:35:55 +03:00
|
|
|
|
pure ["query", G.unName $ G.unVariable varName]
|
2019-09-30 22:50:57 +03:00
|
|
|
|
Nothing -> do
|
|
|
|
|
syntheticVarIndex <- gets (length . snd)
|
|
|
|
|
modifying _2 (|> colVal)
|
|
|
|
|
pure ["synthetic", T.pack $ show syntheticVarIndex]
|
|
|
|
|
pure $ fromResVars (PGTypeScalar $ pstType colVal) varJsonPath
|
2019-10-03 22:35:55 +03:00
|
|
|
|
GR.UVSessVar ty sessVar -> pure $ fromResVars ty ["session", T.toLower sessVar]
|
2019-09-14 09:01:06 +03:00
|
|
|
|
GR.UVSQL sqlExp -> pure sqlExp
|
2019-11-20 09:47:06 +03:00
|
|
|
|
GR.UVSession -> pure $ fromResVars (PGTypeScalar PGJSON) ["session"]
|
2019-08-28 15:19:21 +03:00
|
|
|
|
where
|
|
|
|
|
fromResVars ty jPath =
|
|
|
|
|
flip S.SETyAnn (S.mkTypeAnn ty) $ S.SEOpApp (S.SQLOp "#>>")
|
2020-02-04 18:34:17 +03:00
|
|
|
|
[ S.SEQIden $ S.QIden (S.QualIden (Iden "_subs") Nothing) (Iden "result_vars")
|
2019-08-28 15:19:21 +03:00
|
|
|
|
, S.SEArray $ map S.SELit jPath
|
|
|
|
|
]
|
|
|
|
|
|
2019-09-30 22:50:57 +03:00
|
|
|
|
newtype CohortId = CohortId { unCohortId :: UUID }
|
|
|
|
|
deriving (Show, Eq, Hashable, J.ToJSON, Q.FromCol)
|
|
|
|
|
|
|
|
|
|
newCohortId :: (MonadIO m) => m CohortId
|
|
|
|
|
newCohortId = CohortId <$> liftIO UUID.nextRandom
|
|
|
|
|
|
|
|
|
|
data CohortVariables
|
|
|
|
|
= CohortVariables
|
|
|
|
|
{ _cvSessionVariables :: !UserVars
|
|
|
|
|
, _cvQueryVariables :: !ValidatedQueryVariables
|
|
|
|
|
, _cvSyntheticVariables :: !ValidatedSyntheticVariables
|
|
|
|
|
-- ^ To allow more queries to be multiplexed together, we introduce “synthetic” variables for
|
|
|
|
|
-- /all/ SQL literals in a query, even if they don’t correspond to any GraphQL variable. For
|
|
|
|
|
-- example, the query
|
|
|
|
|
--
|
|
|
|
|
-- > subscription latest_tracks($condition: tracks_bool_exp!) {
|
|
|
|
|
-- > tracks(where: $tracks_bool_exp) {
|
|
|
|
|
-- > id
|
|
|
|
|
-- > title
|
|
|
|
|
-- > }
|
|
|
|
|
-- > }
|
|
|
|
|
--
|
|
|
|
|
-- might be executed with similar values for @$condition@, such as @{"album_id": {"_eq": "1"}}@
|
|
|
|
|
-- and @{"album_id": {"_eq": "2"}}@.
|
|
|
|
|
--
|
|
|
|
|
-- Normally, we wouldn’t bother parameterizing over the @1@ and @2@ literals in the resulting
|
|
|
|
|
-- query because we can’t cache that query plan (since different @$condition@ values could lead to
|
|
|
|
|
-- different SQL). However, for live queries, we can still take advantage of the similarity
|
|
|
|
|
-- between the two queries by multiplexing them together, so we replace them with references to
|
|
|
|
|
-- synthetic variables.
|
|
|
|
|
} deriving (Show, Eq, Generic)
|
|
|
|
|
instance Hashable CohortVariables
|
|
|
|
|
|
|
|
|
|
instance J.ToJSON CohortVariables where
|
|
|
|
|
toJSON (CohortVariables sessionVars queryVars syntheticVars) =
|
|
|
|
|
J.object ["session" J..= sessionVars, "query" J..= queryVars, "synthetic" J..= syntheticVars]
|
|
|
|
|
|
|
|
|
|
-- These types exist only to use the Postgres array encoding.
|
|
|
|
|
newtype CohortIdArray = CohortIdArray { unCohortIdArray :: [CohortId] }
|
|
|
|
|
deriving (Show, Eq)
|
|
|
|
|
instance Q.ToPrepArg CohortIdArray where
|
|
|
|
|
toPrepVal (CohortIdArray l) = Q.toPrepValHelper PTI.unknown encoder $ map unCohortId l
|
|
|
|
|
where
|
|
|
|
|
encoder = PE.array 2950 . PE.dimensionArray foldl' (PE.encodingArray . PE.uuid)
|
|
|
|
|
newtype CohortVariablesArray = CohortVariablesArray { unCohortVariablesArray :: [CohortVariables] }
|
|
|
|
|
deriving (Show, Eq)
|
|
|
|
|
instance Q.ToPrepArg CohortVariablesArray where
|
|
|
|
|
toPrepVal (CohortVariablesArray l) =
|
|
|
|
|
Q.toPrepValHelper PTI.unknown encoder (map J.toJSON l)
|
|
|
|
|
where
|
|
|
|
|
encoder = PE.array 114 . PE.dimensionArray foldl' (PE.encodingArray . PE.json_ast)
|
|
|
|
|
|
|
|
|
|
executeMultiplexedQuery
|
|
|
|
|
:: (MonadTx m) => MultiplexedQuery -> [(CohortId, CohortVariables)] -> m [(CohortId, EncJSON)]
|
|
|
|
|
executeMultiplexedQuery (MultiplexedQuery query) = executeQuery query
|
|
|
|
|
|
|
|
|
|
-- | Internal; used by both 'executeMultiplexedQuery' and 'explainLiveQueryPlan'.
|
|
|
|
|
executeQuery :: (MonadTx m, Q.FromRow a) => Q.Query -> [(CohortId, CohortVariables)] -> m [a]
|
|
|
|
|
executeQuery query cohorts =
|
|
|
|
|
let (cohortIds, cohortVars) = unzip cohorts
|
|
|
|
|
preparedArgs = (CohortIdArray cohortIds, CohortVariablesArray cohortVars)
|
|
|
|
|
in liftTx $ Q.listQE defaultTxErrorHandler query preparedArgs True
|
|
|
|
|
|
2019-08-28 15:19:21 +03:00
|
|
|
|
-- -------------------------------------------------------------------------------------------------
|
|
|
|
|
-- Variable validation
|
|
|
|
|
|
|
|
|
|
-- | When running multiplexed queries, we have to be especially careful about user input, since
|
|
|
|
|
-- invalid values will cause the query to fail, causing collateral damage for anyone else
|
|
|
|
|
-- multiplexed into the same query. Therefore, we pre-validate variables against Postgres by
|
|
|
|
|
-- executing a no-op query of the shape
|
|
|
|
|
--
|
|
|
|
|
-- > SELECT 'v1'::t1, 'v2'::t2, ..., 'vn'::tn
|
|
|
|
|
--
|
|
|
|
|
-- so if any variable values are invalid, the error will be caught early.
|
2019-09-30 22:50:57 +03:00
|
|
|
|
newtype ValidatedVariables f = ValidatedVariables (f TxtEncodedPGVal)
|
|
|
|
|
deriving instance (Show (f TxtEncodedPGVal)) => Show (ValidatedVariables f)
|
|
|
|
|
deriving instance (Eq (f TxtEncodedPGVal)) => Eq (ValidatedVariables f)
|
|
|
|
|
deriving instance (Hashable (f TxtEncodedPGVal)) => Hashable (ValidatedVariables f)
|
|
|
|
|
deriving instance (J.ToJSON (f TxtEncodedPGVal)) => J.ToJSON (ValidatedVariables f)
|
|
|
|
|
|
|
|
|
|
type ValidatedQueryVariables = ValidatedVariables (Map.HashMap G.Variable)
|
|
|
|
|
type ValidatedSyntheticVariables = ValidatedVariables []
|
2019-08-28 15:19:21 +03:00
|
|
|
|
|
|
|
|
|
-- | Checks if the provided arguments are valid values for their corresponding types.
|
|
|
|
|
-- Generates SQL of the format "select 'v1'::t1, 'v2'::t2 ..."
|
2019-09-30 22:50:57 +03:00
|
|
|
|
validateVariables
|
|
|
|
|
:: (Traversable f, MonadError QErr m, MonadIO m)
|
2019-08-28 15:19:21 +03:00
|
|
|
|
=> PGExecCtx
|
2019-09-30 22:50:57 +03:00
|
|
|
|
-> f (WithScalarType PGScalarValue)
|
|
|
|
|
-> m (ValidatedVariables f)
|
|
|
|
|
validateVariables pgExecCtx variableValues = do
|
|
|
|
|
let valSel = mkValidationSel $ toList variableValues
|
2019-08-28 15:19:21 +03:00
|
|
|
|
Q.Discard () <- runTx' $ liftTx $
|
|
|
|
|
Q.rawQE dataExnErrHandler (Q.fromBuilder $ toSQL valSel) [] False
|
2019-09-30 22:50:57 +03:00
|
|
|
|
pure . ValidatedVariables $ fmap (txtEncodedPGVal . pstValue) variableValues
|
2019-08-28 15:19:21 +03:00
|
|
|
|
where
|
|
|
|
|
mkExtrs = map (flip S.Extractor Nothing . toTxtValue)
|
|
|
|
|
mkValidationSel vars =
|
|
|
|
|
S.mkSelect { S.selExtr = mkExtrs vars }
|
|
|
|
|
runTx' tx = do
|
|
|
|
|
res <- liftIO $ runExceptT (runLazyTx' pgExecCtx tx)
|
|
|
|
|
liftEither res
|
|
|
|
|
|
|
|
|
|
-- Explicitly look for the class of errors raised when the format of a value provided
|
|
|
|
|
-- for a type is incorrect.
|
|
|
|
|
dataExnErrHandler = mkTxErrorHandler (has _PGDataException)
|
|
|
|
|
|
|
|
|
|
-- -------------------------------------------------------------------------------------------------
|
|
|
|
|
-- Live query plans
|
|
|
|
|
|
|
|
|
|
-- | A self-contained, ready-to-execute live query plan. Contains enough information to find an
|
|
|
|
|
-- existing poller that this can be added to /or/ to create a new poller if necessary.
|
|
|
|
|
data LiveQueryPlan
|
|
|
|
|
= LiveQueryPlan
|
|
|
|
|
{ _lqpParameterizedPlan :: !ParameterizedLiveQueryPlan
|
2019-09-30 22:50:57 +03:00
|
|
|
|
, _lqpVariables :: !CohortVariables
|
2019-08-28 15:19:21 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
data ParameterizedLiveQueryPlan
|
|
|
|
|
= ParameterizedLiveQueryPlan
|
|
|
|
|
{ _plqpRole :: !RoleName
|
|
|
|
|
, _plqpAlias :: !G.Alias
|
|
|
|
|
, _plqpQuery :: !MultiplexedQuery
|
|
|
|
|
} deriving (Show)
|
|
|
|
|
$(J.deriveToJSON (J.aesonDrop 4 J.snakeCase) ''ParameterizedLiveQueryPlan)
|
|
|
|
|
|
|
|
|
|
data ReusableLiveQueryPlan
|
|
|
|
|
= ReusableLiveQueryPlan
|
2019-09-30 22:50:57 +03:00
|
|
|
|
{ _rlqpParameterizedPlan :: !ParameterizedLiveQueryPlan
|
|
|
|
|
, _rlqpSyntheticVariableValues :: !ValidatedSyntheticVariables
|
|
|
|
|
, _rlqpQueryVariableTypes :: !GV.ReusableVariableTypes
|
2019-08-28 15:19:21 +03:00
|
|
|
|
} deriving (Show)
|
|
|
|
|
$(J.deriveToJSON (J.aesonDrop 4 J.snakeCase) ''ReusableLiveQueryPlan)
|
|
|
|
|
|
|
|
|
|
-- | Constructs a new execution plan for a live query and returns a reusable version of the plan if
|
|
|
|
|
-- possible.
|
|
|
|
|
buildLiveQueryPlan
|
|
|
|
|
:: ( MonadError QErr m
|
|
|
|
|
, MonadReader r m
|
|
|
|
|
, Has UserInfo r
|
|
|
|
|
, MonadIO m
|
|
|
|
|
)
|
|
|
|
|
=> PGExecCtx
|
|
|
|
|
-> G.Alias
|
|
|
|
|
-> GR.QueryRootFldUnresolved
|
|
|
|
|
-> Maybe GV.ReusableVariableTypes
|
|
|
|
|
-> m (LiveQueryPlan, Maybe ReusableLiveQueryPlan)
|
|
|
|
|
buildLiveQueryPlan pgExecCtx fieldAlias astUnresolved varTypes = do
|
|
|
|
|
userInfo <- asks getter
|
|
|
|
|
|
2019-09-30 22:50:57 +03:00
|
|
|
|
(astResolved, (queryVariableValues, syntheticVariableValues)) <- flip runStateT mempty $
|
|
|
|
|
GR.traverseQueryRootFldAST resolveMultiplexedValue astUnresolved
|
2019-08-28 15:19:21 +03:00
|
|
|
|
let pgQuery = mkMultiplexedQuery $ GR.toPGQuery astResolved
|
|
|
|
|
parameterizedPlan = ParameterizedLiveQueryPlan (userRole userInfo) fieldAlias pgQuery
|
|
|
|
|
|
|
|
|
|
-- We need to ensure that the values provided for variables
|
|
|
|
|
-- are correct according to Postgres. Without this check
|
|
|
|
|
-- an invalid value for a variable for one instance of the
|
|
|
|
|
-- subscription will take down the entire multiplexed query
|
2019-09-30 22:50:57 +03:00
|
|
|
|
validatedQueryVars <- validateVariables pgExecCtx queryVariableValues
|
|
|
|
|
validatedSyntheticVars <- validateVariables pgExecCtx (toList syntheticVariableValues)
|
|
|
|
|
let cohortVariables = CohortVariables (userVars userInfo) validatedQueryVars validatedSyntheticVars
|
|
|
|
|
plan = LiveQueryPlan parameterizedPlan cohortVariables
|
|
|
|
|
reusablePlan = ReusableLiveQueryPlan parameterizedPlan validatedSyntheticVars <$> varTypes
|
2019-08-28 15:19:21 +03:00
|
|
|
|
pure (plan, reusablePlan)
|
|
|
|
|
|
|
|
|
|
reuseLiveQueryPlan
|
|
|
|
|
:: (MonadError QErr m, MonadIO m)
|
|
|
|
|
=> PGExecCtx
|
|
|
|
|
-> UserVars
|
|
|
|
|
-> Maybe GH.VariableValues
|
|
|
|
|
-> ReusableLiveQueryPlan
|
|
|
|
|
-> m LiveQueryPlan
|
|
|
|
|
reuseLiveQueryPlan pgExecCtx sessionVars queryVars reusablePlan = do
|
2019-09-30 22:50:57 +03:00
|
|
|
|
let ReusableLiveQueryPlan parameterizedPlan syntheticVars queryVarTypes = reusablePlan
|
|
|
|
|
annVarVals <- GV.validateVariablesForReuse queryVarTypes queryVars
|
|
|
|
|
validatedVars <- validateVariables pgExecCtx annVarVals
|
|
|
|
|
pure $ LiveQueryPlan parameterizedPlan (CohortVariables sessionVars validatedVars syntheticVars)
|
|
|
|
|
|
|
|
|
|
data LiveQueryPlanExplanation
|
|
|
|
|
= LiveQueryPlanExplanation
|
|
|
|
|
{ _lqpeSql :: !Text
|
|
|
|
|
, _lqpePlan :: ![Text]
|
|
|
|
|
} deriving (Show)
|
|
|
|
|
$(J.deriveToJSON (J.aesonDrop 5 J.snakeCase) ''LiveQueryPlanExplanation)
|
|
|
|
|
|
|
|
|
|
explainLiveQueryPlan :: (MonadTx m, MonadIO m) => LiveQueryPlan -> m LiveQueryPlanExplanation
|
|
|
|
|
explainLiveQueryPlan plan = do
|
|
|
|
|
let parameterizedPlan = _lqpParameterizedPlan plan
|
|
|
|
|
queryText = Q.getQueryText . unMultiplexedQuery $ _plqpQuery parameterizedPlan
|
|
|
|
|
explainQuery = Q.fromText $ "EXPLAIN (FORMAT TEXT) " <> queryText
|
|
|
|
|
cohortId <- newCohortId
|
|
|
|
|
explanationLines <- map runIdentity <$> executeQuery explainQuery [(cohortId, _lqpVariables plan)]
|
|
|
|
|
pure $ LiveQueryPlanExplanation queryText explanationLines
|