2019-07-07 11:18:12 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2018-11-08 19:23:39 +05:30
|
|
|
require 'spec_helper'
|
|
|
|
|
|
|
|
describe GitlabSchema do
|
2020-04-08 14:13:33 +05:30
|
|
|
let_it_be(:implementations) { GraphQL::Relay::BaseConnection::CONNECTION_IMPLEMENTATIONS }
|
2019-09-04 21:01:54 +05:30
|
|
|
let(:user) { build :user }
|
|
|
|
|
2018-11-08 19:23:39 +05:30
|
|
|
it 'uses batch loading' do
|
|
|
|
expect(field_instrumenters).to include(BatchLoader::GraphQL)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'enables the preload instrumenter' do
|
|
|
|
expect(field_instrumenters).to include(BatchLoader::GraphQL)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'enables the authorization instrumenter' do
|
|
|
|
expect(field_instrumenters).to include(instance_of(::Gitlab::Graphql::Authorize::Instrumentation))
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'enables using presenters' do
|
|
|
|
expect(field_instrumenters).to include(instance_of(::Gitlab::Graphql::Present::Instrumentation))
|
|
|
|
end
|
|
|
|
|
2019-09-30 21:07:59 +05:30
|
|
|
it 'enables using gitaly call checker' do
|
|
|
|
expect(field_instrumenters).to include(instance_of(::Gitlab::Graphql::CallsGitaly::Instrumentation))
|
|
|
|
end
|
|
|
|
|
2018-11-08 19:23:39 +05:30
|
|
|
it 'has the base mutation' do
|
|
|
|
expect(described_class.mutation).to eq(::Types::MutationType.to_graphql)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'has the base query' do
|
|
|
|
expect(described_class.query).to eq(::Types::QueryType.to_graphql)
|
|
|
|
end
|
|
|
|
|
2020-04-08 14:13:33 +05:30
|
|
|
it 'paginates active record relations using `Connections::Keyset::Connection`' do
|
|
|
|
connection = implementations[ActiveRecord::Relation.name]
|
2018-11-08 19:23:39 +05:30
|
|
|
|
2019-12-26 22:10:19 +05:30
|
|
|
expect(connection).to eq(Gitlab::Graphql::Connections::Keyset::Connection)
|
2018-11-08 19:23:39 +05:30
|
|
|
end
|
|
|
|
|
2020-04-08 14:13:33 +05:30
|
|
|
it 'paginates ExternallyPaginatedArray using `Connections::ExternallyPaginatedArrayConnection`' do
|
|
|
|
connection = implementations[Gitlab::Graphql::ExternallyPaginatedArray.name]
|
|
|
|
|
|
|
|
expect(connection).to eq(Gitlab::Graphql::Connections::ExternallyPaginatedArrayConnection)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'paginates FilterableArray using `Connections::FilterableArrayConnection`' do
|
|
|
|
connection = implementations[Gitlab::Graphql::FilterableArray.name]
|
|
|
|
|
|
|
|
expect(connection).to eq(Gitlab::Graphql::Connections::FilterableArrayConnection)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'paginates OffsetActiveRecordRelation using `Pagination::OffsetActiveRecordRelationConnection`' do
|
|
|
|
connection = implementations[Gitlab::Graphql::Pagination::Relations::OffsetActiveRecordRelation.name]
|
|
|
|
|
|
|
|
expect(connection).to eq(Gitlab::Graphql::Pagination::OffsetActiveRecordRelationConnection)
|
|
|
|
end
|
|
|
|
|
2019-09-04 21:01:54 +05:30
|
|
|
describe '.execute' do
|
|
|
|
context 'for different types of users' do
|
|
|
|
context 'when no context' do
|
|
|
|
it 'returns DEFAULT_MAX_COMPLEXITY' do
|
|
|
|
expect(GraphQL::Schema)
|
|
|
|
.to receive(:execute)
|
|
|
|
.with('query', hash_including(max_complexity: GitlabSchema::DEFAULT_MAX_COMPLEXITY))
|
|
|
|
|
|
|
|
described_class.execute('query')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when no user' do
|
|
|
|
it 'returns DEFAULT_MAX_COMPLEXITY' do
|
|
|
|
expect(GraphQL::Schema)
|
|
|
|
.to receive(:execute)
|
|
|
|
.with('query', hash_including(max_complexity: GitlabSchema::DEFAULT_MAX_COMPLEXITY))
|
|
|
|
|
|
|
|
described_class.execute('query', context: {})
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns DEFAULT_MAX_DEPTH' do
|
|
|
|
expect(GraphQL::Schema)
|
|
|
|
.to receive(:execute)
|
|
|
|
.with('query', hash_including(max_depth: GitlabSchema::DEFAULT_MAX_DEPTH))
|
|
|
|
|
|
|
|
described_class.execute('query', context: {})
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when a logged in user' do
|
|
|
|
it 'returns AUTHENTICATED_COMPLEXITY' do
|
|
|
|
expect(GraphQL::Schema).to receive(:execute).with('query', hash_including(max_complexity: GitlabSchema::AUTHENTICATED_COMPLEXITY))
|
|
|
|
|
|
|
|
described_class.execute('query', context: { current_user: user })
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns AUTHENTICATED_MAX_DEPTH' do
|
|
|
|
expect(GraphQL::Schema).to receive(:execute).with('query', hash_including(max_depth: GitlabSchema::AUTHENTICATED_MAX_DEPTH))
|
|
|
|
|
|
|
|
described_class.execute('query', context: { current_user: user })
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when an admin user' do
|
|
|
|
it 'returns ADMIN_COMPLEXITY' do
|
|
|
|
user = build :user, :admin
|
|
|
|
|
|
|
|
expect(GraphQL::Schema).to receive(:execute).with('query', hash_including(max_complexity: GitlabSchema::ADMIN_COMPLEXITY))
|
2019-07-07 11:18:12 +05:30
|
|
|
|
2019-09-04 21:01:54 +05:30
|
|
|
described_class.execute('query', context: { current_user: user })
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when max_complexity passed on the query' do
|
|
|
|
it 'returns what was passed on the query' do
|
|
|
|
expect(GraphQL::Schema).to receive(:execute).with('query', hash_including(max_complexity: 1234))
|
|
|
|
|
|
|
|
described_class.execute('query', max_complexity: 1234)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when max_depth passed on the query' do
|
|
|
|
it 'returns what was passed on the query' do
|
|
|
|
expect(GraphQL::Schema).to receive(:execute).with('query', hash_including(max_depth: 1234))
|
|
|
|
|
|
|
|
described_class.execute('query', max_depth: 1234)
|
|
|
|
end
|
|
|
|
end
|
2019-07-07 11:18:12 +05:30
|
|
|
end
|
2019-09-04 21:01:54 +05:30
|
|
|
end
|
2019-07-07 11:18:12 +05:30
|
|
|
|
2019-09-04 21:01:54 +05:30
|
|
|
describe '.id_from_object' do
|
|
|
|
it 'returns a global id' do
|
|
|
|
expect(described_class.id_from_object(build(:project, id: 1))).to be_a(GlobalID)
|
|
|
|
end
|
2019-07-07 11:18:12 +05:30
|
|
|
|
2019-09-04 21:01:54 +05:30
|
|
|
it "raises a meaningful error if a global id couldn't be generated" do
|
2019-09-30 21:07:59 +05:30
|
|
|
expect { described_class.id_from_object(build(:wiki_directory)) }
|
2019-09-04 21:01:54 +05:30
|
|
|
.to raise_error(RuntimeError, /include `GlobalID::Identification` into/i)
|
2019-07-07 11:18:12 +05:30
|
|
|
end
|
2019-09-04 21:01:54 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
describe '.object_from_id' do
|
|
|
|
context 'for subclasses of `ApplicationRecord`' do
|
2020-03-09 13:42:32 +05:30
|
|
|
let_it_be(:user) { create(:user) }
|
2019-09-04 21:01:54 +05:30
|
|
|
|
2020-03-09 13:42:32 +05:30
|
|
|
it 'returns the correct record' do
|
2019-09-04 21:01:54 +05:30
|
|
|
result = described_class.object_from_id(user.to_global_id.to_s)
|
2019-07-07 11:18:12 +05:30
|
|
|
|
2019-12-04 20:38:33 +05:30
|
|
|
expect(result.sync).to eq(user)
|
2019-09-04 21:01:54 +05:30
|
|
|
end
|
2019-07-07 11:18:12 +05:30
|
|
|
|
2020-03-09 13:42:32 +05:30
|
|
|
it 'returns the correct record, of the expected type' do
|
|
|
|
result = described_class.object_from_id(user.to_global_id.to_s, expected_type: ::User)
|
|
|
|
|
|
|
|
expect(result.sync).to eq(user)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'fails if the type does not match' do
|
|
|
|
expect do
|
|
|
|
described_class.object_from_id(user.to_global_id.to_s, expected_type: ::Project)
|
|
|
|
end.to raise_error(Gitlab::Graphql::Errors::ArgumentError)
|
|
|
|
end
|
|
|
|
|
2019-09-04 21:01:54 +05:30
|
|
|
it 'batchloads the queries' do
|
|
|
|
user1 = create(:user)
|
|
|
|
user2 = create(:user)
|
2019-07-07 11:18:12 +05:30
|
|
|
|
2019-09-04 21:01:54 +05:30
|
|
|
expect do
|
|
|
|
[described_class.object_from_id(user1.to_global_id),
|
2019-12-04 20:38:33 +05:30
|
|
|
described_class.object_from_id(user2.to_global_id)].map(&:sync)
|
2019-09-04 21:01:54 +05:30
|
|
|
end.not_to exceed_query_limit(1)
|
|
|
|
end
|
2019-07-07 11:18:12 +05:30
|
|
|
end
|
|
|
|
|
2019-09-30 21:07:59 +05:30
|
|
|
context 'for classes that are not ActiveRecord subclasses and have implemented .lazy_find' do
|
|
|
|
it 'returns the correct record' do
|
|
|
|
note = create(:discussion_note_on_merge_request)
|
|
|
|
|
|
|
|
result = described_class.object_from_id(note.to_global_id)
|
|
|
|
|
2019-12-04 20:38:33 +05:30
|
|
|
expect(result.sync).to eq(note)
|
2019-09-30 21:07:59 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'batchloads the queries' do
|
|
|
|
note1 = create(:discussion_note_on_merge_request)
|
|
|
|
note2 = create(:discussion_note_on_merge_request)
|
|
|
|
|
|
|
|
expect do
|
|
|
|
[described_class.object_from_id(note1.to_global_id),
|
2019-12-04 20:38:33 +05:30
|
|
|
described_class.object_from_id(note2.to_global_id)].map(&:sync)
|
2019-09-30 21:07:59 +05:30
|
|
|
end.not_to exceed_query_limit(1)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-09-04 21:01:54 +05:30
|
|
|
context 'for other classes' do
|
|
|
|
# We cannot use an anonymous class here as `GlobalID` expects `.name` not
|
|
|
|
# to return `nil`
|
|
|
|
class TestGlobalId
|
|
|
|
include GlobalID::Identification
|
|
|
|
attr_accessor :id
|
2019-07-07 11:18:12 +05:30
|
|
|
|
2019-09-04 21:01:54 +05:30
|
|
|
def initialize(id)
|
|
|
|
@id = id
|
|
|
|
end
|
|
|
|
end
|
2019-07-07 11:18:12 +05:30
|
|
|
|
2019-09-04 21:01:54 +05:30
|
|
|
it 'falls back to a regular find' do
|
|
|
|
result = TestGlobalId.new(123)
|
|
|
|
|
|
|
|
expect(TestGlobalId).to receive(:find).with("123").and_return(result)
|
2019-07-07 11:18:12 +05:30
|
|
|
|
2019-09-04 21:01:54 +05:30
|
|
|
expect(described_class.object_from_id(result.to_global_id)).to eq(result)
|
|
|
|
end
|
|
|
|
end
|
2019-07-07 11:18:12 +05:30
|
|
|
|
2019-09-04 21:01:54 +05:30
|
|
|
it 'raises the correct error on invalid input' do
|
|
|
|
expect { described_class.object_from_id("bogus id") }.to raise_error(Gitlab::Graphql::Errors::ArgumentError)
|
2019-07-07 11:18:12 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-11-08 19:23:39 +05:30
|
|
|
def field_instrumenters
|
2019-07-07 11:18:12 +05:30
|
|
|
described_class.instrumenters[:field] + described_class.instrumenters[:field_after_built_ins]
|
2018-11-08 19:23:39 +05:30
|
|
|
end
|
|
|
|
end
|