2013-02-25 14:21:38 -05:00
|
|
|
require 'spec_helper'
|
|
|
|
|
2015-12-09 05:55:49 -05:00
|
|
|
describe GitPushService, services: true do
|
2014-08-01 08:11:27 -04:00
|
|
|
include RepoHelpers
|
|
|
|
|
2015-06-22 08:39:07 -04:00
|
|
|
let(:user) { create :user }
|
|
|
|
let(:project) { create :project }
|
|
|
|
let(:service) { GitPushService.new }
|
2013-02-25 14:21:38 -05:00
|
|
|
|
|
|
|
before do
|
2014-11-03 14:35:06 -05:00
|
|
|
@blankrev = Gitlab::Git::BLANK_SHA
|
2014-08-01 08:11:27 -04:00
|
|
|
@oldrev = sample_commit.parent_id
|
|
|
|
@newrev = sample_commit.id
|
2013-02-25 14:21:38 -05:00
|
|
|
@ref = 'refs/heads/master'
|
|
|
|
end
|
|
|
|
|
2014-09-26 05:55:57 -04:00
|
|
|
describe 'Push branches' do
|
|
|
|
context 'new branch' do
|
|
|
|
subject do
|
|
|
|
service.execute(project, user, @blankrev, @newrev, @ref)
|
|
|
|
end
|
|
|
|
|
2015-02-12 13:17:35 -05:00
|
|
|
it { is_expected.to be_truthy }
|
2014-09-26 05:55:57 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
context 'existing branch' do
|
|
|
|
subject do
|
|
|
|
service.execute(project, user, @oldrev, @newrev, @ref)
|
|
|
|
end
|
|
|
|
|
2015-02-12 13:17:35 -05:00
|
|
|
it { is_expected.to be_truthy }
|
2014-09-26 05:55:57 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
context 'rm branch' do
|
|
|
|
subject do
|
|
|
|
service.execute(project, user, @oldrev, @blankrev, @ref)
|
|
|
|
end
|
|
|
|
|
2015-02-12 13:17:35 -05:00
|
|
|
it { is_expected.to be_truthy }
|
2014-09-26 05:55:57 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-02-25 14:21:38 -05:00
|
|
|
describe "Git Push Data" do
|
|
|
|
before do
|
|
|
|
service.execute(project, user, @oldrev, @newrev, @ref)
|
|
|
|
@push_data = service.push_data
|
2015-04-21 09:13:40 -04:00
|
|
|
@commit = project.commit(@newrev)
|
2013-02-25 14:21:38 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
subject { @push_data }
|
|
|
|
|
2015-02-19 00:02:57 -05:00
|
|
|
it { is_expected.to include(object_kind: 'push') }
|
2015-02-12 13:17:35 -05:00
|
|
|
it { is_expected.to include(before: @oldrev) }
|
|
|
|
it { is_expected.to include(after: @newrev) }
|
|
|
|
it { is_expected.to include(ref: @ref) }
|
|
|
|
it { is_expected.to include(user_id: user.id) }
|
|
|
|
it { is_expected.to include(user_name: user.name) }
|
|
|
|
it { is_expected.to include(project_id: project.id) }
|
2013-02-25 14:21:38 -05:00
|
|
|
|
|
|
|
context "with repository data" do
|
|
|
|
subject { @push_data[:repository] }
|
|
|
|
|
2015-02-12 13:17:35 -05:00
|
|
|
it { is_expected.to include(name: project.name) }
|
|
|
|
it { is_expected.to include(url: project.url_to_repo) }
|
|
|
|
it { is_expected.to include(description: project.description) }
|
|
|
|
it { is_expected.to include(homepage: project.web_url) }
|
2013-02-25 14:21:38 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
context "with commits" do
|
|
|
|
subject { @push_data[:commits] }
|
|
|
|
|
2015-02-12 13:17:35 -05:00
|
|
|
it { is_expected.to be_an(Array) }
|
|
|
|
it 'has 1 element' do
|
|
|
|
expect(subject.size).to eq(1)
|
|
|
|
end
|
2013-02-25 14:21:38 -05:00
|
|
|
|
|
|
|
context "the commit" do
|
|
|
|
subject { @push_data[:commits].first }
|
|
|
|
|
2015-02-12 13:17:35 -05:00
|
|
|
it { is_expected.to include(id: @commit.id) }
|
|
|
|
it { is_expected.to include(message: @commit.safe_message) }
|
|
|
|
it { is_expected.to include(timestamp: @commit.date.xmlschema) }
|
2015-01-24 13:02:58 -05:00
|
|
|
it do
|
|
|
|
is_expected.to include(
|
|
|
|
url: [
|
|
|
|
Gitlab.config.gitlab.url,
|
|
|
|
project.namespace.to_param,
|
|
|
|
project.to_param,
|
|
|
|
'commit',
|
|
|
|
@commit.id
|
|
|
|
].join('/')
|
|
|
|
)
|
|
|
|
end
|
2013-02-25 14:21:38 -05:00
|
|
|
|
|
|
|
context "with a author" do
|
|
|
|
subject { @push_data[:commits].first[:author] }
|
|
|
|
|
2015-02-12 13:17:35 -05:00
|
|
|
it { is_expected.to include(name: @commit.author_name) }
|
|
|
|
it { is_expected.to include(email: @commit.author_email) }
|
2013-02-25 14:21:38 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "Push Event" do
|
|
|
|
before do
|
|
|
|
service.execute(project, user, @oldrev, @newrev, @ref)
|
|
|
|
@event = Event.last
|
|
|
|
end
|
|
|
|
|
2015-02-12 13:17:35 -05:00
|
|
|
it { expect(@event).not_to be_nil }
|
|
|
|
it { expect(@event.project).to eq(project) }
|
|
|
|
it { expect(@event.action).to eq(Event::PUSHED) }
|
|
|
|
it { expect(@event.data).to eq(service.push_data) }
|
2015-10-15 04:41:46 -04:00
|
|
|
|
|
|
|
context "Updates merge requests" do
|
|
|
|
it "when pushing a new branch for the first time" do
|
|
|
|
expect(project).to receive(:update_merge_requests).
|
|
|
|
with(@blankrev, 'newrev', 'refs/heads/master', user)
|
|
|
|
service.execute(project, user, @blankrev, 'newrev', 'refs/heads/master')
|
|
|
|
end
|
|
|
|
end
|
2013-02-25 14:21:38 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
describe "Web Hooks" do
|
2013-08-27 17:04:04 -04:00
|
|
|
context "execute web hooks" do
|
2013-02-25 14:21:38 -05:00
|
|
|
it "when pushing a branch for the first time" do
|
2015-02-12 13:17:35 -05:00
|
|
|
expect(project).to receive(:execute_hooks)
|
|
|
|
expect(project.default_branch).to eq("master")
|
|
|
|
expect(project.protected_branches).to receive(:create).with({ name: "master", developers_can_push: false })
|
2013-05-30 19:16:49 -04:00
|
|
|
service.execute(project, user, @blankrev, 'newrev', 'refs/heads/master')
|
2015-01-31 03:10:17 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it "when pushing a branch for the first time with default branch protection disabled" do
|
2015-07-01 17:21:51 -04:00
|
|
|
stub_application_setting(default_branch_protection: Gitlab::Access::PROTECTION_NONE)
|
2015-01-31 03:10:17 -05:00
|
|
|
|
2015-02-12 13:17:35 -05:00
|
|
|
expect(project).to receive(:execute_hooks)
|
|
|
|
expect(project.default_branch).to eq("master")
|
|
|
|
expect(project.protected_branches).not_to receive(:create)
|
2015-01-31 03:10:17 -05:00
|
|
|
service.execute(project, user, @blankrev, 'newrev', 'refs/heads/master')
|
|
|
|
end
|
|
|
|
|
|
|
|
it "when pushing a branch for the first time with default branch protection set to 'developers can push'" do
|
2015-07-01 17:21:51 -04:00
|
|
|
stub_application_setting(default_branch_protection: Gitlab::Access::PROTECTION_DEV_CAN_PUSH)
|
2015-01-31 03:10:17 -05:00
|
|
|
|
2015-02-12 13:17:35 -05:00
|
|
|
expect(project).to receive(:execute_hooks)
|
|
|
|
expect(project.default_branch).to eq("master")
|
|
|
|
expect(project.protected_branches).to receive(:create).with({ name: "master", developers_can_push: true })
|
2015-01-31 03:10:17 -05:00
|
|
|
service.execute(project, user, @blankrev, 'newrev', 'refs/heads/master')
|
2013-02-25 14:21:38 -05:00
|
|
|
end
|
|
|
|
|
2013-12-03 04:34:06 -05:00
|
|
|
it "when pushing new commits to existing branch" do
|
2015-02-12 13:17:35 -05:00
|
|
|
expect(project).to receive(:execute_hooks)
|
2013-12-03 04:34:06 -05:00
|
|
|
service.execute(project, user, 'oldrev', 'newrev', 'refs/heads/master')
|
|
|
|
end
|
2013-02-25 14:21:38 -05:00
|
|
|
end
|
|
|
|
end
|
2013-05-30 19:16:49 -04:00
|
|
|
|
|
|
|
describe "cross-reference notes" do
|
|
|
|
let(:issue) { create :issue, project: project }
|
|
|
|
let(:commit_author) { create :user }
|
2015-04-21 09:13:40 -04:00
|
|
|
let(:commit) { project.commit }
|
2013-05-30 19:16:49 -04:00
|
|
|
|
|
|
|
before do
|
2015-05-21 17:49:06 -04:00
|
|
|
allow(commit).to receive_messages(
|
2015-10-13 05:37:42 -04:00
|
|
|
safe_message: "this commit \n mentions #{issue.to_reference}",
|
2013-05-30 19:16:49 -04:00
|
|
|
references: [issue],
|
|
|
|
author_name: commit_author.name,
|
|
|
|
author_email: commit_author.email
|
2015-05-21 17:49:06 -04:00
|
|
|
)
|
|
|
|
allow(project.repository).to receive(:commits_between).and_return([commit])
|
2013-05-30 19:16:49 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it "creates a note if a pushed commit mentions an issue" do
|
2015-06-17 20:59:19 -04:00
|
|
|
expect(SystemNoteService).to receive(:cross_reference).with(issue, commit, commit_author)
|
2013-05-30 19:16:49 -04:00
|
|
|
|
|
|
|
service.execute(project, user, @oldrev, @newrev, @ref)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "only creates a cross-reference note if one doesn't already exist" do
|
2015-06-17 20:59:19 -04:00
|
|
|
SystemNoteService.cross_reference(issue, commit, user)
|
2013-05-30 19:16:49 -04:00
|
|
|
|
2015-06-17 20:59:19 -04:00
|
|
|
expect(SystemNoteService).not_to receive(:cross_reference).with(issue, commit, commit_author)
|
2013-05-30 19:16:49 -04:00
|
|
|
|
|
|
|
service.execute(project, user, @oldrev, @newrev, @ref)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "defaults to the pushing user if the commit's author is not known" do
|
2015-05-21 17:49:06 -04:00
|
|
|
allow(commit).to receive_messages(
|
|
|
|
author_name: 'unknown name',
|
|
|
|
author_email: 'unknown@email.com'
|
|
|
|
)
|
2015-06-17 20:59:19 -04:00
|
|
|
expect(SystemNoteService).to receive(:cross_reference).with(issue, commit, user)
|
2013-05-30 19:16:49 -04:00
|
|
|
|
|
|
|
service.execute(project, user, @oldrev, @newrev, @ref)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "finds references in the first push to a non-default branch" do
|
2015-02-12 13:17:35 -05:00
|
|
|
allow(project.repository).to receive(:commits_between).with(@blankrev, @newrev).and_return([])
|
|
|
|
allow(project.repository).to receive(:commits_between).with("master", @newrev).and_return([commit])
|
2013-05-30 19:16:49 -04:00
|
|
|
|
2015-06-17 20:59:19 -04:00
|
|
|
expect(SystemNoteService).to receive(:cross_reference).with(issue, commit, commit_author)
|
2013-05-30 19:16:49 -04:00
|
|
|
|
|
|
|
service.execute(project, user, @blankrev, @newrev, 'refs/heads/other')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2015-08-21 09:05:43 -04:00
|
|
|
describe "closing issues from pushed commits containing a closing reference" do
|
2013-05-30 19:16:49 -04:00
|
|
|
let(:issue) { create :issue, project: project }
|
|
|
|
let(:other_issue) { create :issue, project: project }
|
|
|
|
let(:commit_author) { create :user }
|
2015-04-21 09:13:40 -04:00
|
|
|
let(:closing_commit) { project.commit }
|
2013-05-30 19:16:49 -04:00
|
|
|
|
|
|
|
before do
|
2015-05-21 17:49:06 -04:00
|
|
|
allow(closing_commit).to receive_messages(
|
2013-05-30 19:16:49 -04:00
|
|
|
issue_closing_regex: /^([Cc]loses|[Ff]ixes) #\d+/,
|
|
|
|
safe_message: "this is some work.\n\ncloses ##{issue.iid}",
|
|
|
|
author_name: commit_author.name,
|
|
|
|
author_email: commit_author.email
|
2015-05-21 17:49:06 -04:00
|
|
|
)
|
2013-05-30 19:16:49 -04:00
|
|
|
|
2015-05-21 17:49:06 -04:00
|
|
|
allow(project.repository).to receive(:commits_between).
|
|
|
|
and_return([closing_commit])
|
2013-05-30 19:16:49 -04:00
|
|
|
end
|
|
|
|
|
2015-08-21 09:05:43 -04:00
|
|
|
context "to default branches" do
|
|
|
|
it "closes issues" do
|
|
|
|
service.execute(project, user, @oldrev, @newrev, @ref)
|
|
|
|
expect(Issue.find(issue.id)).to be_closed
|
|
|
|
end
|
2013-05-30 19:16:49 -04:00
|
|
|
|
2015-08-21 09:05:43 -04:00
|
|
|
it "adds a note indicating that the issue is now closed" do
|
|
|
|
expect(SystemNoteService).to receive(:change_status).with(issue, project, commit_author, "closed", closing_commit)
|
2013-05-30 19:16:49 -04:00
|
|
|
service.execute(project, user, @oldrev, @newrev, @ref)
|
2015-08-21 09:05:43 -04:00
|
|
|
end
|
2013-05-30 19:16:49 -04:00
|
|
|
|
2015-08-21 09:05:43 -04:00
|
|
|
it "doesn't create additional cross-reference notes" do
|
|
|
|
expect(SystemNoteService).not_to receive(:cross_reference)
|
|
|
|
service.execute(project, user, @oldrev, @newrev, @ref)
|
|
|
|
end
|
2013-05-30 19:16:49 -04:00
|
|
|
|
2015-08-21 09:05:43 -04:00
|
|
|
it "doesn't close issues when external issue tracker is in use" do
|
|
|
|
allow(project).to receive(:default_issues_tracker?).and_return(false)
|
2013-05-30 19:16:49 -04:00
|
|
|
|
2015-08-21 09:05:43 -04:00
|
|
|
# The push still shouldn't create cross-reference notes.
|
|
|
|
expect do
|
|
|
|
service.execute(project, user, @oldrev, @newrev, 'refs/heads/hurf')
|
|
|
|
end.not_to change { Note.where(project_id: project.id, system: true).count }
|
|
|
|
end
|
2013-05-30 19:16:49 -04:00
|
|
|
end
|
2015-06-12 01:39:50 -04:00
|
|
|
|
2015-08-21 09:05:43 -04:00
|
|
|
context "to non-default branches" do
|
|
|
|
before do
|
|
|
|
# Make sure the "default" branch is different
|
|
|
|
allow(project).to receive(:default_branch).and_return('not-master')
|
|
|
|
end
|
|
|
|
|
|
|
|
it "creates cross-reference notes" do
|
|
|
|
expect(SystemNoteService).to receive(:cross_reference).with(issue, closing_commit, commit_author)
|
|
|
|
service.execute(project, user, @oldrev, @newrev, @ref)
|
|
|
|
end
|
2015-06-12 01:39:50 -04:00
|
|
|
|
2015-08-21 09:05:43 -04:00
|
|
|
it "doesn't close issues" do
|
|
|
|
service.execute(project, user, @oldrev, @newrev, @ref)
|
|
|
|
expect(Issue.find(issue.id)).to be_opened
|
|
|
|
end
|
2015-06-12 01:39:50 -04:00
|
|
|
end
|
2015-12-17 17:08:14 -05:00
|
|
|
|
|
|
|
# EE-only tests
|
|
|
|
context "for jira issue tracker" do
|
|
|
|
include JiraServiceHelper
|
|
|
|
|
|
|
|
let(:jira_tracker) { project.create_jira_service if project.jira_service.nil? }
|
|
|
|
|
|
|
|
before do
|
|
|
|
jira_service_settings
|
|
|
|
|
|
|
|
WebMock.stub_request(:post, jira_api_transition_url)
|
|
|
|
WebMock.stub_request(:post, jira_api_comment_url)
|
|
|
|
WebMock.stub_request(:get, jira_api_comment_url).to_return(body: jira_issue_comments)
|
|
|
|
WebMock.stub_request(:get, jira_api_test_url)
|
|
|
|
|
|
|
|
allow(closing_commit).to receive_messages({
|
|
|
|
issue_closing_regex: Regexp.new(Gitlab.config.gitlab.issue_closing_pattern),
|
|
|
|
safe_message: message,
|
|
|
|
author_name: commit_author.name,
|
|
|
|
author_email: commit_author.email
|
|
|
|
})
|
|
|
|
|
|
|
|
allow(project.repository).to receive_messages(commits_between: [closing_commit])
|
|
|
|
end
|
|
|
|
|
|
|
|
after do
|
|
|
|
jira_tracker.destroy!
|
|
|
|
end
|
|
|
|
|
|
|
|
context "mentioning an issue" do
|
|
|
|
let(:message) { "this is some work.\n\nrelated to JIRA-1" }
|
|
|
|
|
|
|
|
it "should initiate one api call to jira server to mention the issue" do
|
|
|
|
service.execute(project, user, @oldrev, @newrev, @ref)
|
|
|
|
|
|
|
|
expect(WebMock).to have_requested(:post, jira_api_comment_url).with(
|
|
|
|
body: /mentioned this issue in/
|
|
|
|
).once
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context "closing an issue" do
|
|
|
|
let(:message) { "this is some work.\n\ncloses JIRA-1" }
|
|
|
|
|
|
|
|
it "should initiate one api call to jira server to close the issue" do
|
|
|
|
transition_body = {
|
|
|
|
transition: {
|
|
|
|
id: '2'
|
|
|
|
}
|
|
|
|
}.to_json
|
|
|
|
|
|
|
|
service.execute(project, user, @oldrev, @newrev, @ref)
|
|
|
|
expect(WebMock).to have_requested(:post, jira_api_transition_url).with(
|
|
|
|
body: transition_body
|
|
|
|
).once
|
|
|
|
end
|
|
|
|
|
|
|
|
it "should initiate one api call to jira server to comment on the issue" do
|
|
|
|
comment_body = {
|
|
|
|
body: "Issue solved with [#{closing_commit.id}|http://localhost/#{project.path_with_namespace}/commit/#{closing_commit.id}]."
|
|
|
|
}.to_json
|
|
|
|
|
|
|
|
service.execute(project, user, @oldrev, @newrev, @ref)
|
|
|
|
expect(WebMock).to have_requested(:post, jira_api_comment_url).with(
|
|
|
|
body: comment_body
|
|
|
|
).once
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2013-05-30 19:16:49 -04:00
|
|
|
end
|
2013-02-25 14:21:38 -05:00
|
|
|
|
2015-05-06 02:49:30 -04:00
|
|
|
describe "empty project" do
|
|
|
|
let(:project) { create(:project_empty_repo) }
|
|
|
|
let(:new_ref) { 'refs/heads/feature'}
|
|
|
|
|
|
|
|
before do
|
|
|
|
allow(project).to receive(:default_branch).and_return('feature')
|
|
|
|
expect(project).to receive(:change_head) { 'feature'}
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'push to first branch updates HEAD' do
|
|
|
|
service.execute(project, user, @blankrev, @newrev, new_ref)
|
|
|
|
end
|
|
|
|
end
|
2015-10-14 01:07:58 -04:00
|
|
|
end
|