graphql 1.13.0 → 1.13.24
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/lib/generators/graphql/core.rb +3 -8
- data/lib/generators/graphql/enum_generator.rb +4 -10
- data/lib/generators/graphql/field_extractor.rb +31 -0
- data/lib/generators/graphql/input_generator.rb +50 -0
- data/lib/generators/graphql/install/mutation_root_generator.rb +34 -0
- data/lib/generators/graphql/install_generator.rb +10 -3
- data/lib/generators/graphql/interface_generator.rb +7 -7
- data/lib/generators/graphql/mutation_create_generator.rb +22 -0
- data/lib/generators/graphql/mutation_delete_generator.rb +22 -0
- data/lib/generators/graphql/mutation_generator.rb +5 -30
- data/lib/generators/graphql/mutation_update_generator.rb +22 -0
- data/lib/generators/graphql/object_generator.rb +8 -37
- data/lib/generators/graphql/orm_mutations_base.rb +40 -0
- data/lib/generators/graphql/scalar_generator.rb +4 -2
- data/lib/generators/graphql/templates/enum.erb +5 -1
- data/lib/generators/graphql/templates/input.erb +9 -0
- data/lib/generators/graphql/templates/interface.erb +4 -2
- data/lib/generators/graphql/templates/mutation.erb +1 -1
- data/lib/generators/graphql/templates/mutation_create.erb +20 -0
- data/lib/generators/graphql/templates/mutation_delete.erb +20 -0
- data/lib/generators/graphql/templates/mutation_update.erb +21 -0
- data/lib/generators/graphql/templates/object.erb +4 -2
- data/lib/generators/graphql/templates/scalar.erb +3 -1
- data/lib/generators/graphql/templates/union.erb +4 -2
- data/lib/generators/graphql/type_generator.rb +46 -9
- data/lib/generators/graphql/union_generator.rb +5 -5
- data/lib/graphql/analysis/ast/field_usage.rb +6 -2
- data/lib/graphql/analysis/ast/visitor.rb +2 -1
- data/lib/graphql/argument.rb +1 -1
- data/lib/graphql/base_type.rb +5 -3
- data/lib/graphql/boolean_type.rb +1 -1
- data/lib/graphql/dataloader/source.rb +2 -2
- data/lib/graphql/date_encoding_error.rb +16 -0
- data/lib/graphql/define/instance_definable.rb +15 -0
- data/lib/graphql/directive/deprecated_directive.rb +1 -1
- data/lib/graphql/directive/include_directive.rb +1 -1
- data/lib/graphql/directive/skip_directive.rb +1 -1
- data/lib/graphql/directive.rb +1 -1
- data/lib/graphql/enum_type.rb +2 -2
- data/lib/graphql/execution/interpreter/arguments_cache.rb +4 -2
- data/lib/graphql/execution/interpreter/runtime.rb +48 -28
- data/lib/graphql/execution/multiplex.rb +3 -0
- data/lib/graphql/field.rb +1 -1
- data/lib/graphql/float_type.rb +1 -1
- data/lib/graphql/id_type.rb +1 -1
- data/lib/graphql/input_object_type.rb +1 -1
- data/lib/graphql/int_type.rb +1 -1
- data/lib/graphql/interface_type.rb +1 -1
- data/lib/graphql/introspection/directive_location_enum.rb +2 -2
- data/lib/graphql/introspection/directive_type.rb +4 -2
- data/lib/graphql/introspection/field_type.rb +1 -1
- data/lib/graphql/introspection/schema_type.rb +7 -2
- data/lib/graphql/introspection/type_type.rb +14 -8
- data/lib/graphql/introspection.rb +4 -1
- data/lib/graphql/language/block_string.rb +2 -2
- data/lib/graphql/language/document_from_schema_definition.rb +8 -3
- data/lib/graphql/language/lexer.rb +50 -25
- data/lib/graphql/language/lexer.rl +2 -0
- data/lib/graphql/language/nodes.rb +15 -3
- data/lib/graphql/language/parser.rb +829 -816
- data/lib/graphql/language/parser.y +8 -2
- data/lib/graphql/language/printer.rb +4 -0
- data/lib/graphql/object_type.rb +2 -2
- data/lib/graphql/pagination/active_record_relation_connection.rb +43 -6
- data/lib/graphql/pagination/relation_connection.rb +59 -29
- data/lib/graphql/query/context.rb +10 -0
- data/lib/graphql/query/input_validation_result.rb +9 -0
- data/lib/graphql/query/validation_pipeline.rb +2 -3
- data/lib/graphql/query/variable_validation_error.rb +2 -2
- data/lib/graphql/query/variables.rb +30 -3
- data/lib/graphql/query.rb +0 -1
- data/lib/graphql/relay/connection_type.rb +15 -2
- data/lib/graphql/relay/global_id_resolve.rb +1 -2
- data/lib/graphql/relay/mutation.rb +1 -1
- data/lib/graphql/relay/page_info.rb +1 -1
- data/lib/graphql/relay/range_add.rb +4 -0
- data/lib/graphql/rubocop/graphql/default_required_true.rb +4 -4
- data/lib/graphql/scalar_type.rb +1 -1
- data/lib/graphql/schema/argument.rb +29 -16
- data/lib/graphql/schema/build_from_definition.rb +9 -7
- data/lib/graphql/schema/directive.rb +25 -2
- data/lib/graphql/schema/enum.rb +4 -3
- data/lib/graphql/schema/enum_value.rb +3 -1
- data/lib/graphql/schema/field.rb +196 -92
- data/lib/graphql/schema/field_extension.rb +89 -2
- data/lib/graphql/schema/input_object.rb +27 -9
- data/lib/graphql/schema/interface.rb +8 -2
- data/lib/graphql/schema/introspection_system.rb +1 -1
- data/lib/graphql/schema/list.rb +21 -4
- data/lib/graphql/schema/loader.rb +3 -0
- data/lib/graphql/schema/member/accepts_definition.rb +7 -2
- data/lib/graphql/schema/member/base_dsl_methods.rb +1 -1
- data/lib/graphql/schema/member/cached_graphql_definition.rb +29 -2
- data/lib/graphql/schema/member/has_arguments.rb +2 -2
- data/lib/graphql/schema/member/has_fields.rb +1 -1
- data/lib/graphql/schema/member/has_interfaces.rb +11 -1
- data/lib/graphql/schema/member/validates_input.rb +2 -2
- data/lib/graphql/schema/non_null.rb +9 -3
- data/lib/graphql/schema/object.rb +3 -1
- data/lib/graphql/schema/relay_classic_mutation.rb +8 -0
- data/lib/graphql/schema/resolver.rb +19 -13
- data/lib/graphql/schema/scalar.rb +15 -1
- data/lib/graphql/schema/traversal.rb +1 -1
- data/lib/graphql/schema/union.rb +2 -0
- data/lib/graphql/schema/validator/required_validator.rb +29 -15
- data/lib/graphql/schema/validator.rb +4 -7
- data/lib/graphql/schema/warden.rb +11 -2
- data/lib/graphql/schema.rb +34 -10
- data/lib/graphql/static_validation/all_rules.rb +1 -0
- data/lib/graphql/static_validation/base_visitor.rb +1 -1
- data/lib/graphql/static_validation/rules/arguments_are_defined.rb +1 -1
- data/lib/graphql/static_validation/rules/directives_are_in_valid_locations.rb +1 -1
- data/lib/graphql/static_validation/rules/fields_will_merge.rb +14 -7
- data/lib/graphql/static_validation/rules/query_root_exists.rb +17 -0
- data/lib/graphql/static_validation/rules/query_root_exists_error.rb +26 -0
- data/lib/graphql/static_validation/rules/required_arguments_are_present.rb +3 -1
- data/lib/graphql/static_validation/rules/unique_directives_per_location.rb +1 -1
- data/lib/graphql/static_validation/rules/variable_usages_are_allowed.rb +6 -0
- data/lib/graphql/static_validation/validation_context.rb +4 -0
- data/lib/graphql/string_type.rb +1 -1
- data/lib/graphql/subscriptions/action_cable_subscriptions.rb +2 -0
- data/lib/graphql/subscriptions/serialize.rb +22 -2
- data/lib/graphql/tracing/active_support_notifications_tracing.rb +6 -20
- data/lib/graphql/tracing/data_dog_tracing.rb +24 -15
- data/lib/graphql/tracing/notifications_tracing.rb +59 -0
- data/lib/graphql/tracing/platform_tracing.rb +20 -10
- data/lib/graphql/types/iso_8601_date.rb +13 -5
- data/lib/graphql/types/iso_8601_date_time.rb +8 -1
- data/lib/graphql/types/relay/connection_behaviors.rb +28 -10
- data/lib/graphql/types/relay/default_relay.rb +5 -1
- data/lib/graphql/types/relay/edge_behaviors.rb +13 -2
- data/lib/graphql/types/relay/node_field.rb +2 -3
- data/lib/graphql/types/relay/nodes_field.rb +19 -3
- data/lib/graphql/types/string.rb +1 -1
- data/lib/graphql/union_type.rb +1 -1
- data/lib/graphql/version.rb +1 -1
- data/lib/graphql.rb +14 -1
- metadata +56 -30
- /data/lib/generators/graphql/{templates → install/templates}/base_mutation.erb +0 -0
- /data/lib/generators/graphql/{templates → install/templates}/mutation_type.erb +0 -0
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: e6d252cf12d6c3f2ccd6f20a139e0c2f3c460a6c25b8c1cb663e3cacf5e12855
|
4
|
+
data.tar.gz: 0d6eda9b7e23c0c58677b44ed1de968c6d60acd4a5f5a49883935ea6e326d691
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 302f1e1fe6e7a31ede6c3eb4e3f63f3a2818b47286e4b2482e5109b47c8fcd1e229815c964fdd11ceed8084db4ccac9578f8374e53b2080bbb8cc03473a50ffb
|
7
|
+
data.tar.gz: 85daddd620840df2718f2730b475f3c2f9df7b318145044c7fb1aada43c4dee1a00c503440953d8711e58265a7f6eb768f58ac8d1af60fe6e50bf7078d7c86b5
|
@@ -19,17 +19,12 @@ module Graphql
|
|
19
19
|
sentinel = /< GraphQL::Schema\s*\n/m
|
20
20
|
|
21
21
|
in_root do
|
22
|
-
|
22
|
+
if File.exist?(schema_file_path)
|
23
|
+
inject_into_file schema_file_path, " #{type}(Types::#{name})\n", after: sentinel, verbose: false, force: false
|
24
|
+
end
|
23
25
|
end
|
24
26
|
end
|
25
27
|
|
26
|
-
def create_mutation_root_type
|
27
|
-
create_dir("#{options[:directory]}/mutations")
|
28
|
-
template("base_mutation.erb", "#{options[:directory]}/mutations/base_mutation.rb", { skip: true })
|
29
|
-
template("mutation_type.erb", "#{options[:directory]}/types/mutation_type.rb", { skip: true })
|
30
|
-
insert_root_type('mutation', 'MutationType')
|
31
|
-
end
|
32
|
-
|
33
28
|
def schema_file_path
|
34
29
|
"#{options[:directory]}/#{schema_name.underscore}.rb"
|
35
30
|
end
|
@@ -13,20 +13,14 @@ module Graphql
|
|
13
13
|
desc "Create a GraphQL::EnumType with the given name and values"
|
14
14
|
source_root File.expand_path('../templates', __FILE__)
|
15
15
|
|
16
|
-
|
17
|
-
type: :array,
|
18
|
-
default: [],
|
19
|
-
banner: "value{:ruby_value} value{:ruby_value} ...",
|
20
|
-
desc: "Values for this enum (if present, ruby_value will be inserted verbatim)"
|
16
|
+
private
|
21
17
|
|
22
|
-
def
|
23
|
-
|
18
|
+
def graphql_type
|
19
|
+
"enum"
|
24
20
|
end
|
25
21
|
|
26
|
-
private
|
27
|
-
|
28
22
|
def prepared_values
|
29
|
-
|
23
|
+
custom_fields.map { |v| v.split(":", 2) }
|
30
24
|
end
|
31
25
|
end
|
32
26
|
end
|
@@ -0,0 +1,31 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
require 'rails/generators/base'
|
3
|
+
|
4
|
+
module Graphql
|
5
|
+
module Generators
|
6
|
+
module FieldExtractor
|
7
|
+
def fields
|
8
|
+
columns = []
|
9
|
+
columns += (klass&.columns&.map { |c| generate_column_string(c) } || [])
|
10
|
+
columns + custom_fields
|
11
|
+
end
|
12
|
+
|
13
|
+
def generate_column_string(column)
|
14
|
+
name = column.name
|
15
|
+
required = column.null ? "" : "!"
|
16
|
+
type = column_type_string(column)
|
17
|
+
"#{name}:#{required}#{type}"
|
18
|
+
end
|
19
|
+
|
20
|
+
def column_type_string(column)
|
21
|
+
column.name == "id" ? "ID" : column.type.to_s.camelize
|
22
|
+
end
|
23
|
+
|
24
|
+
def klass
|
25
|
+
@klass ||= Module.const_get(name.camelize)
|
26
|
+
rescue NameError
|
27
|
+
@klass = nil
|
28
|
+
end
|
29
|
+
end
|
30
|
+
end
|
31
|
+
end
|
@@ -0,0 +1,50 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
require 'generators/graphql/type_generator'
|
3
|
+
require 'generators/graphql/field_extractor'
|
4
|
+
|
5
|
+
module Graphql
|
6
|
+
module Generators
|
7
|
+
# Generate an input type by name,
|
8
|
+
# with the specified fields.
|
9
|
+
#
|
10
|
+
# ```
|
11
|
+
# rails g graphql:object PostType name:string!
|
12
|
+
# ```
|
13
|
+
class InputGenerator < TypeGeneratorBase
|
14
|
+
desc "Create a GraphQL::InputObjectType with the given name and fields"
|
15
|
+
source_root File.expand_path('../templates', __FILE__)
|
16
|
+
include FieldExtractor
|
17
|
+
|
18
|
+
def self.normalize_type_expression(type_expression, mode:, null: true)
|
19
|
+
case type_expression.camelize
|
20
|
+
when "Text", "Citext"
|
21
|
+
["String", null]
|
22
|
+
when "Decimal"
|
23
|
+
["Float", null]
|
24
|
+
when "DateTime", "Datetime"
|
25
|
+
["GraphQL::Types::ISO8601DateTime", null]
|
26
|
+
when "Date"
|
27
|
+
["GraphQL::Types::ISO8601Date", null]
|
28
|
+
when "Json", "Jsonb", "Hstore"
|
29
|
+
["GraphQL::Types::JSON", null]
|
30
|
+
else
|
31
|
+
super
|
32
|
+
end
|
33
|
+
end
|
34
|
+
|
35
|
+
private
|
36
|
+
|
37
|
+
def graphql_type
|
38
|
+
"input"
|
39
|
+
end
|
40
|
+
|
41
|
+
def type_ruby_name
|
42
|
+
super.gsub(/Type\z/, "InputType")
|
43
|
+
end
|
44
|
+
|
45
|
+
def type_file_name
|
46
|
+
super.gsub(/_type\z/, "_input_type")
|
47
|
+
end
|
48
|
+
end
|
49
|
+
end
|
50
|
+
end
|
@@ -0,0 +1,34 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "rails/generators/base"
|
4
|
+
require_relative "../core"
|
5
|
+
|
6
|
+
module Graphql
|
7
|
+
module Generators
|
8
|
+
module Install
|
9
|
+
class MutationRootGenerator < Rails::Generators::Base
|
10
|
+
include Core
|
11
|
+
|
12
|
+
desc "Create mutation base type, mutation root tipe, and adds the latter to the schema"
|
13
|
+
source_root File.expand_path('../templates', __FILE__)
|
14
|
+
|
15
|
+
class_option :schema,
|
16
|
+
type: :string,
|
17
|
+
default: nil,
|
18
|
+
desc: "Name for the schema constant (default: {app_name}Schema)"
|
19
|
+
|
20
|
+
class_option :skip_keeps,
|
21
|
+
type: :boolean,
|
22
|
+
default: false,
|
23
|
+
desc: "Skip .keep files for source control"
|
24
|
+
|
25
|
+
def generate
|
26
|
+
create_dir("#{options[:directory]}/mutations")
|
27
|
+
template("base_mutation.erb", "#{options[:directory]}/mutations/base_mutation.rb", { skip: true })
|
28
|
+
template("mutation_type.erb", "#{options[:directory]}/types/mutation_type.rb", { skip: true })
|
29
|
+
insert_root_type('mutation', 'MutationType')
|
30
|
+
end
|
31
|
+
end
|
32
|
+
end
|
33
|
+
end
|
34
|
+
end
|
@@ -109,7 +109,7 @@ module Graphql
|
|
109
109
|
template("query_type.erb", "#{options[:directory]}/types/query_type.rb")
|
110
110
|
insert_root_type('query', 'QueryType')
|
111
111
|
|
112
|
-
|
112
|
+
invoke "graphql:install:mutation_root" unless options.skip_mutation_root_type?
|
113
113
|
|
114
114
|
template("graphql_controller.erb", "app/controllers/graphql_controller.rb")
|
115
115
|
route('post "/graphql", to: "graphql#execute"')
|
@@ -122,8 +122,15 @@ module Graphql
|
|
122
122
|
if options.api?
|
123
123
|
say("Skipped graphiql, as this rails project is API only")
|
124
124
|
say(" You may wish to use GraphiQL.app for development: https://github.com/skevy/graphiql-app")
|
125
|
-
elsif !options[:skip_graphiql]
|
126
|
-
gem(
|
125
|
+
elsif !options[:skip_graphiql]
|
126
|
+
# `gem(...)` uses `gsub_file(...)` under the hood, which is a no-op for `rails destroy...` (when `behavior == :revoke`).
|
127
|
+
# So handle that case by calling `gsub_file` with `force: true`.
|
128
|
+
if behavior == :invoke && !File.read(Rails.root.join("Gemfile")).include?("graphiql-rails")
|
129
|
+
gem("graphiql-rails", group: :development)
|
130
|
+
elsif behavior == :revoke
|
131
|
+
gemfile_pattern = /\n\s*gem ('|")graphiql-rails('|"), :?group(:| =>) :development/
|
132
|
+
gsub_file Rails.root.join("Gemfile"), gemfile_pattern, "", { force: true }
|
133
|
+
end
|
127
134
|
|
128
135
|
# This is a little cheat just to get cleaner shell output:
|
129
136
|
log :route, 'graphiql-rails'
|
@@ -13,14 +13,14 @@ module Graphql
|
|
13
13
|
desc "Create a GraphQL::InterfaceType with the given name and fields"
|
14
14
|
source_root File.expand_path('../templates', __FILE__)
|
15
15
|
|
16
|
-
|
17
|
-
type: :array,
|
18
|
-
default: [],
|
19
|
-
banner: "name:type name:type ...",
|
20
|
-
desc: "Fields for this interface (type may be expressed as Ruby or GraphQL)"
|
16
|
+
private
|
21
17
|
|
22
|
-
def
|
23
|
-
|
18
|
+
def graphql_type
|
19
|
+
"interface"
|
20
|
+
end
|
21
|
+
|
22
|
+
def fields
|
23
|
+
custom_fields
|
24
24
|
end
|
25
25
|
end
|
26
26
|
end
|
@@ -0,0 +1,22 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
require_relative 'orm_mutations_base'
|
3
|
+
|
4
|
+
module Graphql
|
5
|
+
module Generators
|
6
|
+
# TODO: What other options should be supported?
|
7
|
+
#
|
8
|
+
# @example Generate a `GraphQL::Schema::RelayClassicMutation` by name
|
9
|
+
# rails g graphql:mutation CreatePostMutation
|
10
|
+
class MutationCreateGenerator < OrmMutationsBase
|
11
|
+
|
12
|
+
desc "Scaffold a Relay Classic ORM create mutation for the given model class"
|
13
|
+
source_root File.expand_path('../templates', __FILE__)
|
14
|
+
|
15
|
+
private
|
16
|
+
|
17
|
+
def operation_type
|
18
|
+
"create"
|
19
|
+
end
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
@@ -0,0 +1,22 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
require_relative 'orm_mutations_base'
|
3
|
+
|
4
|
+
module Graphql
|
5
|
+
module Generators
|
6
|
+
# TODO: What other options should be supported?
|
7
|
+
#
|
8
|
+
# @example Generate a `GraphQL::Schema::RelayClassicMutation` by name
|
9
|
+
# rails g graphql:mutation CreatePostMutation
|
10
|
+
class MutationDeleteGenerator < OrmMutationsBase
|
11
|
+
|
12
|
+
desc "Scaffold a Relay Classic ORM delete mutation for the given model class"
|
13
|
+
source_root File.expand_path('../templates', __FILE__)
|
14
|
+
|
15
|
+
private
|
16
|
+
|
17
|
+
def operation_type
|
18
|
+
"delete"
|
19
|
+
end
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
@@ -9,47 +9,22 @@ module Graphql
|
|
9
9
|
#
|
10
10
|
# @example Generate a `GraphQL::Schema::RelayClassicMutation` by name
|
11
11
|
# rails g graphql:mutation CreatePostMutation
|
12
|
-
class MutationGenerator < Rails::Generators::
|
12
|
+
class MutationGenerator < Rails::Generators::NamedBase
|
13
13
|
include Core
|
14
14
|
|
15
15
|
desc "Create a Relay Classic mutation by name"
|
16
16
|
source_root File.expand_path('../templates', __FILE__)
|
17
17
|
|
18
|
-
argument :name, type: :string
|
19
|
-
|
20
|
-
def initialize(args, *options) # :nodoc:
|
21
|
-
# Unfreeze name in case it's given as a frozen string
|
22
|
-
args[0] = args[0].dup if args[0].is_a?(String) && args[0].frozen?
|
23
|
-
super
|
24
|
-
|
25
|
-
assign_names!(name)
|
26
|
-
end
|
27
|
-
|
28
|
-
attr_reader :file_name, :mutation_name, :field_name
|
29
|
-
|
30
18
|
def create_mutation_file
|
31
|
-
|
32
|
-
create_mutation_root_type
|
33
|
-
else
|
34
|
-
log :gsub, "#{options[:directory]}/types/mutation_type.rb"
|
35
|
-
end
|
36
|
-
|
37
|
-
template "mutation.erb", "#{options[:directory]}/mutations/#{file_name}.rb"
|
19
|
+
template "mutation.erb", File.join(options[:directory], "/mutations/", class_path, "#{file_name}.rb")
|
38
20
|
|
39
21
|
sentinel = /class .*MutationType\s*<\s*[^\s]+?\n/m
|
40
22
|
in_root do
|
41
|
-
|
42
|
-
|
23
|
+
path = "#{options[:directory]}/types/mutation_type.rb"
|
24
|
+
invoke "graphql:install:mutation_root" unless File.exist?(path)
|
25
|
+
inject_into_file "#{options[:directory]}/types/mutation_type.rb", " field :#{file_name}, mutation: Mutations::#{class_name}\n", after: sentinel, verbose: false, force: false
|
43
26
|
end
|
44
27
|
end
|
45
|
-
|
46
|
-
private
|
47
|
-
|
48
|
-
def assign_names!(name)
|
49
|
-
@field_name = name.camelize.underscore
|
50
|
-
@mutation_name = name.camelize(:upper)
|
51
|
-
@file_name = name.camelize.underscore
|
52
|
-
end
|
53
28
|
end
|
54
29
|
end
|
55
30
|
end
|
@@ -0,0 +1,22 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
require_relative 'orm_mutations_base'
|
3
|
+
|
4
|
+
module Graphql
|
5
|
+
module Generators
|
6
|
+
# TODO: What other options should be supported?
|
7
|
+
#
|
8
|
+
# @example Generate a `GraphQL::Schema::RelayClassicMutation` by name
|
9
|
+
# rails g graphql:mutation CreatePostMutation
|
10
|
+
class MutationUpdateGenerator < OrmMutationsBase
|
11
|
+
|
12
|
+
desc "Scaffold a Relay Classic ORM update mutation for the given model class"
|
13
|
+
source_root File.expand_path('../templates', __FILE__)
|
14
|
+
|
15
|
+
private
|
16
|
+
|
17
|
+
def operation_type
|
18
|
+
"update"
|
19
|
+
end
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
@@ -1,5 +1,6 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
require 'generators/graphql/type_generator'
|
3
|
+
require 'generators/graphql/field_extractor'
|
3
4
|
|
4
5
|
module Graphql
|
5
6
|
module Generators
|
@@ -15,31 +16,16 @@ module Graphql
|
|
15
16
|
desc "Create a GraphQL::ObjectType with the given name and fields." \
|
16
17
|
"If the given type name matches an existing ActiveRecord model, the generated type will automatically include fields for the models database columns."
|
17
18
|
source_root File.expand_path('../templates', __FILE__)
|
18
|
-
|
19
|
-
argument :custom_fields,
|
20
|
-
type: :array,
|
21
|
-
default: [],
|
22
|
-
banner: "name:type name:type ...",
|
23
|
-
desc: "Fields for this object (type may be expressed as Ruby or GraphQL)"
|
19
|
+
include FieldExtractor
|
24
20
|
|
25
21
|
class_option :node,
|
26
22
|
type: :boolean,
|
27
23
|
default: false,
|
28
24
|
desc: "Include the Relay Node interface"
|
29
25
|
|
30
|
-
def create_type_file
|
31
|
-
template "object.erb", "#{options[:directory]}/types/#{type_file_name}.rb"
|
32
|
-
end
|
33
|
-
|
34
|
-
def fields
|
35
|
-
columns = []
|
36
|
-
columns += klass.columns.map { |c| generate_column_string(c) } if class_exists?
|
37
|
-
columns + custom_fields
|
38
|
-
end
|
39
|
-
|
40
26
|
def self.normalize_type_expression(type_expression, mode:, null: true)
|
41
|
-
case type_expression
|
42
|
-
when "Text"
|
27
|
+
case type_expression.camelize
|
28
|
+
when "Text", "Citext"
|
43
29
|
["String", null]
|
44
30
|
when "Decimal"
|
45
31
|
["Float", null]
|
@@ -47,6 +33,8 @@ module Graphql
|
|
47
33
|
["GraphQL::Types::ISO8601DateTime", null]
|
48
34
|
when "Date"
|
49
35
|
["GraphQL::Types::ISO8601Date", null]
|
36
|
+
when "Json", "Jsonb", "Hstore"
|
37
|
+
["GraphQL::Types::JSON", null]
|
50
38
|
else
|
51
39
|
super
|
52
40
|
end
|
@@ -54,25 +42,8 @@ module Graphql
|
|
54
42
|
|
55
43
|
private
|
56
44
|
|
57
|
-
def
|
58
|
-
|
59
|
-
required = column.null ? "" : "!"
|
60
|
-
type = column_type_string(column)
|
61
|
-
"#{name}:#{required}#{type}"
|
62
|
-
end
|
63
|
-
|
64
|
-
def column_type_string(column)
|
65
|
-
column.name == "id" ? "ID" : column.type.to_s.camelize
|
66
|
-
end
|
67
|
-
|
68
|
-
def class_exists?
|
69
|
-
klass.is_a?(Class) && klass.ancestors.include?(ActiveRecord::Base)
|
70
|
-
rescue NameError
|
71
|
-
return false
|
72
|
-
end
|
73
|
-
|
74
|
-
def klass
|
75
|
-
@klass ||= Module.const_get(type_name.camelize)
|
45
|
+
def graphql_type
|
46
|
+
"object"
|
76
47
|
end
|
77
48
|
end
|
78
49
|
end
|
@@ -0,0 +1,40 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
require 'rails/generators'
|
3
|
+
require 'rails/generators/named_base'
|
4
|
+
require_relative 'core'
|
5
|
+
|
6
|
+
module Graphql
|
7
|
+
module Generators
|
8
|
+
# TODO: What other options should be supported?
|
9
|
+
#
|
10
|
+
# @example Generate a `GraphQL::Schema::RelayClassicMutation` by name
|
11
|
+
# rails g graphql:mutation CreatePostMutation
|
12
|
+
class OrmMutationsBase < Rails::Generators::NamedBase
|
13
|
+
include Core
|
14
|
+
include Rails::Generators::ResourceHelpers
|
15
|
+
|
16
|
+
desc "Create a Relay Classic mutation by name"
|
17
|
+
|
18
|
+
class_option :orm, banner: "NAME", type: :string, required: true,
|
19
|
+
desc: "ORM to generate the controller for"
|
20
|
+
|
21
|
+
class_option 'namespaced_types',
|
22
|
+
type: :boolean,
|
23
|
+
required: false,
|
24
|
+
default: false,
|
25
|
+
banner: "Namespaced",
|
26
|
+
desc: "If the generated types will be namespaced"
|
27
|
+
|
28
|
+
def create_mutation_file
|
29
|
+
template "mutation_#{operation_type}.erb", File.join(options[:directory], "/mutations/", class_path, "#{file_name}_#{operation_type}.rb")
|
30
|
+
|
31
|
+
sentinel = /class .*MutationType\s*<\s*[^\s]+?\n/m
|
32
|
+
in_root do
|
33
|
+
path = "#{options[:directory]}/types/mutation_type.rb"
|
34
|
+
invoke "graphql:install:mutation_root" unless File.exist?(path)
|
35
|
+
inject_into_file "#{options[:directory]}/types/mutation_type.rb", " field :#{file_name}_#{operation_type}, mutation: Mutations::#{class_name}#{operation_type.classify}\n", after: sentinel, verbose: false, force: false
|
36
|
+
end
|
37
|
+
end
|
38
|
+
end
|
39
|
+
end
|
40
|
+
end
|
@@ -12,8 +12,10 @@ module Graphql
|
|
12
12
|
desc "Create a GraphQL::ScalarType with the given name"
|
13
13
|
source_root File.expand_path('../templates', __FILE__)
|
14
14
|
|
15
|
-
|
16
|
-
|
15
|
+
private
|
16
|
+
|
17
|
+
def graphql_type
|
18
|
+
"scalar"
|
17
19
|
end
|
18
20
|
end
|
19
21
|
end
|
@@ -1,6 +1,10 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
<% module_namespacing_when_supported do -%>
|
2
4
|
module Types
|
3
|
-
class <%=
|
5
|
+
class <%= ruby_class_name %> < Types::BaseEnum
|
6
|
+
description "<%= human_name %> enum"
|
7
|
+
|
4
8
|
<% prepared_values.each do |v| %> value "<%= v[0] %>"<%= v.length > 1 ? ", value: #{v[1]}" : "" %>
|
5
9
|
<% end %> end
|
6
10
|
end
|
@@ -1,8 +1,10 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
<% module_namespacing_when_supported do -%>
|
2
4
|
module Types
|
3
|
-
module <%=
|
5
|
+
module <%= ruby_class_name %>
|
4
6
|
include Types::BaseInterface
|
5
|
-
<% normalized_fields.each do |f| %> <%= f.
|
7
|
+
<% normalized_fields.each do |f| %> <%= f.to_object_field %>
|
6
8
|
<% end %> end
|
7
9
|
end
|
8
10
|
<% end -%>
|
@@ -0,0 +1,20 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
<% module_namespacing_when_supported do -%>
|
4
|
+
module Mutations
|
5
|
+
class <%= class_name %>Create < BaseMutation
|
6
|
+
description "Creates a new <%= file_name %>"
|
7
|
+
|
8
|
+
field :<%= file_name %>, Types::<%= options[:namespaced_types] ? 'Objects::' : '' %><%= class_name %>Type, null: false
|
9
|
+
|
10
|
+
argument :<%= file_name %>_input, Types::<%= options[:namespaced_types] ? 'Inputs::' : '' %><%= class_name %>InputType, required: true
|
11
|
+
|
12
|
+
def resolve(<%= file_name %>_input:)
|
13
|
+
<%= singular_table_name %> = ::<%= orm_class.build(class_name, "**#{file_name}_input") %>
|
14
|
+
raise GraphQL::ExecutionError.new "Error creating <%= file_name %>", extensions: <%= singular_table_name %>.errors.to_hash unless <%= orm_instance.save %>
|
15
|
+
|
16
|
+
{ <%= file_name %>: <%= singular_table_name %> }
|
17
|
+
end
|
18
|
+
end
|
19
|
+
end
|
20
|
+
<% end -%>
|
@@ -0,0 +1,20 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
<% module_namespacing_when_supported do -%>
|
4
|
+
module Mutations
|
5
|
+
class <%= class_name %>Delete < BaseMutation
|
6
|
+
description "Deletes a <%= file_name %> by ID"
|
7
|
+
|
8
|
+
field :<%= file_name %>, Types::<%= options[:namespaced_types] ? 'Objects::' : '' %><%= class_name %>Type, null: false
|
9
|
+
|
10
|
+
argument :id, ID, required: true
|
11
|
+
|
12
|
+
def resolve(id:)
|
13
|
+
<%= singular_table_name %> = ::<%= orm_class.find(class_name, "id") %>
|
14
|
+
raise GraphQL::ExecutionError.new "Error deleting <%= file_name %>", extensions: <%= singular_table_name %>.errors.to_hash unless <%= orm_instance.destroy %>
|
15
|
+
|
16
|
+
{ <%= file_name %>: <%= singular_table_name %> }
|
17
|
+
end
|
18
|
+
end
|
19
|
+
end
|
20
|
+
<% end -%>
|
@@ -0,0 +1,21 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
<% module_namespacing_when_supported do -%>
|
4
|
+
module Mutations
|
5
|
+
class <%= class_name %>Update < BaseMutation
|
6
|
+
description "Updates a <%= file_name %> by id"
|
7
|
+
|
8
|
+
field :<%= file_name %>, Types::<%= options[:namespaced_types] ? 'Objects::' : '' %><%= class_name %>Type, null: false
|
9
|
+
|
10
|
+
argument :id, ID, required: true
|
11
|
+
argument :<%= file_name %>_input, Types::<%= options[:namespaced_types] ? 'Inputs::' : '' %><%= class_name %>InputType, required: true
|
12
|
+
|
13
|
+
def resolve(id:, <%= file_name %>_input:)
|
14
|
+
<%= singular_table_name %> = ::<%= orm_class.find(class_name, "id") %>
|
15
|
+
raise GraphQL::ExecutionError.new "Error updating <%= file_name %>", extensions: <%= singular_table_name %>.errors.to_hash unless <%= orm_instance.update("**#{file_name}_input") %>
|
16
|
+
|
17
|
+
{ <%= file_name %>: <%= singular_table_name %> }
|
18
|
+
end
|
19
|
+
end
|
20
|
+
end
|
21
|
+
<% end -%>
|
@@ -1,8 +1,10 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
<% module_namespacing_when_supported do -%>
|
2
4
|
module Types
|
3
|
-
class <%=
|
5
|
+
class <%= ruby_class_name %> < Types::BaseObject
|
4
6
|
<% if options.node %> implements GraphQL::Types::Relay::Node
|
5
|
-
<% end %><% normalized_fields.each do |f| %> <%= f.
|
7
|
+
<% end %><% normalized_fields.each do |f| %> <%= f.to_object_field %>
|
6
8
|
<% end %> end
|
7
9
|
end
|
8
10
|
<% end -%>
|
@@ -1,6 +1,8 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
<% module_namespacing_when_supported do -%>
|
2
4
|
module Types
|
3
|
-
class <%=
|
5
|
+
class <%= ruby_class_name %> < Types::BaseScalar
|
4
6
|
def self.coerce_input(input_value, context)
|
5
7
|
# Override this to prepare a client-provided GraphQL value for your Ruby code
|
6
8
|
input_value
|
@@ -1,7 +1,9 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
<% module_namespacing_when_supported do -%>
|
2
4
|
module Types
|
3
|
-
class <%=
|
4
|
-
<% if
|
5
|
+
class <%= ruby_class_name %> < Types::BaseUnion
|
6
|
+
<% if custom_fields.any? %> possible_types <%= normalized_possible_types.join(", ") %>
|
5
7
|
<% end %> end
|
6
8
|
end
|
7
9
|
<% end -%>
|