rails-graphql 0.1.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (266) hide show
  1. checksums.yaml +7 -0
  2. data/MIT-LICENSE +20 -0
  3. data/README.rdoc +19 -0
  4. data/Rakefile +31 -0
  5. data/ext/depend +3 -0
  6. data/ext/extconf.rb +57 -0
  7. data/ext/graphqlparser/Ast.cpp +346 -0
  8. data/ext/graphqlparser/Ast.h +1214 -0
  9. data/ext/graphqlparser/AstNode.h +36 -0
  10. data/ext/graphqlparser/AstVisitor.h +137 -0
  11. data/ext/graphqlparser/GraphQLParser.cpp +76 -0
  12. data/ext/graphqlparser/GraphQLParser.h +55 -0
  13. data/ext/graphqlparser/JsonVisitor.cpp +161 -0
  14. data/ext/graphqlparser/JsonVisitor.cpp.inc +456 -0
  15. data/ext/graphqlparser/JsonVisitor.h +121 -0
  16. data/ext/graphqlparser/JsonVisitor.h.inc +110 -0
  17. data/ext/graphqlparser/VERSION +1 -0
  18. data/ext/graphqlparser/c/GraphQLAst.cpp +324 -0
  19. data/ext/graphqlparser/c/GraphQLAst.h +180 -0
  20. data/ext/graphqlparser/c/GraphQLAstForEachConcreteType.h +44 -0
  21. data/ext/graphqlparser/c/GraphQLAstNode.cpp +25 -0
  22. data/ext/graphqlparser/c/GraphQLAstNode.h +33 -0
  23. data/ext/graphqlparser/c/GraphQLAstToJSON.cpp +21 -0
  24. data/ext/graphqlparser/c/GraphQLAstToJSON.h +24 -0
  25. data/ext/graphqlparser/c/GraphQLAstVisitor.cpp +55 -0
  26. data/ext/graphqlparser/c/GraphQLAstVisitor.h +53 -0
  27. data/ext/graphqlparser/c/GraphQLParser.cpp +35 -0
  28. data/ext/graphqlparser/c/GraphQLParser.h +54 -0
  29. data/ext/graphqlparser/dump_json_ast.cpp +48 -0
  30. data/ext/graphqlparser/lexer.lpp +324 -0
  31. data/ext/graphqlparser/parser.ypp +693 -0
  32. data/ext/graphqlparser/parsergen/lexer.cpp +2633 -0
  33. data/ext/graphqlparser/parsergen/lexer.h +528 -0
  34. data/ext/graphqlparser/parsergen/location.hh +189 -0
  35. data/ext/graphqlparser/parsergen/parser.tab.cpp +3300 -0
  36. data/ext/graphqlparser/parsergen/parser.tab.hpp +646 -0
  37. data/ext/graphqlparser/parsergen/position.hh +179 -0
  38. data/ext/graphqlparser/parsergen/stack.hh +156 -0
  39. data/ext/graphqlparser/syntaxdefs.h +19 -0
  40. data/ext/libgraphqlparser/AstNode.h +36 -0
  41. data/ext/libgraphqlparser/CMakeLists.txt +148 -0
  42. data/ext/libgraphqlparser/CONTRIBUTING.md +23 -0
  43. data/ext/libgraphqlparser/GraphQLParser.cpp +76 -0
  44. data/ext/libgraphqlparser/GraphQLParser.h +55 -0
  45. data/ext/libgraphqlparser/JsonVisitor.cpp +161 -0
  46. data/ext/libgraphqlparser/JsonVisitor.h +121 -0
  47. data/ext/libgraphqlparser/LICENSE +22 -0
  48. data/ext/libgraphqlparser/README.clang-tidy +7 -0
  49. data/ext/libgraphqlparser/README.md +84 -0
  50. data/ext/libgraphqlparser/ast/ast.ast +203 -0
  51. data/ext/libgraphqlparser/ast/ast.py +61 -0
  52. data/ext/libgraphqlparser/ast/c.py +100 -0
  53. data/ext/libgraphqlparser/ast/c.pyc +0 -0
  54. data/ext/libgraphqlparser/ast/c_impl.py +61 -0
  55. data/ext/libgraphqlparser/ast/c_impl.pyc +0 -0
  56. data/ext/libgraphqlparser/ast/c_visitor_impl.py +39 -0
  57. data/ext/libgraphqlparser/ast/c_visitor_impl.pyc +0 -0
  58. data/ext/libgraphqlparser/ast/casing.py +26 -0
  59. data/ext/libgraphqlparser/ast/casing.pyc +0 -0
  60. data/ext/libgraphqlparser/ast/cxx.py +197 -0
  61. data/ext/libgraphqlparser/ast/cxx.pyc +0 -0
  62. data/ext/libgraphqlparser/ast/cxx_impl.py +61 -0
  63. data/ext/libgraphqlparser/ast/cxx_impl.pyc +0 -0
  64. data/ext/libgraphqlparser/ast/cxx_json_visitor_header.py +42 -0
  65. data/ext/libgraphqlparser/ast/cxx_json_visitor_header.pyc +0 -0
  66. data/ext/libgraphqlparser/ast/cxx_json_visitor_impl.py +80 -0
  67. data/ext/libgraphqlparser/ast/cxx_json_visitor_impl.pyc +0 -0
  68. data/ext/libgraphqlparser/ast/cxx_visitor.py +64 -0
  69. data/ext/libgraphqlparser/ast/cxx_visitor.pyc +0 -0
  70. data/ext/libgraphqlparser/ast/js.py +65 -0
  71. data/ext/libgraphqlparser/ast/license.py +10 -0
  72. data/ext/libgraphqlparser/ast/license.pyc +0 -0
  73. data/ext/libgraphqlparser/c/GraphQLAstNode.cpp +25 -0
  74. data/ext/libgraphqlparser/c/GraphQLAstNode.h +33 -0
  75. data/ext/libgraphqlparser/c/GraphQLAstToJSON.cpp +21 -0
  76. data/ext/libgraphqlparser/c/GraphQLAstToJSON.h +24 -0
  77. data/ext/libgraphqlparser/c/GraphQLAstVisitor.cpp +55 -0
  78. data/ext/libgraphqlparser/c/GraphQLAstVisitor.h +53 -0
  79. data/ext/libgraphqlparser/c/GraphQLParser.cpp +35 -0
  80. data/ext/libgraphqlparser/c/GraphQLParser.h +54 -0
  81. data/ext/libgraphqlparser/clang-tidy-all.sh +3 -0
  82. data/ext/libgraphqlparser/cmake/version.cmake +16 -0
  83. data/ext/libgraphqlparser/dump_json_ast.cpp +48 -0
  84. data/ext/libgraphqlparser/go/README.md +20 -0
  85. data/ext/libgraphqlparser/go/callbacks.go +18 -0
  86. data/ext/libgraphqlparser/go/gotest.go +64 -0
  87. data/ext/libgraphqlparser/lexer.lpp +324 -0
  88. data/ext/libgraphqlparser/libgraphqlparser.pc.in +11 -0
  89. data/ext/libgraphqlparser/parser.ypp +693 -0
  90. data/ext/libgraphqlparser/parsergen/lexer.cpp +2633 -0
  91. data/ext/libgraphqlparser/parsergen/lexer.h +528 -0
  92. data/ext/libgraphqlparser/parsergen/location.hh +189 -0
  93. data/ext/libgraphqlparser/parsergen/parser.tab.cpp +3300 -0
  94. data/ext/libgraphqlparser/parsergen/parser.tab.hpp +646 -0
  95. data/ext/libgraphqlparser/parsergen/position.hh +179 -0
  96. data/ext/libgraphqlparser/parsergen/stack.hh +156 -0
  97. data/ext/libgraphqlparser/python/CMakeLists.txt +14 -0
  98. data/ext/libgraphqlparser/python/README.md +5 -0
  99. data/ext/libgraphqlparser/python/example.py +31 -0
  100. data/ext/libgraphqlparser/syntaxdefs.h +19 -0
  101. data/ext/libgraphqlparser/test/BuildCAPI.c +5 -0
  102. data/ext/libgraphqlparser/test/CMakeLists.txt +25 -0
  103. data/ext/libgraphqlparser/test/JsonVisitorTests.cpp +28 -0
  104. data/ext/libgraphqlparser/test/ParserTests.cpp +352 -0
  105. data/ext/libgraphqlparser/test/kitchen-sink.graphql +59 -0
  106. data/ext/libgraphqlparser/test/kitchen-sink.json +1 -0
  107. data/ext/libgraphqlparser/test/schema-kitchen-sink.graphql +78 -0
  108. data/ext/libgraphqlparser/test/schema-kitchen-sink.json +1 -0
  109. data/ext/libgraphqlparser/test/valgrind.supp +33 -0
  110. data/ext/version.cpp +21 -0
  111. data/lib/generators/graphql/controller_generator.rb +22 -0
  112. data/lib/generators/graphql/schema_generator.rb +22 -0
  113. data/lib/generators/graphql/templates/controller.erb +5 -0
  114. data/lib/generators/graphql/templates/schema.erb +6 -0
  115. data/lib/graphqlparser.so +0 -0
  116. data/lib/rails-graphql.rb +2 -0
  117. data/lib/rails/graphql.rake +1 -0
  118. data/lib/rails/graphql.rb +185 -0
  119. data/lib/rails/graphql/adapters/mysql_adapter.rb +0 -0
  120. data/lib/rails/graphql/adapters/pg_adapter.rb +50 -0
  121. data/lib/rails/graphql/adapters/sqlite_adapter.rb +39 -0
  122. data/lib/rails/graphql/argument.rb +220 -0
  123. data/lib/rails/graphql/callback.rb +124 -0
  124. data/lib/rails/graphql/collectors.rb +14 -0
  125. data/lib/rails/graphql/collectors/hash_collector.rb +83 -0
  126. data/lib/rails/graphql/collectors/idented_collector.rb +73 -0
  127. data/lib/rails/graphql/collectors/json_collector.rb +114 -0
  128. data/lib/rails/graphql/config.rb +61 -0
  129. data/lib/rails/graphql/directive.rb +203 -0
  130. data/lib/rails/graphql/directive/deprecated_directive.rb +59 -0
  131. data/lib/rails/graphql/directive/include_directive.rb +24 -0
  132. data/lib/rails/graphql/directive/skip_directive.rb +24 -0
  133. data/lib/rails/graphql/errors.rb +42 -0
  134. data/lib/rails/graphql/event.rb +141 -0
  135. data/lib/rails/graphql/field.rb +318 -0
  136. data/lib/rails/graphql/field/input_field.rb +92 -0
  137. data/lib/rails/graphql/field/mutation_field.rb +52 -0
  138. data/lib/rails/graphql/field/output_field.rb +96 -0
  139. data/lib/rails/graphql/field/proxied_field.rb +131 -0
  140. data/lib/rails/graphql/field/resolved_field.rb +96 -0
  141. data/lib/rails/graphql/field/scoped_config.rb +22 -0
  142. data/lib/rails/graphql/field/typed_field.rb +104 -0
  143. data/lib/rails/graphql/helpers.rb +40 -0
  144. data/lib/rails/graphql/helpers/attribute_delegator.rb +39 -0
  145. data/lib/rails/graphql/helpers/inherited_collection.rb +152 -0
  146. data/lib/rails/graphql/helpers/leaf_from_ar.rb +141 -0
  147. data/lib/rails/graphql/helpers/registerable.rb +103 -0
  148. data/lib/rails/graphql/helpers/with_arguments.rb +125 -0
  149. data/lib/rails/graphql/helpers/with_assignment.rb +113 -0
  150. data/lib/rails/graphql/helpers/with_callbacks.rb +55 -0
  151. data/lib/rails/graphql/helpers/with_directives.rb +126 -0
  152. data/lib/rails/graphql/helpers/with_events.rb +81 -0
  153. data/lib/rails/graphql/helpers/with_fields.rb +141 -0
  154. data/lib/rails/graphql/helpers/with_namespace.rb +40 -0
  155. data/lib/rails/graphql/helpers/with_owner.rb +35 -0
  156. data/lib/rails/graphql/helpers/with_schema_fields.rb +230 -0
  157. data/lib/rails/graphql/helpers/with_validator.rb +52 -0
  158. data/lib/rails/graphql/introspection.rb +53 -0
  159. data/lib/rails/graphql/native.rb +56 -0
  160. data/lib/rails/graphql/native/functions.rb +38 -0
  161. data/lib/rails/graphql/native/location.rb +41 -0
  162. data/lib/rails/graphql/native/pointers.rb +23 -0
  163. data/lib/rails/graphql/native/visitor.rb +349 -0
  164. data/lib/rails/graphql/railtie.rb +85 -0
  165. data/lib/rails/graphql/railties/base_generator.rb +35 -0
  166. data/lib/rails/graphql/railties/controller.rb +101 -0
  167. data/lib/rails/graphql/railties/controller_runtime.rb +40 -0
  168. data/lib/rails/graphql/railties/log_subscriber.rb +62 -0
  169. data/lib/rails/graphql/request.rb +343 -0
  170. data/lib/rails/graphql/request/arguments.rb +93 -0
  171. data/lib/rails/graphql/request/component.rb +100 -0
  172. data/lib/rails/graphql/request/component/field.rb +225 -0
  173. data/lib/rails/graphql/request/component/fragment.rb +118 -0
  174. data/lib/rails/graphql/request/component/operation.rb +178 -0
  175. data/lib/rails/graphql/request/component/operation/subscription.rb +16 -0
  176. data/lib/rails/graphql/request/component/spread.rb +119 -0
  177. data/lib/rails/graphql/request/component/typename.rb +82 -0
  178. data/lib/rails/graphql/request/context.rb +51 -0
  179. data/lib/rails/graphql/request/errors.rb +54 -0
  180. data/lib/rails/graphql/request/event.rb +112 -0
  181. data/lib/rails/graphql/request/helpers/directives.rb +64 -0
  182. data/lib/rails/graphql/request/helpers/selection_set.rb +87 -0
  183. data/lib/rails/graphql/request/helpers/value_writers.rb +115 -0
  184. data/lib/rails/graphql/request/steps/organizable.rb +146 -0
  185. data/lib/rails/graphql/request/steps/prepareable.rb +33 -0
  186. data/lib/rails/graphql/request/steps/resolveable.rb +32 -0
  187. data/lib/rails/graphql/request/strategy.rb +249 -0
  188. data/lib/rails/graphql/request/strategy/dynamic_instance.rb +41 -0
  189. data/lib/rails/graphql/request/strategy/multi_query_strategy.rb +36 -0
  190. data/lib/rails/graphql/request/strategy/sequenced_strategy.rb +28 -0
  191. data/lib/rails/graphql/schema.rb +272 -0
  192. data/lib/rails/graphql/shortcuts.rb +77 -0
  193. data/lib/rails/graphql/source.rb +371 -0
  194. data/lib/rails/graphql/source/active_record/builders.rb +154 -0
  195. data/lib/rails/graphql/source/active_record_source.rb +231 -0
  196. data/lib/rails/graphql/source/scoped_arguments.rb +87 -0
  197. data/lib/rails/graphql/to_gql.rb +368 -0
  198. data/lib/rails/graphql/type.rb +138 -0
  199. data/lib/rails/graphql/type/enum.rb +206 -0
  200. data/lib/rails/graphql/type/enum/directive_location_enum.rb +30 -0
  201. data/lib/rails/graphql/type/enum/type_kind_enum.rb +57 -0
  202. data/lib/rails/graphql/type/input.rb +134 -0
  203. data/lib/rails/graphql/type/interface.rb +82 -0
  204. data/lib/rails/graphql/type/object.rb +111 -0
  205. data/lib/rails/graphql/type/object/directive_object.rb +34 -0
  206. data/lib/rails/graphql/type/object/enum_value_object.rb +25 -0
  207. data/lib/rails/graphql/type/object/field_object.rb +54 -0
  208. data/lib/rails/graphql/type/object/input_value_object.rb +49 -0
  209. data/lib/rails/graphql/type/object/schema_object.rb +40 -0
  210. data/lib/rails/graphql/type/object/type_object.rb +136 -0
  211. data/lib/rails/graphql/type/scalar.rb +71 -0
  212. data/lib/rails/graphql/type/scalar/bigint_scalar.rb +34 -0
  213. data/lib/rails/graphql/type/scalar/binary_scalar.rb +30 -0
  214. data/lib/rails/graphql/type/scalar/boolean_scalar.rb +37 -0
  215. data/lib/rails/graphql/type/scalar/date_scalar.rb +34 -0
  216. data/lib/rails/graphql/type/scalar/date_time_scalar.rb +32 -0
  217. data/lib/rails/graphql/type/scalar/decimal_scalar.rb +35 -0
  218. data/lib/rails/graphql/type/scalar/float_scalar.rb +32 -0
  219. data/lib/rails/graphql/type/scalar/id_scalar.rb +39 -0
  220. data/lib/rails/graphql/type/scalar/int_scalar.rb +36 -0
  221. data/lib/rails/graphql/type/scalar/string_scalar.rb +28 -0
  222. data/lib/rails/graphql/type/scalar/time_scalar.rb +40 -0
  223. data/lib/rails/graphql/type/union.rb +87 -0
  224. data/lib/rails/graphql/type_map.rb +347 -0
  225. data/lib/rails/graphql/version.rb +7 -0
  226. data/test/assets/introspection-db.json +0 -0
  227. data/test/assets/introspection-mem.txt +1 -0
  228. data/test/assets/introspection.gql +91 -0
  229. data/test/assets/luke.jpg +0 -0
  230. data/test/assets/mem.gql +428 -0
  231. data/test/assets/sqlite.gql +423 -0
  232. data/test/config.rb +80 -0
  233. data/test/graphql/request/context_test.rb +70 -0
  234. data/test/graphql/schema_test.rb +190 -0
  235. data/test/graphql/source_test.rb +237 -0
  236. data/test/graphql/type/enum_test.rb +203 -0
  237. data/test/graphql/type/input_test.rb +138 -0
  238. data/test/graphql/type/interface_test.rb +72 -0
  239. data/test/graphql/type/object_test.rb +104 -0
  240. data/test/graphql/type/scalar/bigint_scalar_test.rb +42 -0
  241. data/test/graphql/type/scalar/binary_scalar_test.rb +17 -0
  242. data/test/graphql/type/scalar/boolean_scalar_test.rb +40 -0
  243. data/test/graphql/type/scalar/date_scalar_test.rb +29 -0
  244. data/test/graphql/type/scalar/date_time_scalar_test.rb +29 -0
  245. data/test/graphql/type/scalar/decimal_scalar_test.rb +28 -0
  246. data/test/graphql/type/scalar/float_scalar_test.rb +22 -0
  247. data/test/graphql/type/scalar/id_scalar_test.rb +26 -0
  248. data/test/graphql/type/scalar/int_scalar_test.rb +26 -0
  249. data/test/graphql/type/scalar/string_scalar_test.rb +17 -0
  250. data/test/graphql/type/scalar/time_scalar_test.rb +36 -0
  251. data/test/graphql/type/scalar_test.rb +45 -0
  252. data/test/graphql/type/union_test.rb +82 -0
  253. data/test/graphql/type_map_test.rb +362 -0
  254. data/test/graphql/type_test.rb +68 -0
  255. data/test/graphql_test.rb +55 -0
  256. data/test/integration/config.rb +56 -0
  257. data/test/integration/memory/star_wars_introspection_test.rb +144 -0
  258. data/test/integration/memory/star_wars_query_test.rb +184 -0
  259. data/test/integration/memory/star_wars_validation_test.rb +99 -0
  260. data/test/integration/schemas/memory.rb +232 -0
  261. data/test/integration/schemas/sqlite.rb +82 -0
  262. data/test/integration/sqlite/star_wars_introspection_test.rb +15 -0
  263. data/test/integration/sqlite/star_wars_mutation_test.rb +82 -0
  264. data/test/integration/sqlite/star_wars_query_test.rb +71 -0
  265. data/test/test_ext.rb +48 -0
  266. metadata +509 -0
@@ -0,0 +1,54 @@
1
+ /**
2
+ * Copyright 2019-present, GraphQL Foundation
3
+ *
4
+ * This source code is licensed under the MIT license found in the
5
+ * LICENSE file in the root directory of this source tree.
6
+ */
7
+
8
+ #pragma once
9
+
10
+ #include <stdio.h>
11
+
12
+ #ifdef __cplusplus
13
+ extern "C" {
14
+ #endif
15
+
16
+ /**
17
+ * This file provides C wrappers for ../GraphQLParser.h.
18
+ */
19
+
20
+ struct GraphQLAstNode;
21
+
22
+ /**
23
+ * Parse the given GraphQL source string, returning an AST. Returns
24
+ * NULL on error. Return value must be freed with
25
+ * graphql_node_free(). If NULL is returned and error is not NULL, an
26
+ * error message is placed in error and must be freed with
27
+ * graphql_error_free().
28
+ */
29
+ struct GraphQLAstNode *graphql_parse_string(
30
+ const char *text, const char **error);
31
+
32
+ struct GraphQLAstNode *graphql_parse_string_with_experimental_schema_support(
33
+ const char *text, const char **error);
34
+
35
+ /**
36
+ * Read and parse GraphQL source from the given file, returning an
37
+ * AST. Returns nullptr on error. Return value must be freed with
38
+ * graphql_node_free(). If NULL is returned and error is not NULL, an
39
+ * error message is placed in error and must be freed with
40
+ * graphql_error_free().
41
+ */
42
+ struct GraphQLAstNode *graphql_parse_file(FILE *file, const char **error);
43
+
44
+ struct GraphQLAstNode *graphql_parse_file_with_experimental_schema_support(
45
+ FILE *file, const char **error);
46
+
47
+ /**
48
+ * Frees an error.
49
+ */
50
+ void graphql_error_free(const char *error);
51
+
52
+ #ifdef __cplusplus
53
+ }
54
+ #endif
@@ -0,0 +1,48 @@
1
+ /**
2
+ * Copyright 2019-present, GraphQL Foundation
3
+ *
4
+ * This source code is licensed under the MIT license found in the
5
+ * LICENSE file in the root directory of this source tree.
6
+ */
7
+
8
+ #include "AstNode.h"
9
+ #include "GraphQLParser.h"
10
+ #include "c/GraphQLAstToJSON.h"
11
+
12
+ #include <cstdio>
13
+ #include <cstdlib>
14
+ #include <iostream>
15
+
16
+
17
+ using std::cout;
18
+ using std::cerr;
19
+ using std::endl;
20
+ using std::fopen;
21
+ using std::fclose;
22
+ using std::free;
23
+
24
+
25
+ int main(int argc, char **argv) {
26
+ const char *error;
27
+ FILE * in;
28
+ if (argc > 1) {
29
+ in = fopen(argv[1], "r"); // NOLINT
30
+ } else {
31
+ in = stdin;
32
+ }
33
+ auto AST = facebook::graphql::parseFile(in, &error);
34
+ if (argc > 1) {
35
+ fclose(in);
36
+ }
37
+ if (!AST) {
38
+ cerr << "Parser failed with error: " << error << endl;
39
+ free((void *)error); // NOLINT
40
+ return 1;
41
+ }
42
+
43
+ const char *json = graphql_ast_to_json(reinterpret_cast<const struct GraphQLAstNode *>(AST.get()));
44
+ puts(json);
45
+ free((void *)json); // NOLINT
46
+
47
+ return 0;
48
+ }
@@ -0,0 +1,324 @@
1
+ /**
2
+ * Copyright 2019-present, GraphQL Foundation
3
+ *
4
+ * This source code is licensed under the MIT license found in the
5
+ * LICENSE file in the root directory of this source tree.
6
+ */
7
+
8
+ %{
9
+ #include <algorithm>
10
+ #include <cassert>
11
+ #include <cctype>
12
+ #include <climits>
13
+ #include <cstdio>
14
+ #include <string>
15
+ #include <vector>
16
+ #include "location.hh"
17
+ #include "position.hh"
18
+ #include "parser.tab.hpp"
19
+ #include "syntaxdefs.h"
20
+
21
+ // Keep track of token lengths.
22
+ #define YY_USER_ACTION yyextra->loc.columns(yyleng);
23
+
24
+ static void escape(char c, char *buf);
25
+
26
+ static std::string clean_up_block_string(const std::string &str);
27
+
28
+ %}
29
+
30
+ %option bison-bridge bison-locations
31
+ %option noyywrap batch noinput nounput
32
+ %option reentrant
33
+ %option extra-type="struct LexerExtra *"
34
+
35
+ %x STRING_STATE
36
+ %x BLOCK_STRING_STATE
37
+ %x C_COMMENT_STATE
38
+ %x LINE_COMMENT_STATE
39
+
40
+ FLOAT -?(0|[1-9][0-9]*)(\.[0-9]+)?([eE][+-]?[0-9]+)?
41
+ INTEGER -?(0|[1-9][0-9]*)
42
+ IDENTIFIER [_A-Za-z][_0-9A-Za-z]*
43
+ VARIABLE $[_0-9A-Za-z]+
44
+ BOM \xef\xbb\xbf
45
+ CRLF \r\n
46
+ BADCHAR [\x00-\x08\x0b\x0c\x0e-\x1f]
47
+ GOODCHAR [^\x00-\x08\x0b\x0c\x0e-\x1f]
48
+ STRINGCHAR [^\x00-\x1f\\\x22]
49
+
50
+ blank [ \t,]
51
+ newline [\n\r]
52
+ notnewline [^\n\r]
53
+
54
+ %%
55
+
56
+ %{
57
+ yyextra->loc.step();
58
+ %}
59
+
60
+ <STRING_STATE>{
61
+ \" {
62
+ BEGIN(INITIAL);
63
+ yylval->str = yyextra->str.c_str();
64
+ *yylloc = yyextra->loc;
65
+ return yy::GraphQLParserImpl::token::TOK_STRING;
66
+ }
67
+
68
+ {newline} {
69
+ throw make_error(yyextra->loc, "Unterminated string");
70
+ }
71
+
72
+ <<EOF>> {
73
+ throw make_error(yyextra->loc, "Unterminated string at EOF");
74
+ }
75
+
76
+ {STRINGCHAR}+ {
77
+ char *p = yytext;
78
+ while (*p) {
79
+ yyextra->str.push_back(*p++);
80
+ }
81
+ }
82
+
83
+ \\\" { yyextra->str.push_back('"'); }
84
+ \\\\ { yyextra->str.push_back('\\'); }
85
+ \\\/ { yyextra->str.push_back('/'); }
86
+ \\n { yyextra->str.push_back('\n'); }
87
+ \\t { yyextra->str.push_back('\t'); }
88
+ \\r { yyextra->str.push_back('\r'); }
89
+ \\b { yyextra->str.push_back('\b'); }
90
+ \\f { yyextra->str.push_back('\f'); }
91
+
92
+ \\u[0-9A-Fa-f]{4} {
93
+ int ch;
94
+ sscanf(yytext + 2, "%x", &ch);
95
+ yyextra->str.push_back(ch);
96
+ }
97
+
98
+ \\u { throw make_error(yyextra->loc, "bad Unicode escape sequence"); }
99
+ \\. { throw make_error(yyextra->loc, std::string("bad escape sequence \\") + yytext[1]); }
100
+
101
+ }
102
+
103
+ <BLOCK_STRING_STATE>{
104
+ <<EOF>> {
105
+ throw make_error(yyextra->loc, "Unterminated block string at EOF");
106
+ }
107
+
108
+ {BADCHAR} {
109
+ throw make_error(yyextra->loc, std::string("Invalid character ") + yytext[0]);
110
+ }
111
+
112
+ {GOODCHAR} {
113
+ /* Can't use {GOODCHAR}+ because that would be a better match for
114
+ """ than the explicit rule! */
115
+ yyextra->str.push_back(*yytext);
116
+ }
117
+
118
+ \\\"\"\" {
119
+ yyextra->str.append(3, '"');
120
+ }
121
+
122
+ \"\"\" {
123
+ BEGIN(INITIAL);
124
+ yyextra->str = clean_up_block_string(yyextra->str);
125
+ yylval->str = yyextra->str.c_str();
126
+ *yylloc = yyextra->loc;
127
+ return yy::GraphQLParserImpl::token::TOK_STRING;
128
+ }
129
+ }
130
+
131
+ <LINE_COMMENT_STATE>{
132
+ {CRLF} { yyextra->loc.lines(yyleng / 2); yyextra->loc.step(); BEGIN(INITIAL); }
133
+ {newline} { yyextra->loc.lines(yyleng); yyextra->loc.step(); BEGIN(INITIAL); }
134
+ {notnewline}+ /* eat comment character */
135
+ }
136
+
137
+ <INITIAL>{
138
+ {blank}+ { yyextra->loc.step(); }
139
+ {BOM}+ { yyextra->loc.step(); yyextra->loc.step(); yyextra->loc.step(); }
140
+ {CRLF}+ { yyextra->loc.lines(yyleng / 2); yyextra->loc.step(); }
141
+ {newline}+ { yyextra->loc.lines(yyleng); yyextra->loc.step(); }
142
+
143
+ # {yyextra->loc.step(); BEGIN(LINE_COMMENT_STATE); }
144
+
145
+ directive { yylval->str = yytext; *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_DIRECTIVE; }
146
+ enum { yylval->str = yytext; *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_ENUM; }
147
+ extend { yylval->str = yytext; *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_EXTEND; }
148
+ false { yylval->str = yytext; *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_FALSE; }
149
+ fragment { yylval->str = yytext; *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_FRAGMENT; }
150
+ implements { yylval->str = yytext; *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_IMPLEMENTS; }
151
+ input { yylval->str = yytext; *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_INPUT; }
152
+ interface { yylval->str = yytext; *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_INTERFACE; }
153
+ mutation { yylval->str = yytext; *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_MUTATION; }
154
+ null { yylval->str = yytext; *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_NULL; }
155
+ on { yylval->str = yytext; *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_ON; }
156
+ query { yylval->str = yytext; *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_QUERY; }
157
+ scalar { yylval->str = yytext; *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_SCALAR; }
158
+ schema { yylval->str = yytext; *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_SCHEMA; }
159
+ subscription { yylval->str = yytext; *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_SUBSCRIPTION; }
160
+ true { yylval->str = yytext; *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_TRUE; }
161
+ type { yylval->str = yytext; *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_TYPE; }
162
+ union { yylval->str = yytext; *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_UNION; }
163
+
164
+ {INTEGER} { yylval->str = yytext; *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_INTEGER; }
165
+ {FLOAT} { yylval->str = yytext; *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_FLOAT; }
166
+ {IDENTIFIER} { yylval->str = yytext; *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_IDENTIFIER; }
167
+ {VARIABLE} { yylval->str = yytext + 1; *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_VARIABLE; }
168
+
169
+ "!" { *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_BANG; }
170
+ "(" { *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_LPAREN; }
171
+ ")" { *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_RPAREN; }
172
+ "..." { *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_ELLIPSIS; }
173
+ ":" { *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_COLON; }
174
+ "=" { *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_EQUAL; }
175
+ "@" { *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_AT; }
176
+ "[" { *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_LBRACKET; }
177
+ "]" { *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_RBRACKET; }
178
+ "{" { *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_LBRACE; }
179
+ "|" { *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_PIPE; }
180
+ "}" { *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_RBRACE; }
181
+
182
+
183
+ <<EOF>> { *yylloc = yyextra->loc; return yy::GraphQLParserImpl::token::TOK_EOF; }
184
+
185
+ \"\"\" {
186
+ BEGIN(BLOCK_STRING_STATE);
187
+ yyextra->str.clear();
188
+ }
189
+
190
+ \" {
191
+ BEGIN(STRING_STATE);
192
+ yyextra->str.clear();
193
+ }
194
+ }
195
+
196
+ <INITIAL,STRING_STATE,LINE_COMMENT_STATE>. {
197
+ char buf[6];
198
+ escape(yytext[0], buf);
199
+ throw make_error(
200
+ yyextra->loc,
201
+ std::string("unrecognized character ") + buf);
202
+ }
203
+
204
+ %%
205
+
206
+ static void escape(char c, char *buf) {
207
+ if (std::isgraph(c)) {
208
+ *buf = c;
209
+ buf[1] = '\0';
210
+ } else {
211
+ buf[0] = '\\';
212
+ buf[2] = '\0';
213
+ switch (c) {
214
+ case '\a':
215
+ buf[1] = 'a';
216
+ break;
217
+ case '\b':
218
+ buf[1] = 'b';
219
+ break;
220
+ case '\f':
221
+ buf[1] = 'f';
222
+ break;
223
+ case '\n':
224
+ buf[1] = 'n';
225
+ break;
226
+ case '\r':
227
+ buf[1] = 'r';
228
+ break;
229
+ case '\t':
230
+ buf[1] = 't';
231
+ break;
232
+ case '\v':
233
+ buf[1] = 'v';
234
+ break;
235
+ default:
236
+ buf[1] = 'x';
237
+ std::snprintf(buf + 2, 3, "%x", ((int)c & 0xFF));
238
+ break;
239
+ }
240
+ }
241
+ }
242
+
243
+ static std::vector<std::string> splitLines(const std::string &str) {
244
+ std::vector<std::string> lines;
245
+ auto it = str.begin();
246
+ while (it != str.end()) {
247
+ static char terminators[2] = {'\r', '\n'};
248
+ auto nextIt = std::find_first_of(it, str.end(), terminators, terminators + sizeof(terminators));
249
+ lines.emplace_back(str.data() + (it - str.begin()), nextIt - it);
250
+ if (nextIt != str.end()) {
251
+ auto advancedIt = nextIt + 1;
252
+ if (advancedIt != str.end()) {
253
+ if (*nextIt == '\r' && *advancedIt == '\n') {
254
+ ++advancedIt;
255
+ }
256
+ }
257
+ nextIt = std::move(advancedIt);
258
+ }
259
+ it = std::move(nextIt);
260
+ }
261
+ return lines;
262
+ }
263
+
264
+ static int count_leading_whitespace(const std::string &str) {
265
+ auto pos = str.find_first_not_of(" \t", 0, strlen(" \t"));
266
+ if (pos == std::string::npos) {
267
+ return str.length();
268
+ }
269
+ return pos;
270
+ }
271
+
272
+ static bool is_all_whitespace(const std::string &str) {
273
+ return count_leading_whitespace(str) == str.length();
274
+ }
275
+
276
+ static std::string clean_up_block_string(const std::string &str) {
277
+ auto lines = splitLines(str);
278
+ bool first = true;
279
+ int commonIndent = INT_MAX;
280
+ for (const auto &line : lines) {
281
+ if (first) {
282
+ first = false;
283
+ continue;
284
+ }
285
+ const auto indent = count_leading_whitespace(line);
286
+ if (indent < line.length()) {
287
+ if (indent < commonIndent) {
288
+ commonIndent = indent;
289
+ }
290
+ }
291
+ }
292
+ if (commonIndent != INT_MAX) {
293
+ first = true;
294
+ for (auto &line : lines) {
295
+ if (first) {
296
+ first = false;
297
+ continue;
298
+ }
299
+ line.erase(0, commonIndent);
300
+ }
301
+ }
302
+
303
+ const auto firstNonBlankIt = std::find_if(lines.begin(), lines.end(), [](const std::string &line) {
304
+ return !is_all_whitespace(line);
305
+ });
306
+ lines.erase(lines.begin(), firstNonBlankIt);
307
+
308
+ const auto firstNonBlankReverseIt = std::find_if(lines.rbegin(), lines.rend(), [](const std::string &line) {
309
+ return !is_all_whitespace(line);
310
+ });
311
+ lines.erase(lines.end() - (firstNonBlankReverseIt - lines.rbegin()), lines.end());
312
+
313
+ std::string formatted;
314
+ first = true;
315
+ for (const auto &line: lines) {
316
+ if (first) {
317
+ first = false;
318
+ } else {
319
+ formatted.push_back('\n');
320
+ }
321
+ formatted.append(line);
322
+ }
323
+ return formatted;
324
+ }
@@ -0,0 +1,693 @@
1
+ /**
2
+ * Copyright 2019-present, GraphQL Foundation
3
+ *
4
+ * This source code is licensed under the MIT license found in the
5
+ * LICENSE file in the root directory of this source tree.
6
+ */
7
+
8
+ %require "3"
9
+
10
+ %skeleton "lalr1.cc"
11
+
12
+ %defines
13
+ %define parser_class_name {GraphQLParserImpl}
14
+
15
+ %define api.token.prefix {TOK_}
16
+
17
+ %define parse.error verbose
18
+
19
+ %code requires
20
+ {
21
+ #include <cstdlib>
22
+ #include <cstring>
23
+ #include <iostream>
24
+ #include <sstream>
25
+ #include <string>
26
+
27
+ #include "Ast.h"
28
+
29
+ using facebook::graphql::ast::Node;
30
+ using facebook::graphql::ast::Name;
31
+ using facebook::graphql::ast::Definition;
32
+ using facebook::graphql::ast::Document;
33
+ using facebook::graphql::ast::OperationDefinition;
34
+ using facebook::graphql::ast::VariableDefinition;
35
+ using facebook::graphql::ast::Variable;
36
+ using facebook::graphql::ast::SelectionSet;
37
+ using facebook::graphql::ast::Selection;
38
+ using facebook::graphql::ast::Field;
39
+ using facebook::graphql::ast::Argument;
40
+ using facebook::graphql::ast::FragmentSpread;
41
+ using facebook::graphql::ast::InlineFragment;
42
+ using facebook::graphql::ast::FragmentDefinition;
43
+ using facebook::graphql::ast::Value;
44
+ using facebook::graphql::ast::IntValue;
45
+ using facebook::graphql::ast::FloatValue;
46
+ using facebook::graphql::ast::StringValue;
47
+ using facebook::graphql::ast::BooleanValue;
48
+ using facebook::graphql::ast::NullValue;
49
+ using facebook::graphql::ast::EnumValue;
50
+ using facebook::graphql::ast::ListValue;
51
+ using facebook::graphql::ast::ObjectValue;
52
+ using facebook::graphql::ast::ObjectField;
53
+ using facebook::graphql::ast::Directive;
54
+ using facebook::graphql::ast::Type;
55
+ using facebook::graphql::ast::NamedType;
56
+ using facebook::graphql::ast::ListType;
57
+ using facebook::graphql::ast::NonNullType;
58
+
59
+ // Experimental schema support.
60
+ using facebook::graphql::ast::SchemaDefinition;
61
+ using facebook::graphql::ast::ScalarTypeDefinition;
62
+ using facebook::graphql::ast::ObjectTypeDefinition;
63
+ using facebook::graphql::ast::InterfaceTypeDefinition;
64
+ using facebook::graphql::ast::UnionTypeDefinition;
65
+ using facebook::graphql::ast::EnumTypeDefinition;
66
+ using facebook::graphql::ast::InputObjectTypeDefinition;
67
+ using facebook::graphql::ast::TypeExtensionDefinition;
68
+ using facebook::graphql::ast::DirectiveDefinition;
69
+ using facebook::graphql::ast::SchemaDefinition;
70
+ using facebook::graphql::ast::OperationTypeDefinition;
71
+ using facebook::graphql::ast::ScalarTypeDefinition;
72
+ using facebook::graphql::ast::ObjectTypeDefinition;
73
+ using facebook::graphql::ast::FieldDefinition;
74
+ using facebook::graphql::ast::InputValueDefinition;
75
+ using facebook::graphql::ast::InterfaceTypeDefinition;
76
+ using facebook::graphql::ast::UnionTypeDefinition;
77
+ using facebook::graphql::ast::EnumTypeDefinition;
78
+ using facebook::graphql::ast::EnumValueDefinition;
79
+ using facebook::graphql::ast::InputObjectTypeDefinition;
80
+ using facebook::graphql::ast::TypeExtensionDefinition;
81
+ using facebook::graphql::ast::DirectiveDefinition;
82
+
83
+ union yystype { \
84
+ const char *str; \
85
+ const char *heapStr; \
86
+ Name *name; \
87
+ Definition *definition; \
88
+ Document *document; \
89
+ OperationDefinition *operationDefinition; \
90
+ VariableDefinition *variableDefinition; \
91
+ Variable *variable; \
92
+ SelectionSet *selectionSet; \
93
+ Selection *selection; \
94
+ Field *field; \
95
+ Argument *argument; \
96
+ FragmentSpread *fragmentSpread; \
97
+ InlineFragment *inlineFragment; \
98
+ FragmentDefinition *fragmentDefinition; \
99
+ Value *value; \
100
+ IntValue *intValue; \
101
+ FloatValue *floatValue; \
102
+ StringValue *stringValue; \
103
+ BooleanValue *booleanValue; \
104
+ NullValue *nullValue; \
105
+ EnumValue *enumValue; \
106
+ ListValue *arrayValue; \
107
+ ObjectValue *objectValue; \
108
+ ObjectField *objectField; \
109
+ Directive *directive; \
110
+ Type *type; \
111
+ NamedType *namedType; \
112
+ ListType *listType; \
113
+ NonNullType *nonNullType; \
114
+ \
115
+ std::vector<std::unique_ptr<Definition>> *definitionList; \
116
+ std::vector<std::unique_ptr<VariableDefinition>> *variableDefinitionList; \
117
+ std::vector<std::unique_ptr<Selection>> *selectionList; \
118
+ std::vector<std::unique_ptr<Field>> *fieldList; \
119
+ std::vector<std::unique_ptr<Argument>> *argumentList; \
120
+ std::vector<std::unique_ptr<Value>> *valueList; \
121
+ std::vector<std::unique_ptr<ObjectField>> *objectFieldList; \
122
+ std::vector<std::unique_ptr<Directive>> *directiveList; \
123
+ \
124
+ SchemaDefinition *schemaDefinition; \
125
+ ScalarTypeDefinition *scalarTypeDefinition; \
126
+ ObjectTypeDefinition *objectTypeDefinition; \
127
+ InterfaceTypeDefinition *interfaceTypeDefinition; \
128
+ UnionTypeDefinition *unionTypeDefinition; \
129
+ EnumTypeDefinition *enumTypeDefinition; \
130
+ InputObjectTypeDefinition *inputObjectTypeDefinition; \
131
+ TypeExtensionDefinition *typeExtensionDefinition; \
132
+ DirectiveDefinition *directiveDefinition; \
133
+ OperationTypeDefinition *operationTypeDefinition; \
134
+ InputValueDefinition *inputValueDefinition; \
135
+ FieldDefinition *fieldDefinition; \
136
+ EnumValueDefinition *enumValueDefinition; \
137
+ \
138
+ std::vector<std::unique_ptr<OperationTypeDefinition>> *operationTypeDefinitionList; \
139
+ std::vector<std::unique_ptr<NamedType>> *typeNameList; \
140
+ std::vector<std::unique_ptr<InputValueDefinition>> *inputValueDefinitionList; \
141
+ std::vector<std::unique_ptr<FieldDefinition>> *fieldDefinitionList; \
142
+ std::vector<std::unique_ptr<Name>> *nameList; \
143
+ std::vector<std::unique_ptr<EnumValueDefinition>> *enumValueDefinitionList; \
144
+ };
145
+
146
+ #define YYSTYPE union yystype
147
+ #define YYLTYPE yy::location
148
+
149
+ }
150
+
151
+ %lex-param { void *scanner }
152
+ %parse-param { bool enableSchema } { Node **outAST } { const char **outError } { void *scanner }
153
+
154
+ %locations
155
+
156
+ %code
157
+ {
158
+ #include "lexer.h"
159
+ #include "syntaxdefs.h"
160
+ }
161
+
162
+ %token EOF 0
163
+ %token <str> DIRECTIVE "directive"
164
+ %token <str> ENUM "enum"
165
+ %token <str> EXTEND "extend"
166
+ %token <str> FALSE "false"
167
+ %token <str> FRAGMENT "fragment"
168
+ %token <str> IMPLEMENTS "implements"
169
+ %token <str> INPUT "input"
170
+ %token <str> INTERFACE "interface"
171
+ %token <str> MUTATION "mutation"
172
+ %token <str> NULL "null"
173
+ %token <str> QUERY "query"
174
+ %token <str> ON "on"
175
+ %token <str> SCALAR "scalar"
176
+ %token <str> SCHEMA "schema"
177
+ %token <str> SUBSCRIPTION "subscription"
178
+ %token <str> TRUE "true"
179
+ %token <str> TYPE "type"
180
+ %token <str> UNION "union"
181
+ %token BANG "!"
182
+ %token LPAREN "("
183
+ %token RPAREN ")"
184
+ %token ELLIPSIS "..."
185
+ %token COLON ":"
186
+ %token EQUAL "="
187
+ %token AT "@"
188
+ %token LBRACKET "["
189
+ %token RBRACKET "]"
190
+ %token LBRACE "{"
191
+ %token PIPE "|"
192
+ %token RBRACE "}"
193
+
194
+ %token <str> VARIABLE
195
+ %token <str> INTEGER
196
+ %token <str> FLOAT
197
+ %token <str> STRING
198
+ %token <str> IDENTIFIER
199
+
200
+ %type <variable> variable
201
+ %type <intValue> int_value
202
+ %type <floatValue> float_value
203
+ %type <stringValue> string_value
204
+
205
+ %type <document> start
206
+ %type <document> document
207
+ %type <name> fragment_name
208
+ %type <name> name
209
+ %type <name> name_opt
210
+
211
+ %type <definitionList> definition_list
212
+ %type <definition> definition
213
+ %type <definition> schema_gate
214
+
215
+ %type <operationDefinition> operation_definition
216
+ %type <variableDefinitionList> variable_definitions
217
+ %type <variableDefinitionList> variable_definition_list
218
+ %type <variableDefinition> variable_definition
219
+ %type <value> default_value_opt
220
+ %type <value> default_value
221
+ %type <selectionSet> selection_set
222
+ %type <selectionSet> selection_set_opt
223
+ %type <selectionList> selection_list
224
+ %type <selection> selection
225
+ %type <field> field
226
+ %type <argumentList> arguments_opt
227
+ %type <argumentList> arguments
228
+ %type <argumentList> argument_list
229
+ %type <argument> argument
230
+
231
+ %type <fragmentSpread> fragment_spread
232
+ %type <inlineFragment> inline_fragment
233
+ %type <fragmentDefinition> fragment_definition
234
+ %type <namedType> type_condition
235
+
236
+ %type <value> value
237
+ %type <value> value_const
238
+ %type <booleanValue> boolean_value
239
+ %type <nullValue> null_value
240
+ %type <enumValue> enum_value
241
+ %type <arrayValue> list_value
242
+ %type <arrayValue> list_value_const
243
+ %type <valueList> value_list
244
+ %type <valueList> value_const_list
245
+ %type <objectValue> object_value
246
+ %type <objectValue> object_value_const
247
+ %type <objectFieldList> object_field_list
248
+ %type <objectFieldList> object_field_const_list
249
+ %type <objectField> object_field
250
+ %type <objectField> object_field_const
251
+
252
+
253
+ %type <directiveList> directives
254
+ %type <directiveList> directives_opt
255
+ %type <directiveList> directive_list
256
+ %type <directive> directive
257
+
258
+ %type <type> type
259
+ %type <namedType> type_name
260
+ %type <listType> list_type
261
+ %type <nonNullType> non_null_type
262
+
263
+ %type <heapStr> operation_type
264
+
265
+ %type <schemaDefinition> schema_definition;
266
+ %type <scalarTypeDefinition> scalar_type_definition;
267
+ %type <objectTypeDefinition> object_type_definition;
268
+ %type <interfaceTypeDefinition> interface_type_definition;
269
+ %type <unionTypeDefinition> union_type_definition;
270
+ %type <enumTypeDefinition> enum_type_definition;
271
+ %type <inputObjectTypeDefinition> input_object_type_definition;
272
+ %type <typeExtensionDefinition> type_extension_definition;
273
+ %type <directiveDefinition> directive_definition;
274
+ %type <operationTypeDefinition> operation_type_definition;
275
+ %type <operationTypeDefinitionList> operation_type_definition_list;
276
+ %type <typeNameList> type_name_list;
277
+ %type <typeNameList> implements_interfaces_opt;
278
+ %type <typeNameList> union_members;
279
+ %type <fieldDefinition> field_definition;
280
+ %type <fieldDefinitionList> field_definition_list;
281
+ %type <inputValueDefinitionList> arguments_definition_opt;
282
+ %type <inputValueDefinitionList> arguments_definition;
283
+ %type <inputValueDefinitionList> input_value_definition_list;
284
+ %type <inputValueDefinition> input_value_definition;
285
+ %type <enumValueDefinition> enum_value_definition;
286
+ %type <nameList> directive_locations;
287
+ %type <enumValueDefinitionList> enum_value_definition_list;
288
+
289
+ %destructor { } <str>
290
+ %destructor { free((void *)$$); } <heapStr>
291
+ %destructor { } <document> /* we steal it and put it in outAST, don't free! */
292
+ %destructor { delete $$; } <*>
293
+
294
+ %printer { yyoutput << $$; } <str>
295
+
296
+ %%
297
+
298
+ start: document { *outAST = $1; }
299
+ ;
300
+
301
+ /* All of the non-identifier tokens are to accommodate various flavors
302
+ of name that don't include those tokens. */
303
+ fragment_name: DIRECTIVE { $$ = new Name(@1, strdup($1)); }
304
+ | ENUM { $$ = new Name(@1, strdup($1)); }
305
+ | EXTEND { $$ = new Name(@1, strdup($1)); }
306
+ | FALSE { $$ = new Name(@1, strdup($1)); }
307
+ | FRAGMENT { $$ = new Name(@1, strdup($1)); }
308
+ | IDENTIFIER { $$ = new Name(@1, strdup($1)); }
309
+ | IMPLEMENTS { $$ = new Name(@1, strdup($1)); }
310
+ | INPUT { $$ = new Name(@1, strdup($1)); }
311
+ | INTERFACE { $$ = new Name(@1, strdup($1)); }
312
+ | MUTATION { $$ = new Name(@1, strdup($1)); }
313
+ | NULL { $$ = new Name(@1, strdup($1)); }
314
+ | QUERY { $$ = new Name(@1, strdup($1)); }
315
+ | SCALAR { $$ = new Name(@1, strdup($1)); }
316
+ | SCHEMA { $$ = new Name(@1, strdup($1)); }
317
+ | SUBSCRIPTION { $$ = new Name(@1, strdup($1)); }
318
+ | TRUE { $$ = new Name(@1, strdup($1)); }
319
+ | TYPE { $$ = new Name(@1, strdup($1)); }
320
+ | UNION { $$ = new Name(@1, strdup($1)); }
321
+ ;
322
+
323
+ name: fragment_name
324
+ | ON { $$ = new Name(@1, strdup($1)); }
325
+ ;
326
+
327
+ name_opt:
328
+ %empty {$$ = nullptr;}
329
+ | name
330
+ ;
331
+
332
+ /* 2.2 Document */
333
+
334
+ document: definition_list { $$ = new Document(@$, $1); }
335
+ ;
336
+
337
+ definition_list:definition { $$ = new std::vector<std::unique_ptr<Definition>>(); $$->emplace_back($1); }
338
+ | definition_list definition { $1->emplace_back($2); $$ = $1; }
339
+ ;
340
+
341
+ definition: operation_definition { $$ = static_cast<Definition *>($1); }
342
+ | fragment_definition { $$ = static_cast<Definition *>($1); }
343
+ | schema_gate {
344
+ if (!enableSchema) {
345
+ error(@$, "schema support disabled");
346
+ // %destructor doesn't work with YYERROR. See
347
+ // https://www.gnu.org/software/bison/manual/html_node/Destructor-Decl.html
348
+ delete $$;
349
+ YYERROR;
350
+ }
351
+ $$ = static_cast<Definition *>($1);
352
+ }
353
+ ;
354
+
355
+ schema_gate: schema_definition { $$ = static_cast<Definition *>($1); }
356
+ | scalar_type_definition { $$ = static_cast<Definition *>($1); }
357
+ | object_type_definition { $$ = static_cast<Definition *>($1); }
358
+ | interface_type_definition { $$ = static_cast<Definition *>($1); }
359
+ | union_type_definition { $$ = static_cast<Definition *>($1); }
360
+ | enum_type_definition { $$ = static_cast<Definition *>($1); }
361
+ | input_object_type_definition { $$ = static_cast<Definition *>($1); }
362
+ | type_extension_definition { $$ = static_cast<Definition *>($1); }
363
+ | directive_definition { $$ = static_cast<Definition *>($1); }
364
+ ;
365
+
366
+
367
+ /* 2.2.1 Operations */
368
+ operation_definition:
369
+ selection_set { $$ = new OperationDefinition(@$, strdup("query"), nullptr, nullptr, nullptr, $1); }
370
+ | operation_type name_opt selection_set { $$ = new OperationDefinition(@$, $1, $2, nullptr, nullptr, $3); }
371
+ | operation_type name_opt variable_definitions selection_set { $$ = new OperationDefinition(@$, $1, $2, $3, nullptr, $4); }
372
+ | operation_type name_opt directives selection_set { $$ = new OperationDefinition(@$, $1, $2, nullptr, $3, $4); }
373
+ | operation_type name_opt variable_definitions directives selection_set { $$ = new OperationDefinition(@$, $1, $2, $3, $4, $5); }
374
+ ;
375
+
376
+ operation_type: QUERY { $$ = strdup($1); }
377
+ | MUTATION { $$ = strdup($1); }
378
+ | SUBSCRIPTION { $$ = strdup($1); }
379
+ ;
380
+
381
+ variable_definitions:
382
+ "(" variable_definition_list ")" { $$ = $2; }
383
+ ;
384
+
385
+ variable_definition_list:
386
+ variable_definition { $$ = new std::vector<std::unique_ptr<VariableDefinition>>(); $$->emplace_back($1); }
387
+ | variable_definition_list variable_definition { $1->emplace_back($2); $$ = $1; }
388
+ ;
389
+
390
+ variable: VARIABLE { $$ = new Variable(@$, new Name(@1, strdup($1))); }
391
+ ;
392
+
393
+ variable_definition:
394
+ variable ":" type default_value_opt { $$ = new VariableDefinition(@$, $1, $3, $4); }
395
+ ;
396
+
397
+ default_value_opt:
398
+ %empty { $$ = nullptr; }
399
+ | default_value
400
+ ;
401
+
402
+ default_value: "=" value_const { $$ = $2; }
403
+ ;
404
+
405
+ selection_set:
406
+ "{" selection_list "}" { $$ = new SelectionSet(@$, $2); }
407
+ ;
408
+
409
+ selection_set_opt:
410
+ %empty { $$ = nullptr; }
411
+ | selection_set
412
+ ;
413
+ selection_list: selection { $$ = new std::vector<std::unique_ptr<Selection>>(); $$->emplace_back($1); }
414
+ | selection_list selection { $1->emplace_back($2); $$ = $1; }
415
+ ;
416
+
417
+ selection: field { $$ = static_cast<Selection *>($1); }
418
+ | fragment_spread { $$ = static_cast<Selection *>($1); }
419
+ | inline_fragment { $$ = static_cast<Selection *>($1); }
420
+ ;
421
+
422
+ field: name arguments_opt directives_opt selection_set_opt { $$ = new Field(@$, nullptr, $1, $2, $3, $4); }
423
+ | name ":" name arguments_opt directives_opt selection_set_opt { $$ = new Field(@$, $1, $3, $4, $5, $6); }
424
+ ;
425
+
426
+ arguments: "(" argument_list ")" { $$ = $2; }
427
+ ;
428
+
429
+ arguments_opt: %empty { $$ = nullptr; }
430
+ | arguments { $$ = $1; }
431
+ ;
432
+
433
+ argument_list: argument { $$ = new std::vector<std::unique_ptr<Argument>>(); $$->emplace_back($1); }
434
+ | argument_list argument { $1->emplace_back($2); $$ = $1; }
435
+ ;
436
+
437
+ argument: name ":" value { $$ = new Argument(@$, $1, $3); }
438
+ ;
439
+
440
+ /* 2.2.6 Fragments */
441
+ fragment_spread:
442
+ "..." fragment_name directives_opt { $$ = new FragmentSpread(@$, $2, $3); }
443
+ ;
444
+
445
+ inline_fragment:
446
+ "..." "on" type_condition directives_opt selection_set { $$ = new InlineFragment(@$, $3, $4, $5); }
447
+ | "..." directives_opt selection_set { $$ = new InlineFragment(@$, nullptr, $2, $3); }
448
+ ;
449
+
450
+ fragment_definition:
451
+ "fragment" fragment_name "on" type_condition directives_opt selection_set { $$ = new FragmentDefinition(@$, $2, $4, $5, $6); }
452
+ ;
453
+
454
+ type_condition: type_name
455
+ ;
456
+
457
+ /* 2.2.7 Input Values */
458
+ value: variable { $$ = static_cast<Value *>($1); }
459
+ | int_value { $$ = static_cast<Value *>($1); }
460
+ | float_value { $$ = static_cast<Value *>($1); }
461
+ | string_value { $$ = static_cast<Value *>($1); }
462
+ | boolean_value { $$ = static_cast<Value *>($1); }
463
+ | null_value { $$ = static_cast<Value *>($1); }
464
+ | enum_value { $$ = static_cast<Value *>($1); }
465
+ | list_value { $$ = static_cast<Value *>($1); }
466
+ | object_value { $$ = static_cast<Value *>($1); }
467
+ ;
468
+
469
+ int_value: INTEGER { $$ = new IntValue(@$, strdup($1)); }
470
+ ;
471
+
472
+ float_value: FLOAT { $$ = new FloatValue(@$, strdup($1)); }
473
+ ;
474
+
475
+ string_value: STRING { $$ = new StringValue(@$, strdup($1)); }
476
+ ;
477
+
478
+ value_const: int_value { $$ = static_cast<Value *>($1); }
479
+ | float_value { $$ = static_cast<Value *>($1); }
480
+ | string_value { $$ = static_cast<Value *>($1); }
481
+ | boolean_value { $$ = static_cast<Value *>($1); }
482
+ | null_value { $$ = static_cast<Value *>($1); }
483
+ | enum_value { $$ = static_cast<Value *>($1); }
484
+ | list_value_const { $$ = static_cast<Value *>($1); }
485
+ | object_value_const { $$ = static_cast<Value *>($1); }
486
+ ;
487
+
488
+ boolean_value: TRUE { $$ = new BooleanValue(@$, true); }
489
+ | FALSE { $$ = new BooleanValue(@$, false); }
490
+ ;
491
+
492
+ null_value: NULL { $$ = new NullValue(@$); }
493
+ ;
494
+
495
+ enum_value: DIRECTIVE { $$ = new EnumValue(@$, strdup($1)); }
496
+ | ENUM { $$ = new EnumValue(@$, strdup($1)); }
497
+ | EXTEND { $$ = new EnumValue(@$, strdup($1)); }
498
+ | FRAGMENT { $$ = new EnumValue(@$, strdup($1)); }
499
+ | IDENTIFIER { $$ = new EnumValue(@$, strdup($1)); }
500
+ | IMPLEMENTS { $$ = new EnumValue(@$, strdup($1)); }
501
+ | INPUT { $$ = new EnumValue(@$, strdup($1)); }
502
+ | INTERFACE { $$ = new EnumValue(@$, strdup($1)); }
503
+ | MUTATION { $$ = new EnumValue(@$, strdup($1)); }
504
+ | ON { $$ = new EnumValue(@$, strdup($1)); }
505
+ | QUERY { $$ = new EnumValue(@$, strdup($1)); }
506
+ | SCALAR { $$ = new EnumValue(@$, strdup($1)); }
507
+ | SCHEMA { $$ = new EnumValue(@$, strdup($1)); }
508
+ | SUBSCRIPTION { $$ = new EnumValue(@$, strdup($1)); }
509
+ | TYPE { $$ = new EnumValue(@$, strdup($1)); }
510
+ | UNION { $$ = new EnumValue(@$, strdup($1)); }
511
+ ;
512
+
513
+ /* 2.2.7.6 List Value */
514
+
515
+ /* REVIEW: the empty case is inefficient; consider implementing
516
+ ListValue manually. Don't forget to also do list_value_const. */
517
+ list_value: "[" "]" { $$ = new ListValue(@$, new std::vector<std::unique_ptr<Value>>()); }
518
+ | "[" value_list "]" { $$ = new ListValue(@$, $2); }
519
+ ;
520
+
521
+ value_list: value { $$ = new std::vector<std::unique_ptr<Value>>(); $$->emplace_back($1); }
522
+ | value_list value { $1->emplace_back($2); $$ = $1; }
523
+ ;
524
+
525
+ list_value_const:
526
+ "[" "]" { $$ = new ListValue(@$, new std::vector<std::unique_ptr<Value>>()); }
527
+ | "[" value_const_list "]" { $$ = new ListValue(@$, $2); }
528
+ ;
529
+
530
+ value_const_list:
531
+ value_const { $$ = new std::vector<std::unique_ptr<Value>>(); $$->emplace_back($1); }
532
+ | value_const_list value_const { $1->emplace_back($2); $$ = $1; }
533
+ ;
534
+
535
+ /* 2.2.7.7 Object Value */
536
+ /* REVIEW: Inefficient, like ListValue. */
537
+ object_value: "{" "}" { $$ = new ObjectValue(@$, new std::vector<std::unique_ptr<ObjectField>>()); }
538
+ | "{" object_field_list "}" { $$ = new ObjectValue(@$, $2); }
539
+ ;
540
+
541
+ object_field_list:
542
+ object_field { $$ = new std::vector<std::unique_ptr<ObjectField>>(); $$->emplace_back($1); }
543
+ | object_field_list object_field { $1->emplace_back($2); $$ = $1; }
544
+ ;
545
+
546
+ object_field: name ":" value { $$ = new ObjectField(@$, $1, $3); }
547
+ ;
548
+
549
+ object_value_const:
550
+ "{" "}" { $$ = new ObjectValue(@$, new std::vector<std::unique_ptr<ObjectField>>()); }
551
+ | "{" object_field_const_list "}" { $$ = new ObjectValue(@$, $2); }
552
+ ;
553
+
554
+ object_field_const_list:
555
+ object_field_const { $$ = new std::vector<std::unique_ptr<ObjectField>>(); $$->emplace_back($1); }
556
+ | object_field_const_list object_field_const { $1->emplace_back($2); $$ = $1; }
557
+ ;
558
+
559
+ object_field_const: name ":" value_const { $$ = new ObjectField(@$, $1, $3); }
560
+ ;
561
+
562
+ /* 2.2.10 Directives */
563
+
564
+ directives: directive_list
565
+ ;
566
+
567
+ directives_opt: %empty { $$ = nullptr; }
568
+ | directives
569
+ ;
570
+
571
+ directive_list: directive { $$ = new std::vector<std::unique_ptr<Directive>>(); $$->emplace_back($1); }
572
+ | directive_list directive { $1->emplace_back($2); $$ = $1; }
573
+ ;
574
+
575
+ directive: "@" name arguments_opt { $$ = new Directive(@$, $2, $3); }
576
+ ;
577
+
578
+ /* 2.2.9 Types */
579
+
580
+ type: type_name { $$ = static_cast<Type *>($1); }
581
+ | list_type { $$ = static_cast<Type *>($1); }
582
+ | non_null_type { $$ = static_cast<Type *>($1); }
583
+ ;
584
+
585
+ type_name: name { $$ = new NamedType(@$, $1); }
586
+ ;
587
+
588
+ list_type: "[" type "]" { $$ = new ListType(@$, $2); }
589
+ ;
590
+
591
+ non_null_type: type_name "!" { $$ = new NonNullType(@$, $1); }
592
+ | list_type "!" { $$ = new NonNullType(@$, $1); }
593
+ ;
594
+
595
+ /* Experimental schema parsing support. */
596
+
597
+ schema_definition: SCHEMA directives_opt "{" operation_type_definition_list "}" { $$ = new SchemaDefinition(@$, $2, $4); }
598
+ ;
599
+
600
+ operation_type_definition_list:
601
+ operation_type_definition { $$ = new std::vector<std::unique_ptr<OperationTypeDefinition>>(); $$->emplace_back($1); }
602
+ | operation_type_definition_list operation_type_definition { $1->emplace_back($2); $$ = $1; }
603
+ ;
604
+
605
+ operation_type_definition:
606
+ operation_type ":" type_name { $$ = new OperationTypeDefinition(@$, $1, $3); }
607
+ ;
608
+
609
+ scalar_type_definition: SCALAR name directives_opt { $$ = new ScalarTypeDefinition(@$, $2, $3); }
610
+ ;
611
+
612
+ object_type_definition: TYPE name implements_interfaces_opt directives_opt "{" field_definition_list "}" { $$ = new ObjectTypeDefinition(@$, $2, $3, $4, $6); }
613
+ ;
614
+
615
+ implements_interfaces_opt: %empty { $$ = nullptr; }
616
+ | IMPLEMENTS type_name_list { $$ = $2; }
617
+ ;
618
+
619
+ type_name_list: type_name { $$ = new std::vector<std::unique_ptr<NamedType>>(); $$->emplace_back($1); }
620
+ | type_name_list type_name { $1->emplace_back($2); $$ = $1; }
621
+ ;
622
+
623
+ field_definition: name arguments_definition_opt ":" type directives_opt { $$ = new FieldDefinition(@$, $1, $2, $4, $5); }
624
+ ;
625
+
626
+ field_definition_list:
627
+ field_definition { $$ = new std::vector<std::unique_ptr<FieldDefinition>>(); $$->emplace_back($1); }
628
+ | field_definition_list field_definition { $1->emplace_back($2); $$ = $1; }
629
+ ;
630
+
631
+ arguments_definition_opt: %empty { $$ = nullptr; }
632
+ | arguments_definition { $$ = $1; }
633
+ ;
634
+
635
+ arguments_definition: "(" input_value_definition_list ")" { $$ = $2; }
636
+ ;
637
+
638
+ input_value_definition_list: input_value_definition { $$ = new std::vector<std::unique_ptr<InputValueDefinition>>(); $$->emplace_back($1); }
639
+ | input_value_definition_list input_value_definition { $1->emplace_back($2); $$ = $1; }
640
+ ;
641
+
642
+ input_value_definition: name ":" type default_value_opt directives_opt { $$ = new InputValueDefinition(@$, $1, $3, $4, $5); }
643
+
644
+ interface_type_definition: INTERFACE name directives_opt "{" field_definition_list "}" { $$ = new InterfaceTypeDefinition(@$, $2, $3, $5); }
645
+ ;
646
+
647
+ union_type_definition: UNION name directives_opt "=" union_members { $$ = new UnionTypeDefinition(@$, $2, $3, $5); }
648
+ ;
649
+
650
+ union_members: type_name { $$ = new std::vector<std::unique_ptr<NamedType>>(); $$->emplace_back($1); }
651
+ | union_members "|" type_name { $1->emplace_back($3); $$ = $1; }
652
+ ;
653
+
654
+ enum_type_definition: ENUM name directives_opt "{" enum_value_definition_list "}" { $$ = new EnumTypeDefinition(@$, $2, $3, $5); }
655
+ ;
656
+
657
+ enum_value_definition: name directives_opt { $$ = new EnumValueDefinition(@$, $1, $2); }
658
+ ;
659
+
660
+ enum_value_definition_list:
661
+ enum_value_definition { $$ = new std::vector<std::unique_ptr<EnumValueDefinition>>(); $$->emplace_back($1); }
662
+ | enum_value_definition_list enum_value_definition { $1->emplace_back($2); $$ = $1; }
663
+ ;
664
+
665
+ input_object_type_definition: INPUT name directives_opt "{" input_value_definition_list "}" { $$ = new InputObjectTypeDefinition(@$, $2, $3, $5); }
666
+ ;
667
+
668
+ type_extension_definition: EXTEND object_type_definition { $$ = new TypeExtensionDefinition(@$, $2); }
669
+ ;
670
+
671
+ directive_definition: DIRECTIVE "@" name arguments_definition_opt ON directive_locations { $$ = new DirectiveDefinition(@$, $3, $4, $6); }
672
+ ;
673
+
674
+ directive_locations:
675
+ name { $$ = new std::vector<std::unique_ptr<Name>>(); $$->emplace_back($1); }
676
+ | directive_locations "|" name { $1->emplace_back($3); $$ = $1; }
677
+ ;
678
+
679
+ %%
680
+
681
+ void yy::GraphQLParserImpl::error(const yy::location &loc, const std::string &str) {
682
+ std::ostringstream out;
683
+ out << loc << ": " << str;
684
+ if (outError) {
685
+ *outError = strdup(out.str().c_str());
686
+ }
687
+ }
688
+
689
+ /* Workaround for syntax_error ctor being marked inline, which causes link
690
+ errors if used from lexer.lpp. */
691
+ yy::GraphQLParserImpl::syntax_error make_error(const yy::location &loc, const std::string &str) {
692
+ return yy::GraphQLParserImpl::syntax_error(loc, str);
693
+ }