debian-mirror-gitlab/app/graphql/resolvers/base_resolver.rb

144 lines
3.4 KiB
Ruby
Raw Normal View History

2018-12-05 23:21:45 +05:30
# frozen_string_literal: true
2018-11-08 19:23:39 +05:30
module Resolvers
class BaseResolver < GraphQL::Schema::Resolver
2020-01-01 13:55:28 +05:30
extend ::Gitlab::Utils::Override
2020-06-23 00:09:42 +05:30
include ::Gitlab::Utils::StrongMemoize
2021-01-03 14:25:43 +05:30
include ::Gitlab::Graphql::GlobalIDCompatibility
argument_class ::Types::BaseArgument
2020-01-01 13:55:28 +05:30
2021-01-29 00:20:46 +05:30
def self.singular_type
return unless type
2020-06-23 00:09:42 +05:30
2021-01-29 00:20:46 +05:30
unwrapped = type.unwrap
%i[node_type relay_node_type of_type itself].reduce(nil) do |t, m|
t || unwrapped.try(m)
end
end
2020-03-13 15:44:24 +05:30
2021-01-29 00:20:46 +05:30
def self.when_single(&block)
as_single << block
# Have we been called after defining the single version of this resolver?
if @single.present?
@single.instance_exec(&block)
end
end
def self.as_single
@as_single ||= []
end
def self.single_definition_blocks
ancestors.flat_map { |klass| klass.try(:as_single) || [] }
end
def self.single
@single ||= begin
parent = self
klass = Class.new(self) do
type parent.singular_type, null: true
def ready?(**args)
ready, early_return = super
[ready, select_result(early_return)]
end
def resolve(**args)
select_result(super)
end
def single?
true
end
def select_result(results)
results&.first
end
define_singleton_method :to_s do
"#{parent}.single"
end
2020-03-13 15:44:24 +05:30
end
2020-06-23 00:09:42 +05:30
2021-01-29 00:20:46 +05:30
single_definition_blocks.each do |definition|
klass.instance_exec(&definition)
2020-06-23 00:09:42 +05:30
end
2021-01-29 00:20:46 +05:30
klass
2019-07-07 11:18:12 +05:30
end
end
2019-07-31 22:56:46 +05:30
2019-12-26 22:10:19 +05:30
def self.last
2021-01-29 00:20:46 +05:30
parent = self
2020-06-23 00:09:42 +05:30
@last ||= Class.new(self.single) do
2021-01-29 00:20:46 +05:30
type parent.singular_type, null: true
2020-06-23 00:09:42 +05:30
def select_result(results)
results&.last
2020-03-13 15:44:24 +05:30
end
2021-01-29 00:20:46 +05:30
define_singleton_method :to_s do
"#{parent}.last"
end
2019-12-26 22:10:19 +05:30
end
end
2020-04-08 14:13:33 +05:30
def self.complexity
0
end
2019-09-04 21:01:54 +05:30
def self.resolver_complexity(args, child_complexity:)
2019-07-31 22:56:46 +05:30
complexity = 1
complexity += 1 if args[:sort]
complexity += 5 if args[:search]
complexity
end
def self.complexity_multiplier(args)
# When fetching many items, additional complexity is added to the field
# depending on how many items is fetched. For each item we add 1% of the
# original complexity - this means that loading 100 items (our default
# maxp_age_size limit) doubles the original complexity.
#
# Complexity is not increased when searching by specific ID(s), because
# complexity difference is minimal in this case.
[args[:iid], args[:iids]].any? ? 0 : 0.01
end
2020-01-01 13:55:28 +05:30
override :object
def object
super.tap do |obj|
2020-03-13 15:44:24 +05:30
# If the field this resolver is used in is wrapped in a presenter, unwrap its subject
2020-01-01 13:55:28 +05:30
break obj.subject if obj.is_a?(Gitlab::View::Presenter::Base)
end
end
2020-03-13 15:44:24 +05:30
2021-01-29 00:20:46 +05:30
# TODO: remove! This should never be necessary
# Remove as part of https://gitlab.com/gitlab-org/gitlab/-/issues/13984,
# since once we use that authorization approach, the object is guaranteed to
# be synchronized before any field.
2020-06-23 00:09:42 +05:30
def synchronized_object
strong_memoize(:synchronized_object) do
2021-01-29 00:20:46 +05:30
::Gitlab::Graphql::Lazy.force(object)
2020-06-23 00:09:42 +05:30
end
end
2020-03-13 15:44:24 +05:30
def single?
false
end
def current_user
context[:current_user]
end
2020-07-28 23:09:34 +05:30
# Overridden in sub-classes (see .single, .last)
def select_result(results)
results
end
2018-11-08 19:23:39 +05:30
end
end