2019-03-27 16:02:25 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2017-08-16 09:04:41 -04:00
|
|
|
require 'spec_helper'
|
|
|
|
|
2020-06-16 14:09:01 -04:00
|
|
|
RSpec.describe GitlabSchema do
|
2020-04-07 23:09:31 -04:00
|
|
|
let_it_be(:connections) { GitlabSchema.connections.all_wrappers }
|
2021-06-28 23:07:32 -04:00
|
|
|
|
2019-05-06 10:00:03 -04:00
|
|
|
let(:user) { build :user }
|
|
|
|
|
2017-08-16 09:04:41 -04:00
|
|
|
it 'uses batch loading' do
|
2018-02-23 10:36:40 -05:00
|
|
|
expect(field_instrumenters).to include(BatchLoader::GraphQL)
|
2017-08-16 09:04:41 -04:00
|
|
|
end
|
|
|
|
|
2020-04-04 17:09:33 -04:00
|
|
|
it 'enables the generic instrumenter' do
|
|
|
|
expect(field_instrumenters).to include(instance_of(::Gitlab::Graphql::GenericTracing))
|
2017-08-16 09:04:41 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'has the base mutation' do
|
2020-03-18 14:09:35 -04:00
|
|
|
expect(described_class.mutation).to eq(::Types::MutationType)
|
2017-08-16 09:04:41 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'has the base query' do
|
2020-03-18 14:09:35 -04:00
|
|
|
expect(described_class.query).to eq(::Types::QueryType)
|
2017-08-16 09:04:41 -04:00
|
|
|
end
|
|
|
|
|
2020-04-07 23:09:31 -04:00
|
|
|
it 'paginates active record relations using `Pagination::Keyset::Connection`' do
|
|
|
|
connection = connections[ActiveRecord::Relation]
|
2018-06-26 12:31:05 -04:00
|
|
|
|
2020-04-07 23:09:31 -04:00
|
|
|
expect(connection).to eq(Gitlab::Graphql::Pagination::Keyset::Connection)
|
2018-06-26 12:31:05 -04:00
|
|
|
end
|
|
|
|
|
2020-04-07 23:09:31 -04:00
|
|
|
it 'paginates ExternallyPaginatedArray using `Pagination::ExternallyPaginatedArrayConnection`' do
|
|
|
|
connection = connections[Gitlab::Graphql::ExternallyPaginatedArray]
|
2020-03-02 07:07:57 -05:00
|
|
|
|
2020-04-07 23:09:31 -04:00
|
|
|
expect(connection).to eq(Gitlab::Graphql::Pagination::ExternallyPaginatedArrayConnection)
|
2020-03-02 07:07:57 -05:00
|
|
|
end
|
|
|
|
|
2021-12-06 22:12:22 -05:00
|
|
|
it 'sets an appropriate validation timeout' do
|
|
|
|
expect(described_class.validate_timeout).to be <= 0.2.seconds
|
|
|
|
end
|
|
|
|
|
2019-05-06 10:00:03 -04:00
|
|
|
describe '.execute' do
|
2021-08-08 23:10:10 -04:00
|
|
|
describe 'setting query `max_complexity` and `max_depth`' do
|
2021-08-11 05:10:00 -04:00
|
|
|
subject(:result) { described_class.execute('query', **kwargs).query }
|
2021-08-08 23:10:10 -04:00
|
|
|
|
|
|
|
shared_examples 'sets default limits' do
|
|
|
|
specify do
|
|
|
|
expect(result).to have_attributes(
|
|
|
|
max_complexity: GitlabSchema::DEFAULT_MAX_COMPLEXITY,
|
|
|
|
max_depth: GitlabSchema::DEFAULT_MAX_DEPTH
|
|
|
|
)
|
2019-05-06 10:00:03 -04:00
|
|
|
end
|
|
|
|
end
|
2019-03-27 16:02:25 -04:00
|
|
|
|
2021-08-08 23:10:10 -04:00
|
|
|
context 'with no context' do
|
|
|
|
let(:kwargs) { {} }
|
2019-04-04 13:52:29 -04:00
|
|
|
|
2021-08-08 23:10:10 -04:00
|
|
|
include_examples 'sets default limits'
|
2019-05-06 10:00:03 -04:00
|
|
|
end
|
2019-03-27 16:02:25 -04:00
|
|
|
|
2021-08-08 23:10:10 -04:00
|
|
|
context 'with no :current_user' do
|
|
|
|
let(:kwargs) { { context: {} } }
|
2019-03-27 16:02:25 -04:00
|
|
|
|
2021-08-08 23:10:10 -04:00
|
|
|
include_examples 'sets default limits'
|
|
|
|
end
|
2019-03-27 16:02:25 -04:00
|
|
|
|
2021-08-08 23:10:10 -04:00
|
|
|
context 'with anonymous user' do
|
|
|
|
let(:kwargs) { { context: { current_user: nil } } }
|
2019-03-27 16:02:25 -04:00
|
|
|
|
2021-08-08 23:10:10 -04:00
|
|
|
include_examples 'sets default limits'
|
2019-05-06 10:00:03 -04:00
|
|
|
end
|
|
|
|
|
2021-08-08 23:10:10 -04:00
|
|
|
context 'with a logged in user' do
|
|
|
|
let(:kwargs) { { context: { current_user: user } } }
|
2019-05-06 10:00:03 -04:00
|
|
|
|
2021-08-08 23:10:10 -04:00
|
|
|
it 'sets authenticated user limits' do
|
|
|
|
expect(result).to have_attributes(
|
|
|
|
max_complexity: GitlabSchema::AUTHENTICATED_MAX_COMPLEXITY,
|
|
|
|
max_depth: GitlabSchema::AUTHENTICATED_MAX_DEPTH
|
|
|
|
)
|
2019-05-06 10:00:03 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-08-08 23:10:10 -04:00
|
|
|
context 'with an admin user' do
|
|
|
|
let(:kwargs) { { context: { current_user: build(:user, :admin) } } }
|
2019-05-06 10:00:03 -04:00
|
|
|
|
2021-08-08 23:10:10 -04:00
|
|
|
it 'sets admin/authenticated user limits' do
|
|
|
|
expect(result).to have_attributes(
|
|
|
|
max_complexity: GitlabSchema::ADMIN_MAX_COMPLEXITY,
|
|
|
|
max_depth: GitlabSchema::AUTHENTICATED_MAX_DEPTH
|
|
|
|
)
|
2019-05-06 10:00:03 -04:00
|
|
|
end
|
|
|
|
end
|
2019-03-27 16:02:25 -04:00
|
|
|
|
2021-08-08 23:10:10 -04:00
|
|
|
context 'when limits passed as kwargs' do
|
|
|
|
let(:kwargs) { { max_complexity: 1234, max_depth: 4321 } }
|
2019-03-27 16:02:25 -04:00
|
|
|
|
2021-08-08 23:10:10 -04:00
|
|
|
it 'sets limits from the kwargs' do
|
|
|
|
expect(result).to have_attributes(
|
|
|
|
max_complexity: 1234,
|
|
|
|
max_depth: 4321
|
|
|
|
)
|
2019-05-06 10:00:03 -04:00
|
|
|
end
|
|
|
|
end
|
2019-03-27 16:02:25 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-06-03 13:38:16 -04:00
|
|
|
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
|
|
|
|
|
|
|
|
it "raises a meaningful error if a global id couldn't be generated" do
|
2019-06-28 03:30:29 -04:00
|
|
|
expect { described_class.id_from_object(build(:wiki_directory)) }
|
2019-06-03 13:38:16 -04:00
|
|
|
.to raise_error(RuntimeError, /include `GlobalID::Identification` into/i)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '.object_from_id' do
|
2021-03-11 07:09:28 -05:00
|
|
|
context 'with subclasses of `ApplicationRecord`' do
|
2020-01-08 16:08:08 -05:00
|
|
|
let_it_be(:user) { create(:user) }
|
2019-06-03 13:38:16 -04:00
|
|
|
|
2020-01-08 16:08:08 -05:00
|
|
|
it 'returns the correct record' do
|
2019-06-03 13:38:16 -04:00
|
|
|
result = described_class.object_from_id(user.to_global_id.to_s)
|
|
|
|
|
2019-09-04 13:42:48 -04:00
|
|
|
expect(result.sync).to eq(user)
|
2019-06-03 13:38:16 -04:00
|
|
|
end
|
|
|
|
|
2020-01-08 16:08:08 -05:00
|
|
|
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-06-03 13:38:16 -04:00
|
|
|
it 'batchloads the queries' do
|
|
|
|
user1 = create(:user)
|
|
|
|
user2 = create(:user)
|
|
|
|
|
|
|
|
expect do
|
|
|
|
[described_class.object_from_id(user1.to_global_id),
|
2019-09-04 13:42:48 -04:00
|
|
|
described_class.object_from_id(user2.to_global_id)].map(&:sync)
|
2019-06-03 13:38:16 -04:00
|
|
|
end.not_to exceed_query_limit(1)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-03-11 07:09:28 -05:00
|
|
|
context 'with classes that are not ActiveRecord subclasses and have implemented .lazy_find' do
|
2019-07-03 23:33:14 -04:00
|
|
|
it 'returns the correct record' do
|
|
|
|
note = create(:discussion_note_on_merge_request)
|
|
|
|
|
|
|
|
result = described_class.object_from_id(note.to_global_id)
|
|
|
|
|
2019-09-04 13:42:48 -04:00
|
|
|
expect(result.sync).to eq(note)
|
2019-07-03 23:33:14 -04:00
|
|
|
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-09-04 13:42:48 -04:00
|
|
|
described_class.object_from_id(note2.to_global_id)].map(&:sync)
|
2019-07-03 23:33:14 -04:00
|
|
|
end.not_to exceed_query_limit(1)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-03-11 07:09:28 -05:00
|
|
|
context 'with other classes' do
|
2019-06-03 13:38:16 -04:00
|
|
|
# We cannot use an anonymous class here as `GlobalID` expects `.name` not
|
|
|
|
# to return `nil`
|
2020-05-17 20:08:13 -04:00
|
|
|
before do
|
|
|
|
test_global_id = Class.new do
|
|
|
|
include GlobalID::Identification
|
|
|
|
attr_accessor :id
|
|
|
|
|
|
|
|
def initialize(id)
|
|
|
|
@id = id
|
|
|
|
end
|
2019-06-03 13:38:16 -04:00
|
|
|
end
|
2020-05-17 20:08:13 -04:00
|
|
|
|
|
|
|
stub_const('TestGlobalId', test_global_id)
|
2019-06-03 13:38:16 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'falls back to a regular find' do
|
|
|
|
result = TestGlobalId.new(123)
|
|
|
|
|
|
|
|
expect(TestGlobalId).to receive(:find).with("123").and_return(result)
|
|
|
|
|
|
|
|
expect(described_class.object_from_id(result.to_global_id)).to eq(result)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'raises the correct error on invalid input' do
|
|
|
|
expect { described_class.object_from_id("bogus id") }.to raise_error(Gitlab::Graphql::Errors::ArgumentError)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-12-06 22:12:22 -05:00
|
|
|
describe 'validate_max_errors' do
|
|
|
|
it 'reports at most 5 errors' do
|
|
|
|
query = <<~GQL
|
|
|
|
query {
|
|
|
|
currentUser {
|
|
|
|
x: id
|
|
|
|
x: bot
|
|
|
|
x: username
|
|
|
|
x: state
|
|
|
|
x: name
|
|
|
|
|
|
|
|
x: id
|
|
|
|
x: bot
|
|
|
|
x: username
|
|
|
|
x: state
|
|
|
|
x: name
|
|
|
|
|
|
|
|
badField
|
|
|
|
veryBadField
|
|
|
|
alsoNotAGoodField
|
|
|
|
}
|
|
|
|
}
|
|
|
|
GQL
|
|
|
|
|
|
|
|
result = described_class.execute(query)
|
|
|
|
|
|
|
|
expect(result.to_h['errors'].count).to eq 5
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-09-02 11:10:54 -04:00
|
|
|
describe '.parse_gid' do
|
|
|
|
let_it_be(:global_id) { 'gid://gitlab/TestOne/2147483647' }
|
|
|
|
|
2021-04-19 17:09:27 -04:00
|
|
|
subject(:parse_gid) { described_class.parse_gid(global_id) }
|
|
|
|
|
2020-09-02 11:10:54 -04:00
|
|
|
before do
|
|
|
|
test_base = Class.new
|
|
|
|
test_one = Class.new(test_base)
|
|
|
|
test_two = Class.new(test_base)
|
2021-04-19 17:09:27 -04:00
|
|
|
test_three = Class.new(test_base)
|
2020-09-02 11:10:54 -04:00
|
|
|
|
|
|
|
stub_const('TestBase', test_base)
|
|
|
|
stub_const('TestOne', test_one)
|
|
|
|
stub_const('TestTwo', test_two)
|
2021-04-19 17:09:27 -04:00
|
|
|
stub_const('TestThree', test_three)
|
2020-09-02 11:10:54 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'parses the gid' do
|
2021-04-19 17:09:27 -04:00
|
|
|
gid = parse_gid
|
2020-09-02 11:10:54 -04:00
|
|
|
|
|
|
|
expect(gid.model_id).to eq '2147483647'
|
|
|
|
expect(gid.model_class).to eq TestOne
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when gid is malformed' do
|
|
|
|
let_it_be(:global_id) { 'malformed://gitlab/TestOne/2147483647' }
|
|
|
|
|
|
|
|
it 'raises an error' do
|
2021-04-19 17:09:27 -04:00
|
|
|
expect { parse_gid }
|
2020-09-02 11:10:54 -04:00
|
|
|
.to raise_error(Gitlab::Graphql::Errors::ArgumentError, "#{global_id} is not a valid GitLab ID.")
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when using expected_type' do
|
|
|
|
it 'accepts a single type' do
|
|
|
|
gid = described_class.parse_gid(global_id, expected_type: TestOne)
|
|
|
|
|
|
|
|
expect(gid.model_class).to eq TestOne
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'accepts an ancestor type' do
|
|
|
|
gid = described_class.parse_gid(global_id, expected_type: TestBase)
|
|
|
|
|
|
|
|
expect(gid.model_class).to eq TestOne
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'rejects an unknown type' do
|
|
|
|
expect { described_class.parse_gid(global_id, expected_type: TestTwo) }
|
|
|
|
.to raise_error(Gitlab::Graphql::Errors::ArgumentError, "#{global_id} is not a valid ID for TestTwo.")
|
|
|
|
end
|
2021-04-19 17:09:27 -04:00
|
|
|
|
|
|
|
context 'when expected_type is an array' do
|
|
|
|
subject(:parse_gid) { described_class.parse_gid(global_id, expected_type: [TestOne, TestTwo]) }
|
|
|
|
|
|
|
|
context 'when global_id is of type TestOne' do
|
|
|
|
it 'returns an object of an expected type' do
|
|
|
|
expect(parse_gid.model_class).to eq TestOne
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when global_id is of type TestTwo' do
|
|
|
|
let_it_be(:global_id) { 'gid://gitlab/TestTwo/2147483647' }
|
|
|
|
|
|
|
|
it 'returns an object of an expected type' do
|
|
|
|
expect(parse_gid.model_class).to eq TestTwo
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when global_id is of type TestThree' do
|
|
|
|
let_it_be(:global_id) { 'gid://gitlab/TestThree/2147483647' }
|
|
|
|
|
|
|
|
it 'rejects an unknown type' do
|
|
|
|
expect { parse_gid }
|
|
|
|
.to raise_error(Gitlab::Graphql::Errors::ArgumentError, "#{global_id} is not a valid ID for TestOne, TestTwo.")
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2020-09-02 11:10:54 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-08-16 09:04:41 -04:00
|
|
|
def field_instrumenters
|
2019-04-11 05:07:06 -04:00
|
|
|
described_class.instrumenters[:field] + described_class.instrumenters[:field_after_built_ins]
|
2017-08-16 09:04:41 -04:00
|
|
|
end
|
|
|
|
end
|