Add Resolver module with helpers and exceptions
This commit is contained in:
parent
53ce65d713
commit
92463f7c4a
|
@ -9,6 +9,7 @@ and this project adheres to
|
||||||
## [Unreleased]
|
## [Unreleased]
|
||||||
### Added
|
### Added
|
||||||
- `ToGraphQL` and `FromGraphQL` typeclasses with instances for basic types.
|
- `ToGraphQL` and `FromGraphQL` typeclasses with instances for basic types.
|
||||||
|
- `Resolver` module with `argument` and `defaultResolver` helper functions.
|
||||||
|
|
||||||
## [1.0.0.0] - 2022-03-29
|
## [1.0.0.0] - 2022-03-29
|
||||||
### Added
|
### Added
|
||||||
|
|
|
@ -26,6 +26,7 @@ library
|
||||||
exposed-modules:
|
exposed-modules:
|
||||||
Language.GraphQL.Class
|
Language.GraphQL.Class
|
||||||
Language.GraphQL.JSON
|
Language.GraphQL.JSON
|
||||||
|
Language.GraphQL.Resolver
|
||||||
Test.Hspec.GraphQL
|
Test.Hspec.GraphQL
|
||||||
other-modules:
|
other-modules:
|
||||||
hs-source-dirs: src
|
hs-source-dirs: src
|
||||||
|
@ -41,6 +42,7 @@ library
|
||||||
megaparsec >= 9.0 && < 10,
|
megaparsec >= 9.0 && < 10,
|
||||||
scientific ^>= 0.3.7,
|
scientific ^>= 0.3.7,
|
||||||
text >= 1.2 && < 3,
|
text >= 1.2 && < 3,
|
||||||
|
transformers ^>= 0.5.6,
|
||||||
vector ^>= 0.12.3,
|
vector ^>= 0.12.3,
|
||||||
unordered-containers ^>= 0.2.16
|
unordered-containers ^>= 0.2.16
|
||||||
default-language: Haskell2010
|
default-language: Haskell2010
|
||||||
|
|
|
@ -24,7 +24,7 @@ fromGraphQLToIntegral (Type.Int value) = Just $ fromIntegral value
|
||||||
fromGraphQLToIntegral (Type.String value) =
|
fromGraphQLToIntegral (Type.String value) =
|
||||||
case Text.Read.decimal value of
|
case Text.Read.decimal value of
|
||||||
Right (converted, "") -> Just converted
|
Right (converted, "") -> Just converted
|
||||||
_ -> Nothing
|
_conversionError -> Nothing
|
||||||
fromGraphQLToIntegral _ = Nothing
|
fromGraphQLToIntegral _ = Nothing
|
||||||
|
|
||||||
-- | Instances of this typeclass can be converted to GraphQL internal
|
-- | Instances of this typeclass can be converted to GraphQL internal
|
||||||
|
|
61
src/Language/GraphQL/Resolver.hs
Normal file
61
src/Language/GraphQL/Resolver.hs
Normal file
|
@ -0,0 +1,61 @@
|
||||||
|
{- This Source Code Form is subject to the terms of the Mozilla Public License,
|
||||||
|
v. 2.0. If a copy of the MPL was not distributed with this file, You can
|
||||||
|
obtain one at https://mozilla.org/MPL/2.0/. -}
|
||||||
|
|
||||||
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
|
|
||||||
|
-- | Helper functions and exceptions to write resolvers.
|
||||||
|
module Language.GraphQL.Resolver
|
||||||
|
( argument
|
||||||
|
, defaultResolver
|
||||||
|
) where
|
||||||
|
|
||||||
|
import Control.Monad.Catch (Exception(..), MonadCatch(..), MonadThrow(..))
|
||||||
|
import Control.Monad.Trans.Reader (ReaderT, asks)
|
||||||
|
import Data.HashMap.Strict ((!))
|
||||||
|
import Data.Text (Text)
|
||||||
|
import qualified Data.Text as Text
|
||||||
|
import Data.Typeable (cast)
|
||||||
|
import Language.GraphQL.AST.Document (Name)
|
||||||
|
import Language.GraphQL.Error
|
||||||
|
import qualified Language.GraphQL.Type as Type
|
||||||
|
import Language.GraphQL.Class (FromGraphQL(..))
|
||||||
|
|
||||||
|
-- | Exceptions thrown by the functions in this module.
|
||||||
|
data ServerException
|
||||||
|
= FieldNotResolvedException !Text
|
||||||
|
| ErroneousArgumentTypeException !Text
|
||||||
|
|
||||||
|
instance Show ServerException where
|
||||||
|
show (FieldNotResolvedException fieldName) =
|
||||||
|
Text.unpack $ Text.unwords ["Field", fieldName, "not resolved."]
|
||||||
|
show (ErroneousArgumentTypeException argumentName) =
|
||||||
|
Text.unpack $ Text.unwords
|
||||||
|
[ "Unable to convert the argument"
|
||||||
|
, argumentName
|
||||||
|
, "to a user-defined type."
|
||||||
|
]
|
||||||
|
|
||||||
|
instance Exception ServerException where
|
||||||
|
toException = toException . ResolverException
|
||||||
|
fromException x = do
|
||||||
|
ResolverException a <- fromException x
|
||||||
|
cast a
|
||||||
|
|
||||||
|
-- | Default resolver expects that the field value is returned by the parent
|
||||||
|
-- object. If the parent is not an object or it doesn't contain the requested
|
||||||
|
-- field name, an error is thrown.
|
||||||
|
defaultResolver :: MonadCatch m => Name -> Type.Resolve m
|
||||||
|
defaultResolver fieldName = do
|
||||||
|
values' <- asks Type.values
|
||||||
|
case values' of
|
||||||
|
Type.Object objectValue -> pure $ objectValue ! fieldName
|
||||||
|
_nonObject -> throwM $ FieldNotResolvedException fieldName
|
||||||
|
|
||||||
|
-- | Takes an argument name, validates that the argument exists, and optionally
|
||||||
|
-- converts it to a user-defined type.
|
||||||
|
argument :: (MonadCatch m, FromGraphQL a) => Name -> ReaderT Type.Context m a
|
||||||
|
argument argumentName =
|
||||||
|
Type.argument argumentName >>= maybe throwError pure . fromGraphQL
|
||||||
|
where
|
||||||
|
throwError = throwM $ ErroneousArgumentTypeException argumentName
|
Loading…
Reference in a new issue