{-# LANGUAGE ConstraintKinds #-}
{-# LANGUAGE FlexibleContexts #-}
{-# LANGUAGE NamedFieldPuns #-}
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE RankNTypes #-}
{-# LANGUAGE ScopedTypeVariables #-}
module Data.Morpheus.Execution.Server.Resolve
( statelessResolver
, byteStringIO
, streamResolver
, statefulResolver
, RootResCon
, fullSchema
) where
import qualified Codec.Binary.UTF8.String as UTF8
import Control.Monad.Trans.Except (ExceptT (..), runExceptT)
import Data.Aeson (Result (..), encode, fromJSON)
import Data.Aeson.Parser (jsonNoDup)
import Data.Attoparsec.ByteString (parseOnly)
import qualified Data.ByteString as S
import qualified Data.ByteString.Lazy.Char8 as L
import Data.Functor.Identity (Identity (..))
import Data.Proxy
import GHC.Generics
import Data.Morpheus.Error.Utils (badRequestError, renderErrors)
import Data.Morpheus.Execution.Server.Encode (EncodeCon, EncodeMutCon, EncodeSubCon, encodeMut,
encodeQuery, encodeSub)
import Data.Morpheus.Execution.Server.Introspect (ObjectRep (..), resolveTypes)
import Data.Morpheus.Execution.Subscription.ClientRegister (GQLState, publishUpdates)
import Data.Morpheus.Parsing.Request.Parser (parseGQL)
import Data.Morpheus.Schema.SchemaAPI (defaultTypes, hiddenRootFields, schemaAPI)
import Data.Morpheus.Types.Internal.AST.Operation (Operation (..), OperationKind (..))
import Data.Morpheus.Types.Internal.Data (DataArguments, DataFingerprint (..),
DataType (..), DataTypeLib (..), initTypeLib)
import Data.Morpheus.Types.Internal.Stream (Event (..), ResponseEvent (..), ResponseStream,
StreamState (..), StreamT (..), closeStream, mapS)
import Data.Morpheus.Types.Internal.Validation (SchemaValidation)
import Data.Morpheus.Types.Internal.Value (Value (..))
import Data.Morpheus.Types.IO (GQLRequest (..), GQLResponse (..))
import Data.Morpheus.Types.Resolver (GQLRootResolver (..))
import Data.Morpheus.Validation.Internal.Utils (VALIDATION_MODE (..))
import Data.Morpheus.Validation.Query.Validation (validateRequest)
type EventCon event = Eq event
type IntroCon a = (Generic a, ObjectRep (Rep a) DataArguments)
type RootResCon m event cont query mutation subscription
= ( EventCon event
, IntroCon query
, IntroCon mutation
, IntroCon subscription
, EncodeCon m query Value
, EncodeMutCon m event cont mutation
, EncodeSubCon m event cont subscription)
decodeNoDup :: L.ByteString -> Result GQLRequest
decodeNoDup bs =
case parseOnly jsonNoDup (S.pack . UTF8.encode $ L.unpack bs) of
Left e -> Error e
Right v -> fromJSON v
byteStringIO :: Monad m => (GQLRequest -> m GQLResponse) -> L.ByteString -> m L.ByteString
byteStringIO resolver request =
case decodeNoDup request of
Error aesonError' -> return $ badRequestError aesonError'
Success req -> encode <$> resolver req
statelessResolver ::
(Monad m, RootResCon m s cont query mut sub)
=> GQLRootResolver m s cont query mut sub
-> GQLRequest
-> m GQLResponse
statelessResolver root = fmap snd . closeStream . streamResolver root
streamResolver ::
(Monad m, RootResCon m s cont query mut sub)
=> GQLRootResolver m s cont query mut sub
-> GQLRequest
-> ResponseStream m s cont GQLResponse
streamResolver root@GQLRootResolver {queryResolver, mutationResolver, subscriptionResolver} request =
renderResponse <$> runExceptT (ExceptT (pure validRequest) >>= ExceptT . execOperator)
where
renderResponse (Left errors) = Errors $ renderErrors errors
renderResponse (Right value) = Data value
validRequest = do
schema <- fullSchema $ Identity root
query <- parseGQL request >>= validateRequest schema FULL_VALIDATION
return (schema, query)
execOperator (schema, operation@Operation {operationKind = QUERY}) =
StreamT $ StreamState [] <$> encodeQuery (schemaAPI schema) queryResolver operation
execOperator (_, operation@Operation {operationKind = MUTATION}) =
mapS Publish (encodeMut mutationResolver operation)
execOperator (_, operation@Operation {operationKind = SUBSCRIPTION}) =
StreamT $ handleActions <$> closeStream (encodeSub subscriptionResolver operation)
where
handleActions (_, Left gqlError) = StreamState [] (Left gqlError)
handleActions (channels, Right subResolver) =
StreamState [Subscribe $ Event (concat channels) handleRes] (Right Null)
where
handleRes event = renderResponse <$> runExceptT (subResolver event)
statefulResolver ::
EventCon s
=> GQLState IO s cont
-> (L.ByteString -> ResponseStream IO s cont L.ByteString)
-> L.ByteString
-> IO L.ByteString
statefulResolver state streamApi request = do
(actions, value) <- closeStream (streamApi request)
mapM_ execute actions
pure value
where
execute (Publish updates) = publishUpdates state updates
execute Subscribe {} = pure ()
fullSchema ::
forall proxy m s cont query mutation subscription. (IntroCon query, IntroCon mutation, IntroCon subscription)
=> proxy (GQLRootResolver m s cont query mutation subscription)
-> SchemaValidation DataTypeLib
fullSchema _ = querySchema >>= mutationSchema >>= subscriptionSchema
where
querySchema = resolveTypes (initTypeLib (operatorType (hiddenRootFields ++ fields) "Query")) (defaultTypes : types)
where
(fields, types) = unzip $ objectFieldTypes (Proxy :: Proxy (Rep query))
mutationSchema lib = resolveTypes (lib {mutation = maybeOperator fields "Mutation"}) types
where
(fields, types) = unzip $ objectFieldTypes (Proxy :: Proxy (Rep mutation))
subscriptionSchema lib = resolveTypes (lib {subscription = maybeOperator fields "Subscription"}) types
where
(fields, types) = unzip $ objectFieldTypes (Proxy :: Proxy (Rep subscription))
maybeOperator [] = const Nothing
maybeOperator fields = Just . operatorType fields
operatorType typeData typeName =
( typeName
, DataType
{ typeData
, typeVisibility = True
, typeName
, typeFingerprint = SystemFingerprint typeName
, typeDescription = ""
})