2016-06-22 15:30:34 +05:30
|
|
|
require 'spec_helper'
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
describe Gitlab::ImportExport::ProjectTreeSaver do
|
2016-06-22 15:30:34 +05:30
|
|
|
describe 'saves the project tree into a json object' do
|
2018-03-27 19:54:05 +05:30
|
|
|
let(:shared) { project.import_export_shared }
|
2017-08-17 22:00:37 +05:30
|
|
|
let(:project_tree_saver) { described_class.new(project: project, current_user: user, shared: shared) }
|
|
|
|
let(:export_path) { "#{Dir.tmpdir}/project_tree_saver_spec" }
|
2016-06-22 15:30:34 +05:30
|
|
|
let(:user) { create(:user) }
|
2017-08-17 22:00:37 +05:30
|
|
|
let!(:project) { setup_project }
|
2016-06-22 15:30:34 +05:30
|
|
|
|
|
|
|
before do
|
2018-11-18 11:00:15 +05:30
|
|
|
project.add_maintainer(user)
|
2016-06-22 15:30:34 +05:30
|
|
|
allow_any_instance_of(Gitlab::ImportExport).to receive(:storage_path).and_return(export_path)
|
2018-03-17 18:26:18 +05:30
|
|
|
allow_any_instance_of(MergeRequest).to receive(:source_branch_sha).and_return('ABCD')
|
|
|
|
allow_any_instance_of(MergeRequest).to receive(:target_branch_sha).and_return('DCBA')
|
2016-06-22 15:30:34 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
after do
|
|
|
|
FileUtils.rm_rf(export_path)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'saves project successfully' do
|
|
|
|
expect(project_tree_saver.save).to be true
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'JSON' do
|
|
|
|
let(:saved_project_json) do
|
|
|
|
project_tree_saver.save
|
|
|
|
project_json(project_tree_saver.full_path)
|
|
|
|
end
|
|
|
|
|
2018-05-09 12:01:36 +05:30
|
|
|
context 'with description override' do
|
|
|
|
let(:params) { { description: 'Foo Bar' } }
|
|
|
|
let(:project_tree_saver) { described_class.new(project: project, current_user: user, shared: shared, params: params) }
|
|
|
|
|
|
|
|
it 'overrides the project description' do
|
|
|
|
expect(saved_project_json).to include({ 'description' => params[:description] })
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-06-22 15:30:34 +05:30
|
|
|
it 'saves the correct json' do
|
2018-05-09 12:01:36 +05:30
|
|
|
expect(saved_project_json).to include({ 'description' => 'description', 'visibility_level' => 20 })
|
2016-06-22 15:30:34 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'has milestones' do
|
|
|
|
expect(saved_project_json['milestones']).not_to be_empty
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'has merge requests' do
|
|
|
|
expect(saved_project_json['merge_requests']).not_to be_empty
|
|
|
|
end
|
|
|
|
|
2016-08-24 12:49:21 +05:30
|
|
|
it 'has merge request\'s milestones' do
|
|
|
|
expect(saved_project_json['merge_requests'].first['milestone']).not_to be_empty
|
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
it 'has merge request\'s source branch SHA' do
|
|
|
|
expect(saved_project_json['merge_requests'].first['source_branch_sha']).to eq('ABCD')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'has merge request\'s target branch SHA' do
|
|
|
|
expect(saved_project_json['merge_requests'].first['target_branch_sha']).to eq('DCBA')
|
|
|
|
end
|
|
|
|
|
2016-08-24 12:49:21 +05:30
|
|
|
it 'has events' do
|
|
|
|
expect(saved_project_json['merge_requests'].first['milestone']['events']).not_to be_empty
|
2016-06-22 15:30:34 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'has snippets' do
|
|
|
|
expect(saved_project_json['snippets']).not_to be_empty
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'has snippet notes' do
|
|
|
|
expect(saved_project_json['snippets'].first['notes']).not_to be_empty
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'has releases' do
|
|
|
|
expect(saved_project_json['releases']).not_to be_empty
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'has issues' do
|
|
|
|
expect(saved_project_json['issues']).not_to be_empty
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'has issue comments' do
|
|
|
|
expect(saved_project_json['issues'].first['notes']).not_to be_empty
|
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
it 'has issue assignees' do
|
|
|
|
expect(saved_project_json['issues'].first['issue_assignees']).not_to be_empty
|
|
|
|
end
|
|
|
|
|
2016-06-22 15:30:34 +05:30
|
|
|
it 'has author on issue comments' do
|
|
|
|
expect(saved_project_json['issues'].first['notes'].first['author']).not_to be_empty
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'has project members' do
|
|
|
|
expect(saved_project_json['project_members']).not_to be_empty
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'has merge requests diffs' do
|
|
|
|
expect(saved_project_json['merge_requests'].first['merge_request_diff']).not_to be_empty
|
|
|
|
end
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
it 'has merge request diff files' do
|
|
|
|
expect(saved_project_json['merge_requests'].first['merge_request_diff']['merge_request_diff_files']).not_to be_empty
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'has merge request diff commits' do
|
|
|
|
expect(saved_project_json['merge_requests'].first['merge_request_diff']['merge_request_diff_commits']).not_to be_empty
|
|
|
|
end
|
|
|
|
|
2016-06-22 15:30:34 +05:30
|
|
|
it 'has merge requests comments' do
|
|
|
|
expect(saved_project_json['merge_requests'].first['notes']).not_to be_empty
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'has author on merge requests comments' do
|
|
|
|
expect(saved_project_json['merge_requests'].first['notes'].first['author']).not_to be_empty
|
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
it 'has pipeline stages' do
|
|
|
|
expect(saved_project_json.dig('pipelines', 0, 'stages')).not_to be_empty
|
|
|
|
end
|
|
|
|
|
2016-06-22 15:30:34 +05:30
|
|
|
it 'has pipeline statuses' do
|
2018-03-17 18:26:18 +05:30
|
|
|
expect(saved_project_json.dig('pipelines', 0, 'stages', 0, 'statuses')).not_to be_empty
|
2016-06-22 15:30:34 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'has pipeline builds' do
|
2018-03-17 18:26:18 +05:30
|
|
|
builds_count = saved_project_json
|
|
|
|
.dig('pipelines', 0, 'stages', 0, 'statuses')
|
|
|
|
.count { |hash| hash['type'] == 'Ci::Build' }
|
|
|
|
|
|
|
|
expect(builds_count).to eq(1)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'has no when YML attributes but only the DB column' do
|
|
|
|
allow_any_instance_of(Ci::Pipeline).to receive(:ci_yaml_file).and_return(File.read(Rails.root.join('spec/support/gitlab_stubs/gitlab_ci.yml')))
|
|
|
|
expect_any_instance_of(Gitlab::Ci::YamlProcessor).not_to receive(:build_attributes)
|
|
|
|
|
|
|
|
saved_project_json
|
2016-06-22 15:30:34 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'has pipeline commits' do
|
|
|
|
expect(saved_project_json['pipelines']).not_to be_empty
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'has ci pipeline notes' do
|
|
|
|
expect(saved_project_json['pipelines'].first['notes']).not_to be_empty
|
|
|
|
end
|
2016-08-24 12:49:21 +05:30
|
|
|
|
|
|
|
it 'has labels with no associations' do
|
|
|
|
expect(saved_project_json['labels']).not_to be_empty
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'has labels associated to records' do
|
|
|
|
expect(saved_project_json['issues'].first['label_links'].first['label']).not_to be_empty
|
|
|
|
end
|
|
|
|
|
2016-11-03 12:29:30 +05:30
|
|
|
it 'has project and group labels' do
|
2017-08-17 22:00:37 +05:30
|
|
|
label_types = saved_project_json['issues'].first['label_links'].map { |link| link['label']['type'] }
|
2016-11-03 12:29:30 +05:30
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
expect(label_types).to match_array(%w(ProjectLabel GroupLabel))
|
2016-11-03 12:29:30 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'has priorities associated to labels' do
|
2017-08-17 22:00:37 +05:30
|
|
|
priorities = saved_project_json['issues'].first['label_links'].map { |link| link['label']['priorities'] }
|
2016-11-03 12:29:30 +05:30
|
|
|
|
|
|
|
expect(priorities.flatten).not_to be_empty
|
|
|
|
end
|
|
|
|
|
2018-11-20 20:47:30 +05:30
|
|
|
it 'has issue resource label events' do
|
|
|
|
expect(saved_project_json['issues'].first['resource_label_events']).not_to be_empty
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'has merge request resource label events' do
|
|
|
|
expect(saved_project_json['merge_requests'].first['resource_label_events']).not_to be_empty
|
|
|
|
end
|
|
|
|
|
2016-10-01 15:18:49 +05:30
|
|
|
it 'saves the correct service type' do
|
|
|
|
expect(saved_project_json['services'].first['type']).to eq('CustomIssueTrackerService')
|
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
it 'saves the properties for a service' do
|
|
|
|
expect(saved_project_json['services'].first['properties']).to eq('one' => 'value')
|
|
|
|
end
|
|
|
|
|
2016-09-29 09:46:39 +05:30
|
|
|
it 'has project feature' do
|
|
|
|
project_feature = saved_project_json['project_feature']
|
|
|
|
expect(project_feature).not_to be_empty
|
|
|
|
expect(project_feature["issues_access_level"]).to eq(ProjectFeature::DISABLED)
|
|
|
|
expect(project_feature["wiki_access_level"]).to eq(ProjectFeature::ENABLED)
|
|
|
|
expect(project_feature["builds_access_level"]).to eq(ProjectFeature::PRIVATE)
|
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
it 'has custom attributes' do
|
|
|
|
expect(saved_project_json['custom_attributes'].count).to eq(2)
|
2016-08-24 12:49:21 +05:30
|
|
|
end
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2018-03-27 19:54:05 +05:30
|
|
|
it 'has badges' do
|
|
|
|
expect(saved_project_json['project_badges'].count).to eq(2)
|
|
|
|
end
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
it 'does not complain about non UTF-8 characters in MR diff files' do
|
|
|
|
ActiveRecord::Base.connection.execute("UPDATE merge_request_diff_files SET diff = '---\n- :diff: !binary |-\n LS0tIC9kZXYvbnVsbAorKysgYi9pbWFnZXMvbnVjb3IucGRmCkBAIC0wLDAg\n KzEsMTY3OSBAQAorJVBERi0xLjUNJeLjz9MNCisxIDAgb2JqDTw8L01ldGFk\n YXR'")
|
|
|
|
|
|
|
|
expect(project_tree_saver.save).to be true
|
|
|
|
end
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
context 'group members' do
|
|
|
|
let(:user2) { create(:user, email: 'group@member.com') }
|
|
|
|
let(:member_emails) do
|
|
|
|
saved_project_json['project_members'].map do |pm|
|
|
|
|
pm['user']['email']
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
before do
|
|
|
|
Group.first.add_developer(user2)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not export group members if it has no permission' do
|
|
|
|
Group.first.add_developer(user)
|
|
|
|
|
|
|
|
expect(member_emails).not_to include('group@member.com')
|
|
|
|
end
|
|
|
|
|
2018-11-18 11:00:15 +05:30
|
|
|
it 'does not export group members as maintainer' do
|
|
|
|
Group.first.add_maintainer(user)
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(member_emails).not_to include('group@member.com')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'exports group members as group owner' do
|
|
|
|
Group.first.add_owner(user)
|
|
|
|
|
|
|
|
expect(member_emails).to include('group@member.com')
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'as admin' do
|
|
|
|
let(:user) { create(:admin) }
|
|
|
|
|
|
|
|
it 'exports group members as admin' do
|
|
|
|
expect(member_emails).to include('group@member.com')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'exports group members as project members' do
|
|
|
|
member_types = saved_project_json['project_members'].map { |pm| pm['source_type'] }
|
|
|
|
|
|
|
|
expect(member_types).to all(eq('Project'))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'project attributes' do
|
|
|
|
it 'does not contain the runners token' do
|
|
|
|
expect(saved_project_json).not_to include("runners_token" => 'token')
|
|
|
|
end
|
|
|
|
end
|
2016-06-22 15:30:34 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def setup_project
|
2017-08-17 22:00:37 +05:30
|
|
|
issue = create(:issue, assignees: [user])
|
2016-06-22 15:30:34 +05:30
|
|
|
snippet = create(:project_snippet)
|
|
|
|
release = create(:release)
|
2016-11-03 12:29:30 +05:30
|
|
|
group = create(:group)
|
2016-06-22 15:30:34 +05:30
|
|
|
|
|
|
|
project = create(:project,
|
|
|
|
:public,
|
2017-08-17 22:00:37 +05:30
|
|
|
:repository,
|
|
|
|
:issues_disabled,
|
|
|
|
:wiki_enabled,
|
|
|
|
:builds_private,
|
2018-05-09 12:01:36 +05:30
|
|
|
description: 'description',
|
2016-06-22 15:30:34 +05:30
|
|
|
issues: [issue],
|
|
|
|
snippets: [snippet],
|
2016-11-03 12:29:30 +05:30
|
|
|
releases: [release],
|
|
|
|
group: group
|
2016-06-22 15:30:34 +05:30
|
|
|
)
|
2016-11-03 12:29:30 +05:30
|
|
|
project_label = create(:label, project: project)
|
|
|
|
group_label = create(:group_label, group: group)
|
|
|
|
create(:label_link, label: project_label, target: issue)
|
|
|
|
create(:label_link, label: group_label, target: issue)
|
|
|
|
create(:label_priority, label: group_label, priority: 1)
|
2016-08-24 12:49:21 +05:30
|
|
|
milestone = create(:milestone, project: project)
|
|
|
|
merge_request = create(:merge_request, source_project: project, milestone: milestone)
|
2016-06-22 15:30:34 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
ci_build = create(:ci_build, project: project, when: nil)
|
|
|
|
ci_build.pipeline.update(project: project)
|
|
|
|
create(:commit_status, project: project, pipeline: ci_build.pipeline)
|
2016-06-22 15:30:34 +05:30
|
|
|
|
|
|
|
create(:milestone, project: project)
|
|
|
|
create(:note, noteable: issue, project: project)
|
|
|
|
create(:note, noteable: merge_request, project: project)
|
|
|
|
create(:note, noteable: snippet, project: project)
|
|
|
|
create(:note_on_commit,
|
|
|
|
author: user,
|
|
|
|
project: project,
|
2018-03-17 18:26:18 +05:30
|
|
|
commit_id: ci_build.pipeline.sha)
|
2016-08-24 12:49:21 +05:30
|
|
|
|
2018-11-20 20:47:30 +05:30
|
|
|
create(:resource_label_event, label: project_label, issue: issue)
|
|
|
|
create(:resource_label_event, label: group_label, merge_request: merge_request)
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
create(:event, :created, target: milestone, project: project, author: user)
|
2018-03-17 18:26:18 +05:30
|
|
|
create(:service, project: project, type: 'CustomIssueTrackerService', category: 'issue_tracker', properties: { one: 'value' })
|
|
|
|
|
|
|
|
create(:project_custom_attribute, project: project)
|
|
|
|
create(:project_custom_attribute, project: project)
|
2016-08-24 12:49:21 +05:30
|
|
|
|
2018-03-27 19:54:05 +05:30
|
|
|
create(:project_badge, project: project)
|
|
|
|
create(:project_badge, project: project)
|
|
|
|
|
2016-06-22 15:30:34 +05:30
|
|
|
project
|
|
|
|
end
|
|
|
|
|
|
|
|
def project_json(filename)
|
|
|
|
JSON.parse(IO.read(filename))
|
|
|
|
end
|
|
|
|
end
|