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/models/custom_field_spec.rb

487 lines
13 KiB

#-- copyright
# OpenProject is an open source project management software.
# Copyright (C) 2012-2023 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 COPYRIGHT and LICENSE files for more details.
#++
require 'spec_helper'
describe CustomField do
before do
described_class.destroy_all
end
let(:field) { build(:custom_field) }
let(:field2) { build(:custom_field) }
describe '#name' do
it { is_expected.to validate_presence_of(:name) }
it { is_expected.to validate_length_of(:name).is_at_most(256) }
describe 'uniqueness' do
describe 'WHEN value, locale and type are identical' do
before do
field.name = field2.name = 'taken name'
field2.save!
end
it { expect(field).not_to be_valid }
end
describe 'WHEN value and locale are identical and type is different' do
before do
field.name = field2.name = 'taken name'
field2.save!
field.type = 'TestCustomField'
end
it { expect(field).to be_valid }
end
describe 'WHEN type and locale are identical and value is different' do
before do
field.name = 'new name'
field2.name = 'taken name'
field2.save!
end
it { expect(field).to be_valid }
end
end
end
describe '#valid?' do
describe "WITH a text field
WITH minimum length blank" do
before do
field.field_format = 'text'
field.min_length = nil
end
it { expect(field).not_to be_valid }
end
describe "WITH a text field
WITH maximum length blank" do
before do
field.field_format = 'text'
field.max_length = nil
end
it { expect(field).not_to be_valid }
end
describe "WITH a text field
WITH minimum length not an integer" do
before do
field.field_format = 'text'
field.min_length = 'a'
end
it { expect(field).not_to be_valid }
end
describe "WITH a text field
WITH maximum length not an integer" do
before do
field.field_format = 'text'
field.max_length = 'a'
end
it { expect(field).not_to be_valid }
end
describe "WITH a text field
WITH minimum length greater than maximum length" do
before do
field.field_format = 'text'
field.min_length = 2
field.max_length = 1
end
it { expect(field).not_to be_valid }
end
describe "WITH a text field
WITH negative minimum length" do
before do
field.field_format = 'text'
field.min_length = -2
end
it { expect(field).not_to be_valid }
end
describe "WITH a text field
WITH negative maximum length" do
before do
field.field_format = 'text'
field.max_length = -2
end
it { expect(field).not_to be_valid }
end
describe "WITH a text field
WITH an invalid regexp" do
before do
field.field_format = 'text'
field.regexp = '[0-9}'
end
it 'is not valid' do
expect(field).not_to be_valid
expect(field.errors[:regexp].size).to eq(1)
end
end
describe "WITH a list field
WITHOUT a custom option" do
before do
field.field_format = 'list'
end
it 'is not valid' do
expect(field)
.to be_invalid
end
end
describe "WITH a list field
WITH a custom option" do
before do
field.field_format = 'list'
field.custom_options.build(value: 'some value')
end
it 'is valid' do
expect(field)
.to be_valid
end
end
end
describe '#attribute_name' do
let(:field) { build_stubbed(:custom_field) }
subject { field.attribute_name }
it { is_expected.to eq("custom_field_#{field.id}") }
context 'when a format is provided' do
subject { field.attribute_name(:camel_case) }
it { is_expected.to eq("customField#{field.id}") }
end
end
describe '#attribute_getter' do
let(:field) { build_stubbed(:custom_field) }
subject { field.attribute_getter }
it { is_expected.to eq(:"custom_field_#{field.id}") }
end
describe '#attribute_setter' do
let(:field) { build_stubbed(:custom_field) }
subject { field.attribute_setter }
it { is_expected.to eq(:"custom_field_#{field.id}=") }
end
describe '#column_name' do
let(:field) { build_stubbed(:custom_field) }
subject { field.column_name }
it { is_expected.to eq("cf_#{field.id}") }
end
describe '#possible_values_options' do
let(:project) { build_stubbed(:project) }
let(:user1) { build_stubbed(:user) }
let(:user2) { build_stubbed(:user) }
context 'for a user custom field' do
before do
field.field_format = 'user'
allow(project)
.to receive(:principals)
.and_return([user1, user2])
allow(Principal)
.to receive(:in_visible_project_or_me)
.and_return([user2])
end
context 'for a project' do
it 'is a list of name, id pairs' do
expect(field.possible_values_options(project))
.to match_array [[user1.name, user1.id.to_s],
[user2.name, user2.id.to_s]]
end
end
context 'for something that responds to project' do
it 'is a list of name, id pairs' do
object = OpenStruct.new(project:) # rubocop:disable Style/OpenStructUse
expect(field.possible_values_options(object))
.to match_array [[user1.name, user1.id.to_s],
[user2.name, user2.id.to_s]]
end
end
context 'for nil' do
it 'returns all principles visible to me' do
expect(field.possible_values_options)
.to match_array [[user2.name, user2.id.to_s]]
end
end
end
context 'for a list custom field' do
let(:option1) { build_stubbed(:custom_option) }
let(:option2) { build_stubbed(:custom_option) }
before do
field.field_format = 'list'
field.custom_options = [option1, option2]
end
it 'is a list of name, id pairs' do
expect(field.possible_values_options)
.to match_array [[option1.value, option1.id.to_s],
[option2.value, option2.id.to_s]]
end
end
context 'for a version custom field' do
let(:versions) { [build_stubbed(:version), build_stubbed(:version)] }
before do
field.field_format = 'version'
end
context 'with a project provided' do
it 'returns the project\'s shared_versions' do
allow(project)
.to receive(:shared_versions)
.and_return(versions)
expect(field.possible_values_options(project))
.to eql(versions.sort.map { |u| [u.name, u.id.to_s] })
end
end
context 'with a time entry provided' do
let(:time_entry) { build_stubbed(:time_entry, project:) }
it 'returns the project\'s shared_versions' do
allow(project)
.to receive(:shared_versions)
.and_return(versions)
expect(field.possible_values_options(project))
.to eql(versions.sort.map { |u| [u.name, u.id.to_s] })
end
end
context 'with nothing provided' do
it 'returns the systemwide versions' do
allow(Version)
.to receive(:systemwide)
.and_return(versions)
expect(field.possible_values_options)
.to eql(versions.sort.map { |u| [u.name, u.id.to_s] })
end
end
end
end
describe '#possible_values' do
context 'on a list custom field' do
let(:field) { described_class.new field_format: "list" }
context 'on providing an array' do
before do
field.possible_values = ['One value', 'Two values', '']
end
it 'accepts the values' do
expect(field.possible_values.map(&:value))
.to match_array(['One value', 'Two values'])
end
end
context 'on providing a string' do
before do
field.possible_values = 'One value'
end
it 'accepts the values' do
expect(field.possible_values.map(&:value))
.to match_array(['One value'])
end
end
context 'on providing a multiline string' do
before do
field.possible_values = "One value\nTwo values \r\n \n"
end
it 'accepts the values' do
expect(field.possible_values.map(&:value))
.to match_array(['One value', 'Two values'])
end
end
end
end
describe 'nested attributes for custom options' do
let(:option) { build(:custom_option) }
let(:options) { [option] }
let(:field) { build(:custom_field, field_format: 'list', custom_options: options) }
before do
field.save!
end
shared_examples_for 'saving updates field\'s updated_at' do
it 'updates updated_at' do
timestamp_before = field.updated_at
sleep 0.001
field.save
expect(field.updated_at).not_to eql(timestamp_before)
end
end
context 'after adding a custom option' do
before do
field.attributes = { 'custom_options_attributes' => { '0' => option.attributes,
'1' => { value: 'blubs' } } }
end
it_behaves_like 'saving updates field\'s updated_at'
end
context 'after changing a custom option' do
before do
attributes = option.attributes.merge(value: 'new_value')
field.attributes = { 'custom_options_attributes' => { '0' => attributes } }
end
it_behaves_like 'saving updates field\'s updated_at'
end
end
describe '#multi_value_possible?' do
context 'with a wp list cf' do
let(:field) { build_stubbed(:list_wp_custom_field) }
it 'is true' do
expect(field)
.to be_multi_value_possible
end
end
context 'with a wp user cf' do
let(:field) { build_stubbed(:user_wp_custom_field) }
it 'is true' do
expect(field)
.to be_multi_value_possible
end
end
context 'with a wp int cf' do
let(:field) { build_stubbed(:int_wp_custom_field) }
it 'is true' do
expect(field)
.not_to be_multi_value_possible
end
end
context 'with a project list cf' do
let(:field) { build_stubbed(:list_project_custom_field) }
it 'is true' do
expect(field)
.to be_multi_value_possible
end
end
context 'with a project user cf' do
let(:field) { build_stubbed(:user_project_custom_field) }
it 'is true' do
expect(field)
.to be_multi_value_possible
end
end
context 'with a project int cf' do
let(:field) { build_stubbed(:int_project_custom_field) }
it 'is true' do
expect(field)
.not_to be_multi_value_possible
end
end
context 'with a time_entry user cf' do
let(:field) { build_stubbed(:time_entry_custom_field, field_format: 'user') }
it 'is true' do
expect(field)
.not_to be_multi_value_possible
end
end
context 'with a time_entry list cf' do
let(:field) { build_stubbed(:time_entry_custom_field, field_format: 'list') }
it 'is true' do
expect(field)
.not_to be_multi_value_possible
end
end
end
describe '#destroy' do
it 'removes the cf' do
field.save!
field.destroy
expect(described_class.where(id: field.id)).not_to exist
end
end
end