#-- copyright # OpenProject is an open source project management software. # Copyright (C) 2012-2021 the OpenProject GmbH # # 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 docs/COPYRIGHT.rdoc for more details. #++ require File.expand_path(File.dirname(__FILE__) + '/../spec_helper') describe CostEntry, type: :model do include Cost::PluginSpecHelper let(:project) { FactoryBot.create(:project_with_types) } let(:project2) { FactoryBot.create(:project_with_types) } let(:work_package) do FactoryBot.create(:work_package, project: project, type: project.types.first, author: user) end let(:work_package2) do FactoryBot.create(:work_package, project: project2, type: project2.types.first, author: user) end let(:user) { FactoryBot.create(:user) } let(:user2) { FactoryBot.create(:user) } let(:klass) { CostEntry } let(:cost_entry) do member FactoryBot.build(:cost_entry, cost_type: cost_type, project: project, work_package: work_package, spent_on: date, units: units, user: user, comments: 'lorem') end let(:cost_entry2) do FactoryBot.build(:cost_entry, cost_type: cost_type, project: project, work_package: work_package, spent_on: date, units: units, user: user, comments: 'lorem') end let(:cost_type) do cost_type = FactoryBot.create(:cost_type) [first_rate, second_rate, third_rate].each do |rate| rate.cost_type = cost_type rate.save! end cost_type.reload cost_type end let(:first_rate) do FactoryBot.build(:cost_rate, valid_from: date - 6.days, rate: 10.0) end let(:second_rate) do FactoryBot.build(:cost_rate, valid_from: date - 4.days, rate: 100.0) end let(:third_rate) do FactoryBot.build(:cost_rate, valid_from: date - 2.days, rate: 1000.0) end let(:member) do FactoryBot.create(:member, project: project, roles: [role], principal: user) end let(:role) { FactoryBot.create(:role, permissions: []) } let(:units) { 5.0 } let(:date) { Date.today } describe 'class' do describe '#visible' do describe "WHEN having the view_cost_entries permission WHEN querying for a project WHEN a cost entry from another user is defined" do before do is_member(project, user2, [:view_cost_entries]) cost_entry.save! end it { expect(CostEntry.visible(user2, project)).to match_array([cost_entry]) } end describe "WHEN not having the view_cost_entries permission WHEN querying for a project WHEN a cost entry from another user is defined" do before do is_member(project, user2, []) cost_entry.save! end it { expect(CostEntry.visible(user2, project)).to match_array([]) } end describe "WHEN having the view_own_cost_entries permission WHEN querying for a project WHEN a cost entry from another user is defined" do before do is_member(project, user2, [:view_own_cost_entries]) cost_entry.save! end it { expect(CostEntry.visible(user2, project)).to match_array([]) } end describe "WHEN having the view_own_cost_entries permission WHEN querying for a project WHEN a cost entry from the user is defined" do before do is_member(project, cost_entry2.user, [:view_own_cost_entries]) cost_entry2.save! end it { expect(CostEntry.visible(cost_entry2.user, project)).to match_array([cost_entry2]) } end end end describe 'instance' do describe '#costs' do let(:fourth_rate) do FactoryBot.build(:cost_rate, valid_from: date - 1.days, rate: 10000.0, cost_type: cost_type) end describe 'WHEN updating the number of units' do before do cost_entry.spent_on = first_rate.valid_from + 1.day end it 'should update costs' do (0..5).each do |units| cost_entry.units = units cost_entry.save! expect(cost_entry.costs).to eq(first_rate.rate * units) end end end describe 'WHEN a new rate is added at the end' do before do cost_entry.save! fourth_rate.save! cost_entry.reload end it { expect(cost_entry.costs).to eq(fourth_rate.rate * cost_entry.units) } end describe 'WHEN a new rate is added for the future' do before do cost_entry.save! fourth_rate.valid_from = date + 1.day fourth_rate.save! cost_entry.reload end it { expect(cost_entry.costs).to eq(third_rate.rate * cost_entry.units) } end describe 'WHEN a new rate is added in between' do before do cost_entry.save! fourth_rate.valid_from = date - 3.days fourth_rate.save! cost_entry.reload end it { expect(cost_entry.costs).to eq(third_rate.rate * cost_entry.units) } end describe 'WHEN a rate is destroyed' do before do cost_entry.save! third_rate.destroy cost_entry.reload end it { expect(cost_entry.costs).to eq(cost_entry.units * second_rate.rate) } end describe "WHEN a rate's valid from is updated" do before do cost_entry.save! first_rate.update_attribute(:valid_from, date - 1.days) cost_entry.reload end it { expect(cost_entry.costs).to eq(cost_entry.units * first_rate.rate) } end describe 'WHEN spent on is changed' do before do cost_type.save! cost_entry.save! end it 'should take the then active rate to calculate' do (5.days.ago.to_date..Date.today).each do |time| cost_entry.spent_on = time cost_entry.save! rate = CostRate .where(['cost_type_id = ? AND valid_from <= ?', cost_entry.cost_type.id, cost_entry.spent_on]) .order(Arel.sql('valid_from DESC')).first.rate expect(cost_entry.costs).to eq(cost_entry.units * rate) end end end end describe '#overridden_costs' do describe 'WHEN overridden costs are seet' do let(:value) { rand(500) } before do cost_entry.overridden_costs = value end it { expect(cost_entry.overridden_costs).to eq(value) } end end describe '#real_costs' do describe 'WHEN overrridden cost are set' do let(:value) { rand(500) } before do cost_entry.overridden_costs = value end it { expect(cost_entry.real_costs).to eq(value) } end end describe '#valid' do before do cost_entry.save! end it { expect(cost_entry).to be_valid } describe 'WHEN no cost_type is provided' do before { cost_entry.cost_type = nil } it { expect(cost_entry).not_to be_valid } end describe 'WHEN no project is provided' do before do cost_entry.project = nil # unfortunately the project get's set to the work_package's project if no project is provided # TODO: check if that is necessary cost_entry.work_package = nil end it { expect(cost_entry).not_to be_valid } end describe 'WHEN no work_package is provided' do before { cost_entry.work_package = nil } it { expect(cost_entry).not_to be_valid } end describe 'WHEN the work_package is not in the project' do before { cost_entry.work_package = work_package2 } it { expect(cost_entry).not_to be_valid } end describe 'WHEN no units are provided' do before { cost_entry.units = nil } it { expect(cost_entry).not_to be_valid } end describe 'WHEN no spent_on is provided' do before { cost_entry.spent_on = nil } it { expect(cost_entry).not_to be_valid } end describe 'WHEN no user is provided' do before { cost_entry.user = nil } it { expect(cost_entry).not_to be_valid } end describe "WHEN the provided user is no member of the project WHEN the user is unchanged" do before { member.destroy } it { expect(cost_entry).to be_valid } end describe "WHEN the provided user is no member of the project WHEN the user changes" do before do cost_entry.user = user2 member.destroy end it { expect(cost_entry).not_to be_valid } end describe 'WHEN the cost_type is deleted' do before { cost_type.deleted_at = Date.new } it { expect(cost_entry).not_to be_valid } end end describe '#user' do describe 'WHEN a non existing user is provided (i.e. the user has been deleted)' do before do cost_entry.save! user.destroy end it { expect(cost_entry.reload.user).to eq(DeletedUser.first) } end describe 'WHEN an existing user is provided' do it { expect(cost_entry.user).to eq(user) } end end describe '#editable_by?' do describe "WHEN the user has the edit_cost_entries permission WHEN the cost entry is not created by the user" do before do is_member(project, user2, [:edit_cost_entries]) cost_entry end it { expect(cost_entry.editable_by?(user2)).to be_truthy } end describe "WHEN the user has the edit_cost_entries permission WHEN the cost entry is created by the user" do before do is_member(project, cost_entry2.user, [:edit_cost_entries]) end it { expect(cost_entry2.editable_by?(cost_entry2.user)).to be_truthy } end describe "WHEN the user has the edit_own_cost_entries permission WHEN the cost entry is created by the user" do before do is_member(project, cost_entry2.user, [:edit_own_cost_entries]) cost_entry2 end it { expect(cost_entry2.editable_by?(cost_entry2.user)).to be_truthy } end describe "WHEN the user has the edit_own_cost_entries permission WHEN the cost entry is created by another user" do before do is_member(project, user2, [:edit_own_cost_entries]) cost_entry end it { expect(cost_entry.editable_by?(user2)).to be_falsey } end describe "WHEN the user has no cost permission WHEN the cost entry is created by the user" do before do is_member(project, cost_entry2.user, []) cost_entry2 end it { expect(cost_entry2.editable_by?(cost_entry2.user)).to be_falsey } end end describe '#creatable_by?' do describe "WHEN the user has the log costs permission WHEN the cost entry is not associated to the user" do before do is_member(project, user2, [:log_costs]) end it { expect(cost_entry.creatable_by?(user2)).to be_truthy } end describe "WHEN the user has the log_costs permission WHEN the cost entry is associated to user" do before do is_member(project, cost_entry2.user, [:log_costs]) end it { expect(cost_entry2.creatable_by?(cost_entry2.user)).to be_truthy } end describe "WHEN the user has the log own costs permission WHEN the cost entry is associated to the user" do before do is_member(project, cost_entry2.user, [:log_own_costs]) end it { expect(cost_entry2.creatable_by?(cost_entry2.user)).to be_truthy } end describe "WHEN the user has the log_own_costs permission WHEN the cost entry is created by another user" do before do is_member(project, user2, [:log_own_costs]) end it { expect(cost_entry.creatable_by?(user2)).to be_falsey } end describe "WHEN the user has no cost permission WHEN the cost entry is associated to the user" do before do is_member(project, cost_entry2.user, []) end it { expect(cost_entry2.creatable_by?(cost_entry2.user)).to be_falsey } end end describe '#costs_visible_by?' do describe "WHEN the user has the view_cost_rates permission WHEN the cost entry is not associated to the user" do before do is_member(project, user2, [:view_cost_rates]) end it { expect(cost_entry.costs_visible_by?(user2)).to be_truthy } end describe "WHEN the user has the view_cost_rates permission in another project WHEN the cost entry is not associated to the user" do before do is_member(project2, user2, [:view_cost_rates]) end it { expect(cost_entry.costs_visible_by?(user2)).to be_falsey } end describe "WHEN the user lacks the view_cost_rates permission WHEN the cost entry is associated to the user WHEN the costs are overridden" do before do is_member(project, cost_entry2.user, []) cost_entry2.update_attribute(:overridden_costs, 1.0) end it { expect(cost_entry2.costs_visible_by?(cost_entry2.user)).to be_truthy } end describe "WHEN the user lacks the view_cost_rates permission WHEN the cost entry is associated to the user WHEN the costs are not overridden" do before do is_member(project, cost_entry2.user, []) end it { expect(cost_entry2.costs_visible_by?(cost_entry2.user)).to be_falsey } end end end end