7 Commits

Author SHA1 Message Date
c95a5fcd61 Allow graphql 1.4.0.0
All checks were successful
Build / audit (push) Successful in 6s
Build / test (push) Successful in 7m37s
Build / doc (push) Successful in 7m22s
Release / release (push) Successful in 5s
2024-11-21 17:14:03 +01:00
8417be25d7 Release 1.0.5.0
All checks were successful
Build / audit (push) Successful in 8s
Build / test (push) Successful in 7m59s
Build / doc (push) Successful in 8m5s
Release / release (push) Successful in 4s
2024-11-21 15:59:44 +01:00
8aa2e521c4 Add a few missing instances
All checks were successful
Build / audit (pull_request) Successful in 10s
Build / test (pull_request) Successful in 6m59s
Build / doc (pull_request) Successful in 7m7s
Build / audit (push) Successful in 6s
Build / test (push) Successful in 6m59s
Build / doc (push) Successful in 7m2s
2024-11-19 20:21:41 +01:00
e7fbf8b88a Release 1.0.4.0
All checks were successful
Build / audit (push) Successful in 7s
Build / test (push) Successful in 7m21s
Build / doc (push) Successful in 7m17s
Release / release (push) Successful in 5s
2024-10-24 16:23:33 +02:00
d280cd835f Add gql quasi quoter
All checks were successful
Build / audit (push) Successful in 7s
Build / test (push) Successful in 7m8s
Build / doc (push) Successful in 6m58s
2024-10-20 17:13:39 +02:00
ce5fa260f4 Build only after pushing to a branch
All checks were successful
Build / audit (push) Successful in 6s
Build / test (push) Successful in 7m23s
Build / doc (push) Successful in 7m22s
2024-07-26 09:30:38 +02:00
7295681440 Add missing bounds
All checks were successful
Build / audit (push) Successful in 8s
Build / test (push) Successful in 8m8s
Build / doc (push) Successful in 7m49s
2024-07-24 12:11:00 +02:00
7 changed files with 111 additions and 7 deletions

3
.gitea/deploy.awk Normal file
View File

@ -0,0 +1,3 @@
END {
system("cabal upload --username belka --password "ENVIRON["HACKAGE_PASSWORD"]" "$0)
}

View File

@ -2,6 +2,8 @@ name: Build
on: on:
push: push:
branches:
- '**'
pull_request: pull_request:
branches: [master] branches: [master]

View File

@ -0,0 +1,17 @@
name: Release
on:
push:
tags:
- '**'
jobs:
release:
runs-on: buildenv
steps:
- uses: actions/checkout@v4
- name: Upload a candidate
env:
HACKAGE_PASSWORD: ${{ secrets.HACKAGE_PASSWORD }}
run: |
cabal sdist | awk -f .gitea/deploy.awk

View File

@ -6,6 +6,16 @@ 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/).
## [1.0.5.0] - 2024-11-21
### Added
- Add `ToGraphQL` and `FromGraphQL` instances for `Value` and `HashMap`.
## [1.0.4.0] - 2024-10-24
### Added
- `gql` quasi quoter which generates a string literal with the first line
starting at the first column and all following lines indented relative to the
first line.
## [1.0.3.0] - 2024-07-20 ## [1.0.3.0] - 2024-07-20
### Added ### Added
- Add `deriveToGraphQL` for deriving `ToGraphQL` instances automatically. - Add `deriveToGraphQL` for deriving `ToGraphQL` instances automatically.
@ -31,6 +41,8 @@ and this project adheres to
- JSON serialization. - JSON serialization.
- Test helpers. - Test helpers.
[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.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.0.1.0]: https://git.caraus.tech/OSS/graphql-spice/compare/v1.0.0.0...v1.0.1.0 [1.0.1.0]: https://git.caraus.tech/OSS/graphql-spice/compare/v1.0.0.0...v1.0.1.0

View File

@ -1,7 +1,7 @@
cabal-version: 2.4 cabal-version: 3.0
name: graphql-spice name: graphql-spice
version: 1.0.3.0 version: 1.0.5.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-2023 Eugen Wissner copyright: (c) 2021-2024 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.4.8 GHC == 9.8.2
source-repository head source-repository head
type: git type: git
@ -35,15 +35,15 @@ library
aeson >= 2.0.3 && < 2.3, aeson >= 2.0.3 && < 2.3,
base >= 4.7 && < 5, base >= 4.7 && < 5,
conduit ^>= 1.3.4, conduit ^>= 1.3.4,
containers ^>= 0.6.2, 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.2, graphql >= 1.3.0 && < 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,
text >= 1.2 && < 3, text >= 1.2 && < 3,
time >= 1.11.1, time >= 1.12.2 && < 1.15,
transformers >= 0.5.6 && < 0.7, transformers >= 0.5.6 && < 0.7,
vector >= 0.12 && < 0.14, vector >= 0.12 && < 0.14,
unordered-containers ^>= 0.2.16 unordered-containers ^>= 0.2.16

View File

@ -5,6 +5,7 @@
{-# LANGUAGE OverloadedStrings #-} {-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE TemplateHaskell #-} {-# LANGUAGE TemplateHaskell #-}
{-# LANGUAGE TupleSections #-} {-# LANGUAGE TupleSections #-}
{-# LANGUAGE FlexibleInstances #-}
-- | ToGraphQL and FromGraphQL typeclasses used for user-defined type -- | ToGraphQL and FromGraphQL typeclasses used for user-defined type
-- conversion. -- conversion.
@ -13,6 +14,7 @@ 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)
@ -45,6 +47,7 @@ import Language.Haskell.TH
, Dec(..) , Dec(..)
, Exp(..) , Exp(..)
, Info(..) , Info(..)
, Lit(..)
, Quote(..) , Quote(..)
, Name , Name
, Q , Q
@ -72,6 +75,7 @@ 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
@ -98,6 +102,10 @@ class ToGraphQL a
where where
toGraphQL :: a -> Type.Value toGraphQL :: a -> Type.Value
instance ToGraphQL Type.Value
where
toGraphQL a = a
instance ToGraphQL Text instance ToGraphQL Text
where where
toGraphQL = Type.String toGraphQL = Type.String
@ -195,12 +203,20 @@ instance ToGraphQL LocalTime
where where
toGraphQL = iso8601ToGraphQL toGraphQL = iso8601ToGraphQL
instance ToGraphQL a => ToGraphQL (HashMap.HashMap Text a)
where
toGraphQL = Type.Object . fmap toGraphQL
-- | Instances of this typeclass can be used to convert GraphQL internal -- | Instances of this typeclass can be used to convert GraphQL internal
-- representation to user-defined type. -- representation to user-defined type.
class FromGraphQL a class FromGraphQL a
where where
fromGraphQL :: Type.Value -> Maybe a fromGraphQL :: Type.Value -> Maybe a
instance FromGraphQL Type.Value
where
fromGraphQL = Just
instance FromGraphQL Text instance FromGraphQL Text
where where
fromGraphQL (Type.String value) = Just value fromGraphQL (Type.String value) = Just value
@ -308,6 +324,11 @@ instance FromGraphQL LocalTime
where where
fromGraphQL = fromGraphQLToISO8601 fromGraphQL = fromGraphQLToISO8601
instance FromGraphQL a => FromGraphQL (HashMap.HashMap Text a)
where
fromGraphQL (Type.Object hm) = traverse fromGraphQL hm
fromGraphQL _ = Nothing
stringLE :: Name -> Q Exp stringLE :: Name -> Q Exp
stringLE = litE . stringL . nameBase stringLE = litE . stringL . nameBase
@ -438,3 +459,39 @@ 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)"
}

View File

@ -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 #-}
{-# LANGUAGE QuasiQuotes #-}
{-# LANGUAGE TemplateHaskell #-} {-# LANGUAGE TemplateHaskell #-}
module Language.GraphQL.ClassSpec module Language.GraphQL.ClassSpec
( spec ( spec
@ -17,6 +18,7 @@ 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
@ -159,3 +161,14 @@ 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