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

411 lines
13 KiB

#-- 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 CustomField, type: :model do
before do CustomField.destroy_all end
let(:field) { FactoryGirl.build :custom_field }
let(:field2) { FactoryGirl.build :custom_field }
describe '#name' do
it { should validate_presence_of(:name) }
it { should validate_length_of(:name).is_at_most(30) }
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
describe 'WHEN value and type are identical and locale is different' do
before do
I18n.locale = :de
field2.name = 'taken_name'
# this fields needs an explicit english translations
# otherwise it falls back using the german one
I18n.locale = :en
field2.name = 'unique_name'
field2.save!
field.name = 'taken_name'
end
it { expect(field).to be_valid }
end
end
describe 'localization' do
before do
I18n.locale = :de
field.name = 'Feld'
I18n.locale = :en
field.name = 'Field'
end
after do
I18n.locale = :en
end
it 'should return english name when in locale en' do
I18n.locale = :en
expect(field.name).to eq('Field')
end
it 'should return german name when in locale de' do
I18n.locale = :de
expect(field.name).to eq('Feld')
end
end
end
describe '#translations_attributes' do
describe 'WHEN providing a hash with locale and values' do
before do
field.translations_attributes = [{ 'name' => 'Feld',
'default_value' => 'zwei',
'possible_values' => ['eins', 'zwei', 'drei'],
'locale' => 'de' }]
end
it { expect(field.translations.size).to eq(1) }
it { expect(field.name(:de)).to eq('Feld') }
it { expect(field.default_value(:de)).to eq('zwei') }
it { expect(field.possible_values(locale: :de)).to eq(['eins', 'zwei', 'drei']) }
end
describe 'WHEN providing a hash with only a locale' do
before do
field.translations_attributes = [{ 'locale' => 'de' }]
end
it { expect(field.translations.size).to eq(0) }
end
describe 'WHEN providing a hash with a locale and blank values' do
before do
field.translations_attributes = [{ 'name' => '',
'default_value' => '',
'possible_values' => '',
'locale' => 'de' }]
end
it { expect(field.translations.size).to eq(0) }
end
describe 'WHEN providing a hash with a locale and only one values' do
before do
field.translations_attributes = [{ 'name' => 'Feld',
'locale' => 'de' }]
end
it { expect(field.translations.size).to eq(1) }
it { expect(field.name(:de)).to eq('Feld') }
end
describe 'WHEN providing a hash without a locale but with values' do
before do
field.translations_attributes = [{ 'name' => 'Feld',
'default_value' => 'zwei',
'possible_values' => ['eins', 'zwei', 'drei'],
'locale' => '' }]
end
it { expect(field.translations.size).to eq(0) }
end
describe 'WHEN already having a translation and wishing to delete it' do
before do
I18n.locale = :de
field.name = 'Feld'
I18n.locale = :en
field.name = 'Field'
field.save!
field.reload
field.translations_attributes = [{ 'id' => field.translations.first.id.to_s,
'_destroy' => '1' }]
field.save!
end
it { expect(field.translations.size).to eq(1) }
end
end
describe '#default_value' do
describe 'localization' do
before do
I18n.locale = :de
field.default_value = 'Standard'
I18n.locale = :en
field.default_value = 'default'
end
it { expect(field.default_value(:en)).to eq('default') }
it { expect(field.default_value(:de)).to eq('Standard') }
end
end
describe '#possible_values' do
describe 'localization' do
before do
I18n.locale = :de
field.possible_values = ['eins', 'zwei', 'drei']
I18n.locale = :en
field.possible_values = ['one', 'two', 'three']
I18n.locale = :de
field.save!
field.reload
end
after do
I18n.locale = :en
end
it { expect(field.possible_values(locale: :en)).to eq(['one', 'two', 'three']) }
it { expect(field.possible_values(locale: :de)).to eq(['eins', 'zwei', 'drei']) }
end
end
describe '#valid?' do
describe "WITH a list field
WITH two translations
WITH default_value not included in possible_values in the non current locale translation" do
before do
field.field_format = 'list'
field.translations_attributes = [{ 'name' => 'Feld',
'default_value' => 'vier',
'possible_values' => ['eins', 'zwei', 'drei'],
'locale' => 'de' },
{ 'name' => 'Field',
'locale' => 'en',
'possible_values' => "one\ntwo\nthree\n",
'default_value' => 'two' }]
end
it { expect(field).not_to be_valid }
end
describe "WITH a list field
WITH two translations
WITH default_value included in possible_values" do
before do
field.field_format = 'list'
field.translations_attributes = [{ 'name' => 'Feld',
'default_value' => 'zwei',
'possible_values' => ['eins', 'zwei', 'drei'],
'locale' => 'de' },
{ 'name' => 'Field',
'locale' => 'en',
'possible_values' => "one\ntwo\nthree\n",
'default_value' => 'two' }]
end
it { expect(field).to be_valid }
end
describe "WITH a list field
WITH two translations
WITH default_value not included in possible_values in the current locale translation" do
before do
field.field_format = 'list'
field.translations_attributes = [{ 'name' => 'Feld',
'default_value' => 'zwei',
'possible_values' => ['eins', 'zwei', 'drei'],
'locale' => 'de' },
{ 'name' => 'Field',
'locale' => 'en',
'possible_values' => "one\ntwo\nthree\n",
'default_value' => 'four' }]
end
it { expect(field).not_to be_valid }
end
describe "WITH a list field
WITH two translations
WITH possible_values being empty in a fallbacked translation" do
before do
field.field_format = 'list'
field.translations_attributes = [{ 'name' => 'Feld',
'locale' => 'de' },
{ 'name' => 'Field',
'locale' => 'en',
'possible_values' => "one\ntwo\nthree\n",
'default_value' => 'two' }]
end
it { expect(field).to be_valid }
end
describe "WITH a list field
WITH the field being required
WITH two translations
WITH neither translation defining a default_value" do
before do
field.field_format = 'list'
field.is_required = true
field.translations_attributes = [{ 'name' => 'Feld',
'locale' => 'de' },
{ 'name' => 'Field',
'possible_values' => "one\ntwo\nthree\n",
'locale' => 'en' }]
end
it { expect(field).to be_valid }
end
describe "WITH a boolean field
WITH the field being required
WITH two translations being provided
WITH only one translation specifying a default value" do
before do
field.field_format = 'bool'
field.translations_attributes = { '0' => { 'name' => 'name_en',
'default_value' => '1',
'locale' => 'en' },
'1' => { 'name' => 'name_de',
'locale' => 'de' } }
field.is_required = true
end
it { expect(field).to be_valid }
end
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
end
describe '#accessor_name' do
# create the custom field to force assignment of an id
let(:field) { FactoryGirl.create :custom_field }
it 'is formatted as expected' do
expect(field.accessor_name).to eql("custom_field_#{field.id}")
end
it 'returns a string' do
expect(field.accessor_name).to be_a(String)
end
end
end