OpenProject is the leading open source project management software.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
openproject/spec/features/work_packages/custom_actions_spec.rb

440 lines
16 KiB

#-- copyright
# OpenProject is a project management system.
# Copyright (C) 2012-2017 the OpenProject Foundation (OPF)
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License version 3.
#
# OpenProject is a fork of ChiliProject, which is a fork of Redmine. The copyright follows:
# Copyright (C) 2006-2017 Jean-Philippe Lang
# Copyright (C) 2010-2013 the ChiliProject Team
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
#
# See doc/COPYRIGHT.rdoc for more details.
#++
require 'spec_helper'
describe 'Custom actions', type: :feature, js: true do
let(:permissions) { %i(view_work_packages edit_work_packages move_work_packages) }
let(:role) { FactoryBot.create(:role, permissions: permissions) }
let!(:other_role) { FactoryBot.create(:role, permissions: permissions) }
let(:admin) { FactoryBot.create(:admin) }
let(:user) do
user = FactoryBot.create(:user,
firstname: 'A',
lastname: 'User')
FactoryBot.create(:member,
project: project,
roles: [role],
user: user)
FactoryBot.create(:member,
project: other_project,
roles: [role],
user: user)
user
end
let!(:other_member_user) do
FactoryBot.create(:user,
firstname: 'Other member',
lastname: 'User',
member_in_project: project,
member_through_role: role)
end
let(:project) { FactoryBot.create(:project, name: 'This project') }
let(:other_project) { FactoryBot.create(:project, name: 'Other project') }
let!(:work_package) do
FactoryBot.create(:work_package,
project: project,
assigned_to: user,
priority: default_priority,
status: default_status)
end
let(:wp_page) { Pages::FullWorkPackage.new(work_package) }
let(:default_priority) do
FactoryBot.create(:default_priority, name: 'Normal')
end
let!(:immediate_priority) do
FactoryBot.create(:issue_priority,
name: 'At once',
position: IssuePriority.maximum(:position) + 1)
end
let(:default_status) do
FactoryBot.create(:default_status, name: 'Default status')
end
let(:closed_status) do
FactoryBot.create(:closed_status, name: 'Closed')
end
let(:rejected_status) do
FactoryBot.create(:closed_status, name: 'Rejected')
end
let(:other_type) do
type = FactoryBot.create(:type)
other_project.types << type
type
end
let!(:workflows) do
FactoryBot.create(:workflow,
old_status: default_status,
new_status: closed_status,
role: role,
type: work_package.type)
FactoryBot.create(:workflow,
new_status: default_status,
old_status: closed_status,
role: role,
type: work_package.type)
FactoryBot.create(:workflow,
old_status: default_status,
new_status: rejected_status,
role: role,
type: work_package.type)
FactoryBot.create(:workflow,
old_status: rejected_status,
new_status: default_status,
role: role,
type: other_type)
end
let!(:list_custom_field) do
cf = FactoryBot.create(:list_wp_custom_field, multi_value: true)
project.work_package_custom_fields = [cf]
work_package.type.custom_fields = [cf]
cf
end
let!(:int_custom_field) do
FactoryBot.create(:int_wp_custom_field)
end
let(:selected_list_custom_field_options) do
[list_custom_field.custom_options.first, list_custom_field.custom_options.last]
end
let!(:date_custom_field) do
cf = FactoryBot.create(:date_wp_custom_field)
other_project.work_package_custom_fields = [cf]
other_type.custom_fields = [cf]
cf
end
let(:index_ca_page) { Pages::Admin::CustomActions::Index.new }
before do
with_enterprise_token(:custom_actions)
login_as(admin)
end
scenario 'viewing workflow buttons' do
# create custom action 'Unassign'
index_ca_page.visit!
new_ca_page = index_ca_page.new
retry_block do
new_ca_page.visit!
new_ca_page.set_name('Unassign')
new_ca_page.set_description('Removes the assignee')
new_ca_page.add_action('Assignee', '-')
expect(page).to have_selector('#custom-actions-form--active-actions .form--selected-value', text: '-')
end
new_ca_page.create
index_ca_page.expect_current_path
index_ca_page.expect_listed('Unassign')
unassign = CustomAction.last
expect(unassign.actions.length).to eq(1)
expect(unassign.conditions.length).to eq(0)
# create custom action 'Close'
new_ca_page = index_ca_page.new
retry_block do
new_ca_page.visit!
new_ca_page.set_name('Close')
new_ca_page.add_action('Status', 'Close')
page.assert_selector('#custom-actions-form--active-actions .form--selected-value', text: 'Close')
new_ca_page.set_condition('Role', role.name)
page.assert_selector('#custom-actions-form--conditions .form--selected-value', text: role.name)
new_ca_page.set_condition('Status', [default_status.name, rejected_status.name])
page.assert_selector('#custom-actions-form--conditions .form--selected-value', text: default_status.name)
page.assert_selector('#custom-actions-form--conditions .form--selected-value', text: rejected_status.name)
end
new_ca_page.create
index_ca_page.expect_current_path
index_ca_page.expect_listed('Unassign', 'Close')
close = CustomAction.last
expect(close.actions.length).to eq(1)
expect(close.conditions.length).to eq(2)
# create custom action 'Escalate'
new_ca_page = index_ca_page.new
retry_block do
new_ca_page.visit!
new_ca_page.set_name('Escalate')
new_ca_page.add_action('Priority', immediate_priority.name)
page.assert_selector('#custom-actions-form--active-actions .form--selected-value', text: immediate_priority.name)
new_ca_page.add_action('Notify', other_member_user.name)
page.assert_selector('#custom-actions-form--active-actions .form--selected-value', text: other_member_user.name)
new_ca_page.add_action(list_custom_field.name, selected_list_custom_field_options.map(&:name))
page.assert_selector('#custom-actions-form--active-actions .form--selected-value', text: 'A', minimum: 1)
page.assert_selector('#custom-actions-form--active-actions .form--selected-value', text: 'G', minimum: 1)
end
new_ca_page.create
index_ca_page.expect_current_path
index_ca_page.expect_listed('Unassign', 'Close', 'Escalate')
escalate = CustomAction.last
expect(escalate.actions.length).to eq(3)
expect(escalate.conditions.length).to eq(0)
# create custom action 'Reset'
new_ca_page = index_ca_page.new
retry_block do
new_ca_page.visit!
new_ca_page.set_name('Reset')
new_ca_page.add_action('Priority', default_priority.name)
page.assert_selector('#custom-actions-form--active-actions .form--selected-value', text: default_priority.name)
new_ca_page.add_action('Status', default_status.name)
page.assert_selector('#custom-actions-form--active-actions .form--selected-value', text: default_status.name)
new_ca_page.add_action('Assignee', user.name)
page.assert_selector('#custom-actions-form--active-actions .form--selected-value', text: user.name)
# This custom field is not applicable
new_ca_page.add_action(int_custom_field.name, '1')
page.find_field("custom_action_actions_custom_field_#{int_custom_field.id}", with: '1')
new_ca_page.set_condition('Status', closed_status.name)
page.assert_selector('#custom-actions-form--conditions .form--selected-value', text: closed_status.name)
end
new_ca_page.create
index_ca_page.expect_current_path
index_ca_page.expect_listed('Unassign', 'Close', 'Escalate', 'Reset')
reset = CustomAction.last
expect(reset.actions.length).to eq(4)
expect(reset.conditions.length).to eq(1)
# create custom action 'Other roles action'
new_ca_page = index_ca_page.new
retry_block do
new_ca_page.visit!
new_ca_page.set_name('Other roles action')
new_ca_page.add_action('Status', default_status.name)
page.assert_selector('#custom-actions-form--active-actions .form--selected-value', text: default_status.name)
new_ca_page.set_condition('Role', other_role.name)
page.assert_selector('#custom-actions-form--conditions .form--selected-value', text: other_role.name)
end
new_ca_page.create
index_ca_page.expect_current_path
index_ca_page.expect_listed('Unassign', 'Close', 'Escalate', 'Reset', 'Other roles action')
other_roles_action = CustomAction.last
expect(other_roles_action.actions.length).to eq(1)
expect(other_roles_action.conditions.length).to eq(1)
# create custom action 'Move project'
new_ca_page = index_ca_page.new
retry_block do
new_ca_page.visit!
new_ca_page.set_name('Move project')
# Add date custom action which has a different admin layout
select date_custom_field.name, from: 'Add action'
select 'on', from: date_custom_field.name
date = (Date.today + 5.days).to_s
fill_in "custom_action_actions_custom_field_#{date_custom_field.id}_visible", with: date
# Close autocompleter
if page.has_selector? '.ui-datepicker-close'
scroll_to_and_click(find('.ui-datepicker-close'))
end
new_ca_page.add_action('Type', other_type.name)
page.assert_selector('#custom-actions-form--active-actions .form--selected-value', text: other_type.name)
new_ca_page.add_action('Project', other_project.name)
page.assert_selector('#custom-actions-form--active-actions .form--selected-value', text: other_project.name)
new_ca_page.set_condition('Project', project.name)
page.assert_selector('#custom-actions-form--conditions .form--selected-value', text: project.name)
end
new_ca_page.create
index_ca_page.expect_current_path
index_ca_page.expect_listed('Unassign', 'Close', 'Escalate', 'Reset', 'Other roles action', 'Move project')
move_project = CustomAction.last
expect(move_project.actions.length).to eq(3)
expect(move_project.conditions.length).to eq(1)
# use custom actions
login_as(user)
wp_page.visit!
wp_page.expect_custom_action('Unassign')
wp_page.expect_custom_action('Close')
wp_page.expect_custom_action('Escalate')
wp_page.expect_custom_action('Move project')
wp_page.expect_no_custom_action('Reset')
wp_page.expect_no_custom_action('Other roles action')
wp_page.expect_custom_action_order('Unassign', 'Close', 'Escalate', 'Move project')
within('.custom-actions') do
# When hovering over the button, the description is displayed
button = find('.custom-action--button', text: 'Unassign')
expect(button['title'])
.to eql 'Removes the assignee'
end
wp_page.click_custom_action('Unassign')
wp_page.expect_attributes assignee: '-'
wp_page.click_custom_action('Escalate')
wp_page.expect_attributes priority: immediate_priority.name,
status: default_status.name,
assignee: '-',
"customField#{list_custom_field.id}" => selected_list_custom_field_options.map(&:name).join("\n")
expect(page)
.to have_selector('.work-package-details-activities-activity-contents a.user-mention', text: other_member_user.name)
wp_page.click_custom_action('Close')
wp_page.expect_attributes status: closed_status.name,
priority: immediate_priority.name
wp_page.expect_custom_action('Reset')
wp_page.expect_no_custom_action('Close')
wp_page.click_custom_action('Reset')
wp_page.expect_attributes priority: default_priority.name,
status: default_status.name,
assignee: user.name
wp_page.expect_no_attribute "customField#{int_custom_field.id}"
# edit 'Reset' to now be named 'Reject' which sets the status to 'Rejected'
login_as(admin)
index_ca_page.visit!
edit_ca_page = index_ca_page.edit('Reset')
retry_block do
edit_ca_page.visit!
edit_ca_page.set_name 'Reject'
edit_ca_page.remove_action 'Priority'
edit_ca_page.set_action 'Assignee', '-'
find('#custom-actions-form--active-actions .form--selected-value', text: '-')
edit_ca_page.set_action 'Status', rejected_status.name
find('#custom-actions-form--active-actions .form--selected-value', text: rejected_status.name)
edit_ca_page.set_condition 'Status', default_status.name
find('#custom-actions-form--conditions .form--selected-value', text: default_status.name)
end
edit_ca_page.save
index_ca_page.expect_current_path
index_ca_page.expect_listed('Unassign', 'Close', 'Escalate', 'Reject')
reset = CustomAction.find_by(name: 'Reject')
expect(reset.actions.length).to eq(3)
expect(reset.conditions.length).to eq(1)
index_ca_page.move_top 'Move project'
index_ca_page.move_bottom 'Escalate'
index_ca_page.move_up 'Close'
index_ca_page.move_down 'Unassign'
# Check the altered button
login_as(user)
wp_page.visit!
wp_page.expect_custom_action('Unassign')
wp_page.expect_custom_action('Close')
wp_page.expect_custom_action('Escalate')
wp_page.expect_custom_action('Move project')
wp_page.expect_custom_action('Reject')
wp_page.expect_no_custom_action('Reset')
wp_page.expect_custom_action_order('Move project', 'Close', 'Reject', 'Unassign', 'Escalate')
wp_page.click_custom_action('Reject')
wp_page.expect_attributes assignee: '-',
status: rejected_status.name,
priority: default_priority.name
wp_page.expect_custom_action('Close')
wp_page.expect_no_custom_action('Reject')
# Delete 'Reject' from list of actions
login_as(admin)
index_ca_page.visit!
index_ca_page.delete('Unassign')
index_ca_page.expect_current_path
index_ca_page.expect_listed('Close', 'Escalate', 'Reject')
login_as(user)
wp_page.visit!
wp_page.expect_no_custom_action('Unassign')
wp_page.expect_custom_action('Close')
wp_page.expect_custom_action('Escalate')
wp_page.expect_no_custom_action('Reject')
# Move project
wp_page.click_custom_action('Move project')
wp_page.expect_attributes assignee: '-',
status: rejected_status.name,
type: other_type.name.upcase,
"customField#{date_custom_field.id}" => (Date.today + 5.days).strftime('%m/%d/%Y')
expect(page)
.to have_content(I18n.t('js.project.work_package_belongs_to', projectname: other_project.name))
## Bump the lockVersion and by that force a conflict.
work_package.reload.touch
wp_page.click_custom_action('Escalate', expect_success: false)
wp_page.expect_notification type: :error, message: I18n.t('api_v3.errors.code_409')
end
end