2011-10-08 21:36:38 +00:00
|
|
|
require 'spec_helper'
|
|
|
|
|
2017-10-26 11:04:07 +00:00
|
|
|
describe 'Issues' do
|
2017-02-24 21:37:32 +00:00
|
|
|
include DropzoneHelper
|
2016-07-23 23:28:12 +00:00
|
|
|
include IssueHelpers
|
2015-02-06 06:40:35 +00:00
|
|
|
include SortingHelper
|
|
|
|
|
2017-06-20 02:06:13 +00:00
|
|
|
let(:user) { create(:user) }
|
2017-08-02 19:55:11 +00:00
|
|
|
let(:project) { create(:project, :public) }
|
2011-10-08 21:36:38 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
describe 'while user is signed out' do
|
|
|
|
describe 'empty state' do
|
|
|
|
it 'user sees empty state' do
|
|
|
|
visit project_issues_path(project)
|
2011-10-08 21:36:38 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(page).to have_content('Register / Sign In')
|
|
|
|
expect(page).to have_content('The Issue Tracker is the place to add things that need to be improved or solved in a project.')
|
|
|
|
expect(page).to have_content('You can register or sign in to create issues for this project.')
|
|
|
|
end
|
2012-11-14 00:20:37 +00:00
|
|
|
end
|
2017-12-18 15:54:44 +00:00
|
|
|
end
|
2012-11-14 00:20:37 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
describe 'while user is signed in' do
|
2011-10-26 13:46:25 +00:00
|
|
|
before do
|
2017-12-18 15:54:44 +00:00
|
|
|
sign_in(user)
|
|
|
|
user2 = create(:user)
|
2017-10-26 11:04:07 +00:00
|
|
|
|
2017-12-22 08:18:28 +00:00
|
|
|
project.add_developer(user)
|
|
|
|
project.add_developer(user2)
|
2017-10-26 11:04:07 +00:00
|
|
|
end
|
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
describe 'empty state' do
|
|
|
|
it 'user sees empty state' do
|
|
|
|
visit project_issues_path(project)
|
2017-10-26 11:04:07 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(page).to have_content('The Issue Tracker is the place to add things that need to be improved or solved in a project')
|
|
|
|
expect(page).to have_content('Issues can be bugs, tasks or ideas to be discussed. Also, issues are searchable and filterable.')
|
|
|
|
expect(page).to have_content('New issue')
|
|
|
|
end
|
|
|
|
end
|
2017-10-26 11:04:07 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
describe 'Edit issue' do
|
|
|
|
let!(:issue) do
|
|
|
|
create(:issue,
|
|
|
|
author: user,
|
|
|
|
assignees: [user],
|
|
|
|
project: project)
|
|
|
|
end
|
2017-10-26 11:04:07 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
before do
|
|
|
|
visit edit_project_issue_path(project, issue)
|
|
|
|
find('.js-zen-enter').click
|
2017-10-26 11:04:07 +00:00
|
|
|
end
|
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'opens new issue popup' do
|
|
|
|
expect(page).to have_content("Issue ##{issue.iid}")
|
|
|
|
end
|
2017-10-26 11:04:07 +00:00
|
|
|
end
|
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
describe 'Editing issue assignee' do
|
|
|
|
let!(:issue) do
|
|
|
|
create(:issue,
|
|
|
|
author: user,
|
|
|
|
assignees: [user],
|
|
|
|
project: project)
|
2017-10-26 11:04:07 +00:00
|
|
|
end
|
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'allows user to select unassigned', :js do
|
|
|
|
visit edit_project_issue_path(project, issue)
|
2017-10-26 11:04:07 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(page).to have_content "Assignee #{user.name}"
|
2017-10-26 11:04:07 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
first('.js-user-search').click
|
|
|
|
click_link 'Unassigned'
|
2017-10-26 11:04:07 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
click_button 'Save changes'
|
2017-10-26 11:04:07 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within('.assignee') do
|
|
|
|
expect(page).to have_content 'No assignee - assign yourself'
|
2017-10-26 11:04:07 +00:00
|
|
|
end
|
2017-12-18 15:54:44 +00:00
|
|
|
|
|
|
|
expect(issue.reload.assignees).to be_empty
|
2017-10-26 11:04:07 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
describe 'due date', :js do
|
|
|
|
context 'on new form' do
|
|
|
|
before do
|
|
|
|
visit new_project_issue_path(project)
|
|
|
|
end
|
2017-10-26 11:04:07 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'saves with due date' do
|
|
|
|
date = Date.today.at_beginning_of_month
|
2017-10-26 11:04:07 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
fill_in 'issue_title', with: 'bug 345'
|
|
|
|
fill_in 'issue_description', with: 'bug description'
|
|
|
|
find('#issuable-due-date').click
|
2017-10-26 11:04:07 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within '.pika-single' do
|
|
|
|
click_button date.day
|
|
|
|
end
|
2017-10-26 11:04:07 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(find('#issuable-due-date').value).to eq date.to_s
|
2017-10-26 11:04:07 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
click_button 'Submit issue'
|
2017-10-26 11:04:07 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within '.issuable-sidebar' do
|
|
|
|
expect(page).to have_content date.to_s(:medium)
|
|
|
|
end
|
2017-10-26 11:04:07 +00:00
|
|
|
end
|
2017-12-18 15:54:44 +00:00
|
|
|
end
|
2017-10-26 11:04:07 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
context 'on edit form' do
|
|
|
|
let(:issue) { create(:issue, author: user, project: project, due_date: Date.today.at_beginning_of_month.to_s) }
|
2017-10-26 11:04:07 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
before do
|
|
|
|
visit edit_project_issue_path(project, issue)
|
2017-10-26 11:04:07 +00:00
|
|
|
end
|
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'saves with due date' do
|
|
|
|
date = Date.today.at_beginning_of_month
|
2017-10-26 11:04:07 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(find('#issuable-due-date').value).to eq date.to_s
|
2017-10-26 11:04:07 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
date = date.tomorrow
|
2017-10-26 11:04:07 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
fill_in 'issue_title', with: 'bug 345'
|
|
|
|
fill_in 'issue_description', with: 'bug description'
|
|
|
|
find('#issuable-due-date').click
|
|
|
|
|
|
|
|
page.within '.pika-single' do
|
|
|
|
click_button date.day
|
|
|
|
end
|
2016-04-22 20:48:45 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(find('#issuable-due-date').value).to eq date.to_s
|
2017-09-27 15:22:12 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
click_button 'Save changes'
|
2017-09-27 15:22:12 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within '.issuable-sidebar' do
|
|
|
|
expect(page).to have_content date.to_s(:medium)
|
|
|
|
end
|
|
|
|
end
|
2017-09-27 15:22:12 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'warns about version conflict' do
|
|
|
|
issue.update(title: "New title")
|
2016-04-07 22:49:35 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
fill_in 'issue_title', with: 'bug 345'
|
|
|
|
fill_in 'issue_description', with: 'bug description'
|
2016-04-07 22:49:35 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
click_button 'Save changes'
|
2016-04-07 22:49:35 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(page).to have_content 'Someone edited the issue the same time you did'
|
|
|
|
end
|
2012-08-13 05:38:00 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
describe 'Issue info' do
|
|
|
|
it 'links to current issue in breadcrubs' do
|
|
|
|
issue = create(:issue, project: project)
|
2012-11-14 00:20:37 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
visit project_issue_path(project, issue)
|
2012-08-14 00:49:18 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(find('.breadcrumbs-sub-title a')[:href]).to end_with(issue_path(issue))
|
|
|
|
end
|
2012-08-14 00:49:18 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'excludes award_emoji from comment count' do
|
|
|
|
issue = create(:issue, author: user, assignees: [user], project: project, title: 'foobar')
|
|
|
|
create(:award_emoji, awardable: issue)
|
2012-08-14 00:49:18 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
visit project_issues_path(project, assignee_id: user.id)
|
2013-11-20 22:59:50 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(page).to have_content 'foobar'
|
|
|
|
expect(page.all('.no-comments').first.text).to eq "0"
|
2015-02-12 18:53:23 +00:00
|
|
|
end
|
2013-11-20 22:59:50 +00:00
|
|
|
end
|
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
describe 'Filter issue' do
|
|
|
|
before do
|
|
|
|
%w(foobar barbaz gitlab).each do |title|
|
|
|
|
create(:issue,
|
|
|
|
author: user,
|
|
|
|
assignees: [user],
|
|
|
|
project: project,
|
|
|
|
title: title)
|
|
|
|
end
|
2013-11-27 08:04:00 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
@issue = Issue.find_by(title: 'foobar')
|
|
|
|
@issue.milestone = create(:milestone, project: project)
|
|
|
|
@issue.assignees = []
|
|
|
|
@issue.save
|
|
|
|
end
|
2013-11-20 22:59:50 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
let(:issue) { @issue }
|
2013-11-20 22:59:50 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'allows filtering by issues with no specified assignee' do
|
|
|
|
visit project_issues_path(project, assignee_id: IssuableFinder::NONE)
|
2013-11-20 22:59:50 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(page).to have_content 'foobar'
|
|
|
|
expect(page).not_to have_content 'barbaz'
|
|
|
|
expect(page).not_to have_content 'gitlab'
|
2016-03-10 14:26:56 +00:00
|
|
|
end
|
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'allows filtering by a specified assignee' do
|
|
|
|
visit project_issues_path(project, assignee_id: user.id)
|
2016-04-19 11:03:28 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(page).not_to have_content 'foobar'
|
|
|
|
expect(page).to have_content 'barbaz'
|
|
|
|
expect(page).to have_content 'gitlab'
|
2016-03-10 14:26:56 +00:00
|
|
|
end
|
2017-12-18 15:54:44 +00:00
|
|
|
end
|
2016-03-10 14:26:56 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
describe 'filter issue' do
|
|
|
|
titles = %w[foo bar baz]
|
|
|
|
titles.each_with_index do |title, index|
|
|
|
|
let!(title.to_sym) do
|
|
|
|
create(:issue, title: title,
|
|
|
|
project: project,
|
|
|
|
created_at: Time.now - (index * 60))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
let(:newer_due_milestone) { create(:milestone, due_date: '2013-12-11') }
|
|
|
|
let(:later_due_milestone) { create(:milestone, due_date: '2013-12-12') }
|
2016-04-19 11:03:28 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'sorts by newest' do
|
|
|
|
visit project_issues_path(project, sort: sort_value_created_date)
|
2016-04-19 11:03:28 +00:00
|
|
|
|
2016-03-10 14:26:56 +00:00
|
|
|
expect(first_issue).to include('foo')
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(last_issue).to include('baz')
|
2016-03-10 14:26:56 +00:00
|
|
|
end
|
2016-04-21 16:49:08 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'sorts by most recently updated' do
|
|
|
|
baz.updated_at = Time.now + 100
|
|
|
|
baz.save
|
|
|
|
visit project_issues_path(project, sort: sort_value_recently_updated)
|
2017-06-14 18:18:56 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(first_issue).to include('baz')
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'sorting by due date' do
|
2017-06-14 18:18:56 +00:00
|
|
|
before do
|
2017-12-18 15:54:44 +00:00
|
|
|
foo.update(due_date: 1.day.from_now)
|
|
|
|
bar.update(due_date: 6.days.from_now)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'sorts by due date' do
|
|
|
|
visit project_issues_path(project, sort: sort_value_due_date)
|
|
|
|
|
|
|
|
expect(first_issue).to include('foo')
|
2017-06-14 18:18:56 +00:00
|
|
|
end
|
2016-04-21 16:49:08 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'sorts by due date by excluding nil due dates' do
|
2016-04-21 16:49:08 +00:00
|
|
|
bar.update(due_date: nil)
|
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
visit project_issues_path(project, sort: sort_value_due_date)
|
2016-04-21 16:49:08 +00:00
|
|
|
|
|
|
|
expect(first_issue).to include('foo')
|
|
|
|
end
|
2016-03-10 14:26:56 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
context 'with a filter on labels' do
|
|
|
|
let(:label) { create(:label, project: project) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
create(:label_link, label: label, target: foo)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'sorts by least recently due date by excluding nil due dates' do
|
|
|
|
bar.update(due_date: nil)
|
2016-03-10 14:26:56 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
visit project_issues_path(project, label_names: [label.name], sort: sort_value_due_date_later)
|
2016-04-19 11:03:28 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(first_issue).to include('foo')
|
|
|
|
end
|
2017-08-17 08:51:19 +00:00
|
|
|
end
|
2016-03-10 14:26:56 +00:00
|
|
|
end
|
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
describe 'filtering by due date' do
|
|
|
|
before do
|
|
|
|
foo.update(due_date: 1.day.from_now)
|
|
|
|
bar.update(due_date: 6.days.from_now)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'filters by none' do
|
|
|
|
visit project_issues_path(project, due_date: Issue::NoDueDate.name)
|
2016-04-19 11:03:28 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within '.issues-holder' do
|
|
|
|
expect(page).not_to have_content('foo')
|
|
|
|
expect(page).not_to have_content('bar')
|
|
|
|
expect(page).to have_content('baz')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'filters by any' do
|
|
|
|
visit project_issues_path(project, due_date: Issue::AnyDueDate.name)
|
|
|
|
|
|
|
|
page.within '.issues-holder' do
|
|
|
|
expect(page).to have_content('foo')
|
|
|
|
expect(page).to have_content('bar')
|
|
|
|
expect(page).to have_content('baz')
|
|
|
|
end
|
2017-08-17 08:51:19 +00:00
|
|
|
end
|
2016-03-10 14:26:56 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'filters by due this week' do
|
|
|
|
foo.update(due_date: Date.today.beginning_of_week + 2.days)
|
|
|
|
bar.update(due_date: Date.today.end_of_week)
|
|
|
|
baz.update(due_date: Date.today - 8.days)
|
2016-04-19 11:03:28 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
visit project_issues_path(project, due_date: Issue::DueThisWeek.name)
|
2016-04-19 11:03:28 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within '.issues-holder' do
|
|
|
|
expect(page).to have_content('foo')
|
|
|
|
expect(page).to have_content('bar')
|
|
|
|
expect(page).not_to have_content('baz')
|
|
|
|
end
|
2017-08-17 08:51:19 +00:00
|
|
|
end
|
2016-03-10 14:26:56 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'filters by due this month' do
|
|
|
|
foo.update(due_date: Date.today.beginning_of_month + 2.days)
|
|
|
|
bar.update(due_date: Date.today.end_of_month)
|
|
|
|
baz.update(due_date: Date.today - 50.days)
|
2016-04-19 11:03:28 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
visit project_issues_path(project, due_date: Issue::DueThisMonth.name)
|
2016-04-19 11:03:28 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within '.issues-holder' do
|
|
|
|
expect(page).to have_content('foo')
|
|
|
|
expect(page).to have_content('bar')
|
|
|
|
expect(page).not_to have_content('baz')
|
|
|
|
end
|
2017-08-17 08:51:19 +00:00
|
|
|
end
|
2016-03-10 14:26:56 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'filters by overdue' do
|
|
|
|
foo.update(due_date: Date.today + 2.days)
|
|
|
|
bar.update(due_date: Date.today + 20.days)
|
|
|
|
baz.update(due_date: Date.yesterday)
|
2016-03-18 17:17:01 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
visit project_issues_path(project, due_date: Issue::Overdue.name)
|
2016-04-19 11:03:28 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within '.issues-holder' do
|
|
|
|
expect(page).not_to have_content('foo')
|
|
|
|
expect(page).not_to have_content('bar')
|
|
|
|
expect(page).to have_content('baz')
|
|
|
|
end
|
2017-08-17 08:51:19 +00:00
|
|
|
end
|
2016-04-19 11:03:28 +00:00
|
|
|
end
|
2016-03-10 14:26:56 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
describe 'sorting by milestone' do
|
|
|
|
before do
|
|
|
|
foo.milestone = newer_due_milestone
|
|
|
|
foo.save
|
|
|
|
bar.milestone = later_due_milestone
|
|
|
|
bar.save
|
|
|
|
end
|
2013-11-20 22:59:50 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'sorts by milestone' do
|
|
|
|
visit project_issues_path(project, sort: sort_value_milestone)
|
2013-11-20 22:59:50 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(first_issue).to include('foo')
|
|
|
|
expect(last_issue).to include('baz')
|
|
|
|
end
|
2013-11-20 22:59:50 +00:00
|
|
|
end
|
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
describe 'combine filter and sort' do
|
|
|
|
let(:user2) { create(:user) }
|
2013-11-20 22:59:50 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
before do
|
|
|
|
foo.assignees << user2
|
|
|
|
foo.save
|
|
|
|
bar.assignees << user2
|
|
|
|
bar.save
|
|
|
|
end
|
2013-11-20 22:59:50 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'sorts with a filter applied' do
|
|
|
|
visit project_issues_path(project, sort: sort_value_created_date, assignee_id: user2.id)
|
2013-11-20 22:59:50 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(first_issue).to include('foo')
|
|
|
|
expect(last_issue).to include('bar')
|
|
|
|
expect(page).not_to have_content('baz')
|
|
|
|
end
|
2013-11-20 22:59:50 +00:00
|
|
|
end
|
|
|
|
end
|
2013-11-27 13:00:57 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
describe 'when I want to reset my incoming email token' do
|
|
|
|
let(:project1) { create(:project, namespace: user.namespace) }
|
|
|
|
let!(:issue) { create(:issue, project: project1) }
|
2016-10-18 18:03:31 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
before do
|
|
|
|
stub_incoming_email_setting(enabled: true, address: "p+%{key}@gl.ab")
|
2017-12-22 08:18:28 +00:00
|
|
|
project1.add_master(user)
|
2017-12-18 15:54:44 +00:00
|
|
|
visit namespace_project_issues_path(user.namespace, project1)
|
|
|
|
end
|
2016-10-18 18:03:31 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'changes incoming email address token', :js do
|
|
|
|
find('.issuable-email-modal-btn').click
|
|
|
|
previous_token = find('input#issuable_email').value
|
|
|
|
find('.incoming-email-token-reset').click
|
2017-02-16 19:36:59 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
wait_for_requests
|
2016-10-18 18:03:31 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(page).to have_no_field('issuable_email', with: previous_token)
|
|
|
|
new_token = project1.new_issuable_address(user.reload, 'issue')
|
|
|
|
expect(page).to have_field(
|
|
|
|
'issuable_email',
|
|
|
|
with: new_token
|
|
|
|
)
|
|
|
|
end
|
2016-10-18 18:03:31 +00:00
|
|
|
end
|
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
describe 'update labels from issue#show', :js do
|
|
|
|
let(:issue) { create(:issue, project: project, author: user, assignees: [user]) }
|
|
|
|
let!(:label) { create(:label, project: project) }
|
2016-09-27 15:23:15 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
before do
|
|
|
|
visit project_issue_path(project, issue)
|
|
|
|
end
|
2016-09-27 15:23:15 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'will not send ajax request when no data is changed' do
|
|
|
|
page.within '.labels' do
|
|
|
|
click_link 'Edit'
|
2017-04-21 15:05:38 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
find('.dropdown-menu-close', match: :first).click
|
2016-09-27 15:23:15 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(page).not_to have_selector('.block-loading')
|
|
|
|
end
|
2016-09-27 15:23:15 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
describe 'update assignee from issue#show' do
|
|
|
|
let(:issue) { create(:issue, project: project, author: user, assignees: [user]) }
|
2013-12-17 14:39:39 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
context 'by authorized user' do
|
|
|
|
it 'allows user to select unassigned', :js do
|
|
|
|
visit project_issue_path(project, issue)
|
2013-12-17 14:39:39 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within('.assignee') do
|
|
|
|
expect(page).to have_content "#{user.name}"
|
2016-03-22 19:18:29 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
click_link 'Edit'
|
|
|
|
click_link 'Unassigned'
|
|
|
|
first('.title').click
|
|
|
|
expect(page).to have_content 'No assignee'
|
|
|
|
end
|
2013-12-17 14:39:39 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
# wait_for_requests does not work with vue-resource at the moment
|
|
|
|
sleep 1
|
2017-05-04 12:11:15 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(issue.reload.assignees).to be_empty
|
|
|
|
end
|
2016-04-22 12:30:58 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'allows user to select an assignee', :js do
|
|
|
|
issue2 = create(:issue, project: project, author: user)
|
|
|
|
visit project_issue_path(project, issue2)
|
2016-04-22 12:30:58 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within('.assignee') do
|
|
|
|
expect(page).to have_content "No assignee"
|
|
|
|
end
|
2016-04-22 12:30:58 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within '.assignee' do
|
|
|
|
click_link 'Edit'
|
|
|
|
end
|
2016-04-22 20:48:45 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within '.dropdown-menu-user' do
|
|
|
|
click_link user.name
|
|
|
|
end
|
2016-04-22 12:30:58 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within('.assignee') do
|
|
|
|
expect(page).to have_content user.name
|
|
|
|
end
|
2016-04-22 12:30:58 +00:00
|
|
|
end
|
2016-06-15 11:40:22 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'allows user to unselect themselves', :js do
|
|
|
|
issue2 = create(:issue, project: project, author: user)
|
|
|
|
visit project_issue_path(project, issue2)
|
2016-06-15 11:40:22 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within '.assignee' do
|
|
|
|
click_link 'Edit'
|
|
|
|
click_link user.name
|
2016-06-15 11:40:22 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within '.value .author' do
|
|
|
|
expect(page).to have_content user.name
|
|
|
|
end
|
2016-06-15 11:40:22 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
click_link 'Edit'
|
|
|
|
click_link user.name
|
2016-06-15 11:40:22 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within '.value .assign-yourself' do
|
|
|
|
expect(page).to have_content "No assignee"
|
|
|
|
end
|
2016-06-15 11:40:22 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2013-12-17 14:39:39 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
context 'by unauthorized user' do
|
|
|
|
let(:guest) { create(:user) }
|
2013-12-20 06:56:39 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
before do
|
2017-12-22 08:18:28 +00:00
|
|
|
project.add_guest(guest)
|
2017-12-18 15:54:44 +00:00
|
|
|
end
|
2013-12-17 14:39:39 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'shows assignee text', :js do
|
|
|
|
sign_out(:user)
|
|
|
|
sign_in(guest)
|
2013-12-17 14:39:39 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
visit project_issue_path(project, issue)
|
|
|
|
expect(page).to have_content issue.assignees.first.name
|
|
|
|
end
|
2013-12-17 14:39:39 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
describe 'update milestone from issue#show' do
|
|
|
|
let!(:issue) { create(:issue, project: project, author: user) }
|
|
|
|
let!(:milestone) { create(:milestone, project: project) }
|
2013-12-17 14:39:39 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
context 'by authorized user' do
|
|
|
|
it 'allows user to select unassigned', :js do
|
|
|
|
visit project_issue_path(project, issue)
|
2013-12-17 14:39:39 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within('.milestone') do
|
|
|
|
expect(page).to have_content "None"
|
|
|
|
end
|
2015-12-10 20:06:26 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
find('.block.milestone .edit-link').click
|
|
|
|
sleep 2 # wait for ajax stuff to complete
|
|
|
|
first('.dropdown-content li').click
|
|
|
|
sleep 2
|
|
|
|
page.within('.milestone') do
|
|
|
|
expect(page).to have_content 'None'
|
|
|
|
end
|
|
|
|
|
|
|
|
expect(issue.reload.milestone).to be_nil
|
2015-12-10 20:06:26 +00:00
|
|
|
end
|
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'allows user to de-select milestone', :js do
|
|
|
|
visit project_issue_path(project, issue)
|
2016-06-15 11:40:22 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within('.milestone') do
|
|
|
|
click_link 'Edit'
|
|
|
|
click_link milestone.title
|
2016-06-15 11:40:22 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within '.value' do
|
|
|
|
expect(page).to have_content milestone.title
|
|
|
|
end
|
2016-06-15 11:40:22 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
click_link 'Edit'
|
|
|
|
click_link milestone.title
|
2016-06-15 11:40:22 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within '.value' do
|
|
|
|
expect(page).to have_content 'None'
|
|
|
|
end
|
2016-06-15 11:40:22 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2013-12-17 14:39:39 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
context 'by unauthorized user' do
|
|
|
|
let(:guest) { create(:user) }
|
2013-12-20 06:56:39 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
before do
|
2017-12-22 08:18:28 +00:00
|
|
|
project.add_guest(guest)
|
2017-12-18 15:54:44 +00:00
|
|
|
issue.milestone = milestone
|
|
|
|
issue.save
|
|
|
|
end
|
2013-12-17 14:39:39 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'shows milestone text', :js do
|
|
|
|
sign_out(:user)
|
|
|
|
sign_in(guest)
|
2013-12-17 14:39:39 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
visit project_issue_path(project, issue)
|
|
|
|
expect(page).to have_content milestone.title
|
|
|
|
end
|
2013-12-17 14:39:39 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
describe 'new issue' do
|
|
|
|
let!(:issue) { create(:issue, project: project) }
|
2017-05-12 02:40:56 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
context 'by unauthenticated user' do
|
|
|
|
before do
|
|
|
|
sign_out(:user)
|
|
|
|
end
|
2017-03-06 20:18:39 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'redirects to signin then back to new issue after signin' do
|
|
|
|
visit project_issues_path(project)
|
2017-03-06 20:18:39 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within '.nav-controls' do
|
|
|
|
click_link 'New issue'
|
|
|
|
end
|
2017-03-06 20:18:39 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(current_path).to eq new_user_session_path
|
2017-03-06 20:18:39 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
gitlab_sign_in(create(:user))
|
2017-03-06 20:18:39 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(current_path).to eq new_project_issue_path(project)
|
|
|
|
end
|
2017-03-06 20:18:39 +00:00
|
|
|
end
|
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
context 'dropzone upload file', :js do
|
|
|
|
before do
|
|
|
|
visit new_project_issue_path(project)
|
|
|
|
end
|
2016-04-17 11:15:45 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'uploads file when dragging into textarea' do
|
|
|
|
dropzone_file Rails.root.join('spec', 'fixtures', 'banana_sample.gif')
|
2016-04-17 11:15:45 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(page.find_field("issue_description").value).to have_content 'banana_sample'
|
|
|
|
end
|
2017-02-21 22:48:02 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it "doesn't add double newline to end of a single attachment markdown" do
|
|
|
|
dropzone_file Rails.root.join('spec', 'fixtures', 'banana_sample.gif')
|
2017-02-21 22:48:02 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(page.find_field("issue_description").value).not_to match /\n\n$/
|
|
|
|
end
|
2017-11-01 17:58:11 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it "cancels a file upload correctly" do
|
|
|
|
slow_requests do
|
|
|
|
dropzone_file([Rails.root.join('spec', 'fixtures', 'dk.png')], 0, false)
|
2017-11-01 17:58:11 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
click_button 'Cancel'
|
|
|
|
end
|
2017-11-01 17:58:11 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(page).to have_button('Attach a file')
|
|
|
|
expect(page).not_to have_button('Cancel')
|
|
|
|
expect(page).not_to have_selector('.uploading-progress-container', visible: true)
|
|
|
|
end
|
2017-11-01 17:58:11 +00:00
|
|
|
end
|
2017-03-22 16:26:47 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
context 'form filled by URL parameters' do
|
|
|
|
let(:project) { create(:project, :public, :repository) }
|
2017-03-28 21:13:16 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
before do
|
|
|
|
project.repository.create_file(
|
|
|
|
user,
|
|
|
|
'.gitlab/issue_templates/bug.md',
|
|
|
|
'this is a test "bug" template',
|
|
|
|
message: 'added issue template',
|
|
|
|
branch_name: 'master')
|
|
|
|
|
|
|
|
visit new_project_issue_path(project, issuable_template: 'bug')
|
|
|
|
end
|
2017-03-22 16:26:47 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'fills in template' do
|
|
|
|
expect(find('.js-issuable-selector .dropdown-toggle-text')).to have_content('bug')
|
|
|
|
end
|
2017-03-22 16:26:47 +00:00
|
|
|
end
|
|
|
|
end
|
2016-04-17 11:15:45 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
describe 'new issue by email' do
|
|
|
|
shared_examples 'show the email in the modal' do
|
|
|
|
let(:issue) { create(:issue, project: project) }
|
2016-11-05 17:28:29 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
before do
|
|
|
|
project.issues << issue
|
|
|
|
stub_incoming_email_setting(enabled: true, address: "p+%{key}@gl.ab")
|
2016-07-26 11:00:36 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
visit project_issues_path(project)
|
|
|
|
click_button('Email a new issue')
|
|
|
|
end
|
2016-07-26 11:00:36 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'click the button to show modal for the new email' do
|
|
|
|
page.within '#issuable-email-modal' do
|
|
|
|
email = project.new_issuable_address(user, 'issue')
|
2016-07-26 11:00:36 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(page).to have_selector("input[value='#{email}']")
|
|
|
|
end
|
2016-07-26 11:00:36 +00:00
|
|
|
end
|
|
|
|
end
|
2016-07-27 10:07:24 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
context 'with existing issues' do
|
|
|
|
let!(:issue) { create(:issue, project: project, author: user) }
|
2016-07-27 10:07:24 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it_behaves_like 'show the email in the modal'
|
|
|
|
end
|
2016-07-27 10:07:24 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
context 'without existing issues' do
|
|
|
|
it_behaves_like 'show the email in the modal'
|
|
|
|
end
|
2016-07-27 10:07:24 +00:00
|
|
|
end
|
2016-07-26 11:00:36 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
describe 'due date' do
|
|
|
|
context 'update due on issue#show', :js do
|
|
|
|
let(:issue) { create(:issue, project: project, author: user, assignees: [user]) }
|
2016-05-19 11:19:19 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
before do
|
|
|
|
visit project_issue_path(project, issue)
|
|
|
|
end
|
2016-05-19 11:19:19 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'adds due date to issue' do
|
|
|
|
date = Date.today.at_beginning_of_month + 2.days
|
2017-01-02 20:20:17 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within '.due_date' do
|
|
|
|
click_link 'Edit'
|
2016-05-19 11:19:19 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within '.pika-single' do
|
|
|
|
click_button date.day
|
|
|
|
end
|
2016-05-19 11:19:19 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
wait_for_requests
|
2017-01-05 22:40:42 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(find('.value').text).to have_content date.strftime('%b %-d, %Y')
|
|
|
|
end
|
2016-05-19 11:19:19 +00:00
|
|
|
end
|
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
it 'removes due date from issue' do
|
|
|
|
date = Date.today.at_beginning_of_month + 2.days
|
2017-02-16 19:36:59 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within '.due_date' do
|
|
|
|
click_link 'Edit'
|
2016-05-19 11:19:19 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
page.within '.pika-single' do
|
|
|
|
click_button date.day
|
|
|
|
end
|
2016-05-19 11:19:19 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
wait_for_requests
|
2017-01-05 22:40:42 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(page).to have_no_content 'No due date'
|
2016-05-19 11:19:19 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
click_link 'remove due date'
|
|
|
|
expect(page).to have_content 'No due date'
|
|
|
|
end
|
2016-05-19 11:19:19 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2017-04-06 01:13:06 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
describe 'title issue#show', :js do
|
|
|
|
it 'updates the title', :js do
|
|
|
|
issue = create(:issue, author: user, assignees: [user], project: project, title: 'new title')
|
2017-04-06 01:13:06 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
visit project_issue_path(project, issue)
|
2017-04-06 01:13:06 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(page).to have_text("new title")
|
2017-04-06 01:13:06 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
issue.update(title: "updated title")
|
2017-04-06 01:13:06 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
wait_for_requests
|
|
|
|
expect(page).to have_text("updated title")
|
|
|
|
end
|
2017-04-06 01:13:06 +00:00
|
|
|
end
|
2017-08-07 23:56:16 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
describe 'confidential issue#show', :js do
|
|
|
|
it 'shows confidential sibebar information as confidential and can be turned off' do
|
|
|
|
issue = create(:issue, :confidential, project: project)
|
2017-08-07 23:56:16 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
visit project_issue_path(project, issue)
|
2017-08-07 23:56:16 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(page).to have_css('.issuable-note-warning')
|
|
|
|
expect(find('.issuable-sidebar-item.confidentiality')).to have_css('.is-active')
|
|
|
|
expect(find('.issuable-sidebar-item.confidentiality')).not_to have_css('.not-active')
|
2017-08-07 23:56:16 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
find('.confidential-edit').click
|
|
|
|
expect(page).to have_css('.sidebar-item-warning-message')
|
2017-08-07 23:56:16 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
within('.sidebar-item-warning-message') do
|
|
|
|
find('.btn-close').click
|
|
|
|
end
|
2017-08-07 23:56:16 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
wait_for_requests
|
2017-08-07 23:56:16 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
visit project_issue_path(project, issue)
|
2017-08-07 23:56:16 +00:00
|
|
|
|
2017-12-18 15:54:44 +00:00
|
|
|
expect(page).not_to have_css('.is-active')
|
|
|
|
end
|
2017-08-07 23:56:16 +00:00
|
|
|
end
|
|
|
|
end
|
2014-02-19 08:35:24 +00:00
|
|
|
end
|