2019-07-22 05:50:00 +02:00
|
|
|
{-# LANGUAGE OverloadedStrings #-}
|
2019-09-06 07:48:01 +02:00
|
|
|
{-# LANGUAGE QuasiQuotes #-}
|
2019-11-03 10:42:10 +01:00
|
|
|
module Language.GraphQL.AST.ParserSpec
|
2019-07-22 05:50:00 +02:00
|
|
|
( spec
|
|
|
|
) where
|
|
|
|
|
2020-01-25 16:37:17 +01:00
|
|
|
import Data.List.NonEmpty (NonEmpty(..))
|
|
|
|
import Language.GraphQL.AST.Document
|
2021-02-20 16:06:27 +01:00
|
|
|
import qualified Language.GraphQL.AST.DirectiveLocation as DirLoc
|
2019-11-03 10:42:10 +01:00
|
|
|
import Language.GraphQL.AST.Parser
|
2019-09-27 10:50:38 +02:00
|
|
|
import Test.Hspec (Spec, describe, it)
|
2020-05-22 10:11:48 +02:00
|
|
|
import Test.Hspec.Megaparsec (shouldParse, shouldFailOn, shouldSucceedOn)
|
2019-07-22 05:50:00 +02:00
|
|
|
import Text.Megaparsec (parse)
|
2019-09-06 07:48:01 +02:00
|
|
|
import Text.RawString.QQ (r)
|
2019-07-22 05:50:00 +02:00
|
|
|
|
|
|
|
spec :: Spec
|
2019-09-06 07:48:01 +02:00
|
|
|
spec = describe "Parser" $ do
|
2019-07-22 05:50:00 +02:00
|
|
|
it "accepts BOM header" $
|
2019-09-27 10:50:38 +02:00
|
|
|
parse document "" `shouldSucceedOn` "\xfeff{foo}"
|
2019-09-06 07:48:01 +02:00
|
|
|
|
|
|
|
it "accepts block strings as argument" $
|
2019-09-27 10:50:38 +02:00
|
|
|
parse document "" `shouldSucceedOn` [r|{
|
2019-09-06 07:48:01 +02:00
|
|
|
hello(text: """Argument""")
|
2019-09-27 10:50:38 +02:00
|
|
|
}|]
|
2019-09-06 07:48:01 +02:00
|
|
|
|
|
|
|
it "accepts strings as argument" $
|
2019-09-27 10:50:38 +02:00
|
|
|
parse document "" `shouldSucceedOn` [r|{
|
2019-09-06 07:48:01 +02:00
|
|
|
hello(text: "Argument")
|
2019-09-27 10:50:38 +02:00
|
|
|
}|]
|
2019-11-19 09:41:52 +01:00
|
|
|
|
|
|
|
it "accepts two required arguments" $
|
|
|
|
parse document "" `shouldSucceedOn` [r|
|
|
|
|
mutation auth($username: String!, $password: String!){
|
2020-01-05 07:42:04 +01:00
|
|
|
test
|
2019-11-19 09:41:52 +01:00
|
|
|
}|]
|
2019-11-27 09:18:20 +01:00
|
|
|
|
|
|
|
it "accepts two string arguments" $
|
|
|
|
parse document "" `shouldSucceedOn` [r|
|
|
|
|
mutation auth{
|
2020-01-05 07:42:04 +01:00
|
|
|
test(username: "username", password: "password")
|
2019-11-27 09:18:20 +01:00
|
|
|
}|]
|
|
|
|
|
|
|
|
it "accepts two block string arguments" $
|
|
|
|
parse document "" `shouldSucceedOn` [r|
|
|
|
|
mutation auth{
|
2020-01-05 07:42:04 +01:00
|
|
|
test(username: """username""", password: """password""")
|
2019-11-27 09:18:20 +01:00
|
|
|
}|]
|
2020-01-03 07:20:48 +01:00
|
|
|
|
|
|
|
it "parses minimal schema definition" $
|
|
|
|
parse document "" `shouldSucceedOn` [r|schema { query: Query }|]
|
2020-01-05 07:42:04 +01:00
|
|
|
|
|
|
|
it "parses minimal scalar definition" $
|
|
|
|
parse document "" `shouldSucceedOn` [r|scalar Time|]
|
|
|
|
|
|
|
|
it "parses ImplementsInterfaces" $
|
|
|
|
parse document "" `shouldSucceedOn` [r|
|
|
|
|
type Person implements NamedEntity & ValuedEntity {
|
|
|
|
name: String
|
|
|
|
}
|
|
|
|
|]
|
|
|
|
|
|
|
|
it "parses a type without ImplementsInterfaces" $
|
|
|
|
parse document "" `shouldSucceedOn` [r|
|
|
|
|
type Person {
|
|
|
|
name: String
|
|
|
|
}
|
|
|
|
|]
|
|
|
|
|
|
|
|
it "parses ArgumentsDefinition in an ObjectDefinition" $
|
|
|
|
parse document "" `shouldSucceedOn` [r|
|
|
|
|
type Person {
|
|
|
|
name(first: String, last: String): String
|
|
|
|
}
|
|
|
|
|]
|
2020-01-07 13:56:58 +01:00
|
|
|
|
|
|
|
it "parses minimal union type definition" $
|
|
|
|
parse document "" `shouldSucceedOn` [r|
|
|
|
|
union SearchResult = Photo | Person
|
|
|
|
|]
|
2020-01-11 08:32:25 +01:00
|
|
|
|
|
|
|
it "parses minimal interface type definition" $
|
|
|
|
parse document "" `shouldSucceedOn` [r|
|
|
|
|
interface NamedEntity {
|
|
|
|
name: String
|
|
|
|
}
|
|
|
|
|]
|
2020-01-12 07:07:04 +01:00
|
|
|
|
|
|
|
it "parses minimal enum type definition" $
|
|
|
|
parse document "" `shouldSucceedOn` [r|
|
|
|
|
enum Direction {
|
|
|
|
NORTH
|
|
|
|
EAST
|
|
|
|
SOUTH
|
|
|
|
WEST
|
|
|
|
}
|
|
|
|
|]
|
|
|
|
|
|
|
|
it "parses minimal enum type definition" $
|
|
|
|
parse document "" `shouldSucceedOn` [r|
|
|
|
|
enum Direction {
|
|
|
|
NORTH
|
|
|
|
EAST
|
|
|
|
SOUTH
|
|
|
|
WEST
|
|
|
|
}
|
|
|
|
|]
|
|
|
|
|
|
|
|
it "parses minimal input object type definition" $
|
|
|
|
parse document "" `shouldSucceedOn` [r|
|
|
|
|
input Point2D {
|
|
|
|
x: Float
|
|
|
|
y: Float
|
|
|
|
}
|
|
|
|
|]
|
2020-01-15 20:20:50 +01:00
|
|
|
|
|
|
|
it "parses minimal input enum definition with an optional pipe" $
|
|
|
|
parse document "" `shouldSucceedOn` [r|
|
|
|
|
directive @example on
|
|
|
|
| FIELD
|
|
|
|
| FRAGMENT_SPREAD
|
|
|
|
|]
|
2020-01-25 16:37:17 +01:00
|
|
|
|
2021-02-20 16:06:27 +01:00
|
|
|
it "parses two minimal directive definitions" $
|
|
|
|
let directive nm loc =
|
|
|
|
TypeSystemDefinition
|
|
|
|
(DirectiveDefinition
|
|
|
|
(Description Nothing)
|
|
|
|
nm
|
|
|
|
(ArgumentsDefinition [])
|
|
|
|
(loc :| []))
|
|
|
|
example1 =
|
|
|
|
directive "example1"
|
|
|
|
(DirLoc.TypeSystemDirectiveLocation DirLoc.FieldDefinition)
|
|
|
|
(Location {line = 2, column = 17})
|
|
|
|
example2 =
|
|
|
|
directive "example2"
|
|
|
|
(DirLoc.ExecutableDirectiveLocation DirLoc.Field)
|
|
|
|
(Location {line = 3, column = 17})
|
|
|
|
testSchemaExtension = example1 :| [ example2 ]
|
|
|
|
query = [r|
|
|
|
|
directive @example1 on FIELD_DEFINITION
|
|
|
|
directive @example2 on FIELD
|
|
|
|
|]
|
|
|
|
in parse document "" query `shouldParse` testSchemaExtension
|
|
|
|
|
|
|
|
it "parses a directive definition with a default empty list argument" $
|
|
|
|
let directive nm loc args =
|
|
|
|
TypeSystemDefinition
|
|
|
|
(DirectiveDefinition
|
|
|
|
(Description Nothing)
|
|
|
|
nm
|
|
|
|
(ArgumentsDefinition
|
|
|
|
[ InputValueDefinition
|
|
|
|
(Description Nothing)
|
|
|
|
argName
|
|
|
|
argType
|
|
|
|
argValue
|
|
|
|
[]
|
|
|
|
| (argName, argType, argValue) <- args])
|
|
|
|
(loc :| []))
|
|
|
|
defn =
|
|
|
|
directive "test"
|
|
|
|
(DirLoc.TypeSystemDirectiveLocation DirLoc.FieldDefinition)
|
|
|
|
[("foo",
|
|
|
|
TypeList (TypeNamed "String"),
|
|
|
|
Just
|
|
|
|
$ Node (ConstList [])
|
|
|
|
$ Location {line = 1, column = 33})]
|
|
|
|
(Location {line = 1, column = 1})
|
|
|
|
query = [r|directive @test(foo: [String] = []) on FIELD_DEFINITION|]
|
|
|
|
in parse document "" query `shouldParse` (defn :| [ ])
|
|
|
|
|
2020-01-25 16:37:17 +01:00
|
|
|
it "parses schema extension with a new directive" $
|
|
|
|
parse document "" `shouldSucceedOn`[r|
|
|
|
|
extend schema @newDirective
|
|
|
|
|]
|
|
|
|
|
|
|
|
it "parses schema extension with an operation type definition" $
|
|
|
|
parse document "" `shouldSucceedOn` [r|extend schema { query: Query }|]
|
|
|
|
|
|
|
|
it "parses schema extension with an operation type and directive" $
|
2020-09-18 07:32:58 +02:00
|
|
|
let newDirective = Directive "newDirective" [] $ Location 1 15
|
2020-07-20 21:29:12 +02:00
|
|
|
schemaExtension = SchemaExtension
|
2020-01-25 16:37:17 +01:00
|
|
|
$ SchemaOperationExtension [newDirective]
|
|
|
|
$ OperationTypeDefinition Query "Query" :| []
|
2020-07-20 21:29:12 +02:00
|
|
|
testSchemaExtension = TypeSystemExtension schemaExtension
|
|
|
|
$ Location 1 1
|
2020-01-25 16:37:17 +01:00
|
|
|
query = [r|extend schema @newDirective { query: Query }|]
|
|
|
|
in parse document "" query `shouldParse` (testSchemaExtension :| [])
|
2020-01-28 11:08:28 +01:00
|
|
|
|
|
|
|
it "parses an object extension" $
|
|
|
|
parse document "" `shouldSucceedOn` [r|
|
|
|
|
extend type Story {
|
|
|
|
isHiddenLocally: Boolean
|
|
|
|
}
|
2020-05-22 10:11:48 +02:00
|
|
|
|]
|
|
|
|
|
|
|
|
it "rejects variables in DefaultValue" $
|
|
|
|
parse document "" `shouldFailOn` [r|
|
|
|
|
query ($book: String = "Zarathustra", $author: String = $book) {
|
|
|
|
title
|
|
|
|
}
|
|
|
|
|]
|
2020-07-09 08:11:12 +02:00
|
|
|
|
|
|
|
it "parses documents beginning with a comment" $
|
|
|
|
parse document "" `shouldSucceedOn` [r|
|
|
|
|
"""
|
|
|
|
Query
|
|
|
|
"""
|
|
|
|
type Query {
|
|
|
|
queryField: String
|
|
|
|
}
|
|
|
|
|]
|
2020-07-11 06:34:10 +02:00
|
|
|
|
|
|
|
it "parses subscriptions" $
|
|
|
|
parse document "" `shouldSucceedOn` [r|
|
|
|
|
subscription NewMessages {
|
|
|
|
newMessage(roomId: 123) {
|
|
|
|
sender
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|]
|