Commit 56b2638f authored by Alex Kalderimis's avatar Alex Kalderimis

Add full-path loader

This is used in the resolves projects concern, which is raised from
mutations to the more general resolver package.
parent 5ce21aa2
...@@ -3,7 +3,7 @@ ...@@ -3,7 +3,7 @@
module Mutations module Mutations
module AlertManagement module AlertManagement
class Base < BaseMutation class Base < BaseMutation
include Mutations::ResolvesProject include ResolvesProject
argument :project_path, GraphQL::ID_TYPE, argument :project_path, GraphQL::ID_TYPE,
required: true, required: true,
......
...@@ -3,7 +3,7 @@ ...@@ -3,7 +3,7 @@
module Mutations module Mutations
module Branches module Branches
class Create < BaseMutation class Create < BaseMutation
include Mutations::ResolvesProject include ResolvesProject
graphql_name 'CreateBranch' graphql_name 'CreateBranch'
......
...@@ -3,7 +3,7 @@ ...@@ -3,7 +3,7 @@
module Mutations module Mutations
module Commits module Commits
class Create < BaseMutation class Create < BaseMutation
include Mutations::ResolvesProject include ResolvesProject
graphql_name 'CommitCreate' graphql_name 'CommitCreate'
......
...@@ -3,7 +3,10 @@ ...@@ -3,7 +3,10 @@
module Mutations module Mutations
module ResolvesIssuable module ResolvesIssuable
extend ActiveSupport::Concern extend ActiveSupport::Concern
include Mutations::ResolvesProject
included do
include ResolvesProject
end
def resolve_issuable(type:, parent_path:, iid:) def resolve_issuable(type:, parent_path:, iid:)
parent = resolve_issuable_parent(type, parent_path) parent = resolve_issuable_parent(type, parent_path)
...@@ -29,7 +32,7 @@ module Mutations ...@@ -29,7 +32,7 @@ module Mutations
def resolve_issuable_parent(type, parent_path) def resolve_issuable_parent(type, parent_path)
return unless type == :issue || type == :merge_request return unless type == :issue || type == :merge_request
resolve_project(full_path: parent_path) resolve_project(full_path: parent_path) if parent_path.present?
end end
end end
end end
......
# frozen_string_literal: true
module Mutations
module ResolvesProject
extend ActiveSupport::Concern
def resolve_project(full_path:)
project_resolver.resolve(full_path: full_path)
end
def project_resolver
Resolvers::ProjectResolver.new(object: nil, context: context, field: nil)
end
end
end
...@@ -3,7 +3,7 @@ ...@@ -3,7 +3,7 @@
module Mutations module Mutations
module JiraImport module JiraImport
class Start < BaseMutation class Start < BaseMutation
include Mutations::ResolvesProject include ResolvesProject
graphql_name 'JiraImportStart' graphql_name 'JiraImportStart'
......
...@@ -3,7 +3,7 @@ ...@@ -3,7 +3,7 @@
module Mutations module Mutations
module MergeRequests module MergeRequests
class Create < BaseMutation class Create < BaseMutation
include Mutations::ResolvesProject include ResolvesProject
graphql_name 'MergeRequestCreate' graphql_name 'MergeRequestCreate'
......
...@@ -3,7 +3,7 @@ ...@@ -3,7 +3,7 @@
module Mutations module Mutations
module Snippets module Snippets
class Create < BaseMutation class Create < BaseMutation
include Mutations::ResolvesProject include ResolvesProject
graphql_name 'CreateSnippet' graphql_name 'CreateSnippet'
......
# frozen_string_literal: true
module ResolvesProject
def resolve_project(full_path: nil, project_id: nil)
unless full_path.present? ^ project_id.present?
raise ::Gitlab::Graphql::Errors::ArgumentError, 'Incompatible arguments: projectId, projectPath.'
end
if full_path.present?
::Gitlab::Graphql::Loaders::FullPathModelLoader.new(Project, full_path).find
else
::GitlabSchema.object_from_id(project_id, expected_type: Project)
end
end
end
...@@ -11,12 +11,7 @@ module Resolvers ...@@ -11,12 +11,7 @@ module Resolvers
end end
def model_by_full_path(model, full_path) def model_by_full_path(model, full_path)
BatchLoader::GraphQL.for(full_path).batch(key: model) do |full_paths, loader, args| ::Gitlab::Graphql::Loaders::FullPathModelLoader.new(model, full_path).find
# `with_route` avoids an N+1 calculating full_path
args[:key].where_full_path_in(full_paths).with_route.each do |model_instance|
loader.call(model_instance.full_path, model_instance)
end
end
end end
end end
end end
...@@ -3,7 +3,7 @@ ...@@ -3,7 +3,7 @@
module Mutations module Mutations
module RequirementsManagement module RequirementsManagement
class CreateRequirement < BaseMutation class CreateRequirement < BaseMutation
include Mutations::ResolvesProject include ResolvesProject
graphql_name 'CreateRequirement' graphql_name 'CreateRequirement'
......
...@@ -3,7 +3,7 @@ ...@@ -3,7 +3,7 @@
module Mutations module Mutations
module RequirementsManagement module RequirementsManagement
class UpdateRequirement < BaseMutation class UpdateRequirement < BaseMutation
include Mutations::ResolvesProject include ResolvesProject
graphql_name 'UpdateRequirement' graphql_name 'UpdateRequirement'
......
# frozen_string_literal: true
module Gitlab
module Graphql
module Loaders
# Suitable for use to find resources that expose `where_full_path_in`,
# such as Project, Group, Namespace
class FullPathModelLoader
attr_reader :model_class, :full_path
def initialize(model_class, full_path)
@model_class, @full_path = model_class, full_path
end
def find
BatchLoader::GraphQL.for(full_path).batch(key: model_class) do |full_paths, loader, args|
# `with_route` avoids an N+1 calculating full_path
args[:key].where_full_path_in(full_paths).with_route.each do |model_instance|
loader.call(model_instance.full_path, model_instance)
end
end
end
end
end
end
end
# frozen_string_literal: true
require 'spec_helper'
describe Mutations::ResolvesProject do
let(:mutation_class) do
Class.new(Mutations::BaseMutation) do
include Mutations::ResolvesProject
end
end
let(:context) { double }
subject(:mutation) { mutation_class.new(object: nil, context: context, field: nil) }
it 'uses the ProjectsResolver to resolve projects by path' do
project = create(:project)
expect(Resolvers::ProjectResolver).to receive(:new).with(object: nil, context: context, field: nil).and_call_original
expect(mutation.resolve_project(full_path: project.full_path).sync).to eq(project)
end
end
# frozen_string_literal: true
require 'spec_helper'
describe ResolvesProject do
include GraphqlHelpers
let(:implementing_class) do
Class.new do
include ResolvesProject
end
end
subject(:instance) { implementing_class.new }
let_it_be(:project) { create(:project) }
it 'can resolve projects by path' do
expect(sync(instance.resolve_project(full_path: project.full_path))).to eq(project)
end
it 'can resolve projects by id' do
expect(sync(instance.resolve_project(project_id: global_id_of(project)))).to eq(project)
end
it 'complains when both are present' do
expect do
instance.resolve_project(full_path: project.full_path, project_id: global_id_of(project))
end.to raise_error(::Gitlab::Graphql::Errors::ArgumentError)
end
it 'complains when neither is present' do
expect do
instance.resolve_project(full_path: nil, project_id: nil)
end.to raise_error(::Gitlab::Graphql::Errors::ArgumentError)
end
end
...@@ -29,16 +29,6 @@ RSpec.shared_examples 'resolving an issuable in GraphQL' do |type| ...@@ -29,16 +29,6 @@ RSpec.shared_examples 'resolving an issuable in GraphQL' do |type|
subject subject
end end
it 'uses correct Resolver to resolve issuable parent' do
resolver_class = type == :epic ? 'Resolvers::GroupResolver' : 'Resolvers::ProjectResolver'
expect(resolver_class.constantize).to receive(:new)
.with(object: nil, context: context, field: nil)
.and_call_original
subject
end
it 'returns nil if issuable is not found' do it 'returns nil if issuable is not found' do
result = mutation.resolve_issuable(type: type, parent_path: parent.full_path, iid: "100") result = mutation.resolve_issuable(type: type, parent_path: parent.full_path, iid: "100")
result = result.respond_to?(:sync) ? result.sync : result result = result.respond_to?(:sync) ? result.sync : result
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment