HEX
Server: Apache/2.4.41 (Ubuntu)
System: Linux vmi1674223.contaboserver.net 5.4.0-182-generic #202-Ubuntu SMP Fri Apr 26 12:29:36 UTC 2024 x86_64
User: root (0)
PHP: 7.4.3-4ubuntu2.22
Disabled: pcntl_alarm,pcntl_fork,pcntl_waitpid,pcntl_wait,pcntl_wifexited,pcntl_wifstopped,pcntl_wifsignaled,pcntl_wifcontinued,pcntl_wexitstatus,pcntl_wtermsig,pcntl_wstopsig,pcntl_signal,pcntl_signal_get_handler,pcntl_signal_dispatch,pcntl_get_last_error,pcntl_strerror,pcntl_sigprocmask,pcntl_sigwaitinfo,pcntl_sigtimedwait,pcntl_exec,pcntl_getpriority,pcntl_setpriority,pcntl_async_signals,pcntl_unshare,
Upload Files
File: //opt/openproject/spec/models/custom_action_spec.rb
#-- copyright
# OpenProject is an open source project management software.
# Copyright (C) 2012-2020 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-2017 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 'spec_helper'

describe CustomAction, type: :model do
  let(:stubbed_instance) { FactoryBot.build_stubbed :custom_action }
  let(:instance) { FactoryBot.create :custom_action, name: 'zzzzzzzzz' }
  let(:other_instance) { FactoryBot.create :custom_action, name: 'aaaaa' }

  describe '#name' do
    it 'can be set and read' do
      stubbed_instance.name = 'blubs'

      expect(stubbed_instance.name)
        .to eql 'blubs'
    end
  end

  describe '#description' do
    it 'can be set and read' do
      stubbed_instance.description = 'blubs'

      expect(stubbed_instance.description)
        .to eql 'blubs'
    end
  end

  describe 'validations' do
    it 'is invalid with a name longer than 255 chars' do
      stubbed_instance.name = 'a' * 256

      expect(stubbed_instance)
        .to be_invalid
    end

    it 'is invalid with a nil name' do
      stubbed_instance.name = nil

      expect(stubbed_instance)
        .to be_invalid
    end

    it 'is invalid with an empty name' do
      stubbed_instance.name = ''

      expect(stubbed_instance)
        .to be_invalid
    end
  end

  describe '.order_by_name' do
    before do
      instance
      other_instance
    end

    it 'returns the actions ordered by name' do
      expect(described_class.order_by_name.to_a)
        .to eql [other_instance, instance]
    end
  end

  describe '.actions' do
    it 'is empty initially' do
      expect(stubbed_instance.actions)
        .to be_empty
    end

    it 'can be set and read' do
      stubbed_instance.actions = [CustomActions::Actions::AssignedTo.new(1)]

      expect(stubbed_instance.actions.map { |a| [a.key, a.values] })
        .to match_array [[:assigned_to, [1]]]
    end

    it 'can be persisted' do
      instance.actions = [CustomActions::Actions::AssignedTo.new(1)]

      instance.save!

      expect(CustomAction.find(instance.id).actions.map { |a| [a.key, a.values] })
        .to match_array [[:assigned_to, [1]]]
    end
  end

  describe '.all_actions' do
    it 'returns all available actions with the default value initialized' do
      expect(stubbed_instance.all_actions.map { |a| [a.key, a.values] })
        .to include([:assigned_to, []], [:status, []])
    end

    it 'returns the activated actions with their selected value and all other with the default value' do
      stubbed_instance.actions = [CustomActions::Actions::AssignedTo.new(1)]

      expect(stubbed_instance.all_actions.map { |a| [a.key, a.values] })
        .to include([:assigned_to, [1]], [:status, []])
    end
  end

  describe '.conditions' do
    let(:status) { FactoryBot.create(:status) }
    let(:role) { FactoryBot.create(:role) }
    let(:project) { FactoryBot.create(:project) }

    it 'is empty initially' do
      expect(stubbed_instance.conditions)
        .to be_empty
    end

    it 'can be set and read' do
      stubbed_instance.conditions = [CustomActions::Conditions::Status.new(status.id),
                                     CustomActions::Conditions::Role.new(role.id)]

      expect(stubbed_instance.conditions.map { |a| [a.key, a.values] })
        .to match_array [[:status, [status.id]],
                         [:role, [role.id]]]
    end

    it 'can be persisted' do
      instance.conditions = [CustomActions::Conditions::Status.new(status.id),
                             CustomActions::Conditions::Role.new(role.id)]

      instance.save!

      expect(CustomAction.find(instance.id).conditions.map { |a| [a.key, a.values] })
        .to match_array [[:status, [status.id]],
                         [:role, [role.id]]]
    end

    it 'existing permissions can be removed' do
      instance.conditions = [CustomActions::Conditions::Project.new(project.id)]

      instance.save!

      instance.conditions = []

      instance.save!

      expect(CustomAction.find(instance.id).conditions.map { |a| [a.key, a.values] })
        .to be_empty
    end
  end

  describe '#conditions_fulfilled?' do
    let(:work_package) { double('work_package') }
    let(:user) { double('user') }

    let(:stubbed_condition1) do
      condition = double('condition1')
      allow(condition)
        .to receive(:fulfilled_by?)
        .with(work_package, user)
        .and_return(condition1_fulfilled)
      condition
    end
    let(:stubbed_condition2) do
      condition = double('condition2')
      allow(condition)
        .to receive(:fulfilled_by?)
        .with(work_package, user)
        .and_return(condition2_fulfilled)
      condition
    end
    let(:condition1_fulfilled) { true }
    let(:condition2_fulfilled) { true }

    before do
      allow(stubbed_instance)
        .to receive(:conditions)
        .and_return [stubbed_condition1, stubbed_condition2]
    end

    context 'all conditions fulfilled' do
      it 'is true' do
        expect(stubbed_instance.conditions_fulfilled?(work_package, user))
          .to be_truthy
      end
    end

    context 'but one condition not fulfilled' do
      let(:condition1_fulfilled) { false }

      it 'is false' do
        expect(stubbed_instance.conditions_fulfilled?(work_package, user))
          .to be_falsey
      end
    end
  end

  describe '.all_conditions' do
    it 'returns all available conditions with the default value initialized' do
      expect(stubbed_instance.all_conditions.map { |a| [a.key, a.values] })
        .to match_array [[:status, []],
                         [:role, []],
                         [:type, []],
                         [:project, []]]
    end
  end
end