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/services/work_packages/update_service_integration_...

868 lines
30 KiB

#-- encoding: UTF-8
#-- 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 WorkPackages::UpdateService, 'integration tests', type: :model do
let(:user) do
FactoryGirl.create(:user,
member_in_project: project,
member_through_role: role)
end
let(:role) { FactoryGirl.create(:role, permissions: permissions) }
let(:permissions) do
%i(view_work_packages edit_work_packages add_work_packages move_work_packages)
end
let(:type) { FactoryGirl.create(:type_standard) }
let(:project_types) { [type] }
let(:project) { FactoryGirl.create(:project, types: project_types) }
let(:status) { FactoryGirl.create(:status) }
let(:priority) { FactoryGirl.create(:priority) }
let(:work_package_attributes) do
{ project_id: project.id,
type_id: type.id,
author_id: user.id,
status_id: status.id,
priority: priority }
end
let(:work_package) do
FactoryGirl.create(:work_package,
work_package_attributes)
end
let(:parent_work_package) do
FactoryGirl.create(:work_package,
work_package_attributes).tap do |w|
w.children << work_package
work_package.reload
end
end
let(:grandparent_work_package) do
FactoryGirl.create(:work_package,
work_package_attributes).tap do |w|
w.children << parent_work_package
end
end
let(:sibling1_attributes) do
work_package_attributes.merge(parent: parent_work_package)
end
let(:sibling2_attributes) do
work_package_attributes.merge(parent: parent_work_package)
end
let(:sibling1_work_package) do
FactoryGirl.create(:work_package,
sibling1_attributes)
end
let(:sibling2_work_package) do
FactoryGirl.create(:work_package,
sibling2_attributes)
end
let(:child_attributes) do
work_package_attributes.merge(parent: work_package)
end
let(:child_work_package) do
FactoryGirl.create(:work_package,
child_attributes)
end
let(:grandchild_attributes) do
work_package_attributes.merge(parent: child_work_package)
end
let(:grandchild_work_package) do
FactoryGirl.create(:work_package,
grandchild_attributes)
end
let(:instance) do
described_class.new(user: user,
work_package: work_package)
end
subject do
instance.call(attributes: attributes,
send_notifications: false)
end
describe '#call' do
describe 'updating subject' do
let(:attributes) { { subject: 'New subject' } }
it 'works' do
expect(subject)
.to be_success
expect(work_package.subject)
.to eql(attributes[:subject])
end
end
describe 'updating project' do
let(:target_project) do
p = FactoryGirl.create(:project,
types: target_types,
parent: target_parent)
FactoryGirl.create(:member,
user: user,
project: p,
roles: [FactoryGirl.create(:role, permissions: target_permissions)])
p
end
let(:attributes) { { project_id: target_project.id } }
let(:target_permissions) { [:move_work_packages] }
let(:target_parent) { nil }
let(:target_types) { [type] }
describe 'with missing permissions' do
let(:target_permissions) { [] }
it 'is failure' do
expect(subject)
.to be_failure
end
end
describe 'time_entries' do
let!(:time_entries) do
[FactoryGirl.create(:time_entry,
project: project,
work_package: work_package),
FactoryGirl.create(:time_entry,
project: project,
work_package: work_package)]
end
it 'moves the time entries along' do
expect(subject)
.to be_success
expect(TimeEntry.where(id: time_entries.map(&:id)).pluck(:project_id).uniq)
.to match_array [target_project.id]
end
describe 'categories' do
let(:category) do
FactoryGirl.create(:category,
project: project)
end
before do
work_package.category = category
work_package.save!
end
context 'with equally named category' do
let!(:target_category) do
FactoryGirl.create(:category,
name: category.name,
project: target_project)
end
it 'replaces the current category by the qually named one' do
expect(subject)
.to be_success
expect(subject.result.category)
.to eql target_category
end
end
context 'w/o target category' do
let!(:other_category) do
FactoryGirl.create(:category,
project: target_project)
end
it 'removes the category' do
expect(subject)
.to be_success
expect(subject.result.category)
.to be_nil
end
end
end
describe 'version' do
let(:sharing) { 'none' }
let(:version) do
FactoryGirl.create(:version,
status: 'open',
project: project,
sharing: sharing)
end
let(:work_package) do
FactoryGirl.create(:work_package,
fixed_version: version,
project: project)
end
context 'unshared version' do
it 'removes the version' do
expect(subject)
.to be_success
expect(subject.result.fixed_version)
.to be_nil
end
end
context 'system wide shared version' do
let(:sharing) { 'system' }
it 'keeps the version' do
expect(subject)
.to be_success
expect(subject.result.fixed_version)
.to eql version
end
end
context 'move work package in project hierarchy' do
let(:target_parent) do
project
end
context 'unshared version' do
it 'removes the version' do
expect(subject)
.to be_success
expect(subject.result.fixed_version)
.to be_nil
end
end
context 'shared version' do
let(:sharing) { 'tree' }
it 'keeps the version' do
expect(subject)
.to be_success
expect(subject.result.fixed_version)
.to eql version
end
end
end
end
describe 'type' do
let(:target_types) { [type, other_type] }
let(:other_type) { FactoryGirl.create(:type) }
let(:default_type) { type }
let(:project_types) { [type, other_type] }
# otherwise ar error
let!(:default_status) { FactoryGirl.create(:default_status) }
context 'with the type existing in the target project' do
it 'keeps the type' do
expect(subject)
.to be_success
expect(subject.result.type)
.to eql type
end
end
context 'with a default type existing in the target project' do
let(:target_types) { [other_type, default_type] }
it 'uses the default type' do
expect(subject)
.to be_success
expect(subject.result.type)
.to eql default_type
end
end
context 'with only non default types' do
let(:target_types) { [other_type] }
it 'uses the first type' do
expect(subject)
.to be_success
expect(subject.result.type)
.to eql other_type
end
end
context 'with an invalid type being provided' do
let(:target_types) { [type] }
let(:attributes) do
{ project: target_project,
type: other_type }
end
it 'is unsuccessful' do
expect(subject)
.to be_failure
end
end
end
end
end
describe 'inheriting dates' do
let(:attributes) { { start_date: Date.today - 8.days, due_date: Date.today + 12.days } }
let(:sibling1_attributes) do
work_package_attributes.merge(start_date: Date.today - 5.days,
due_date: Date.today + 10.days,
parent: parent_work_package)
end
let(:sibling2_attributes) do
work_package_attributes.merge(due_date: Date.today + 16.days,
parent: parent_work_package)
end
before do
parent_work_package
grandparent_work_package
sibling1_work_package
sibling2_work_package
end
it 'works and inherits' do
expect(subject)
.to be_success
# receives the provided start/due date
expect(work_package.start_date)
.to eql(attributes[:start_date])
expect(work_package.due_date)
.to eql(attributes[:due_date])
# receives the min/max of the children's start/due date
[parent_work_package,
grandparent_work_package].each do |wp|
wp.reload
expect(wp.start_date)
.to eql(attributes[:start_date])
expect(wp.due_date)
.to eql(sibling2_work_package.due_date)
end
# sibling dates are unchanged
sibling1_work_package.reload
expect(sibling1_work_package.start_date)
.to eql(sibling1_attributes[:start_date])
expect(sibling1_work_package.due_date)
.to eql(sibling1_attributes[:due_date])
sibling2_work_package.reload
expect(sibling2_work_package.start_date)
.to eql(sibling2_attributes[:start_date])
expect(sibling2_work_package.due_date)
.to eql(sibling2_attributes[:due_date])
end
end
describe 'inheriting done_ratio' do
let(:attributes) { { done_ratio: 50 } }
let(:work_package_attributes) do
{ project_id: project.id,
type_id: type.id,
author_id: user.id,
status_id: status.id,
priority: priority,
estimated_hours: 10 }
end
let(:sibling1_attributes) do
work_package_attributes.merge(estimated_hours: nil,
done_ratio: 20,
parent: parent_work_package)
end
let(:sibling2_attributes) do
work_package_attributes.merge(done_ratio: 0,
estimated_hours: 100,
parent: parent_work_package)
end
before do
parent_work_package
grandparent_work_package
sibling1_work_package
sibling2_work_package
end
it 'works and inherits average done ratio of leaves weighted by estimated times' do
expect(subject)
.to be_success
# set to the provided values
expect(work_package.done_ratio)
.to eql(attributes[:done_ratio])
# calculated
# sibling1 not factored in as its estimated_hours are nil
calculated_ratio = (work_package.done_ratio * work_package.estimated_hours +
sibling2_work_package.done_ratio * sibling2_work_package.estimated_hours) /
(work_package.done_ratio +
sibling2_work_package.done_ratio)
[parent_work_package,
grandparent_work_package].each do |wp|
wp.reload
expect(wp.done_ratio)
.to eql(calculated_ratio.to_i)
end
# unchanged
sibling1_work_package.reload
expect(sibling1_work_package.done_ratio)
.to eql(sibling1_attributes[:done_ratio])
sibling2_work_package.reload
expect(sibling2_work_package.done_ratio)
.to eql(sibling2_attributes[:done_ratio])
end
end
describe 'inheriting estimated_hours' do
let(:attributes) { { estimated_hours: 7 } }
let(:sibling1_attributes) do
# no estimated hours
work_package_attributes.merge(parent: parent_work_package)
end
let(:sibling2_attributes) do
work_package_attributes.merge(estimated_hours: 5,
parent: parent_work_package)
end
before do
parent_work_package
grandparent_work_package
sibling1_work_package
sibling2_work_package
end
it 'works and inherits' do
expect(subject)
.to be_success
# receives the provided value
expect(work_package.estimated_hours)
.to eql(attributes[:estimated_hours].to_f)
# receive the sum of the children's estimated hours
[parent_work_package,
grandparent_work_package].each do |wp|
sum = sibling1_attributes[:estimated_hours].to_f +
sibling2_attributes[:estimated_hours].to_f +
attributes[:estimated_hours].to_f
wp.reload
expect(wp.estimated_hours)
.to eql(sum)
end
# sibling hours are unchanged
sibling1_work_package.reload
expect(sibling1_work_package.estimated_hours)
.to be_nil
sibling2_work_package.reload
expect(sibling2_work_package.estimated_hours)
.to eql(sibling2_attributes[:estimated_hours].to_f)
end
end
describe 'closing duplicates on closing status' do
let(:status_closed) do
FactoryGirl.create(:status,
is_closed: true).tap do |status_closed|
FactoryGirl.create(:workflow,
old_status: status,
new_status: status_closed,
type: type,
role: role)
end
end
let(:duplicate_work_package) do
FactoryGirl.create(:work_package,
work_package_attributes).tap do |wp|
wp.duplicated << work_package
end
end
let(:attributes) { { status: status_closed } }
before do
duplicate_work_package
end
it 'works and closes duplicates' do
expect(subject)
.to be_success
duplicate_work_package.reload
expect(work_package.status)
.to eql(attributes[:status])
expect(duplicate_work_package.status)
.to eql(attributes[:status])
end
end
describe 'rescheduling work packages along follows/hierarchy relations' do
# layout
# following_parent_work_package +-follows- following2_parent_work_package following3_parent_work_package
# | | / |
# hierarchy hierarchy hierarchy hierarchy
# | | / |
# + + + |
# work_package +-follows- following_work_package following2_work_package +-follows- following3_work_package +
# following3_sibling_work_package
let(:work_package_attributes) do
{ project_id: project.id,
type_id: type.id,
author_id: user.id,
status_id: status.id,
priority: priority,
start_date: Date.today,
due_date: Date.today + 5.days }
end
let(:following_attributes) do
work_package_attributes.merge(parent: following_parent_work_package,
subject: 'following',
start_date: Date.today + 6.days,
due_date: Date.today + 20.days)
end
let(:following_work_package) do
FactoryGirl.create(:work_package,
following_attributes).tap do |wp|
wp.follows << work_package
end
end
let(:following_parent_attributes) do
work_package_attributes.merge(subject: 'following_parent',
start_date: Date.today + 6.days,
due_date: Date.today + 20.days)
end
let(:following_parent_work_package) do
FactoryGirl.create(:work_package,
following_parent_attributes)
end
let(:following2_attributes) do
work_package_attributes.merge(parent: following2_parent_work_package,
subject: 'following2',
start_date: Date.today + 21.days,
due_date: Date.today + 25.days)
end
let(:following2_work_package) do
FactoryGirl.create(:work_package,
following2_attributes)
end
let(:following2_parent_attributes) do
work_package_attributes.merge(subject: 'following2_parent',
start_date: Date.today + 21.days,
due_date: Date.today + 25.days)
end
let(:following2_parent_work_package) do
FactoryGirl.create(:work_package,
following2_parent_attributes).tap do |wp|
wp.follows << following_parent_work_package
end
end
let(:following3_attributes) do
work_package_attributes.merge(subject: 'following3',
parent: following3_parent_work_package,
start_date: Date.today + 26.days,
due_date: Date.today + 30.days)
end
let(:following3_work_package) do
FactoryGirl.create(:work_package,
following3_attributes).tap do |wp|
wp.follows << following2_work_package
end
end
let(:following3_parent_attributes) do
work_package_attributes.merge(subject: 'following3_parent',
start_date: Date.today + 26.days,
due_date: Date.today + 36.days)
end
let(:following3_parent_work_package) do
FactoryGirl.create(:work_package,
following3_parent_attributes)
end
let(:following3_sibling_attributes) do
work_package_attributes.merge(parent: following3_parent_work_package,
subject: 'following3_sibling',
start_date: Date.today + 32.days,
due_date: Date.today + 36.days)
end
let(:following3_sibling_work_package) do
FactoryGirl.create(:work_package,
following3_sibling_attributes)
end
before do
work_package
following_parent_work_package
following_work_package
following2_parent_work_package
following2_work_package
following3_parent_work_package
following3_work_package
following3_sibling_work_package
end
let(:attributes) do
{ start_date: Date.today + 5.days,
due_date: Date.today + 10.days }
end
it 'propagates the changes to start/due date along' do
expect(subject)
.to be_success
work_package.reload(select: %i(start_date due_date))
expect(work_package.start_date)
.to eql Date.today + 5.days
expect(work_package.due_date)
.to eql Date.today + 10.days
following_work_package.reload(select: %i(start_date due_date))
expect(following_work_package.start_date)
.to eql Date.today + 11.days
expect(following_work_package.due_date)
.to eql Date.today + 25.days
following_parent_work_package.reload(select: %i(start_date due_date))
expect(following_parent_work_package.start_date)
.to eql Date.today + 11.days
expect(following_parent_work_package.due_date)
.to eql Date.today + 25.days
following2_parent_work_package.reload(select: %i(start_date due_date))
expect(following2_parent_work_package.start_date)
.to eql Date.today + 26.days
expect(following2_parent_work_package.due_date)
.to eql Date.today + 30.days
following2_work_package.reload(select: %i(start_date due_date))
expect(following2_work_package.start_date)
.to eql Date.today + 26.days
expect(following2_work_package.due_date)
.to eql Date.today + 30.days
following3_work_package.reload(select: %i(start_date due_date))
expect(following3_work_package.start_date)
.to eql Date.today + 31.days
expect(following3_work_package.due_date)
.to eql Date.today + 35.days
following3_parent_work_package.reload(select: %i(start_date due_date))
expect(following3_parent_work_package.start_date)
.to eql Date.today + 31.days
expect(following3_parent_work_package.due_date)
.to eql Date.today + 36.days
following3_sibling_work_package.reload(select: %i(start_date due_date))
expect(following3_sibling_work_package.start_date)
.to eql Date.today + 32.days
expect(following3_sibling_work_package.due_date)
.to eql Date.today + 36.days
end
end
describe 'rescheduling work packages forward follows/hierarchy relations' do
# layout
# other_work_package
# +
# |
# follows (delay: 3 days)
# |
# following_parent_work_package +-follows- following2_parent_work_package
# | |
# hierarchy hierarchy
# | |
# + +
# work_package +-follows- following_work_package following2_work_package +-follows- following3_work_package
let(:work_package_attributes) do
{ project_id: project.id,
type_id: type.id,
author_id: user.id,
status_id: status.id,
priority: priority,
start_date: Date.today,
due_date: Date.today + 5.days }
end
let(:following_attributes) do
work_package_attributes.merge(parent: following_parent_work_package,
subject: 'following',
start_date: Date.today + 6.days,
due_date: Date.today + 20.days)
end
let(:following_work_package) do
FactoryGirl.create(:work_package,
following_attributes).tap do |wp|
wp.follows << work_package
end
end
let(:following_parent_attributes) do
work_package_attributes.merge(subject: 'following_parent',
start_date: Date.today + 6.days,
due_date: Date.today + 20.days)
end
let(:following_parent_work_package) do
FactoryGirl.create(:work_package,
following_parent_attributes)
end
let(:other_attributes) do
work_package_attributes.merge(subject: 'other',
start_date: Date.today + 10.days,
due_date: Date.today + 18.days)
end
let(:other_work_package) do
FactoryGirl.create(:work_package,
other_attributes)
end
let(:following2_attributes) do
work_package_attributes.merge(parent: following2_parent_work_package,
subject: 'following2',
start_date: Date.today + 24.days,
due_date: Date.today + 28.days)
end
let(:following2_work_package) do
FactoryGirl.create(:work_package,
following2_attributes)
end
let(:following2_parent_attributes) do
work_package_attributes.merge(subject: 'following2_parent',
start_date: Date.today + 24.days,
due_date: Date.today + 28.days)
end
let(:following2_parent_work_package) do
following2 = FactoryGirl.create(:work_package,
following2_parent_attributes).tap do |wp|
wp.follows << following_parent_work_package
end
FactoryGirl.create(:relation,
relation_type: Relation::TYPE_FOLLOWS,
from: following2,
to: other_work_package,
delay: 3)
following2
end
let(:following3_attributes) do
work_package_attributes.merge(subject: 'following3',
start_date: Date.today + 29.days,
due_date: Date.today + 33.days)
end
let(:following3_work_package) do
FactoryGirl.create(:work_package,
following3_attributes).tap do |wp|
wp.follows << following2_work_package
end
end
before do
work_package
other_work_package
following_parent_work_package
following_work_package
following2_parent_work_package
following2_work_package
following3_work_package
end
let(:attributes) do
{ start_date: Date.today - 5.days,
due_date: Date.today }
end
it 'propagates the changes to start/due date along' do
expect(subject)
.to be_success
work_package.reload(select: %i(start_date due_date))
expect(work_package.start_date)
.to eql Date.today - 5.days
expect(work_package.due_date)
.to eql Date.today
following_work_package.reload(select: %i(start_date due_date))
expect(following_work_package.start_date)
.to eql Date.today + 1.day
expect(following_work_package.due_date)
.to eql Date.today + 15.days
following_parent_work_package.reload(select: %i(start_date due_date))
expect(following_parent_work_package.start_date)
.to eql Date.today + 1.days
expect(following_parent_work_package.due_date)
.to eql Date.today + 15.days
following2_parent_work_package.reload(select: %i(start_date due_date))
expect(following2_parent_work_package.start_date)
.to eql Date.today + 22.days
expect(following2_parent_work_package.due_date)
.to eql Date.today + 26.days
following2_work_package.reload(select: %i(start_date due_date))
expect(following2_work_package.start_date)
.to eql Date.today + 22.days
expect(following2_work_package.due_date)
.to eql Date.today + 26.days
following3_work_package.reload(select: %i(start_date due_date))
expect(following3_work_package.start_date)
.to eql Date.today + 27.days
expect(following3_work_package.due_date)
.to eql Date.today + 31.days
end
end
end
end