Add gql quasi quoter
This commit is contained in:
parent
ce5fa260f4
commit
d280cd835f
|
@ -7,6 +7,10 @@ and this project adheres to
|
||||||
[Haskell Package Versioning Policy](https://pvp.haskell.org/).
|
[Haskell Package Versioning Policy](https://pvp.haskell.org/).
|
||||||
|
|
||||||
## [Unreleased]
|
## [Unreleased]
|
||||||
|
### 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
|
||||||
|
|
|
@ -16,7 +16,7 @@ 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
|
||||||
|
|
|
@ -13,6 +13,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 +46,7 @@ import Language.Haskell.TH
|
||||||
, Dec(..)
|
, Dec(..)
|
||||||
, Exp(..)
|
, Exp(..)
|
||||||
, Info(..)
|
, Info(..)
|
||||||
|
, Lit(..)
|
||||||
, Quote(..)
|
, Quote(..)
|
||||||
, Name
|
, Name
|
||||||
, Q
|
, Q
|
||||||
|
@ -72,6 +74,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
|
||||||
|
@ -438,3 +441,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)"
|
||||||
|
}
|
||||||
|
|
|
@ -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
|
||||||
|
|
Loading…
Reference in a new issue