Add deriveFromGraphQL
For deriving FromGraphQL instances automatically.
This commit is contained in:
parent
11ab7e18e1
commit
cf029961e8
|
@ -7,28 +7,14 @@ on:
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
audit:
|
audit:
|
||||||
runs-on: haskell
|
runs-on: buildenv
|
||||||
steps:
|
steps:
|
||||||
- name: Set up environment
|
|
||||||
run: |
|
|
||||||
apt-get update -y
|
|
||||||
apt-get upgrade -y
|
|
||||||
apt-get install -y nodejs pkg-config
|
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
- name: Install dependencies
|
- run: hlint -- src tests
|
||||||
run: |
|
|
||||||
cabal update
|
|
||||||
cabal install hlint "--constraint=hlint ==3.8"
|
|
||||||
- run: cabal exec hlint -- src tests
|
|
||||||
|
|
||||||
test:
|
test:
|
||||||
runs-on: haskell
|
runs-on: buildenv
|
||||||
steps:
|
steps:
|
||||||
- name: Set up environment
|
|
||||||
run: |
|
|
||||||
apt-get update -y
|
|
||||||
apt-get upgrade -y
|
|
||||||
apt-get install -y nodejs pkg-config
|
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
run: cabal update
|
run: cabal update
|
||||||
|
@ -37,13 +23,8 @@ jobs:
|
||||||
- run: cabal test --test-show-details=streaming
|
- run: cabal test --test-show-details=streaming
|
||||||
|
|
||||||
doc:
|
doc:
|
||||||
runs-on: haskell
|
runs-on: buildenv
|
||||||
steps:
|
steps:
|
||||||
- name: Set up environment
|
|
||||||
run: |
|
|
||||||
apt-get update -y
|
|
||||||
apt-get upgrade -y
|
|
||||||
apt-get install -y nodejs pkg-config
|
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
run: cabal update
|
run: cabal update
|
||||||
|
|
|
@ -9,6 +9,7 @@ and this project adheres to
|
||||||
## [Unreleased]
|
## [Unreleased]
|
||||||
### Added
|
### Added
|
||||||
- Add `deriveToGraphQL` for deriving `ToGraphQL` instances automatically.
|
- Add `deriveToGraphQL` for deriving `ToGraphQL` instances automatically.
|
||||||
|
- Add `deriveFromGraphQL`for deriving `FromGraphQL` instances automatically.
|
||||||
|
|
||||||
## [1.0.2.0] - 2023-07-07
|
## [1.0.2.0] - 2023-07-07
|
||||||
### Added
|
### Added
|
||||||
|
|
|
@ -11,6 +11,7 @@
|
||||||
module Language.GraphQL.Class
|
module Language.GraphQL.Class
|
||||||
( FromGraphQL(..)
|
( FromGraphQL(..)
|
||||||
, ToGraphQL(..)
|
, ToGraphQL(..)
|
||||||
|
, deriveFromGraphQL
|
||||||
, deriveToGraphQL
|
, deriveToGraphQL
|
||||||
) where
|
) where
|
||||||
|
|
||||||
|
@ -66,6 +67,10 @@ import Language.Haskell.TH
|
||||||
, normalB
|
, normalB
|
||||||
, appE
|
, appE
|
||||||
, mkName
|
, mkName
|
||||||
|
, conE
|
||||||
|
, integerL
|
||||||
|
, litP
|
||||||
|
, wildP
|
||||||
)
|
)
|
||||||
import Data.Foldable (Foldable(..))
|
import Data.Foldable (Foldable(..))
|
||||||
import qualified Data.HashMap.Strict as HashMap
|
import qualified Data.HashMap.Strict as HashMap
|
||||||
|
@ -306,6 +311,57 @@ instance FromGraphQL LocalTime
|
||||||
stringLE :: Name -> Q Exp
|
stringLE :: Name -> Q Exp
|
||||||
stringLE = litE . stringL . nameBase
|
stringLE = litE . stringL . nameBase
|
||||||
|
|
||||||
|
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
|
||||||
|
_ -> error "Only records with a single data constructor are supported"
|
||||||
|
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
|
||||||
|
|
||||||
deriveToGraphQL :: Name -> Q [Dec]
|
deriveToGraphQL :: Name -> Q [Dec]
|
||||||
deriveToGraphQL typeName = do
|
deriveToGraphQL typeName = do
|
||||||
TyConI plainConstructor <- reify typeName
|
TyConI plainConstructor <- reify typeName
|
||||||
|
|
|
@ -12,16 +12,22 @@ import Data.Text (Text)
|
||||||
import Data.Time (UTCTime(..))
|
import Data.Time (UTCTime(..))
|
||||||
import Data.Time.Calendar.OrdinalDate (fromOrdinalDate)
|
import Data.Time.Calendar.OrdinalDate (fromOrdinalDate)
|
||||||
import qualified Language.GraphQL.Type as Type
|
import qualified Language.GraphQL.Type as Type
|
||||||
import Language.GraphQL.Class (FromGraphQL(..), ToGraphQL(..), deriveToGraphQL)
|
import Language.GraphQL.Class
|
||||||
|
( FromGraphQL(..)
|
||||||
|
, ToGraphQL(..)
|
||||||
|
, deriveFromGraphQL
|
||||||
|
, deriveToGraphQL
|
||||||
|
)
|
||||||
import Test.Hspec (Spec, describe, it, shouldBe)
|
import Test.Hspec (Spec, describe, it, shouldBe)
|
||||||
import qualified Data.HashMap.Strict as HashMap
|
import qualified Data.HashMap.Strict as HashMap
|
||||||
|
|
||||||
data TwoFieldRecord = TwoFieldRecord
|
data TwoFieldRecord = TwoFieldRecord
|
||||||
{ x :: Int
|
{ x :: Int
|
||||||
, y :: Bool
|
, y :: Bool
|
||||||
}
|
} deriving (Eq, Show)
|
||||||
|
|
||||||
$(deriveToGraphQL ''TwoFieldRecord)
|
$(deriveToGraphQL ''TwoFieldRecord)
|
||||||
|
$(deriveFromGraphQL ''TwoFieldRecord)
|
||||||
|
|
||||||
spec :: Spec
|
spec :: Spec
|
||||||
spec = do
|
spec = do
|
||||||
|
@ -76,7 +82,7 @@ spec = do
|
||||||
in actual `shouldBe` expected
|
in actual `shouldBe` expected
|
||||||
|
|
||||||
describe "deriveToGraphQL" $ do
|
describe "deriveToGraphQL" $ do
|
||||||
it "derives ToGraphQL for a record" $ do
|
it "derives ToGraphQL for a record with multiple fields" $ do
|
||||||
let expected = Type.Object $ HashMap.fromList
|
let expected = Type.Object $ HashMap.fromList
|
||||||
[ ("x", Type.Int 1)
|
[ ("x", Type.Int 1)
|
||||||
, ("y", Type.Boolean True)
|
, ("y", Type.Boolean True)
|
||||||
|
@ -87,3 +93,15 @@ spec = do
|
||||||
, y = True
|
, y = True
|
||||||
}
|
}
|
||||||
in toGraphQL given `shouldBe` expected
|
in toGraphQL given `shouldBe` expected
|
||||||
|
|
||||||
|
describe "deriveFromGraphQL" $ do
|
||||||
|
it "derives FromGraphQL for a record with multiple fields" $ do
|
||||||
|
let given = Type.Object $ HashMap.fromList
|
||||||
|
[ ("x", Type.Int 1)
|
||||||
|
, ("y", Type.Boolean True)
|
||||||
|
]
|
||||||
|
expected = TwoFieldRecord
|
||||||
|
{ x = 1
|
||||||
|
, y = True
|
||||||
|
}
|
||||||
|
in fromGraphQL given `shouldBe` Just expected
|
||||||
|
|
Loading…
Reference in a new issue