hq-graphql 2.1.12 → 2.2.3

Sign up to get free protection for your applications and to get access to all the features.
Files changed (37) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +25 -9
  3. data/lib/hq/graphql.rb +17 -8
  4. data/lib/hq/graphql/association_loader.rb +1 -0
  5. data/lib/hq/graphql/comparator.rb +22 -21
  6. data/lib/hq/graphql/enum/filter_operation.rb +15 -0
  7. data/lib/hq/graphql/enum/sort_by.rb +8 -4
  8. data/lib/hq/graphql/enum/sort_order.rb +8 -4
  9. data/lib/hq/graphql/ext.rb +8 -0
  10. data/lib/hq/graphql/ext/active_record_extensions.rb +148 -0
  11. data/lib/hq/graphql/ext/enum_extensions.rb +81 -0
  12. data/lib/hq/graphql/ext/input_object_extensions.rb +110 -0
  13. data/lib/hq/graphql/ext/mutation_extensions.rb +24 -0
  14. data/lib/hq/graphql/ext/object_extensions.rb +122 -0
  15. data/lib/hq/graphql/ext/schema_extensions.rb +50 -0
  16. data/lib/hq/graphql/field.rb +1 -1
  17. data/lib/hq/graphql/filter_operations.rb +59 -0
  18. data/lib/hq/graphql/filters.rb +150 -0
  19. data/lib/hq/graphql/paginated_association_loader.rb +1 -0
  20. data/lib/hq/graphql/record_loader.rb +1 -0
  21. data/lib/hq/graphql/resource.rb +86 -15
  22. data/lib/hq/graphql/resource/auto_mutation.rb +8 -5
  23. data/lib/hq/graphql/root_mutation.rb +1 -1
  24. data/lib/hq/graphql/root_query.rb +3 -3
  25. data/lib/hq/graphql/scalars.rb +0 -2
  26. data/lib/hq/graphql/types.rb +1 -2
  27. data/lib/hq/graphql/types/uuid.rb +3 -5
  28. data/lib/hq/graphql/util.rb +13 -0
  29. data/lib/hq/graphql/version.rb +1 -1
  30. metadata +28 -38
  31. data/lib/hq/graphql/active_record_extensions.rb +0 -143
  32. data/lib/hq/graphql/enum.rb +0 -78
  33. data/lib/hq/graphql/input_object.rb +0 -104
  34. data/lib/hq/graphql/mutation.rb +0 -15
  35. data/lib/hq/graphql/object.rb +0 -120
  36. data/lib/hq/graphql/schema.rb +0 -22
  37. data/lib/hq/graphql/types/object.rb +0 -35
@@ -0,0 +1,81 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "hq/graphql/types"
4
+
5
+ module HQ::GraphQL
6
+ module Ext
7
+ module EnumExtensions
8
+ ## Auto generate enums from the database using ActiveRecord
9
+ # This comes in handy when we have constants that we want represented as enums.
10
+ #
11
+ # == Example
12
+ # Let's assume we're saving data into a user types table
13
+ # # select * from user_types;
14
+ # id | name
15
+ # --- +-------------
16
+ # 1 | Admin
17
+ # 2 | Support User
18
+ # (2 rows)
19
+ #
20
+ # ```ruby
21
+ # class Enums::UserType < ::HQ::GraphQL::Enum
22
+ # with_model
23
+ # end
24
+ # ```
25
+ #
26
+ # Creates the following enum:
27
+ # ```graphql
28
+ # enum UserType {
29
+ # Admin
30
+ # SupportUser
31
+ # }
32
+ # ```
33
+ def with_model(
34
+ klass = default_model_name.safe_constantize,
35
+ prefix: nil,
36
+ register: true,
37
+ scope: nil,
38
+ strip: /(^[^_a-zA-Z])|([^_a-zA-Z0-9]*)/,
39
+ value_method: :name
40
+ )
41
+ raise ArgumentError.new(<<~ERROR) if !klass
42
+ `::HQ::GraphQL::Enum.with_model {...}' had trouble automatically inferring the class name.
43
+ Avoid this by manually passing in the class name: `::HQ::GraphQL::Enum.with_model(#{default_model_name}) {...}`
44
+ ERROR
45
+
46
+ if register
47
+ ::HQ::GraphQL.enums << klass
48
+ ::HQ::GraphQL::Types.register(klass, self)
49
+ end
50
+
51
+ lazy_load do
52
+ records = scope ? klass.instance_exec(&scope) : klass.all
53
+ records.each do |record|
54
+ value "#{prefix}#{record.send(value_method).gsub(strip, "")}", value: record
55
+ end
56
+ end
57
+ end
58
+
59
+ def lazy_load(&block)
60
+ @lazy_load ||= []
61
+ if block
62
+ ::HQ::GraphQL.lazy_load(self)
63
+ @lazy_load << block
64
+ end
65
+ @lazy_load
66
+ end
67
+
68
+ def lazy_load!
69
+ lazy_load.shift.call while lazy_load.length > 0
70
+ @lazy_load = []
71
+ end
72
+
73
+ def default_model_name
74
+ to_s.sub(/^((::)?\w+)::/, "")
75
+ end
76
+ end
77
+ end
78
+ end
79
+
80
+
81
+ ::GraphQL::Schema::Enum.extend ::HQ::GraphQL::Ext::EnumExtensions
@@ -0,0 +1,110 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "hq/graphql/ext/active_record_extensions"
4
+ require "hq/graphql/inputs"
5
+ require "hq/graphql/types"
6
+
7
+ module HQ
8
+ module GraphQL
9
+ module Ext
10
+ module InputObjectExtensions
11
+ def self.included(klass)
12
+ klass.include Scalars
13
+ klass.include InstanceMethods
14
+ klass.include ActiveRecordExtensions
15
+ klass.extend ActiveRecordExtensions
16
+ klass.extend ClassMethods
17
+ end
18
+
19
+ module InstanceMethods
20
+ # Recursively format attributes so that they are compatible with `accepts_nested_attributes_for`
21
+ def format_nested_attributes
22
+ self.each.inject({}) do |formatted_attrs, (key, value) |
23
+ if self.class.nested_attributes.include?(key.to_s)
24
+ formatted_value =
25
+ if value.is_a?(Array)
26
+ value.map(&:format_nested_attributes)
27
+ elsif value
28
+ value.format_nested_attributes
29
+ end
30
+
31
+ formatted_attrs[:"#{key}_attributes"] = formatted_value if formatted_value
32
+ elsif key.to_s == "x"
33
+ formatted_attrs[:X] = value
34
+ else
35
+ formatted_attrs[key] = value
36
+ end
37
+ formatted_attrs
38
+ end
39
+ end
40
+ end
41
+
42
+ module ClassMethods
43
+ #### Class Methods ####
44
+ def with_model(model_name, attributes: true, associations: false, enums: true, excluded_inputs: [])
45
+ self.model_name = model_name
46
+ self.auto_load_attributes = attributes
47
+ self.auto_load_associations = associations
48
+ self.auto_load_enums = enums
49
+
50
+ lazy_load do
51
+ excluded_inputs += ::HQ::GraphQL.excluded_inputs
52
+
53
+ model_columns.each do |column|
54
+ argument_from_column(column) unless excluded_inputs.include?(column.name.to_sym)
55
+ end
56
+
57
+ model_associations.each do |association|
58
+ argument_from_association(association) unless excluded_inputs.include?(association.name.to_sym)
59
+ end
60
+
61
+ argument :X, String, required: false
62
+ end
63
+ end
64
+
65
+ def nested_attributes
66
+ @nested_attributes ||= Set.new
67
+ end
68
+
69
+ private
70
+
71
+ def argument_from_association(association)
72
+ is_enum = is_enum?(association)
73
+ input_or_type = is_enum ? ::HQ::GraphQL::Types[association.klass] : ::HQ::GraphQL::Inputs[association.klass]
74
+ name = association.name
75
+ return if argument_exists?(name)
76
+
77
+ case association.macro
78
+ when :has_many
79
+ argument name, [input_or_type], required: false
80
+ else
81
+ argument name, input_or_type, required: false
82
+ end
83
+
84
+ return if is_enum
85
+
86
+ if !model_klass.nested_attributes_options.key?(name.to_sym)
87
+ model_klass.accepts_nested_attributes_for name, allow_destroy: true
88
+ end
89
+
90
+ nested_attributes << name.to_s
91
+ rescue ::HQ::GraphQL::Inputs::Error
92
+ nil
93
+ end
94
+
95
+ def argument_from_column(column)
96
+ name = column.name
97
+ return if argument_exists?(name)
98
+ argument name, ::HQ::GraphQL::Types.type_from_column(column), required: false
99
+ end
100
+
101
+ def argument_exists?(name)
102
+ !!arguments[camelize(name)]
103
+ end
104
+ end
105
+ end
106
+ end
107
+ end
108
+ end
109
+
110
+ ::GraphQL::Schema::InputObject.include ::HQ::GraphQL::Ext::InputObjectExtensions
@@ -0,0 +1,24 @@
1
+ # frozen_string_literal: true
2
+
3
+ module HQ
4
+ module GraphQL
5
+ module Ext
6
+ module MutationExtensions
7
+ def self.included(klass)
8
+ klass.include Scalars
9
+ klass.include ActiveRecordExtensions
10
+ klass.singleton_class.prepend PrependMethods
11
+ end
12
+
13
+ module PrependMethods
14
+ def generate_payload_type
15
+ lazy_load!
16
+ super
17
+ end
18
+ end
19
+ end
20
+ end
21
+ end
22
+ end
23
+
24
+ ::GraphQL::Schema::Mutation.include ::HQ::GraphQL::Ext::MutationExtensions
@@ -0,0 +1,122 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "hq/graphql/ext/active_record_extensions"
4
+ require "hq/graphql/field"
5
+ require "hq/graphql/field_extension/association_loader_extension"
6
+ require "hq/graphql/field_extension/paginated_arguments"
7
+ require "hq/graphql/field_extension/paginated_loader"
8
+ require "hq/graphql/object_association"
9
+ require "hq/graphql/types"
10
+
11
+ module HQ
12
+ module GraphQL
13
+ module Ext
14
+ module ObjectExtensions
15
+ def self.included(klass)
16
+ klass.include Scalars
17
+ klass.include ActiveRecordExtensions
18
+ klass.extend ObjectAssociation
19
+ klass.singleton_class.prepend PrependMethods
20
+ klass.field_class Field
21
+ end
22
+
23
+ module PrependMethods
24
+ def authorize_action(action)
25
+ self.authorized_action = action
26
+ end
27
+
28
+ def authorized?(object, context)
29
+ super && ::HQ::GraphQL.authorized?(authorized_action, object, context)
30
+ end
31
+
32
+ def with_model(model_name, attributes: true, associations: true, auto_nil: true, enums: true)
33
+ self.model_name = model_name
34
+ self.auto_load_attributes = attributes
35
+ self.auto_load_associations = associations
36
+ self.auto_load_enums = enums
37
+
38
+ lazy_load do
39
+ model_columns.each do |column|
40
+ field_from_column(column, auto_nil: auto_nil)
41
+ end
42
+
43
+ model_associations.each do |association|
44
+ next if resource_reflections[association.name.to_s]
45
+ field_from_association(association, auto_nil: auto_nil)
46
+ end
47
+
48
+ resource_reflections.values.each do |resource_reflection|
49
+ reflection = resource_reflection.reflection(model_klass)
50
+ next unless reflection
51
+ field_from_association(reflection, auto_nil: auto_nil, internal_association: true, &resource_reflection.block)
52
+ end
53
+ end
54
+ end
55
+
56
+ private
57
+ attr_writer :authorized_action
58
+
59
+ def authorized_action
60
+ @authorized_action ||= :read
61
+ end
62
+
63
+ def field_from_association(association, auto_nil:, internal_association: false, &block)
64
+ association_klass = association.klass
65
+ name = association.name.to_s
66
+ return if field_exists?(name)
67
+
68
+ klass = model_klass
69
+ type = Types[association_klass]
70
+
71
+ case association.macro
72
+ when :has_many
73
+ field name, [type], null: false, klass: model_name do
74
+ if ::HQ::GraphQL.use_experimental_associations?
75
+ extension FieldExtension::PaginatedArguments, klass: association_klass
76
+ extension FieldExtension::PaginatedLoader, klass: klass, association: name, internal_association: internal_association
77
+ else
78
+ extension FieldExtension::AssociationLoaderExtension, klass: klass
79
+ end
80
+ instance_eval(&block) if block
81
+ end
82
+ when :has_one
83
+ field name, type, null: !auto_nil || !has_presence_validation?(association), klass: model_name do
84
+ extension FieldExtension::AssociationLoaderExtension, klass: klass
85
+ end
86
+ else
87
+ field name, type, null: !auto_nil || !association_required?(association), klass: model_name do
88
+ extension FieldExtension::AssociationLoaderExtension, klass: klass
89
+ end
90
+ end
91
+ rescue Types::Error
92
+ nil
93
+ end
94
+
95
+ def field_from_column(column, auto_nil:)
96
+ name = column.name
97
+ return if field_exists?(name)
98
+
99
+ field name, Types.type_from_column(column), null: !auto_nil || column.null
100
+ end
101
+
102
+ def field_exists?(name)
103
+ !!fields[camelize(name)]
104
+ end
105
+
106
+ def association_required?(association)
107
+ !association.options[:optional] || has_presence_validation?(association)
108
+ end
109
+
110
+ def has_presence_validation?(association)
111
+ model_klass.validators.any? do |validation|
112
+ next unless validation.class == ActiveRecord::Validations::PresenceValidator && !(validation.options.include?(:if) || validation.options.include?(:unless))
113
+ validation.attributes.any? { |a| a.to_s == association.name.to_s }
114
+ end
115
+ end
116
+ end
117
+ end
118
+ end
119
+ end
120
+ end
121
+
122
+ ::GraphQL::Schema::Object.include ::HQ::GraphQL::Ext::ObjectExtensions
@@ -0,0 +1,50 @@
1
+ # frozen_string_literal: true
2
+
3
+ module HQ
4
+ module GraphQL
5
+ module Ext
6
+ module SchemaExtensions
7
+ def self.prepended(klass)
8
+ klass.alias_method :add_type_and_traverse_without_types, :add_type_and_traverse
9
+ klass.alias_method :add_type_and_traverse, :add_type_and_traverse_with_types
10
+ end
11
+
12
+ def execute(*args, **options)
13
+ load_types!
14
+ super
15
+ end
16
+
17
+ def dump_directory(directory = Rails.root.join("app/graphql"))
18
+ @dump_directory ||= directory
19
+ end
20
+
21
+ def dump_filename(filename = "#{self.name.underscore}.graphql")
22
+ @dump_filename ||= filename
23
+ end
24
+
25
+ def dump
26
+ load_types!
27
+ ::FileUtils.mkdir_p(dump_directory)
28
+ ::File.open(::File.join(dump_directory, dump_filename), "w") { |file| file.write(self.to_definition) }
29
+ end
30
+
31
+ def load_types!
32
+ ::HQ::GraphQL.load_types!
33
+ return if @add_type_and_traverse_with_types.blank?
34
+ while (args, options = @add_type_and_traverse_with_types.shift)
35
+ add_type_and_traverse_without_types(*args, **options)
36
+ end
37
+ end
38
+
39
+ # Defer adding types until first schema execution
40
+ # https://github.com/rmosolgo/graphql-ruby/blob/345ebb2e3833909963067d81e0e8378717b5bdbf/lib/graphql/schema.rb#L1792
41
+ def add_type_and_traverse_with_types(*args, **options)
42
+ @add_type_and_traverse_with_types ||= []
43
+ @add_type_and_traverse_with_types.push([args, options])
44
+ end
45
+ end
46
+ end
47
+ end
48
+ end
49
+
50
+ ::GraphQL::Schema.singleton_class.prepend ::HQ::GraphQL::Ext::SchemaExtensions
@@ -20,7 +20,7 @@ module HQ
20
20
  end
21
21
  end
22
22
 
23
- def authorized?(object, ctx)
23
+ def authorized?(object, _args, ctx)
24
24
  super &&
25
25
  (!authorize || authorize.call(object, ctx)) &&
26
26
  ::HQ::GraphQL.authorize_field(authorize_action, self, object, ctx)
@@ -0,0 +1,59 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "hq/graphql/enum/filter_operation"
4
+
5
+ module HQ
6
+ module GraphQL
7
+ module FilterOperations
8
+ class Operation
9
+ attr_reader :name, :arel, :check_for_null, :sanitize
10
+
11
+ def initialize(name:, arel:, check_for_null: false, sanitize: nil)
12
+ @name = name
13
+ @arel = arel
14
+ @check_for_null = check_for_null
15
+ @sanitize = sanitize
16
+ end
17
+
18
+ def sanitize_value(value)
19
+ sanitize ? sanitize.call(value) : value
20
+ end
21
+
22
+ def to_arel(table:, column_name:, value:)
23
+ sanitized_value = sanitize_value(value)
24
+
25
+ if arel.is_a?(Proc)
26
+ return arel.call(table: table, column_name: column_name, value: sanitized_value)
27
+ end
28
+
29
+ expression = table[column_name].send(arel, sanitized_value)
30
+
31
+ if check_for_null
32
+ expression = expression.or(table[column_name].eq(nil))
33
+ end
34
+
35
+ expression
36
+ end
37
+ end
38
+
39
+ OPERATIONS = [
40
+ EQUAL = Operation.new(name: "EQUAL", arel: :eq),
41
+ NOT_EQUAL = Operation.new(name: "NOT_EQUAL", arel: :not_eq, check_for_null: true),
42
+ LIKE = Operation.new(name: "LIKE", arel: :matches, sanitize: ->(value) { "%#{ActiveRecord::Base.sanitize_sql_like(value)}%" }),
43
+ NOT_LIKE = Operation.new(name: "NOT_LIKE", arel: :does_not_match, check_for_null: true, sanitize: ->(value) { "%#{ActiveRecord::Base.sanitize_sql_like(value)}%" }),
44
+ GREATER_THAN = Operation.new(name: "GREATER_THAN", arel: :gt),
45
+ LESS_THAN = Operation.new(name: "LESS_THAN", arel: :lt),
46
+ WITH = Operation.new(
47
+ name: "WITH",
48
+ arel: ->(table:, column_name:, value:) do
49
+ if value.casecmp("t") == 0 || value.casecmp("true") == 0
50
+ table[column_name].not_eq(nil)
51
+ else
52
+ table[column_name].eq(nil)
53
+ end
54
+ end
55
+ )
56
+ ]
57
+ end
58
+ end
59
+ end