2022-09-08 19:53:22 +02:00
|
|
|
{- 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 #-}
|
2024-07-07 12:55:42 +02:00
|
|
|
{-# LANGUAGE TemplateHaskell #-}
|
|
|
|
{-# LANGUAGE TupleSections #-}
|
2022-09-08 19:53:22 +02:00
|
|
|
|
|
|
|
-- | ToGraphQL and FromGraphQL typeclasses used for user-defined type
|
|
|
|
-- conversion.
|
|
|
|
module Language.GraphQL.Class
|
|
|
|
( FromGraphQL(..)
|
|
|
|
, ToGraphQL(..)
|
2024-07-10 13:22:48 +02:00
|
|
|
, deriveFromGraphQL
|
2024-07-07 12:55:42 +02:00
|
|
|
, deriveToGraphQL
|
2024-10-20 17:13:39 +02:00
|
|
|
, gql
|
2022-09-08 19:53:22 +02:00
|
|
|
) where
|
|
|
|
|
|
|
|
import Data.Int (Int8, Int16, Int32, Int64)
|
|
|
|
import Data.Text (Text)
|
2023-02-19 11:26:27 +01:00
|
|
|
import Data.Word (Word8, Word16, Word32, Word64)
|
2022-09-08 19:53:22 +02:00
|
|
|
import qualified Data.Text.Read as Text.Read
|
|
|
|
import Data.Vector (Vector)
|
|
|
|
import qualified Data.Vector as Vector
|
2023-06-23 17:31:19 +02:00
|
|
|
import Data.Scientific (Scientific, toRealFloat)
|
2023-06-26 16:50:14 +02:00
|
|
|
import qualified Data.Text as Text
|
|
|
|
import Data.Time
|
|
|
|
( Day
|
|
|
|
, DiffTime
|
2023-06-29 19:18:35 +02:00
|
|
|
, LocalTime(..)
|
2023-06-26 16:50:14 +02:00
|
|
|
, NominalDiffTime
|
2023-06-29 19:18:35 +02:00
|
|
|
, TimeOfDay(..)
|
2023-06-26 16:50:14 +02:00
|
|
|
, UTCTime(..)
|
|
|
|
, showGregorian
|
|
|
|
, secondsToNominalDiffTime
|
|
|
|
, secondsToDiffTime
|
|
|
|
)
|
2023-06-29 19:18:35 +02:00
|
|
|
import Data.Time.Format.ISO8601
|
|
|
|
( ISO8601(..)
|
|
|
|
, formatParseM
|
|
|
|
, iso8601Format
|
|
|
|
, iso8601Show
|
|
|
|
)
|
2024-07-07 12:55:42 +02:00
|
|
|
import Language.Haskell.TH
|
|
|
|
( Con(..)
|
|
|
|
, Dec(..)
|
|
|
|
, Exp(..)
|
|
|
|
, Info(..)
|
2024-10-20 17:13:39 +02:00
|
|
|
, Lit(..)
|
2024-07-07 12:55:42 +02:00
|
|
|
, Quote(..)
|
|
|
|
, Name
|
|
|
|
, Q
|
|
|
|
, VarBangType
|
|
|
|
, appT
|
|
|
|
, conP
|
|
|
|
, conT
|
|
|
|
, instanceD
|
|
|
|
, recP
|
|
|
|
, reify
|
|
|
|
, nameBase
|
|
|
|
, listE
|
|
|
|
, stringL
|
|
|
|
, tupE
|
|
|
|
, litE
|
|
|
|
, varE
|
|
|
|
, varP
|
|
|
|
, funD
|
|
|
|
, clause
|
|
|
|
, normalB
|
|
|
|
, appE
|
|
|
|
, mkName
|
2024-07-10 13:22:48 +02:00
|
|
|
, conE
|
|
|
|
, integerL
|
|
|
|
, litP
|
|
|
|
, wildP
|
2024-07-07 12:55:42 +02:00
|
|
|
)
|
2024-10-20 17:13:39 +02:00
|
|
|
import Language.Haskell.TH.Quote (QuasiQuoter(..))
|
2024-07-07 12:55:42 +02:00
|
|
|
import Data.Foldable (Foldable(..))
|
|
|
|
import qualified Data.HashMap.Strict as HashMap
|
|
|
|
import qualified Language.GraphQL.Type as Type
|
|
|
|
import Prelude hiding (id)
|
2022-09-08 19:53:22 +02:00
|
|
|
|
|
|
|
fromGraphQLToIntegral :: Integral a => Type.Value -> Maybe a
|
|
|
|
fromGraphQLToIntegral (Type.Int value) = Just $ fromIntegral value
|
|
|
|
fromGraphQLToIntegral (Type.String value) =
|
|
|
|
case Text.Read.decimal value of
|
|
|
|
Right (converted, "") -> Just converted
|
2022-08-24 22:33:20 +03:00
|
|
|
_conversionError -> Nothing
|
2022-09-08 19:53:22 +02:00
|
|
|
fromGraphQLToIntegral _ = Nothing
|
|
|
|
|
2023-06-29 19:18:35 +02:00
|
|
|
iso8601ToGraphQL :: ISO8601 t => t -> Type.Value
|
|
|
|
iso8601ToGraphQL = Type.String . Text.pack . iso8601Show
|
|
|
|
|
|
|
|
fromGraphQLToISO8601 :: ISO8601 t => Type.Value -> Maybe t
|
|
|
|
fromGraphQLToISO8601 (Type.String value') = formatParseM iso8601Format $ Text.unpack value'
|
|
|
|
fromGraphQLToISO8601 _ = Nothing
|
|
|
|
|
2022-09-08 19:53:22 +02:00
|
|
|
-- | Instances of this typeclass can be converted to GraphQL internal
|
|
|
|
-- representation.
|
2023-06-23 17:31:19 +02:00
|
|
|
class ToGraphQL a
|
|
|
|
where
|
2022-09-08 19:53:22 +02:00
|
|
|
toGraphQL :: a -> Type.Value
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance ToGraphQL Text
|
|
|
|
where
|
2022-09-08 19:53:22 +02:00
|
|
|
toGraphQL = Type.String
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance ToGraphQL Int
|
|
|
|
where
|
2022-09-08 19:53:22 +02:00
|
|
|
toGraphQL = Type.Int . fromIntegral
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance ToGraphQL Int8
|
|
|
|
where
|
2022-09-08 19:53:22 +02:00
|
|
|
toGraphQL = Type.Int . fromIntegral
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance ToGraphQL Int16
|
|
|
|
where
|
2022-09-08 19:53:22 +02:00
|
|
|
toGraphQL = Type.Int . fromIntegral
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance ToGraphQL Int32
|
|
|
|
where
|
2022-09-08 19:53:22 +02:00
|
|
|
toGraphQL = Type.Int
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance ToGraphQL Int64
|
|
|
|
where
|
2022-09-08 19:53:22 +02:00
|
|
|
toGraphQL = Type.Int . fromIntegral
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance ToGraphQL Word
|
|
|
|
where
|
2023-02-19 11:26:27 +01:00
|
|
|
toGraphQL = Type.Int . fromIntegral
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance ToGraphQL Word8
|
|
|
|
where
|
2023-02-19 11:26:27 +01:00
|
|
|
toGraphQL = Type.Int . fromIntegral
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance ToGraphQL Word16
|
|
|
|
where
|
2023-02-19 11:26:27 +01:00
|
|
|
toGraphQL = Type.Int . fromIntegral
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance ToGraphQL Word32
|
|
|
|
where
|
2023-02-19 11:26:27 +01:00
|
|
|
toGraphQL = Type.Int . fromIntegral
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance ToGraphQL Word64
|
|
|
|
where
|
2023-02-19 11:26:27 +01:00
|
|
|
toGraphQL = Type.Int . fromIntegral
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance ToGraphQL a => ToGraphQL [a]
|
|
|
|
where
|
2022-09-08 19:53:22 +02:00
|
|
|
toGraphQL = Type.List . fmap toGraphQL
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance ToGraphQL a => ToGraphQL (Vector a)
|
|
|
|
where
|
2022-09-08 19:53:22 +02:00
|
|
|
toGraphQL = Type.List . toList . fmap toGraphQL
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance ToGraphQL a => ToGraphQL (Maybe a)
|
|
|
|
where
|
2022-09-08 19:53:22 +02:00
|
|
|
toGraphQL (Just justValue) = toGraphQL justValue
|
|
|
|
toGraphQL Nothing = Type.Null
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance ToGraphQL Bool
|
|
|
|
where
|
2022-09-08 19:53:22 +02:00
|
|
|
toGraphQL = Type.Boolean
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance ToGraphQL Float
|
|
|
|
where
|
2023-05-07 17:19:57 +02:00
|
|
|
toGraphQL = Type.Float . realToFrac
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance ToGraphQL Double
|
|
|
|
where
|
2023-05-07 17:19:57 +02:00
|
|
|
toGraphQL = Type.Float
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance ToGraphQL Scientific
|
|
|
|
where
|
|
|
|
toGraphQL = Type.Float . toRealFloat
|
|
|
|
|
2023-06-26 16:50:14 +02:00
|
|
|
instance ToGraphQL Day
|
|
|
|
where
|
|
|
|
toGraphQL = Type.String . Text.pack . showGregorian
|
|
|
|
|
|
|
|
instance ToGraphQL DiffTime
|
|
|
|
where
|
|
|
|
toGraphQL = Type.Int . truncate . (realToFrac :: DiffTime -> Double)
|
|
|
|
|
|
|
|
instance ToGraphQL NominalDiffTime
|
|
|
|
where
|
|
|
|
toGraphQL = Type.Int . truncate . (realToFrac :: NominalDiffTime -> Double)
|
|
|
|
|
|
|
|
instance ToGraphQL UTCTime
|
|
|
|
where
|
2023-06-29 19:18:35 +02:00
|
|
|
toGraphQL = iso8601ToGraphQL
|
|
|
|
|
|
|
|
instance ToGraphQL TimeOfDay
|
|
|
|
where
|
|
|
|
toGraphQL = iso8601ToGraphQL
|
|
|
|
|
|
|
|
instance ToGraphQL LocalTime
|
|
|
|
where
|
|
|
|
toGraphQL = iso8601ToGraphQL
|
2023-06-26 16:50:14 +02:00
|
|
|
|
2022-09-08 19:53:22 +02:00
|
|
|
-- | Instances of this typeclass can be used to convert GraphQL internal
|
|
|
|
-- representation to user-defined type.
|
2023-06-23 17:31:19 +02:00
|
|
|
class FromGraphQL a
|
|
|
|
where
|
2022-09-08 19:53:22 +02:00
|
|
|
fromGraphQL :: Type.Value -> Maybe a
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance FromGraphQL Text
|
|
|
|
where
|
2022-09-08 19:53:22 +02:00
|
|
|
fromGraphQL (Type.String value) = Just value
|
|
|
|
fromGraphQL _ = Nothing
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance FromGraphQL Int
|
|
|
|
where
|
2022-09-08 19:53:22 +02:00
|
|
|
fromGraphQL = fromGraphQLToIntegral
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance FromGraphQL Int8
|
|
|
|
where
|
2022-09-08 19:53:22 +02:00
|
|
|
fromGraphQL = fromGraphQLToIntegral
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance FromGraphQL Int16
|
|
|
|
where
|
2022-09-08 19:53:22 +02:00
|
|
|
fromGraphQL = fromGraphQLToIntegral
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance FromGraphQL Int32
|
|
|
|
where
|
2022-09-08 19:53:22 +02:00
|
|
|
fromGraphQL = fromGraphQLToIntegral
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance FromGraphQL Int64
|
|
|
|
where
|
2022-09-08 19:53:22 +02:00
|
|
|
fromGraphQL = fromGraphQLToIntegral
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance FromGraphQL Word
|
|
|
|
where
|
2023-02-19 11:26:27 +01:00
|
|
|
fromGraphQL = fromGraphQLToIntegral
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance FromGraphQL Word8
|
|
|
|
where
|
2023-02-19 11:26:27 +01:00
|
|
|
fromGraphQL = fromGraphQLToIntegral
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance FromGraphQL Word16
|
|
|
|
where
|
2023-02-19 11:26:27 +01:00
|
|
|
fromGraphQL = fromGraphQLToIntegral
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance FromGraphQL Word32
|
|
|
|
where
|
2023-02-19 11:26:27 +01:00
|
|
|
fromGraphQL = fromGraphQLToIntegral
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance FromGraphQL Word64
|
|
|
|
where
|
2023-02-19 11:26:27 +01:00
|
|
|
fromGraphQL = fromGraphQLToIntegral
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance FromGraphQL a => FromGraphQL [a]
|
|
|
|
where
|
2022-09-08 19:53:22 +02:00
|
|
|
fromGraphQL (Type.List value) = traverse fromGraphQL value
|
|
|
|
fromGraphQL _ = Nothing
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance FromGraphQL a => FromGraphQL (Vector a)
|
|
|
|
where
|
2022-09-08 19:53:22 +02:00
|
|
|
fromGraphQL (Type.List value) = Vector.fromList
|
|
|
|
<$> traverse fromGraphQL value
|
|
|
|
fromGraphQL _ = Nothing
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance FromGraphQL a => FromGraphQL (Maybe a)
|
|
|
|
where
|
2022-09-08 19:53:22 +02:00
|
|
|
fromGraphQL Type.Null = Just Nothing
|
|
|
|
fromGraphQL value = Just <$> fromGraphQL value
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance FromGraphQL Bool
|
|
|
|
where
|
2022-09-08 19:53:22 +02:00
|
|
|
fromGraphQL (Type.Boolean value) = Just value
|
|
|
|
fromGraphQL _ = Nothing
|
2023-05-07 17:19:57 +02:00
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance FromGraphQL Float
|
|
|
|
where
|
2023-05-07 17:19:57 +02:00
|
|
|
fromGraphQL (Type.Float value) = Just $ realToFrac value
|
|
|
|
fromGraphQL _ = Nothing
|
|
|
|
|
2023-06-23 17:31:19 +02:00
|
|
|
instance FromGraphQL Double
|
|
|
|
where
|
2023-05-07 17:19:57 +02:00
|
|
|
fromGraphQL (Type.Float value) = Just value
|
|
|
|
fromGraphQL _ = Nothing
|
2023-06-23 17:31:19 +02:00
|
|
|
|
|
|
|
instance FromGraphQL Scientific
|
|
|
|
where
|
|
|
|
fromGraphQL (Type.Float value) = Just $ realToFrac value
|
|
|
|
fromGraphQL _ = Nothing
|
2023-06-26 16:50:14 +02:00
|
|
|
|
|
|
|
instance FromGraphQL Day
|
|
|
|
where
|
2023-06-29 19:18:35 +02:00
|
|
|
fromGraphQL = fromGraphQLToISO8601
|
2023-06-26 16:50:14 +02:00
|
|
|
|
|
|
|
instance FromGraphQL DiffTime
|
|
|
|
where
|
|
|
|
fromGraphQL (Type.Int value') = Just $ secondsToDiffTime $ fromIntegral value'
|
|
|
|
fromGraphQL _ = Nothing
|
|
|
|
|
|
|
|
instance FromGraphQL NominalDiffTime
|
|
|
|
where
|
|
|
|
fromGraphQL (Type.Int value') = Just $ secondsToNominalDiffTime $ fromIntegral value'
|
|
|
|
fromGraphQL _ = Nothing
|
|
|
|
|
|
|
|
instance FromGraphQL UTCTime
|
|
|
|
where
|
2023-06-29 19:18:35 +02:00
|
|
|
fromGraphQL = fromGraphQLToISO8601
|
|
|
|
|
|
|
|
instance FromGraphQL TimeOfDay
|
|
|
|
where
|
|
|
|
fromGraphQL = fromGraphQLToISO8601
|
|
|
|
|
|
|
|
instance FromGraphQL LocalTime
|
|
|
|
where
|
|
|
|
fromGraphQL = fromGraphQLToISO8601
|
2024-07-07 12:55:42 +02:00
|
|
|
|
|
|
|
stringLE :: Name -> Q Exp
|
|
|
|
stringLE = litE . stringL . nameBase
|
|
|
|
|
2024-07-19 12:55:36 +02:00
|
|
|
-- | Given a type derives a 'FromGraphQL' instance for it.
|
|
|
|
--
|
|
|
|
-- The derivation can only work when all nested types already have 'FromGraphQL'
|
|
|
|
-- instances.
|
|
|
|
--
|
|
|
|
-- The following cases are supported:
|
|
|
|
--
|
|
|
|
-- * Records encode input objects.
|
|
|
|
-- * Sum types with all data constructors without parameters encode Enums.
|
2024-07-10 13:22:48 +02:00
|
|
|
deriveFromGraphQL :: Name -> Q [Dec]
|
|
|
|
deriveFromGraphQL typeName = do
|
|
|
|
TyConI plainConstructor <- reify typeName
|
|
|
|
case plainConstructor of
|
|
|
|
DataD _ _ _ _ [cons'] _
|
|
|
|
| RecC dataConName varBangTypes <- cons' ->
|
|
|
|
withRecordConstructor dataConName varBangTypes
|
|
|
|
DataD _ _ _ _ cons' _ -> pure <$> generateEnumInstance cons'
|
|
|
|
NewtypeD _ _ _ _ cons' _
|
|
|
|
| RecC dataConName varBangTypes <- cons' ->
|
|
|
|
withRecordConstructor dataConName varBangTypes
|
2024-07-17 09:37:29 +02:00
|
|
|
_ -> error "Only input objects and enums are supported if all member types have a FromGraphQL instance"
|
2024-07-10 13:22:48 +02:00
|
|
|
where
|
|
|
|
enumMemberPattern (NormalC normalName []) =
|
|
|
|
let fromGraphQLF = conP (mkName "Type.Enum") [litP $ stringL $ nameBase normalName]
|
|
|
|
in flip (clause [fromGraphQLF]) []
|
|
|
|
$ normalB [|Just $(conE normalName)|]
|
|
|
|
enumMemberPattern _ =
|
|
|
|
error "Enum member should be a normal constructor without parameters"
|
|
|
|
generateEnumInstance :: [Con] -> Q Dec
|
|
|
|
generateEnumInstance cons'
|
|
|
|
= instanceD mempty (appT (conT ''FromGraphQL) conTName)
|
|
|
|
$ pure $ funD 'fromGraphQL
|
|
|
|
$ (enumMemberPattern <$> cons')
|
|
|
|
<> [clause [wildP] (normalB [|Nothing|]) []]
|
|
|
|
hashMapLookup fieldName objectName =
|
|
|
|
[|HashMap.lookup $(stringLE fieldName) $objectName >>= fromGraphQL|]
|
|
|
|
addRecordField objectName accumulator (name', _, _)
|
|
|
|
= appE (appE (varE $ mkName "<*>") accumulator)
|
|
|
|
$ hashMapLookup name' objectName
|
|
|
|
withRecordConstructor dataConName varBangTypes = do
|
|
|
|
valueName <- newName "value"
|
|
|
|
let objectName = varE valueName
|
|
|
|
toGraphQLF = conP (mkName "Type.Object") [varP valueName]
|
|
|
|
fBody = makeRecordBody (conE dataConName) objectName varBangTypes
|
|
|
|
recordSize = litE $ integerL $ fromIntegral $ length varBangTypes
|
|
|
|
[d|
|
|
|
|
instance FromGraphQL $conTName
|
|
|
|
where
|
|
|
|
fromGraphQL $toGraphQLF
|
|
|
|
| HashMap.size $objectName == $recordSize = $fBody
|
|
|
|
| otherwise = Nothing
|
|
|
|
fromGraphQL _ = Nothing
|
|
|
|
|]
|
|
|
|
makeRecordBody dataConE objectName ((headName, _, _) : varBangTypes') =
|
|
|
|
let initialExpression = appE (appE (varE $ mkName "<$>") dataConE)
|
|
|
|
$ hashMapLookup headName objectName
|
|
|
|
in foldl' (addRecordField objectName) initialExpression varBangTypes'
|
|
|
|
makeRecordBody dataConE _ [] = dataConE
|
|
|
|
conTName = conT typeName
|
|
|
|
|
2024-07-19 12:55:36 +02:00
|
|
|
-- | Given a type derives a 'ToGraphQL' instance for it.
|
|
|
|
--
|
|
|
|
-- The derivation can only work when all nested types already have 'ToGraphQL'
|
|
|
|
-- instances.
|
|
|
|
--
|
|
|
|
-- The following cases are supported:
|
|
|
|
--
|
|
|
|
-- * Records are decoded as objects.
|
|
|
|
-- * Sum types with all data constructors without parameters are decoded as Enums.
|
|
|
|
-- * Sum types whose data constructors have exactly one parameter are decoded as Unions.
|
2024-07-07 12:55:42 +02:00
|
|
|
deriveToGraphQL :: Name -> Q [Dec]
|
|
|
|
deriveToGraphQL typeName = do
|
|
|
|
TyConI plainConstructor <- reify typeName
|
|
|
|
case plainConstructor of
|
|
|
|
DataD _ _ _ _ [cons'] _
|
|
|
|
| RecC dataConName varBangTypes <- cons' ->
|
|
|
|
withRecordConstructor dataConName varBangTypes
|
|
|
|
DataD _ _ _ _ cons' _ -> fmap pure
|
|
|
|
$ instanceD mempty (appT (conT ''ToGraphQL) conTName)
|
|
|
|
$ pure $ funD 'toGraphQL
|
|
|
|
$ generateSumTypeInstance cons'
|
|
|
|
NewtypeD _ _ _ _ cons' _
|
|
|
|
| RecC dataConName varBangTypes <- cons' ->
|
|
|
|
withRecordConstructor dataConName varBangTypes
|
2024-07-17 09:37:29 +02:00
|
|
|
_ -> error "Only objects, unions and enums are supported if all member types have a ToGraphQL instance"
|
2024-07-07 12:55:42 +02:00
|
|
|
where
|
|
|
|
conTName = conT typeName
|
|
|
|
collectEnumMemberNames (NormalC normalName []) = Just normalName
|
|
|
|
collectEnumMemberNames _ = Nothing
|
|
|
|
collectUnionMembers (NormalC normalName [_]) = Just normalName
|
|
|
|
collectUnionMembers _ = Nothing
|
|
|
|
enumMemberPattern normalName
|
|
|
|
= flip (clause [conP normalName mempty]) []
|
|
|
|
$ normalB [|Type.Enum $(stringLE normalName)|]
|
|
|
|
unionMemberPattern normalName = do
|
|
|
|
dataName <- newName "member"
|
|
|
|
flip (clause [conP normalName [varP dataName]]) []
|
|
|
|
$ normalB
|
|
|
|
$ appE (varE $ mkName "toGraphQL")
|
|
|
|
$ varE dataName
|
|
|
|
generateSumTypeInstance cons'
|
|
|
|
| Just enumMemberNames <- traverse collectEnumMemberNames cons' =
|
|
|
|
enumMemberPattern <$> enumMemberNames
|
|
|
|
| Just unionMembers <- traverse collectUnionMembers cons' =
|
|
|
|
unionMemberPattern <$> unionMembers
|
2024-07-17 09:37:29 +02:00
|
|
|
| otherwise = error "All data constructors should have either no parameters (Enum) or one parameter (Union)"
|
2024-07-07 12:55:42 +02:00
|
|
|
withRecordConstructor dataConName varBangTypes = do
|
|
|
|
fieldAliases <- traverse newFieldAliases varBangTypes
|
|
|
|
let fBody =
|
|
|
|
[| Type.Object
|
|
|
|
$ HashMap.insert "__typename" $(stringLE typeName)
|
|
|
|
$ HashMap.fromList $(listE $ resultObjectPairs <$> fieldAliases)
|
|
|
|
|]
|
|
|
|
toGraphQLF = recP dataConName (newFieldPatterns <$> fieldAliases)
|
|
|
|
[d|
|
|
|
|
instance ToGraphQL $conTName
|
|
|
|
where
|
|
|
|
toGraphQL $toGraphQLF = $fBody
|
|
|
|
|]
|
|
|
|
newFieldAliases :: VarBangType -> Q (Name, Name)
|
|
|
|
newFieldAliases (name', _, _) = (name',) <$> newName (nameBase name')
|
|
|
|
newFieldPatterns (name', alias) = (name',) <$> varP alias
|
|
|
|
resultObjectPairs :: (Name, Name) -> Q Exp
|
|
|
|
resultObjectPairs (name', alias) = tupE
|
|
|
|
[ litE (stringL $ nameBase name')
|
|
|
|
, [|toGraphQL $(varE alias)|]
|
|
|
|
]
|
2024-10-20 17:13:39 +02:00
|
|
|
|
|
|
|
stripIndentation :: String -> String
|
|
|
|
stripIndentation code = reverse
|
|
|
|
$ dropWhile isLineBreak
|
|
|
|
$ reverse
|
|
|
|
$ unlines
|
|
|
|
$ indent spaces <$> lines' withoutLeadingNewlines
|
|
|
|
where
|
|
|
|
indent 0 xs = xs
|
|
|
|
indent count (' ' : xs) = indent (count - 1) xs
|
|
|
|
indent _ xs = xs
|
|
|
|
withoutLeadingNewlines = dropWhile isLineBreak code
|
|
|
|
spaces = length $ takeWhile (== ' ') withoutLeadingNewlines
|
|
|
|
lines' "" = []
|
|
|
|
lines' string =
|
|
|
|
let (line, rest) = break isLineBreak string
|
|
|
|
reminder =
|
|
|
|
case rest of
|
|
|
|
[] -> []
|
|
|
|
'\r' : '\n' : strippedString -> lines' strippedString
|
|
|
|
_ : strippedString -> lines' strippedString
|
|
|
|
in line : reminder
|
|
|
|
isLineBreak = flip any ['\n', '\r'] . (==)
|
|
|
|
|
|
|
|
-- | Removes leading and trailing newlines. Indentation of the first line is
|
|
|
|
-- removed from each line of the string.
|
|
|
|
gql :: QuasiQuoter
|
|
|
|
gql = QuasiQuoter
|
|
|
|
{ quoteExp = pure . LitE . StringL . stripIndentation
|
|
|
|
, quotePat = const
|
|
|
|
$ fail "Illegal gql QuasiQuote (allowed as expression only, used as a pattern)"
|
|
|
|
, quoteType = const
|
|
|
|
$ fail "Illegal gql QuasiQuote (allowed as expression only, used as a type)"
|
|
|
|
, quoteDec = const
|
|
|
|
$ fail "Illegal gql QuasiQuote (allowed as expression only, used as a declaration)"
|
|
|
|
}
|