2019-12-21 20:55:43 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2014-09-02 18:07:02 +05:30
|
|
|
require "spec_helper"
|
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
RSpec.describe IssuesHelper do
|
2017-09-10 17:25:29 +05:30
|
|
|
let(:project) { create(:project) }
|
2014-09-02 18:07:02 +05:30
|
|
|
let(:issue) { create :issue, project: project }
|
|
|
|
let(:ext_project) { create :redmine_project }
|
|
|
|
|
2016-09-29 09:46:39 +05:30
|
|
|
describe '#award_user_list' do
|
2016-11-03 12:29:30 +05:30
|
|
|
it "returns a comma-separated list of the first X users" do
|
|
|
|
user = build_stubbed(:user, name: 'Joe')
|
|
|
|
awards = Array.new(3, build_stubbed(:award_emoji, user: user))
|
2016-09-29 09:46:39 +05:30
|
|
|
|
2016-11-03 12:29:30 +05:30
|
|
|
expect(award_user_list(awards, nil, limit: 3))
|
|
|
|
.to eq('Joe, Joe, and Joe')
|
2016-09-29 09:46:39 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it "displays the current user's name as 'You'" do
|
2016-11-03 12:29:30 +05:30
|
|
|
user = build_stubbed(:user, name: 'Joe')
|
|
|
|
award = build_stubbed(:award_emoji, user: user)
|
2016-09-29 09:46:39 +05:30
|
|
|
|
2016-11-03 12:29:30 +05:30
|
|
|
expect(award_user_list([award], user)).to eq('You')
|
|
|
|
expect(award_user_list([award], nil)).to eq 'Joe'
|
2016-09-29 09:46:39 +05:30
|
|
|
end
|
|
|
|
|
2016-11-03 12:29:30 +05:30
|
|
|
it "truncates lists" do
|
|
|
|
user = build_stubbed(:user, name: 'Jane')
|
|
|
|
awards = Array.new(5, build_stubbed(:award_emoji, user: user))
|
|
|
|
|
|
|
|
expect(award_user_list(awards, nil, limit: 3))
|
|
|
|
.to eq('Jane, Jane, Jane, and 2 more.')
|
2016-09-29 09:46:39 +05:30
|
|
|
end
|
|
|
|
|
2016-11-03 12:29:30 +05:30
|
|
|
it "displays the current user in front of other users" do
|
|
|
|
current_user = build_stubbed(:user)
|
|
|
|
my_award = build_stubbed(:award_emoji, user: current_user)
|
|
|
|
award = build_stubbed(:award_emoji, user: build_stubbed(:user, name: 'Jane'))
|
|
|
|
awards = Array.new(5, award).push(my_award)
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
expect(award_user_list(awards, current_user, limit: 2))
|
|
|
|
.to eq("You, Jane, and 4 more.")
|
2016-09-29 09:46:39 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
describe '#award_state_class' do
|
2016-06-16 23:09:34 +05:30
|
|
|
let!(:upvote) { create(:award_emoji) }
|
2018-05-09 12:01:36 +05:30
|
|
|
let(:awardable) { upvote.awardable }
|
|
|
|
let(:user) { upvote.user }
|
|
|
|
|
|
|
|
before do
|
|
|
|
allow(helper).to receive(:can?) do |*args|
|
|
|
|
Ability.allowed?(*args)
|
|
|
|
end
|
|
|
|
end
|
2015-11-26 14:37:03 +05:30
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
it "returns disabled string for unauthenticated user" do
|
2018-05-09 12:01:36 +05:30
|
|
|
expect(helper.award_state_class(awardable, AwardEmoji.all, nil)).to eq("disabled")
|
|
|
|
end
|
|
|
|
|
|
|
|
it "returns disabled for a user that does not have access to the awardable" do
|
|
|
|
expect(helper.award_state_class(awardable, AwardEmoji.all, build(:user))).to eq("disabled")
|
2015-11-26 14:37:03 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it "returns active string for author" do
|
2018-05-09 12:01:36 +05:30
|
|
|
expect(helper.award_state_class(awardable, AwardEmoji.all, upvote.user)).to eq("active")
|
|
|
|
end
|
|
|
|
|
|
|
|
it "is blank for a user that has access to the awardable" do
|
|
|
|
user = build(:user)
|
|
|
|
expect(helper).to receive(:can?).with(user, :award_emoji, awardable).and_return(true)
|
|
|
|
|
|
|
|
expect(helper.award_state_class(awardable, AwardEmoji.all, user)).to be_blank
|
2015-11-26 14:37:03 +05:30
|
|
|
end
|
|
|
|
end
|
2016-01-14 18:37:52 +05:30
|
|
|
|
2016-06-02 11:05:42 +05:30
|
|
|
describe "awards_sort" do
|
2016-01-14 18:37:52 +05:30
|
|
|
it "sorts a hash so thumbsup and thumbsdown are always on top" do
|
|
|
|
data = { "thumbsdown" => "some value", "lifter" => "some value", "thumbsup" => "some value" }
|
2017-08-17 22:00:37 +05:30
|
|
|
expect(awards_sort(data).keys).to eq(%w(thumbsup thumbsdown lifter))
|
2016-01-14 18:37:52 +05:30
|
|
|
end
|
|
|
|
end
|
2016-06-02 11:05:42 +05:30
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
describe "#link_to_discussions_to_resolve" do
|
|
|
|
describe "passing only a merge request" do
|
|
|
|
let(:merge_request) { create(:merge_request) }
|
|
|
|
|
|
|
|
it "links just the merge request" do
|
2017-09-10 17:25:29 +05:30
|
|
|
expected_path = project_merge_request_path(merge_request.project, merge_request)
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(link_to_discussions_to_resolve(merge_request, nil)).to include(expected_path)
|
|
|
|
end
|
|
|
|
|
2019-12-04 20:38:33 +05:30
|
|
|
it "contains the reference to the merge request" do
|
2017-08-17 22:00:37 +05:30
|
|
|
expect(link_to_discussions_to_resolve(merge_request, nil)).to include(merge_request.to_reference)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "when passing a discussion" do
|
2019-03-02 22:35:43 +05:30
|
|
|
let(:diff_note) { create(:diff_note_on_merge_request) }
|
2017-08-17 22:00:37 +05:30
|
|
|
let(:merge_request) { diff_note.noteable }
|
|
|
|
let(:discussion) { diff_note.to_discussion }
|
|
|
|
|
|
|
|
it "links to the merge request with first note if a single discussion was passed" do
|
|
|
|
expected_path = Gitlab::UrlBuilder.build(diff_note)
|
|
|
|
|
|
|
|
expect(link_to_discussions_to_resolve(merge_request, discussion)).to include(expected_path)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "contains both the reference to the merge request and a mention of the discussion" do
|
|
|
|
expect(link_to_discussions_to_resolve(merge_request, discussion)).to include("#{merge_request.to_reference} (discussion #{diff_note.id})")
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2018-05-09 12:01:36 +05:30
|
|
|
|
|
|
|
describe '#show_new_issue_link?' do
|
|
|
|
before do
|
|
|
|
allow(helper).to receive(:current_user)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'is false when no project there is no project' do
|
|
|
|
expect(helper.show_new_issue_link?(nil)).to be_falsey
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'is true when there is a project and no logged in user' do
|
|
|
|
expect(helper.show_new_issue_link?(build(:project))).to be_truthy
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'is true when the current user does not have access to the project' do
|
|
|
|
project = build(:project)
|
|
|
|
allow(helper).to receive(:current_user).and_return(project.owner)
|
|
|
|
|
|
|
|
expect(helper).to receive(:can?).with(project.owner, :create_issue, project).and_return(true)
|
|
|
|
expect(helper.show_new_issue_link?(project)).to be_truthy
|
|
|
|
end
|
|
|
|
end
|
2019-12-04 20:38:33 +05:30
|
|
|
|
|
|
|
describe '#issue_closed_link' do
|
|
|
|
let(:new_issue) { create(:issue, project: project) }
|
|
|
|
let(:guest) { create(:user) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
allow(helper).to receive(:can?) do |*args|
|
|
|
|
Ability.allowed?(*args)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
shared_examples 'successfully displays link to issue and with css class' do |action|
|
|
|
|
it 'returns link' do
|
2020-04-08 14:13:33 +05:30
|
|
|
link = "<a class=\"#{css_class}\" href=\"/#{new_issue.project.full_path}/-/issues/#{new_issue.iid}\">(#{action})</a>"
|
2019-12-04 20:38:33 +05:30
|
|
|
|
|
|
|
expect(helper.issue_closed_link(issue, user, css_class: css_class)).to match(link)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
shared_examples 'does not display link' do
|
|
|
|
it 'returns nil' do
|
|
|
|
expect(helper.issue_closed_link(issue, user)).to be_nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with linked issue' do
|
|
|
|
context 'with moved issue' do
|
|
|
|
before do
|
2020-10-24 23:57:45 +05:30
|
|
|
issue.update!(moved_to: new_issue)
|
2019-12-04 20:38:33 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
context 'when user has permission to see new issue' do
|
|
|
|
let(:user) { project.owner }
|
|
|
|
let(:css_class) { 'text-white text-underline' }
|
|
|
|
|
|
|
|
it_behaves_like 'successfully displays link to issue and with css class', 'moved'
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when user has no permission to see new issue' do
|
|
|
|
let(:user) { guest }
|
|
|
|
|
|
|
|
it_behaves_like 'does not display link'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with duplicated issue' do
|
|
|
|
before do
|
2020-10-24 23:57:45 +05:30
|
|
|
issue.update!(duplicated_to: new_issue)
|
2019-12-04 20:38:33 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
context 'when user has permission to see new issue' do
|
|
|
|
let(:user) { project.owner }
|
|
|
|
let(:css_class) { 'text-white text-underline' }
|
|
|
|
|
|
|
|
it_behaves_like 'successfully displays link to issue and with css class', 'duplicated'
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when user has no permission to see new issue' do
|
|
|
|
let(:user) { guest }
|
|
|
|
|
|
|
|
it_behaves_like 'does not display link'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'without linked issue' do
|
|
|
|
let(:user) { project.owner }
|
|
|
|
|
|
|
|
before do
|
2020-10-24 23:57:45 +05:30
|
|
|
issue.update!(moved_to: nil, duplicated_to: nil)
|
2019-12-04 20:38:33 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it_behaves_like 'does not display link'
|
|
|
|
end
|
|
|
|
end
|
2020-07-28 23:09:34 +05:30
|
|
|
|
|
|
|
describe '#show_moved_service_desk_issue_warning?' do
|
|
|
|
let(:project1) { create(:project, service_desk_enabled: true) }
|
|
|
|
let(:project2) { create(:project, service_desk_enabled: true) }
|
|
|
|
let!(:old_issue) { create(:issue, author: User.support_bot, project: project1) }
|
|
|
|
let!(:new_issue) { create(:issue, author: User.support_bot, project: project2) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
allow(Gitlab::IncomingEmail).to receive(:enabled?) { true }
|
|
|
|
allow(Gitlab::IncomingEmail).to receive(:supports_wildcard?) { true }
|
|
|
|
|
2020-10-24 23:57:45 +05:30
|
|
|
old_issue.update!(moved_to: new_issue)
|
2020-07-28 23:09:34 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'is true when moved issue project has service desk disabled' do
|
|
|
|
project2.update!(service_desk_enabled: false)
|
|
|
|
|
|
|
|
expect(helper.show_moved_service_desk_issue_warning?(new_issue)).to be(true)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'is false when moved issue project has service desk enabled' do
|
|
|
|
expect(helper.show_moved_service_desk_issue_warning?(new_issue)).to be(false)
|
|
|
|
end
|
|
|
|
end
|
2021-01-03 14:25:43 +05:30
|
|
|
|
|
|
|
describe '#use_startup_call' do
|
|
|
|
it "returns false when a query param is present" do
|
|
|
|
allow(controller.request).to receive(:query_parameters).and_return({ foo: 'bar' })
|
|
|
|
|
|
|
|
expect(helper.use_startup_call?).to eq(false)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "returns false when user has stored sort preference" do
|
|
|
|
controller.instance_variable_set(:@sort, 'updated_asc')
|
|
|
|
|
|
|
|
expect(helper.use_startup_call?).to eq(false)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns true when request.query_parameters is empty with default sorting preference' do
|
|
|
|
controller.instance_variable_set(:@sort, 'created_date')
|
|
|
|
allow(controller.request).to receive(:query_parameters).and_return({})
|
|
|
|
|
|
|
|
expect(helper.use_startup_call?).to eq(true)
|
|
|
|
end
|
|
|
|
end
|
2021-03-11 19:13:27 +05:30
|
|
|
|
|
|
|
describe '#issue_header_actions_data' do
|
|
|
|
let(:current_user) { create(:user) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
allow(helper).to receive(:current_user).and_return(current_user)
|
|
|
|
allow(helper).to receive(:can?).and_return(true)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns expected result' do
|
|
|
|
expected = {
|
|
|
|
can_create_issue: "true",
|
|
|
|
can_reopen_issue: "true",
|
|
|
|
can_report_spam: "false",
|
|
|
|
can_update_issue: "true",
|
|
|
|
iid: issue.iid,
|
|
|
|
is_issue_author: "false",
|
|
|
|
issue_type: "issue",
|
|
|
|
new_issue_path: new_project_issue_path(project),
|
|
|
|
project_path: project.full_path,
|
|
|
|
report_abuse_path: new_abuse_report_path(user_id: issue.author.id, ref_url: issue_url(issue)),
|
|
|
|
submit_as_spam_path: mark_as_spam_project_issue_path(project, issue)
|
|
|
|
}
|
|
|
|
|
|
|
|
expect(helper.issue_header_actions_data(project, issue, current_user)).to include(expected)
|
|
|
|
end
|
|
|
|
end
|
2021-04-29 21:17:54 +05:30
|
|
|
|
|
|
|
shared_examples 'issues list data' do
|
|
|
|
it 'returns expected result' do
|
|
|
|
finder = double.as_null_object
|
|
|
|
allow(helper).to receive(:current_user).and_return(current_user)
|
|
|
|
allow(helper).to receive(:finder).and_return(finder)
|
|
|
|
allow(helper).to receive(:can?).and_return(true)
|
|
|
|
allow(helper).to receive(:image_path).and_return('#')
|
|
|
|
allow(helper).to receive(:import_csv_namespace_project_issues_path).and_return('#')
|
|
|
|
allow(helper).to receive(:url_for).and_return('#')
|
|
|
|
|
|
|
|
expected = {
|
2021-06-08 01:23:25 +05:30
|
|
|
autocomplete_award_emojis_path: autocomplete_award_emojis_path,
|
|
|
|
autocomplete_users_path: autocomplete_users_path(active: true, current_user: true, project_id: project.id, format: :json),
|
2021-04-29 21:17:54 +05:30
|
|
|
calendar_path: '#',
|
|
|
|
can_bulk_update: 'true',
|
|
|
|
can_edit: 'true',
|
|
|
|
can_import_issues: 'true',
|
|
|
|
email: current_user&.notification_email,
|
2021-06-08 01:23:25 +05:30
|
|
|
emails_help_page_path: help_page_path('development/emails', anchor: 'email-namespace'),
|
2021-04-29 21:17:54 +05:30
|
|
|
empty_state_svg_path: '#',
|
|
|
|
export_csv_path: export_csv_project_issues_path(project),
|
2021-09-04 01:27:46 +05:30
|
|
|
has_project_issues: project_issues(project).exists?.to_s,
|
2021-04-29 21:17:54 +05:30
|
|
|
import_csv_issues_path: '#',
|
2021-06-08 01:23:25 +05:30
|
|
|
initial_email: project.new_issuable_address(current_user, 'issue'),
|
2021-04-29 21:17:54 +05:30
|
|
|
is_signed_in: current_user.present?.to_s,
|
|
|
|
issues_path: project_issues_path(project),
|
2021-09-04 01:27:46 +05:30
|
|
|
jira_integration_path: help_page_url('integration/jira/', anchor: 'view-jira-issues'),
|
2021-06-08 01:23:25 +05:30
|
|
|
markdown_help_path: help_page_path('user/markdown'),
|
2021-04-29 21:17:54 +05:30
|
|
|
max_attachment_size: number_to_human_size(Gitlab::CurrentSettings.max_attachment_size.megabytes),
|
2021-09-04 01:27:46 +05:30
|
|
|
new_issue_path: new_project_issue_path(project, issue: { milestone_id: finder.milestones.first.id }),
|
2021-04-29 21:17:54 +05:30
|
|
|
project_import_jira_path: project_import_jira_path(project),
|
2021-06-08 01:23:25 +05:30
|
|
|
project_labels_path: project_labels_path(project, include_ancestor_groups: true, format: :json),
|
|
|
|
project_milestones_path: project_milestones_path(project, format: :json),
|
|
|
|
project_path: project.full_path,
|
|
|
|
quick_actions_help_path: help_page_path('user/project/quick_actions'),
|
|
|
|
reset_path: new_issuable_address_project_path(project, issuable_type: 'issue'),
|
2021-04-29 21:17:54 +05:30
|
|
|
rss_path: '#',
|
|
|
|
show_new_issue_link: 'true',
|
|
|
|
sign_in_path: new_user_session_path
|
|
|
|
}
|
|
|
|
|
|
|
|
expect(helper.issues_list_data(project, current_user, finder)).to include(expected)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#issues_list_data' do
|
|
|
|
context 'when user is signed in' do
|
|
|
|
it_behaves_like 'issues list data' do
|
|
|
|
let(:current_user) { double.as_null_object }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when user is anonymous' do
|
|
|
|
it_behaves_like 'issues list data' do
|
|
|
|
let(:current_user) { nil }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2021-06-08 01:23:25 +05:30
|
|
|
|
|
|
|
describe '#issue_manual_ordering_class' do
|
|
|
|
context 'when sorting by relative position' do
|
|
|
|
before do
|
|
|
|
assign(:sort, 'relative_position')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns manual ordering class' do
|
|
|
|
expect(helper.issue_manual_ordering_class).to eq("manual-ordering")
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when manual sorting disabled' do
|
|
|
|
before do
|
|
|
|
allow(helper).to receive(:issue_repositioning_disabled?).and_return(true)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns nil' do
|
|
|
|
expect(helper.issue_manual_ordering_class).to eq(nil)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#issue_repositioning_disabled?' do
|
|
|
|
let_it_be(:group) { create(:group) }
|
|
|
|
let_it_be(:project) { create(:project, group: group) }
|
|
|
|
|
|
|
|
subject { helper.issue_repositioning_disabled? }
|
|
|
|
|
|
|
|
context 'for project' do
|
|
|
|
before do
|
|
|
|
assign(:project, project)
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to eq(false) }
|
|
|
|
|
|
|
|
context 'when block_issue_repositioning feature flag is enabled' do
|
|
|
|
before do
|
|
|
|
stub_feature_flags(block_issue_repositioning: group)
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to eq(true) }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'for group' do
|
|
|
|
before do
|
|
|
|
assign(:group, group)
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to eq(false) }
|
|
|
|
|
|
|
|
context 'when block_issue_repositioning feature flag is enabled' do
|
|
|
|
before do
|
|
|
|
stub_feature_flags(block_issue_repositioning: group)
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to eq(true) }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|