forked from OSS/graphql-spice
Compare commits
6 Commits
Author | SHA1 | Date | |
---|---|---|---|
e3d3cb93b2 | |||
88f41271eb | |||
b399bddb90 | |||
131576e56c | |||
c95a5fcd61 | |||
8417be25d7 |
19
CHANGELOG.md
19
CHANGELOG.md
@ -6,6 +6,22 @@ The format is based on
|
|||||||
and this project adheres to
|
and this project adheres to
|
||||||
[Haskell Package Versioning Policy](https://pvp.haskell.org/).
|
[Haskell Package Versioning Policy](https://pvp.haskell.org/).
|
||||||
|
|
||||||
|
## [Unreleased]
|
||||||
|
### Removed
|
||||||
|
- Deprecated `Language.GraphQL.Class.gql` (moved to `Language.GraphQL.TH`).
|
||||||
|
|
||||||
|
### Added
|
||||||
|
- `String` instances from `ToGraphQL` and `FromGraphQL`.
|
||||||
|
|
||||||
|
## [1.0.6.0] - 2024-12-06
|
||||||
|
### Added
|
||||||
|
- `Language.GraphQL.Class.gql` is moved to `Language.GraphQL.TH` where it was
|
||||||
|
before in `graphql`.
|
||||||
|
|
||||||
|
## [1.0.5.0] - 2024-11-21
|
||||||
|
### Added
|
||||||
|
- Add `ToGraphQL` and `FromGraphQL` instances for `Value` and `HashMap`.
|
||||||
|
|
||||||
## [1.0.4.0] - 2024-10-24
|
## [1.0.4.0] - 2024-10-24
|
||||||
### Added
|
### Added
|
||||||
- `gql` quasi quoter which generates a string literal with the first line
|
- `gql` quasi quoter which generates a string literal with the first line
|
||||||
@ -37,6 +53,9 @@ and this project adheres to
|
|||||||
- JSON serialization.
|
- JSON serialization.
|
||||||
- Test helpers.
|
- Test helpers.
|
||||||
|
|
||||||
|
[Unreleased]: https://git.caraus.tech/OSS/graphql-spice/compare/v1.0.6.0...master
|
||||||
|
[1.0.6.0]: https://git.caraus.tech/OSS/graphql-spice/compare/v1.0.5.0...v1.0.6.0
|
||||||
|
[1.0.5.0]: https://git.caraus.tech/OSS/graphql-spice/compare/v1.0.4.0...v1.0.5.0
|
||||||
[1.0.4.0]: https://git.caraus.tech/OSS/graphql-spice/compare/v1.0.3.0...v1.0.4.0
|
[1.0.4.0]: https://git.caraus.tech/OSS/graphql-spice/compare/v1.0.3.0...v1.0.4.0
|
||||||
[1.0.3.0]: https://git.caraus.tech/OSS/graphql-spice/compare/v1.0.2.0...v1.0.3.0
|
[1.0.3.0]: https://git.caraus.tech/OSS/graphql-spice/compare/v1.0.2.0...v1.0.3.0
|
||||||
[1.0.2.0]: https://git.caraus.tech/OSS/graphql-spice/compare/v1.0.1.0...v1.0.2.0
|
[1.0.2.0]: https://git.caraus.tech/OSS/graphql-spice/compare/v1.0.1.0...v1.0.2.0
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
cabal-version: 3.0
|
cabal-version: 3.0
|
||||||
|
|
||||||
name: graphql-spice
|
name: graphql-spice
|
||||||
version: 1.0.4.0
|
version: 1.0.6.0
|
||||||
synopsis: GraphQL with batteries
|
synopsis: GraphQL with batteries
|
||||||
description: Various extensions and convenience functions for the core
|
description: Various extensions and convenience functions for the core
|
||||||
graphql package.
|
graphql package.
|
||||||
@ -10,13 +10,13 @@ homepage: https://git.caraus.tech/OSS/graphql-spice
|
|||||||
bug-reports: https://git.caraus.tech/OSS/graphql-spice/issues
|
bug-reports: https://git.caraus.tech/OSS/graphql-spice/issues
|
||||||
author: Eugen Wissner <belka@caraus.de>
|
author: Eugen Wissner <belka@caraus.de>
|
||||||
maintainer: belka@caraus.de
|
maintainer: belka@caraus.de
|
||||||
copyright: (c) 2021-2024 Eugen Wissner
|
copyright: (c) 2021-2025 Eugen Wissner
|
||||||
license: MPL-2.0
|
license: MPL-2.0
|
||||||
license-files: LICENSE
|
license-files: LICENSE
|
||||||
build-type: Simple
|
build-type: Simple
|
||||||
extra-source-files: CHANGELOG.md
|
extra-source-files: CHANGELOG.md
|
||||||
tested-with:
|
tested-with:
|
||||||
GHC == 9.8.2
|
GHC == 9.10.1
|
||||||
|
|
||||||
source-repository head
|
source-repository head
|
||||||
type: git
|
type: git
|
||||||
@ -27,6 +27,7 @@ library
|
|||||||
Language.GraphQL.Class
|
Language.GraphQL.Class
|
||||||
Language.GraphQL.JSON
|
Language.GraphQL.JSON
|
||||||
Language.GraphQL.Resolver
|
Language.GraphQL.Resolver
|
||||||
|
Language.GraphQL.TH
|
||||||
Test.Hspec.GraphQL
|
Test.Hspec.GraphQL
|
||||||
other-modules:
|
other-modules:
|
||||||
hs-source-dirs: src
|
hs-source-dirs: src
|
||||||
@ -38,7 +39,7 @@ library
|
|||||||
containers >= 0.6 && < 0.8,
|
containers >= 0.6 && < 0.8,
|
||||||
exceptions ^>= 0.10.4,
|
exceptions ^>= 0.10.4,
|
||||||
hspec-expectations >= 0.8.2 && < 0.9,
|
hspec-expectations >= 0.8.2 && < 0.9,
|
||||||
graphql ^>= 1.3.0,
|
graphql ^>= 1.5.0,
|
||||||
megaparsec >= 9.0 && < 10,
|
megaparsec >= 9.0 && < 10,
|
||||||
scientific ^>= 0.3.7,
|
scientific ^>= 0.3.7,
|
||||||
template-haskell >= 2.16 && < 3,
|
template-haskell >= 2.16 && < 3,
|
||||||
@ -58,6 +59,7 @@ test-suite graphql-test
|
|||||||
Language.GraphQL.DirectiveSpec
|
Language.GraphQL.DirectiveSpec
|
||||||
Language.GraphQL.FragmentSpec
|
Language.GraphQL.FragmentSpec
|
||||||
Language.GraphQL.RootOperationSpec
|
Language.GraphQL.RootOperationSpec
|
||||||
|
Language.GraphQL.THSpec
|
||||||
hs-source-dirs:
|
hs-source-dirs:
|
||||||
tests
|
tests
|
||||||
ghc-options: -threaded -rtsopts -with-rtsopts=-N -Wall
|
ghc-options: -threaded -rtsopts -with-rtsopts=-N -Wall
|
||||||
|
@ -14,7 +14,6 @@ module Language.GraphQL.Class
|
|||||||
, ToGraphQL(..)
|
, ToGraphQL(..)
|
||||||
, deriveFromGraphQL
|
, deriveFromGraphQL
|
||||||
, deriveToGraphQL
|
, deriveToGraphQL
|
||||||
, gql
|
|
||||||
) where
|
) where
|
||||||
|
|
||||||
import Data.Int (Int8, Int16, Int32, Int64)
|
import Data.Int (Int8, Int16, Int32, Int64)
|
||||||
@ -47,7 +46,6 @@ import Language.Haskell.TH
|
|||||||
, Dec(..)
|
, Dec(..)
|
||||||
, Exp(..)
|
, Exp(..)
|
||||||
, Info(..)
|
, Info(..)
|
||||||
, Lit(..)
|
|
||||||
, Quote(..)
|
, Quote(..)
|
||||||
, Name
|
, Name
|
||||||
, Q
|
, Q
|
||||||
@ -75,7 +73,6 @@ import Language.Haskell.TH
|
|||||||
, litP
|
, litP
|
||||||
, wildP
|
, wildP
|
||||||
)
|
)
|
||||||
import Language.Haskell.TH.Quote (QuasiQuoter(..))
|
|
||||||
import Data.Foldable (Foldable(..))
|
import Data.Foldable (Foldable(..))
|
||||||
import qualified Data.HashMap.Strict as HashMap
|
import qualified Data.HashMap.Strict as HashMap
|
||||||
import qualified Language.GraphQL.Type as Type
|
import qualified Language.GraphQL.Type as Type
|
||||||
@ -110,6 +107,10 @@ instance ToGraphQL Text
|
|||||||
where
|
where
|
||||||
toGraphQL = Type.String
|
toGraphQL = Type.String
|
||||||
|
|
||||||
|
instance ToGraphQL String
|
||||||
|
where
|
||||||
|
toGraphQL = Type.String . Text.pack
|
||||||
|
|
||||||
instance ToGraphQL Int
|
instance ToGraphQL Int
|
||||||
where
|
where
|
||||||
toGraphQL = Type.Int . fromIntegral
|
toGraphQL = Type.Int . fromIntegral
|
||||||
@ -222,6 +223,11 @@ instance FromGraphQL Text
|
|||||||
fromGraphQL (Type.String value) = Just value
|
fromGraphQL (Type.String value) = Just value
|
||||||
fromGraphQL _ = Nothing
|
fromGraphQL _ = Nothing
|
||||||
|
|
||||||
|
instance FromGraphQL String
|
||||||
|
where
|
||||||
|
fromGraphQL (Type.String value) = Just $ Text.unpack value
|
||||||
|
fromGraphQL _ = Nothing
|
||||||
|
|
||||||
instance FromGraphQL Int
|
instance FromGraphQL Int
|
||||||
where
|
where
|
||||||
fromGraphQL = fromGraphQLToIntegral
|
fromGraphQL = fromGraphQLToIntegral
|
||||||
@ -459,39 +465,3 @@ deriveToGraphQL typeName = do
|
|||||||
[ litE (stringL $ nameBase name')
|
[ litE (stringL $ nameBase name')
|
||||||
, [|toGraphQL $(varE alias)|]
|
, [|toGraphQL $(varE alias)|]
|
||||||
]
|
]
|
||||||
|
|
||||||
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)"
|
|
||||||
}
|
|
||||||
|
49
src/Language/GraphQL/TH.hs
Normal file
49
src/Language/GraphQL/TH.hs
Normal file
@ -0,0 +1,49 @@
|
|||||||
|
{- 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/. -}
|
||||||
|
|
||||||
|
module Language.GraphQL.TH
|
||||||
|
( gql
|
||||||
|
) where
|
||||||
|
|
||||||
|
import Language.Haskell.TH
|
||||||
|
( Exp(..)
|
||||||
|
, Lit(..)
|
||||||
|
)
|
||||||
|
import Language.Haskell.TH.Quote (QuasiQuoter(..))
|
||||||
|
|
||||||
|
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)"
|
||||||
|
}
|
@ -5,6 +5,7 @@
|
|||||||
{-# LANGUAGE OverloadedStrings #-}
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
{-# LANGUAGE QuasiQuotes #-}
|
{-# LANGUAGE QuasiQuotes #-}
|
||||||
{-# LANGUAGE TemplateHaskell #-}
|
{-# LANGUAGE TemplateHaskell #-}
|
||||||
|
|
||||||
module Language.GraphQL.ClassSpec
|
module Language.GraphQL.ClassSpec
|
||||||
( spec
|
( spec
|
||||||
) where
|
) where
|
||||||
@ -18,7 +19,6 @@ import Language.GraphQL.Class
|
|||||||
, ToGraphQL(..)
|
, ToGraphQL(..)
|
||||||
, deriveFromGraphQL
|
, deriveFromGraphQL
|
||||||
, deriveToGraphQL
|
, deriveToGraphQL
|
||||||
, gql
|
|
||||||
)
|
)
|
||||||
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
|
||||||
@ -161,14 +161,3 @@ spec = do
|
|||||||
let given = Type.Enum "TWO_FIELD_ENUM_2"
|
let given = Type.Enum "TWO_FIELD_ENUM_2"
|
||||||
expected = TWO_FIELD_ENUM_2
|
expected = TWO_FIELD_ENUM_2
|
||||||
in fromGraphQL given `shouldBe` Just expected
|
in fromGraphQL given `shouldBe` Just expected
|
||||||
|
|
||||||
describe "gql" $
|
|
||||||
it "replaces CRNL with NL" $
|
|
||||||
let expected :: Text
|
|
||||||
expected = "line1\nline2\nline3"
|
|
||||||
actual = [gql|
|
|
||||||
line1
|
|
||||||
line2
|
|
||||||
line3
|
|
||||||
|]
|
|
||||||
in actual `shouldBe` expected
|
|
||||||
|
@ -3,6 +3,7 @@
|
|||||||
obtain one at https://mozilla.org/MPL/2.0/. -}
|
obtain one at https://mozilla.org/MPL/2.0/. -}
|
||||||
|
|
||||||
{-# LANGUAGE OverloadedStrings #-}
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
|
|
||||||
module Language.GraphQL.CoerceSpec
|
module Language.GraphQL.CoerceSpec
|
||||||
( spec
|
( spec
|
||||||
) where
|
) where
|
||||||
|
@ -4,7 +4,6 @@
|
|||||||
|
|
||||||
{-# LANGUAGE OverloadedStrings #-}
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
{-# LANGUAGE QuasiQuotes #-}
|
{-# LANGUAGE QuasiQuotes #-}
|
||||||
{-# LANGUAGE PackageImports #-}
|
|
||||||
|
|
||||||
module Language.GraphQL.DirectiveSpec
|
module Language.GraphQL.DirectiveSpec
|
||||||
( spec
|
( spec
|
||||||
@ -18,7 +17,7 @@ import Language.GraphQL.TH
|
|||||||
import Language.GraphQL.Type
|
import Language.GraphQL.Type
|
||||||
import qualified Language.GraphQL.Type.Out as Out
|
import qualified Language.GraphQL.Type.Out as Out
|
||||||
import Test.Hspec (Spec, describe, it)
|
import Test.Hspec (Spec, describe, it)
|
||||||
import "graphql-spice" Test.Hspec.GraphQL
|
import Test.Hspec.GraphQL
|
||||||
|
|
||||||
experimentalResolver :: Schema IO
|
experimentalResolver :: Schema IO
|
||||||
experimentalResolver = schema queryType Nothing Nothing mempty
|
experimentalResolver = schema queryType Nothing Nothing mempty
|
||||||
|
@ -4,7 +4,6 @@
|
|||||||
|
|
||||||
{-# LANGUAGE OverloadedStrings #-}
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
{-# LANGUAGE QuasiQuotes #-}
|
{-# LANGUAGE QuasiQuotes #-}
|
||||||
{-# LANGUAGE PackageImports #-}
|
|
||||||
|
|
||||||
module Language.GraphQL.FragmentSpec
|
module Language.GraphQL.FragmentSpec
|
||||||
( spec
|
( spec
|
||||||
@ -20,7 +19,7 @@ import qualified Language.GraphQL.Type.Out as Out
|
|||||||
import Language.GraphQL.TH
|
import Language.GraphQL.TH
|
||||||
import qualified Language.GraphQL as GraphQL
|
import qualified Language.GraphQL as GraphQL
|
||||||
import Test.Hspec (Spec, describe, it)
|
import Test.Hspec (Spec, describe, it)
|
||||||
import "graphql-spice" Test.Hspec.GraphQL
|
import Test.Hspec.GraphQL
|
||||||
|
|
||||||
size :: (Text, Value)
|
size :: (Text, Value)
|
||||||
size = ("size", String "L")
|
size = ("size", String "L")
|
||||||
|
@ -4,7 +4,6 @@
|
|||||||
|
|
||||||
{-# LANGUAGE OverloadedStrings #-}
|
{-# LANGUAGE OverloadedStrings #-}
|
||||||
{-# LANGUAGE QuasiQuotes #-}
|
{-# LANGUAGE QuasiQuotes #-}
|
||||||
{-# LANGUAGE PackageImports #-}
|
|
||||||
|
|
||||||
module Language.GraphQL.RootOperationSpec
|
module Language.GraphQL.RootOperationSpec
|
||||||
( spec
|
( spec
|
||||||
@ -18,7 +17,7 @@ import Test.Hspec (Spec, describe, it)
|
|||||||
import Language.GraphQL.TH
|
import Language.GraphQL.TH
|
||||||
import Language.GraphQL.Type
|
import Language.GraphQL.Type
|
||||||
import qualified Language.GraphQL.Type.Out as Out
|
import qualified Language.GraphQL.Type.Out as Out
|
||||||
import "graphql-spice" Test.Hspec.GraphQL
|
import Test.Hspec.GraphQL
|
||||||
|
|
||||||
hatType :: Out.ObjectType IO
|
hatType :: Out.ObjectType IO
|
||||||
hatType = Out.ObjectType "Hat" Nothing []
|
hatType = Out.ObjectType "Hat" Nothing []
|
||||||
|
27
tests/Language/GraphQL/THSpec.hs
Normal file
27
tests/Language/GraphQL/THSpec.hs
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
{- 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 #-}
|
||||||
|
{-# LANGUAGE QuasiQuotes #-}
|
||||||
|
|
||||||
|
module Language.GraphQL.THSpec
|
||||||
|
( spec
|
||||||
|
) where
|
||||||
|
|
||||||
|
import Data.Text (Text)
|
||||||
|
import Language.GraphQL.TH (gql)
|
||||||
|
import Test.Hspec (Spec, describe, it, shouldBe)
|
||||||
|
|
||||||
|
spec :: Spec
|
||||||
|
spec = do
|
||||||
|
describe "gql" $
|
||||||
|
it "replaces CRNL with NL" $
|
||||||
|
let expected :: Text
|
||||||
|
expected = "line1\nline2\nline3"
|
||||||
|
actual = [gql|
|
||||||
|
line1
|
||||||
|
line2
|
||||||
|
line3
|
||||||
|
|]
|
||||||
|
in actual `shouldBe` expected
|
Loading…
x
Reference in New Issue
Block a user