kanbanworkflowstimelinescrumrubyroadmapproject-planningproject-managementopenprojectangularissue-trackerifcgantt-chartganttbug-trackerboardsbcf
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.
498 lines
15 KiB
498 lines
15 KiB
#-- copyright
|
|
# OpenProject Costs Plugin
|
|
#
|
|
# Copyright (C) 2009 - 2014 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.
|
|
#
|
|
# 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.
|
|
#++
|
|
|
|
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) {
|
|
FactoryBot.create(:work_package, project: project,
|
|
type: project.types.first,
|
|
author: user)
|
|
}
|
|
let(:work_package2) {
|
|
FactoryBot.create(:work_package, project: project2,
|
|
type: project2.types.first,
|
|
author: user)
|
|
}
|
|
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) {
|
|
FactoryBot.build(:cost_rate, valid_from: date - 6.days,
|
|
rate: 10.0)
|
|
}
|
|
let(:second_rate) {
|
|
FactoryBot.build(:cost_rate, valid_from: date - 4.days,
|
|
rate: 100.0)
|
|
}
|
|
let(:third_rate) {
|
|
FactoryBot.build(:cost_rate, valid_from: date - 2.days,
|
|
rate: 1000.0)
|
|
}
|
|
let(:member) {
|
|
FactoryBot.create(:member, project: project,
|
|
roles: [role],
|
|
principal: user)
|
|
}
|
|
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) {
|
|
FactoryBot.build(:cost_rate, valid_from: date - 1.days,
|
|
rate: 10000.0,
|
|
cost_type: cost_type)
|
|
}
|
|
|
|
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
|
|
|