mirror of
https://github.com/hasura/graphql-engine.git
synced 2025-01-08 08:44:24 +03:00
42e5205eb5
### Description This monster of a PR took way too long. As the title suggests, it reduces the schema context carried in the readers to the very strict minimum. In practice, that means that to build a source, we only require: - the global `SchemaContext` - the global `SchemaOptions` (soon to be renamed `SchemaSourceOptions`) - that source's `SourceInfo` Furthermore, _we no longer carry "default" customization options throughout the schema_. All customization information is extracted from the `SourceInfo`, when required. This prevents an entire category of bugs we had previously encountered, such as parts of the code using uninitialized / unupdated customization info. In turn, this meant that we could remove the explicit threading of the `SourceInfo` throughout the schema, since it is now always available through the reader context. Finally, this meant making a few adjustments to relay and actions as well, such as the introduction of a new separate "context" for actions, and a change to how we create some of the action-specific postgres scalar parsers. I'll highlight with review comments the areas of interest. PR-URL: https://github.com/hasura/graphql-engine-mono/pull/6709 GitOrigin-RevId: ea80fddcb24e2513779dd04b0b700a55f0028dd1
167 lines
6.6 KiB
Haskell
167 lines
6.6 KiB
Haskell
-- | This module defines the monads required to run parser tests.
|
|
--
|
|
-- Warning: a lot of the implementations are currently 'undefined'. As we write
|
|
-- more advanced tests, they might require implementations.
|
|
module Test.Parser.Monad
|
|
( ParserTest (..),
|
|
SchemaEnvironment (..),
|
|
SchemaTest,
|
|
runSchemaTest,
|
|
notImplementedYet,
|
|
)
|
|
where
|
|
|
|
import Control.Monad.Memoize
|
|
import Data.Aeson.Internal (JSONPathElement)
|
|
import Data.Has (Has (..))
|
|
import Data.Text qualified as T
|
|
import GHC.Stack
|
|
import Hasura.Base.Error (QErr)
|
|
import Hasura.Base.ErrorMessage
|
|
import Hasura.GraphQL.Parser.Class
|
|
import Hasura.GraphQL.Parser.ErrorCode
|
|
import Hasura.GraphQL.Schema.Common
|
|
import Hasura.GraphQL.Schema.NamingCase
|
|
import Hasura.GraphQL.Schema.Options (SchemaOptions (..))
|
|
import Hasura.GraphQL.Schema.Options qualified as Options
|
|
import Hasura.GraphQL.Schema.Typename
|
|
import Hasura.Prelude
|
|
import Hasura.RQL.Types.Source (SourceInfo)
|
|
import Hasura.RQL.Types.SourceCustomization (MkRootFieldName)
|
|
import Hasura.RemoteSchema.SchemaCache (CustomizeRemoteFieldName)
|
|
import Hasura.SQL.Backend
|
|
import Hasura.Session (adminRoleName)
|
|
import Language.Haskell.TH.Syntax qualified as TH
|
|
import Test.HUnit.Lang (assertFailure)
|
|
|
|
-- | Placeholder value for test inputs that are not relevant yet.
|
|
notImplementedYet :: HasCallStack => String -> a
|
|
notImplementedYet thing =
|
|
withFrozenCallStack $
|
|
error $
|
|
( unlines
|
|
[ "\"" ++ thing ++ "\" is not yet defined, because it hasn't been touched by tests yet.",
|
|
"If you see this message you likely need to provide/mock a value here"
|
|
]
|
|
)
|
|
|
|
-- | Monad builder environment.
|
|
--
|
|
-- Parser functions generally have a return type of @m (Parser n)@. The @m@
|
|
-- parameter is mocked through 'SchemaTestM', which requires a bunch of 'Has'
|
|
-- instances, as well as a 'ReaderT' instance for environment
|
|
-- settings/configurations. This type repesents these settings.
|
|
--
|
|
-- SchemaEnvironment: currently void. This is subject to change if we require
|
|
-- more complex setup.
|
|
data SchemaEnvironment = SchemaEnvironment
|
|
{ seSchemaOptions :: SchemaOptions,
|
|
seSourceInfo :: SourceInfo ('Postgres 'Vanilla)
|
|
}
|
|
|
|
defaultSchemaOptions :: SchemaOptions
|
|
defaultSchemaOptions =
|
|
SchemaOptions
|
|
{ soStringifyNumbers = Options.Don'tStringifyNumbers,
|
|
soDangerousBooleanCollapse = Options.Don'tDangerouslyCollapseBooleans,
|
|
soInferFunctionPermissions = Options.InferFunctionPermissions,
|
|
soOptimizePermissionFilters = Options.Don'tOptimizePermissionFilters,
|
|
soIncludeUpdateManyFields = Options.IncludeUpdateManyFields,
|
|
soIncludeAggregationPredicates = Options.IncludeAggregationPredicates,
|
|
soIncludeStreamFields = Options.IncludeStreamFields,
|
|
soBigQueryStringNumericInput = Options.EnableBigQueryStringNumericInput
|
|
}
|
|
|
|
instance Has NamingCase SchemaEnvironment where
|
|
getter :: SchemaEnvironment -> NamingCase
|
|
getter = const HasuraCase
|
|
|
|
modifier :: (NamingCase -> NamingCase) -> SchemaEnvironment -> SchemaEnvironment
|
|
modifier = notImplementedYet "modifier<Has NamingCase SchemaEnvironment>"
|
|
|
|
instance Has SchemaOptions SchemaEnvironment where
|
|
getter :: SchemaEnvironment -> SchemaOptions
|
|
getter = seSchemaOptions
|
|
|
|
modifier :: (SchemaOptions -> SchemaOptions) -> SchemaEnvironment -> SchemaEnvironment
|
|
modifier f env = env {seSchemaOptions = f (seSchemaOptions env)}
|
|
|
|
instance Has (SourceInfo ('Postgres 'Vanilla)) SchemaEnvironment where
|
|
getter :: SchemaEnvironment -> SourceInfo ('Postgres 'Vanilla)
|
|
getter = seSourceInfo
|
|
|
|
modifier :: (SourceInfo ('Postgres 'Vanilla) -> SourceInfo ('Postgres 'Vanilla)) -> SchemaEnvironment -> SchemaEnvironment
|
|
modifier = notImplementedYet "modifier<Has (SourceInfo ('Postgres 'Vanilla)) SchemaEnvironment>"
|
|
|
|
instance Has SchemaContext SchemaEnvironment where
|
|
getter :: SchemaEnvironment -> SchemaContext
|
|
getter =
|
|
const
|
|
SchemaContext
|
|
{ scSchemaKind = HasuraSchema,
|
|
scRemoteRelationshipParserBuilder = ignoreRemoteRelationship,
|
|
scRole = adminRoleName
|
|
}
|
|
|
|
modifier :: (SchemaContext -> SchemaContext) -> SchemaEnvironment -> SchemaEnvironment
|
|
modifier = notImplementedYet "modifier<Has SchemaContext SchemaEnvironment>"
|
|
|
|
instance Has MkTypename SchemaEnvironment where
|
|
getter :: SchemaEnvironment -> MkTypename
|
|
getter = const (MkTypename id)
|
|
|
|
modifier :: (MkTypename -> MkTypename) -> SchemaEnvironment -> SchemaEnvironment
|
|
modifier = notImplementedYet "modifier<Has MkTypeName SchemaEnvironment>"
|
|
|
|
instance Has MkRootFieldName SchemaEnvironment where
|
|
getter :: SchemaEnvironment -> MkRootFieldName
|
|
getter = const mempty
|
|
|
|
modifier :: (MkRootFieldName -> MkRootFieldName) -> SchemaEnvironment -> SchemaEnvironment
|
|
modifier = notImplementedYet "modifier<Has MkRootFieldName SchemaEnvironment>"
|
|
|
|
instance Has CustomizeRemoteFieldName SchemaEnvironment where
|
|
getter :: SchemaEnvironment -> CustomizeRemoteFieldName
|
|
getter = notImplementedYet "getter<Has CustomizeRemoteFieldName SchemaEnvironment>"
|
|
|
|
modifier :: (CustomizeRemoteFieldName -> CustomizeRemoteFieldName) -> SchemaEnvironment -> SchemaEnvironment
|
|
modifier = notImplementedYet "modifier<Has CustomizeRemoteFieldName SchemaEnvironment>"
|
|
|
|
-------------------------------------------------------------------------------
|
|
|
|
-- | SchemaTest
|
|
type SchemaTest = SchemaT SchemaEnvironment SchemaTestInternal
|
|
|
|
runSchemaTest :: SourceInfo ('Postgres 'Vanilla) -> SchemaTest a -> a
|
|
runSchemaTest sourceInfo = runSchemaTestInternal . flip runReaderT (SchemaEnvironment defaultSchemaOptions sourceInfo) . runSchemaT
|
|
|
|
newtype SchemaTestInternal a = SchemaTestInternal {runSchemaTestInternal :: a}
|
|
deriving stock (Functor)
|
|
deriving (Applicative, Monad) via Identity
|
|
|
|
instance MonadError QErr SchemaTestInternal where
|
|
throwError :: forall a. QErr -> SchemaTestInternal a
|
|
throwError = notImplementedYet "throwError<MonadError QErr SchemaTestT>"
|
|
|
|
catchError :: forall a. SchemaTestInternal a -> (QErr -> SchemaTestInternal a) -> SchemaTestInternal a
|
|
catchError = notImplementedYet "catchError<MonadError QErr SchemaTestInternal>"
|
|
|
|
instance MonadMemoize SchemaTestInternal where
|
|
memoizeOn :: TH.Name -> a -> SchemaTestInternal p -> SchemaTestInternal p
|
|
memoizeOn _ _ = id
|
|
|
|
-------------------------------------------------------------------------------
|
|
|
|
-- | ParserTest
|
|
newtype ParserTest a = ParserTest {runParserTest :: IO a}
|
|
deriving stock (Functor)
|
|
deriving newtype (Applicative, Monad)
|
|
|
|
instance MonadParse ParserTest where
|
|
withKey :: JSONPathElement -> ParserTest a -> ParserTest a
|
|
withKey = const id
|
|
|
|
parseErrorWith :: ParseErrorCode -> ErrorMessage -> ParserTest a
|
|
parseErrorWith code text =
|
|
ParserTest . assertFailure $ show code <> ": " <> T.unpack (fromErrorMessage text)
|