#-- copyright # OpenProject is a project management system. # Copyright (C) 2012-2015 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-2013 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 MoveWorkPackageService, type: :model do let(:user) { FactoryGirl.create(:user) } let(:type) { FactoryGirl.create(:type_standard) } let(:project) { FactoryGirl.create(:project, types: [type]) } let(:work_package) { FactoryGirl.create(:work_package, project: project, type: type) } let(:instance) { MoveWorkPackageService.new(work_package, user) } before do login_as(user) end def mock_allowed_to_move_to_project(project, is_allowed = true) allowed_scope = double('allowed_scope', :'exists?' => is_allowed) allow(WorkPackage) .to receive(:allowed_target_projects_on_move) .with(user) .and_return(allowed_scope) allow(allowed_scope) .to receive(:where) .with(id: project.id) .and_return(allowed_scope) end describe '#call' do context 'when moving' do let(:target_project) { FactoryGirl.create(:project) } before do work_package mock_allowed_to_move_to_project(target_project, true) end shared_examples_for 'moved work package' do subject { work_package.project } it { is_expected.to eq(target_project) } end context 'the project the work package is moved to' do it_behaves_like 'moved work package' do before do instance.call(target_project) end end it 'will not move if the user does not have the permission' do mock_allowed_to_move_to_project(target_project, false) instance.call(target_project) expect(work_package.project).to eql(project) end end describe '#time_entries' do let(:time_entry_1) { FactoryGirl.create(:time_entry, project: project, work_package: work_package) } let(:time_entry_2) { FactoryGirl.create(:time_entry, project: project, work_package: work_package) } before do time_entry_1 time_entry_2 work_package.reload instance.call(target_project) time_entry_1.reload time_entry_2.reload end context 'time entry 1' do subject { work_package.time_entries } it { is_expected.to include(time_entry_1) } end context 'time entry 2' do subject { work_package.time_entries } it { is_expected.to include(time_entry_2) } end it_behaves_like 'moved work package' end describe '#category' do let(:category) { FactoryGirl.create(:category, project: project) } before do work_package.category = category work_package.save! work_package.reload end context 'with same category' do let(:target_category) { FactoryGirl.create(:category, name: category.name, project: target_project) } before do target_category instance.call(target_project) end describe 'category moved' do subject { work_package.category_id } it { is_expected.to eq(target_category.id) } end it_behaves_like 'moved work package' end context 'w/o target category' do before do instance.call(target_project) end describe 'category discarded' do subject { work_package.category_id } it { is_expected.to be_nil } end it_behaves_like 'moved work package' end end describe '#version' do let(:sharing) { 'none' } let(:version) { FactoryGirl.create(:version, status: 'open', project: project, sharing: sharing) } let(:work_package) { FactoryGirl.create(:work_package, fixed_version: version, project: project) } before do instance.call(target_project) end it_behaves_like 'moved work package' context 'unshared version' do subject { work_package.fixed_version } it { is_expected.to be_nil } end context 'system wide shared version' do let(:sharing) { 'system' } subject { work_package.fixed_version } it { is_expected.to eq(version) } end context 'move work package in project hierarchy' do let(:target_project) { FactoryGirl.create(:project, parent: project) } context 'unshared version' do subject { work_package.fixed_version } it { is_expected.to be_nil } end context 'shared version' do let(:sharing) { 'tree' } subject { work_package.fixed_version } it { is_expected.to eq(version) } end end end describe '#type' do let(:target_type) { FactoryGirl.create(:type) } let(:target_project) { FactoryGirl.create(:project, types: [target_type]) } it 'is false if the current type is not defined for the new project' do expect(instance.call(target_project)).to be_falsey end end end describe 'when copying' do let(:custom_field) { FactoryGirl.create(:work_package_custom_field) } let(:source_type) { FactoryGirl.create(:type, custom_fields: [custom_field]) } let(:source_project) { FactoryGirl.create(:project, types: [source_type]) } let(:work_package) { FactoryGirl.create(:work_package, project: source_project, type: source_type, author: user) } let(:custom_value) { FactoryGirl.create(:work_package_custom_value, custom_field: custom_field, customized: work_package, value: false) } shared_examples_for 'copied work package' do subject { copy.id } it { is_expected.not_to eq(work_package.id) } end describe 'to the same project' do let(:copy) { mock_allowed_to_move_to_project(source_project) instance.call(source_project, nil, copy: true) } it_behaves_like 'copied work package' context 'project' do subject { copy.project } it { is_expected.to eq(source_project) } end end describe 'to a different project' do let(:target_type) { FactoryGirl.create(:type) } let(:target_project) { FactoryGirl.create(:project, types: [target_type]) } let(:copy) do mock_allowed_to_move_to_project(target_project) instance.call(target_project, target_type, copy: true) end it_behaves_like 'copied work package' context 'project' do subject { copy.project_id } it { is_expected.to eq(target_project.id) } end context 'type' do subject { copy.type_id } it { is_expected.to eq(target_type.id) } end context 'custom_fields' do before do custom_value end subject { copy.custom_value_for(custom_field.id) } it { is_expected.to be_nil } end describe '#attributes' do let(:copy) { mock_allowed_to_move_to_project(target_project) instance.call(target_project, target_type, copy: true, attributes: attributes) } context 'assigned_to' do let(:target_user) { FactoryGirl.create(:user) } let(:target_project_member) { FactoryGirl.create(:member, project: target_project, principal: target_user, roles: [FactoryGirl.create(:role)]) } let(:attributes) { { assigned_to_id: target_user.id } } before do target_project_member end it_behaves_like 'copied work package' subject { copy.assigned_to_id } it { is_expected.to eq(target_user.id) } end context 'status' do let(:target_status) { FactoryGirl.create(:status) } let(:attributes) { { status_id: target_status.id } } it_behaves_like 'copied work package' subject { copy.status_id } it { is_expected.to eq(target_status.id) } end context 'date' do let(:target_date) { Date.today + 14 } context 'start' do let(:attributes) { { start_date: target_date } } it_behaves_like 'copied work package' subject { copy.start_date } it { is_expected.to eq(target_date) } end context 'end' do let(:attributes) { { due_date: target_date } } it_behaves_like 'copied work package' subject { copy.due_date } it { is_expected.to eq(target_date) } end end end describe 'private project' do let(:role) { FactoryGirl.create(:role, permissions: [:view_work_packages]) } let(:target_project) { FactoryGirl.create(:project, is_public: false, types: [target_type]) } let(:source_project_member) { FactoryGirl.create(:member, project: source_project, principal: user, roles: [role]) } before do source_project_member allow(User).to receive(:current).and_return user end it_behaves_like 'copied work package' context 'pre-condition' do subject { work_package.recipients } it { is_expected.to include(work_package.author) } end subject { copy.recipients } it { is_expected.not_to include(copy.author) } end describe 'with children' do let(:target_project) { FactoryGirl.create(:project, types: [source_type]) } let(:instance) { MoveWorkPackageService.new(child, user) } let(:copy) do mock_allowed_to_move_to_project(target_project) child.reload instance.call(target_project) end let!(:child) { FactoryGirl.create(:work_package, parent: work_package, project: source_project) } let!(:grandchild) { FactoryGirl.create(:work_package, parent: child, project: source_project) } context 'cross project relations deactivated' do before do allow(Setting).to receive(:cross_project_work_package_relations?).and_return(false) end it do expect(copy).to be_falsy end it do expect(child.reload.project).to eql(source_project) end describe 'grandchild' do before do copy end it { expect(grandchild.reload.project).to eql(source_project) } end end context 'cross project relations activated' do before do allow(Setting).to receive(:cross_project_work_package_relations?).and_return(true) end it do expect(copy).to be_truthy end it do expect(copy.project).to eql(target_project) end describe 'grandchild' do before do copy end it { expect(grandchild.reload.project).to eql(target_project) } end end end end end end end