mirror of
https://github.com/hasura/graphql-engine.git
synced 2025-01-06 23:56:36 +03:00
c945b2d391
PR-URL: https://github.com/hasura/graphql-engine-mono/pull/4267 GitOrigin-RevId: 2d93c35a7e34dbada3b72aabcae5fc2858bbfc29
280 lines
10 KiB
Haskell
280 lines
10 KiB
Haskell
-- | Definition of all supported GraphQL directives.
|
|
module Hasura.GraphQL.Parser.Directives
|
|
( -- list of directives, for the schema
|
|
directivesInfo,
|
|
inclusionDirectives,
|
|
customDirectives,
|
|
-- Custom Directive Types
|
|
CachedDirective (..),
|
|
DirectiveMap,
|
|
-- lookup keys for directives
|
|
include,
|
|
skip,
|
|
cached,
|
|
multipleRootFields,
|
|
-- parsing utilities
|
|
parseDirectives,
|
|
withDirective,
|
|
-- exposed for tests
|
|
Directive (..),
|
|
DirectiveKey (..),
|
|
skipDirective,
|
|
includeDirective,
|
|
cachedDirective,
|
|
multipleRootFieldsDirective,
|
|
)
|
|
where
|
|
|
|
import Data.Dependent.Map qualified as DM
|
|
import Data.Dependent.Sum (DSum (..))
|
|
import Data.GADT.Compare.Extended
|
|
import Data.HashMap.Strict.Extended qualified as M
|
|
import Data.HashSet qualified as S
|
|
import Data.List.Extended (duplicates)
|
|
import Data.Parser.JSONPath
|
|
import Data.Text.Extended
|
|
import Data.Typeable (eqT)
|
|
import Hasura.GraphQL.Parser.Class
|
|
import Hasura.GraphQL.Parser.Constants qualified as G
|
|
import Hasura.GraphQL.Parser.Internal.Input
|
|
import Hasura.GraphQL.Parser.Internal.Scalars
|
|
import Hasura.GraphQL.Parser.Schema
|
|
import Hasura.Prelude
|
|
import Language.GraphQL.Draft.Syntax qualified as G
|
|
import Type.Reflection (Typeable, typeRep, (:~:) (..))
|
|
|
|
-- | Returns the schema information for all supported directives, for each of
|
|
-- which a @DirectiveDefinition@ will be inserted into the schema.
|
|
--
|
|
-- To add a new custom directives to the schema, add it to @customDirectives@.
|
|
-- Inclusion directives are treated separately, as they are not concerned with
|
|
-- altering the result of the parsers, but about whether a given field should be
|
|
-- parsed at all (see @collectFields@ for more information).
|
|
--
|
|
-- Alternatively, you could manually add a @DirectiveInfo@ to this list;
|
|
-- however, if you do so, and do not provide a corresponding definition in
|
|
-- @customDirectives@, calls to @parseDirectives@ will NOT attempt to parse that
|
|
-- new directive: they will only validate that it is used correctly. You can use
|
|
-- this option to implement directives that do not match our simple inclusion /
|
|
-- custom dichotomy.
|
|
--
|
|
-- To create a directive definition, you need to create a @Directive@ using
|
|
-- @mkDirective@, and export from this module its corresponding @DirectiveKey@
|
|
-- (see @skip@ below for an example). Make sure the type in the key matches the
|
|
-- type of the parser!
|
|
--
|
|
-- Directives may be "hidden", in which case they won't advertised in the
|
|
-- schema, but silently accepted. This is un-advisable and should only be used
|
|
-- when there's no other way around it.
|
|
directivesInfo :: forall m. MonadParse m => [DirectiveInfo]
|
|
directivesInfo = do
|
|
dir <- inclusionDirectives @m <> customDirectives @m
|
|
guard $ dAdvertised dir
|
|
pure $ dDefinition dir
|
|
|
|
-- | Not exported, only used internally; identical to 'directivesInfo', but also
|
|
-- contains hidden directives.
|
|
allDirectives :: forall m. MonadParse m => [DirectiveInfo]
|
|
allDirectives = map dDefinition $ inclusionDirectives @m <> customDirectives @m
|
|
|
|
inclusionDirectives :: forall m. MonadParse m => [Directive m]
|
|
inclusionDirectives = [includeDirective @m, skipDirective @m]
|
|
|
|
customDirectives :: forall m. MonadParse m => [Directive m]
|
|
customDirectives = [cachedDirective @m, multipleRootFieldsDirective @m]
|
|
|
|
-- | Parses directives, given a location. Ensures that all directives are known
|
|
-- and match the location; subsequently builds a dependent map of the results,
|
|
-- that can be then introspected with @withDirective@. The list of parsers that
|
|
-- should be applied is given as an argument: if a valid directive is found, but
|
|
-- for which no parser is provided, it will be ignored.
|
|
--
|
|
-- Example use:
|
|
--
|
|
-- dMap <- parseDirectives customDirectives (DLExecutable EDLQUERY) directives
|
|
-- withDirective dMap cached $ onJust \_ -> tagAsCached
|
|
parseDirectives ::
|
|
forall m.
|
|
MonadParse m =>
|
|
[Directive m] ->
|
|
G.DirectiveLocation ->
|
|
[G.Directive Variable] ->
|
|
m DirectiveMap
|
|
parseDirectives directiveParsers location givenDirectives = do
|
|
result <-
|
|
catMaybes <$> for givenDirectives \directive -> do
|
|
let name = G._dName directive
|
|
-- check the directive has a matching definition
|
|
DirectiveInfo {diLocations} <-
|
|
find (\di -> diName di == name) (allDirectives @m)
|
|
`onNothing` parseError ("directive " <> name <<> " is not defined in the schema")
|
|
-- check that it is allowed at the current location
|
|
unless (location `elem` diLocations) $
|
|
parseError $ "directive " <> name <<> " is not allowed on " <> humanReadable location
|
|
-- if we are expecting to parse it now, create a dmap entry
|
|
case find (\d -> diName (dDefinition d) == name) directiveParsers of
|
|
Nothing -> pure Nothing
|
|
Just (Directive {dParser}) -> do
|
|
result <- dParser directive
|
|
pure $ Just (name, DirectiveKey name :=> pure result)
|
|
-- check that the result does not contain duplicates
|
|
let dups = duplicates $ fst <$> result
|
|
unless (null dups) $
|
|
parseError $ "the following directives are used more than once: " <> commaSeparated dups
|
|
pure $ DM.fromList $ snd <$> result
|
|
where
|
|
humanReadable = \case
|
|
G.DLExecutable G.EDLQUERY -> "a query"
|
|
G.DLExecutable G.EDLMUTATION -> "a mutation"
|
|
G.DLExecutable G.EDLSUBSCRIPTION -> "a subscription"
|
|
G.DLExecutable G.EDLFIELD -> "a field"
|
|
G.DLExecutable G.EDLFRAGMENT_DEFINITION -> "a fragment definition"
|
|
G.DLExecutable G.EDLFRAGMENT_SPREAD -> "a fragment spread"
|
|
G.DLExecutable G.EDLINLINE_FRAGMENT -> "an inline fragment"
|
|
G.DLTypeSystem G.TSDLSCHEMA -> "the schema"
|
|
G.DLTypeSystem G.TSDLSCALAR -> "a scalar definition"
|
|
G.DLTypeSystem G.TSDLOBJECT -> "an object definition"
|
|
G.DLTypeSystem G.TSDLFIELD_DEFINITION -> "a field definition"
|
|
G.DLTypeSystem G.TSDLARGUMENT_DEFINITION -> "an argument definition"
|
|
G.DLTypeSystem G.TSDLINTERFACE -> "an interface definition"
|
|
G.DLTypeSystem G.TSDLUNION -> "an union definition"
|
|
G.DLTypeSystem G.TSDLENUM -> "an enum definition"
|
|
G.DLTypeSystem G.TSDLENUM_VALUE -> "an enum value definition"
|
|
G.DLTypeSystem G.TSDLINPUT_OBJECT -> "an input object definition"
|
|
G.DLTypeSystem G.TSDLINPUT_FIELD_DEFINITION -> "an input field definition"
|
|
|
|
withDirective ::
|
|
DirectiveMap ->
|
|
DirectiveKey a ->
|
|
(Maybe a -> m b) ->
|
|
m b
|
|
withDirective dmap key callback = callback $ runIdentity <$> DM.lookup key dmap
|
|
|
|
-- Cached custom directive.
|
|
|
|
cachedDirective :: forall m. MonadParse m => Directive m
|
|
cachedDirective =
|
|
mkDirective
|
|
G._cached
|
|
(Just "whether this query should be cached (Hasura Cloud only)")
|
|
True
|
|
[G.DLExecutable G.EDLQUERY]
|
|
(CachedDirective <$> ttlArgument <*> forcedArgument)
|
|
where
|
|
-- Optionally set the cache entry time to live
|
|
ttlArgument :: InputFieldsParser m Int
|
|
ttlArgument = fieldWithDefault G._ttl (Just "measured in seconds") (G.VInt 60) $ fromIntegral <$> int
|
|
|
|
-- Optionally Force a refresh of the cache entry
|
|
forcedArgument :: InputFieldsParser m Bool
|
|
forcedArgument = fieldWithDefault G._refresh (Just "refresh the cache entry") (G.VBoolean False) boolean
|
|
|
|
data CachedDirective = CachedDirective {cdTtl :: Int, cdRefresh :: Bool}
|
|
|
|
cached :: DirectiveKey CachedDirective
|
|
cached = DirectiveKey G._cached
|
|
|
|
-- Subscription tests custom directive.
|
|
|
|
multipleRootFieldsDirective :: forall m. MonadParse m => Directive m
|
|
multipleRootFieldsDirective =
|
|
mkDirective
|
|
G.__multiple_top_level_fields
|
|
(Just "INTERNAL TESTING TOOL DO NOT USE")
|
|
False -- not advertised in the schema
|
|
[G.DLExecutable G.EDLSUBSCRIPTION]
|
|
(pure ())
|
|
|
|
multipleRootFields :: DirectiveKey ()
|
|
multipleRootFields = DirectiveKey G.__multiple_top_level_fields
|
|
|
|
-- Built-in inclusion directives
|
|
|
|
skipDirective :: MonadParse m => Directive m
|
|
skipDirective =
|
|
mkDirective
|
|
G._skip
|
|
(Just "whether this query should be skipped")
|
|
True
|
|
[ G.DLExecutable G.EDLFIELD,
|
|
G.DLExecutable G.EDLFRAGMENT_SPREAD,
|
|
G.DLExecutable G.EDLINLINE_FRAGMENT
|
|
]
|
|
ifArgument
|
|
|
|
includeDirective :: MonadParse m => Directive m
|
|
includeDirective =
|
|
mkDirective
|
|
G._include
|
|
(Just "whether this query should be included")
|
|
True
|
|
[ G.DLExecutable G.EDLFIELD,
|
|
G.DLExecutable G.EDLFRAGMENT_SPREAD,
|
|
G.DLExecutable G.EDLINLINE_FRAGMENT
|
|
]
|
|
ifArgument
|
|
|
|
skip :: DirectiveKey Bool
|
|
skip = DirectiveKey G._skip
|
|
|
|
include :: DirectiveKey Bool
|
|
include = DirectiveKey G._include
|
|
|
|
ifArgument :: MonadParse m => InputFieldsParser m Bool
|
|
ifArgument = field G._if Nothing boolean
|
|
|
|
-- Parser type for directives.
|
|
|
|
data Directive m where
|
|
Directive ::
|
|
forall m a.
|
|
(MonadParse m, Typeable a) =>
|
|
{ dDefinition :: DirectiveInfo,
|
|
dAdvertised :: Bool,
|
|
dParser :: G.Directive Variable -> m a
|
|
} ->
|
|
Directive m
|
|
|
|
data DirectiveKey a where
|
|
DirectiveKey :: Typeable a => G.Name -> DirectiveKey a
|
|
|
|
instance GEq DirectiveKey where
|
|
geq
|
|
(DirectiveKey name1 :: DirectiveKey a1)
|
|
(DirectiveKey name2 :: DirectiveKey a2)
|
|
| name1 == name2,
|
|
Just Refl <- eqT @a1 @a2 =
|
|
Just Refl
|
|
| otherwise = Nothing
|
|
|
|
instance GCompare DirectiveKey where
|
|
gcompare
|
|
(DirectiveKey name1 :: DirectiveKey a1)
|
|
(DirectiveKey name2 :: DirectiveKey a2) =
|
|
strengthenOrdering (compare name1 name2)
|
|
`extendGOrdering` gcompare (typeRep @a1) (typeRep @a2)
|
|
`extendGOrdering` GEQ
|
|
|
|
type DirectiveMap = DM.DMap DirectiveKey Identity
|
|
|
|
mkDirective ::
|
|
(MonadParse m, Typeable a) =>
|
|
G.Name ->
|
|
Maybe G.Description ->
|
|
Bool ->
|
|
[G.DirectiveLocation] ->
|
|
InputFieldsParser m a ->
|
|
Directive m
|
|
mkDirective name description advertised location argsParser =
|
|
Directive
|
|
{ dDefinition = DirectiveInfo name description (ifDefinitions argsParser) location,
|
|
dAdvertised = advertised,
|
|
dParser = \(G.Directive _name arguments) -> withPath (++ [Key $ G.unName name]) $ do
|
|
for_ (M.keys arguments) \argumentName ->
|
|
unless (argumentName `S.member` argumentNames) $
|
|
parseError $ name <<> " has no argument named " <>> argumentName
|
|
withPath (++ [Key "args"]) $ ifParser argsParser $ GraphQLValue <$> arguments
|
|
}
|
|
where
|
|
argumentNames = S.fromList (dName <$> ifDefinitions argsParser)
|