Attach the field location to resolver exceptions
This commit is contained in:
		| @@ -98,6 +98,7 @@ test-suite graphql-test | |||||||
|     , aeson |     , aeson | ||||||
|     , base >= 4.7 && < 5 |     , base >= 4.7 && < 5 | ||||||
|     , conduit |     , conduit | ||||||
|  |     , exceptions | ||||||
|     , graphql |     , graphql | ||||||
|     , hspec >= 2.8.2 && < 2.9 |     , hspec >= 2.8.2 && < 2.9 | ||||||
|     , hspec-megaparsec >= 2.2.0 && < 2.3 |     , hspec-megaparsec >= 2.2.0 && < 2.3 | ||||||
|   | |||||||
| @@ -37,15 +37,17 @@ resolveFieldValue :: MonadCatch m | |||||||
|     => Type.Value |     => Type.Value | ||||||
|     -> Type.Subs |     -> Type.Subs | ||||||
|     -> Type.Resolve m |     -> Type.Resolve m | ||||||
|  |     -> Full.Location | ||||||
|     -> CollectErrsT m Type.Value |     -> CollectErrsT m Type.Value | ||||||
| resolveFieldValue result args resolver = | resolveFieldValue result args resolver location' = | ||||||
|     catch (lift $ runReaderT resolver context) handleFieldError |     catch (lift $ runReaderT resolver context) handleFieldError | ||||||
|   where |   where | ||||||
|     handleFieldError :: MonadCatch m |     handleFieldError :: MonadCatch m | ||||||
|         => ResolverException |         => ResolverException | ||||||
|         -> CollectErrsT m Type.Value |         -> CollectErrsT m Type.Value | ||||||
|     handleFieldError e = |     handleFieldError e | ||||||
|         addError Type.Null $ Error (Text.pack $ displayException e) [] [] |         = addError Type.Null | ||||||
|  |         $ Error (Text.pack $ displayException e) [location'] [] | ||||||
|     context = Type.Context |     context = Type.Context | ||||||
|         { Type.arguments = Type.Arguments args |         { Type.arguments = Type.Arguments args | ||||||
|         , Type.values = result |         , Type.values = result | ||||||
| @@ -106,7 +108,7 @@ executeField fieldResolver prev fields | |||||||
|             Left errorLocations -> addError null |             Left errorLocations -> addError null | ||||||
|                 $ Error "Argument coercing failed." errorLocations [] |                 $ Error "Argument coercing failed." errorLocations [] | ||||||
|             Right argumentValues -> do |             Right argumentValues -> do | ||||||
|                 answer <- resolveFieldValue prev argumentValues resolver |                 answer <- resolveFieldValue prev argumentValues resolver location' | ||||||
|                 completeValue fieldType fields answer |                 completeValue fieldType fields answer | ||||||
|  |  | ||||||
| completeValue :: (MonadCatch m, Serialize a) | completeValue :: (MonadCatch m, Serialize a) | ||||||
|   | |||||||
| @@ -8,13 +8,15 @@ module Language.GraphQL.ExecuteSpec | |||||||
|     ( spec |     ( spec | ||||||
|     ) where |     ) where | ||||||
|  |  | ||||||
| import Control.Exception (SomeException) | import Control.Exception (Exception(..), SomeException) | ||||||
|  | import Control.Monad.Catch (throwM) | ||||||
| import Data.Aeson ((.=)) | import Data.Aeson ((.=)) | ||||||
| import qualified Data.Aeson as Aeson | import qualified Data.Aeson as Aeson | ||||||
| import Data.Aeson.Types (emptyObject) | import Data.Aeson.Types (emptyObject) | ||||||
| import Data.Conduit | import Data.Conduit | ||||||
| import Data.HashMap.Strict (HashMap) | import Data.HashMap.Strict (HashMap) | ||||||
| import qualified Data.HashMap.Strict as HashMap | import qualified Data.HashMap.Strict as HashMap | ||||||
|  | import Data.Typeable (cast) | ||||||
| import Language.GraphQL.AST (Document, Location(..), Name) | import Language.GraphQL.AST (Document, Location(..), Name) | ||||||
| import Language.GraphQL.AST.Parser (document) | import Language.GraphQL.AST.Parser (document) | ||||||
| import Language.GraphQL.Error | import Language.GraphQL.Error | ||||||
| @@ -28,6 +30,15 @@ import Test.Hspec (Spec, context, describe, it, shouldBe) | |||||||
| import Text.Megaparsec (parse) | import Text.Megaparsec (parse) | ||||||
| import Text.RawString.QQ (r) | import Text.RawString.QQ (r) | ||||||
|  |  | ||||||
|  | data PhilosopherException = PhilosopherException | ||||||
|  |     deriving Show | ||||||
|  |  | ||||||
|  | instance Exception PhilosopherException where | ||||||
|  |     toException = toException. ResolverException | ||||||
|  |     fromException e = do | ||||||
|  |         ResolverException resolverException <- fromException e | ||||||
|  |         cast resolverException | ||||||
|  |  | ||||||
| philosopherSchema :: Schema (Either SomeException) | philosopherSchema :: Schema (Either SomeException) | ||||||
| philosopherSchema = | philosopherSchema = | ||||||
|     schemaWithTypes Nothing queryType Nothing subscriptionRoot extraTypes mempty |     schemaWithTypes Nothing queryType Nothing subscriptionRoot extraTypes mempty | ||||||
| @@ -40,14 +51,21 @@ philosopherSchema = | |||||||
|  |  | ||||||
| queryType :: Out.ObjectType (Either SomeException) | queryType :: Out.ObjectType (Either SomeException) | ||||||
| queryType = Out.ObjectType "Query" Nothing [] | queryType = Out.ObjectType "Query" Nothing [] | ||||||
|     $ HashMap.singleton "philosopher" |     $ HashMap.fromList | ||||||
|     $ ValueResolver philosopherField |     [ ("philosopher", ValueResolver philosopherField philosopherResolver) | ||||||
|     $ pure $ Object mempty |     , ("genres", ValueResolver genresField genresResolver) | ||||||
|  |     ] | ||||||
|   where |   where | ||||||
|     philosopherField = |     philosopherField = | ||||||
|         Out.Field Nothing (Out.NonNullObjectType philosopherType) |         Out.Field Nothing (Out.NonNullObjectType philosopherType) | ||||||
|         $ HashMap.singleton "id" |         $ HashMap.singleton "id" | ||||||
|         $ In.Argument Nothing (In.NamedScalarType id) Nothing |         $ In.Argument Nothing (In.NamedScalarType id) Nothing | ||||||
|  |     philosopherResolver = pure $ Object mempty | ||||||
|  |     genresField = | ||||||
|  |       let fieldType = Out.ListType $ Out.NonNullScalarType string | ||||||
|  |        in Out.Field Nothing fieldType HashMap.empty | ||||||
|  |     genresResolver :: Resolve (Either SomeException) | ||||||
|  |     genresResolver = throwM PhilosopherException | ||||||
|  |  | ||||||
| musicType :: Out.ObjectType (Either SomeException) | musicType :: Out.ObjectType (Either SomeException) | ||||||
| musicType = Out.ObjectType "Music" Nothing [] | musicType = Out.ObjectType "Music" Nothing [] | ||||||
| @@ -288,6 +306,20 @@ spec = | |||||||
|                         $ parse document "" "{ philosopher(id: \"1\") { century } }" |                         $ parse document "" "{ philosopher(id: \"1\") { century } }" | ||||||
|                 in actual `shouldBe` expected |                 in actual `shouldBe` expected | ||||||
|  |  | ||||||
|  |             it "gives location information for failed result coercion" $ | ||||||
|  |                 let data'' = Aeson.object | ||||||
|  |                         [ "genres" .= Aeson.Null | ||||||
|  |                         ] | ||||||
|  |                     executionErrors = pure $ Error | ||||||
|  |                         { message = "PhilosopherException" | ||||||
|  |                         , locations = [Location 1 3] | ||||||
|  |                         , path = [] | ||||||
|  |                         } | ||||||
|  |                     expected = Response data'' executionErrors | ||||||
|  |                     Right (Right actual) = either (pure . parseError) execute' | ||||||
|  |                         $ parse document "" "{ genres }" | ||||||
|  |                 in actual `shouldBe` expected | ||||||
|  |  | ||||||
|         context "Subscription" $ |         context "Subscription" $ | ||||||
|             it "subscribes" $ |             it "subscribes" $ | ||||||
|                 let data'' = Aeson.object |                 let data'' = Aeson.object | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user