2023-02-22 00:55:31 +00:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2022-07-05 00:41:40 +00:00
|
|
|
require 'rails_helper'
|
|
|
|
|
2023-05-04 03:49:53 +00:00
|
|
|
RSpec.describe UserRole do
|
2024-11-25 08:18:07 +00:00
|
|
|
describe 'Validations' do
|
|
|
|
describe 'name' do
|
|
|
|
context 'when everyone' do
|
|
|
|
subject { described_class.everyone }
|
|
|
|
|
|
|
|
it { is_expected.to_not validate_presence_of(:name) }
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when not everyone' do
|
|
|
|
subject { Fabricate.build :user_role }
|
|
|
|
|
|
|
|
it { is_expected.to validate_presence_of(:name) }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2024-12-18 08:29:27 +00:00
|
|
|
describe 'position' do
|
|
|
|
subject { Fabricate.build :user_role }
|
|
|
|
|
2024-12-19 08:10:39 +00:00
|
|
|
let(:limit) { described_class::POSITION_LIMIT }
|
2024-12-18 08:29:27 +00:00
|
|
|
|
2024-12-19 08:10:39 +00:00
|
|
|
it { is_expected.to validate_numericality_of(:position).is_in(-limit..limit) }
|
2024-12-18 08:29:27 +00:00
|
|
|
end
|
|
|
|
|
2024-11-25 08:18:07 +00:00
|
|
|
describe 'color' do
|
|
|
|
it { is_expected.to allow_values('#112233', '#aabbcc', '').for(:color) }
|
|
|
|
it { is_expected.to_not allow_values('x', '112233445566', '#xxyyzz').for(:color) }
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when current_account is set' do
|
|
|
|
subject { Fabricate :user_role }
|
|
|
|
|
|
|
|
let(:account) { Fabricate :account }
|
|
|
|
|
|
|
|
before { subject.current_account = account }
|
|
|
|
|
|
|
|
it { is_expected.to_not allow_value(999_999).for(:position).with_message(:elevated) }
|
|
|
|
|
|
|
|
it { is_expected.to_not allow_value(999_999).for(:permissions).against(:permissions_as_keys).with_message(:elevated) }
|
|
|
|
|
|
|
|
context 'when current_account is changing their own role' do
|
|
|
|
let(:account) { Fabricate :account, user: Fabricate(:user, role: subject) }
|
|
|
|
|
|
|
|
it { is_expected.to_not allow_value(100).for(:permissions).against(:permissions_as_keys).with_message(:own_role) }
|
|
|
|
it { is_expected.to_not allow_value(100).for(:position).with_message(:own_role) }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'Callback for position' do
|
|
|
|
context 'when everyone' do
|
|
|
|
subject { Fabricate.build :user_role, id: described_class::EVERYONE_ROLE_ID }
|
|
|
|
|
|
|
|
it 'sets the position to nobody position' do
|
|
|
|
expect { subject.valid? }
|
|
|
|
.to change(subject, :position).to(described_class::NOBODY_POSITION)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when not everyone' do
|
|
|
|
subject { Fabricate.build :user_role }
|
|
|
|
|
|
|
|
it 'does not change the position' do
|
|
|
|
expect { subject.valid? }
|
|
|
|
.to_not change(subject, :position)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2022-07-05 00:41:40 +00:00
|
|
|
|
|
|
|
describe '#can?' do
|
2024-11-25 08:18:07 +00:00
|
|
|
subject { Fabricate :user_role }
|
|
|
|
|
2022-07-05 00:41:40 +00:00
|
|
|
context 'with a single flag' do
|
|
|
|
it 'returns true if any of them are present' do
|
2024-03-13 15:42:39 +00:00
|
|
|
subject.permissions = described_class::FLAGS[:manage_reports]
|
2022-07-05 00:41:40 +00:00
|
|
|
expect(subject.can?(:manage_reports)).to be true
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns false if it is not set' do
|
|
|
|
expect(subject.can?(:manage_reports)).to be false
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with multiple flags' do
|
|
|
|
it 'returns true if any of them are present' do
|
2024-03-13 15:42:39 +00:00
|
|
|
subject.permissions = described_class::FLAGS[:manage_users]
|
2022-07-05 00:41:40 +00:00
|
|
|
expect(subject.can?(:manage_reports, :manage_users)).to be true
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns false if none of them are present' do
|
|
|
|
expect(subject.can?(:manage_reports, :manage_users)).to be false
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with an unknown flag' do
|
|
|
|
it 'raises an error' do
|
|
|
|
expect { subject.can?(:foo) }.to raise_error ArgumentError
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#overrides?' do
|
|
|
|
it 'returns true if other role has lower position' do
|
|
|
|
expect(subject.overrides?(described_class.new(position: subject.position - 1))).to be true
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns true if other role is nil' do
|
|
|
|
expect(subject.overrides?(nil)).to be true
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns false if other role has higher position' do
|
|
|
|
expect(subject.overrides?(described_class.new(position: subject.position + 1))).to be false
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#permissions_as_keys' do
|
|
|
|
before do
|
2024-03-13 15:42:39 +00:00
|
|
|
subject.permissions = described_class::FLAGS[:invite_users] | described_class::FLAGS[:view_dashboard] | described_class::FLAGS[:manage_reports]
|
2022-07-05 00:41:40 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns an array' do
|
|
|
|
expect(subject.permissions_as_keys).to match_array %w(invite_users view_dashboard manage_reports)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#permissions_as_keys=' do
|
2023-11-07 09:11:04 +00:00
|
|
|
let(:input) { nil }
|
2022-07-05 00:41:40 +00:00
|
|
|
|
|
|
|
before do
|
|
|
|
subject.permissions_as_keys = input
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with a single value' do
|
|
|
|
let(:input) { %w(manage_users) }
|
|
|
|
|
|
|
|
it 'sets permission flags' do
|
2024-03-13 15:42:39 +00:00
|
|
|
expect(subject.permissions).to eq described_class::FLAGS[:manage_users]
|
2022-07-05 00:41:40 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with multiple values' do
|
|
|
|
let(:input) { %w(manage_users manage_reports) }
|
|
|
|
|
|
|
|
it 'sets permission flags' do
|
2024-03-13 15:42:39 +00:00
|
|
|
expect(subject.permissions).to eq described_class::FLAGS[:manage_users] | described_class::FLAGS[:manage_reports]
|
2022-07-05 00:41:40 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with an unknown value' do
|
|
|
|
let(:input) { %w(foo) }
|
|
|
|
|
|
|
|
it 'does not set permission flags' do
|
2024-03-13 15:42:39 +00:00
|
|
|
expect(subject.permissions).to eq described_class::Flags::NONE
|
2022-07-05 00:41:40 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#computed_permissions' do
|
2024-11-25 08:18:07 +00:00
|
|
|
subject { Fabricate :user_role }
|
|
|
|
|
2022-07-05 00:41:40 +00:00
|
|
|
context 'when the role is nobody' do
|
2023-07-12 07:49:33 +00:00
|
|
|
subject { described_class.nobody }
|
2022-07-05 00:41:40 +00:00
|
|
|
|
|
|
|
it 'returns none' do
|
2024-03-13 15:42:39 +00:00
|
|
|
expect(subject.computed_permissions).to eq described_class::Flags::NONE
|
2022-07-05 00:41:40 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when the role is everyone' do
|
2023-07-12 07:49:33 +00:00
|
|
|
subject { described_class.everyone }
|
2022-07-05 00:41:40 +00:00
|
|
|
|
|
|
|
it 'returns permissions' do
|
|
|
|
expect(subject.computed_permissions).to eq subject.permissions
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when role has the administrator flag' do
|
|
|
|
before do
|
2024-03-13 15:42:39 +00:00
|
|
|
subject.permissions = described_class::FLAGS[:administrator]
|
2022-07-05 00:41:40 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns all permissions' do
|
2024-03-13 15:42:39 +00:00
|
|
|
expect(subject.computed_permissions).to eq described_class::Flags::ALL
|
2022-07-05 00:41:40 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2023-06-06 13:51:42 +00:00
|
|
|
it 'returns permissions combined with the everyone role' do
|
|
|
|
expect(subject.computed_permissions).to eq described_class.everyone.permissions
|
2022-07-05 00:41:40 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '.everyone' do
|
|
|
|
subject { described_class.everyone }
|
|
|
|
|
|
|
|
it 'returns a role' do
|
2023-02-18 02:24:16 +00:00
|
|
|
expect(subject).to be_a(described_class)
|
2022-07-05 00:41:40 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'is identified as the everyone role' do
|
|
|
|
expect(subject.everyone?).to be true
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'has default permissions' do
|
2024-03-13 15:42:39 +00:00
|
|
|
expect(subject.permissions).to eq described_class::FLAGS[:invite_users]
|
2022-07-05 00:41:40 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'has negative position' do
|
2024-03-01 11:05:24 +00:00
|
|
|
expect(subject.position).to eq(described_class::NOBODY_POSITION)
|
2022-07-05 00:41:40 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '.nobody' do
|
|
|
|
subject { described_class.nobody }
|
|
|
|
|
|
|
|
it 'returns a role' do
|
2023-02-18 02:24:16 +00:00
|
|
|
expect(subject).to be_a(described_class)
|
2022-07-05 00:41:40 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'is identified as the nobody role' do
|
|
|
|
expect(subject.nobody?).to be true
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'has no permissions' do
|
2024-03-13 15:42:39 +00:00
|
|
|
expect(subject.permissions).to eq described_class::Flags::NONE
|
2022-07-05 00:41:40 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
it 'has negative position' do
|
2024-03-01 11:05:24 +00:00
|
|
|
expect(subject.position).to eq(described_class::NOBODY_POSITION)
|
2022-07-05 00:41:40 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#everyone?' do
|
2024-02-19 11:09:43 +00:00
|
|
|
it 'returns true when id matches the everyone id' do
|
|
|
|
subject.id = described_class::EVERYONE_ROLE_ID
|
2022-07-05 00:41:40 +00:00
|
|
|
expect(subject.everyone?).to be true
|
|
|
|
end
|
|
|
|
|
2024-02-19 11:09:43 +00:00
|
|
|
it 'returns false when id does not match the everyone id' do
|
2022-07-05 00:41:40 +00:00
|
|
|
subject.id = 123
|
|
|
|
expect(subject.everyone?).to be false
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#nobody?' do
|
|
|
|
it 'returns true when id is nil' do
|
|
|
|
subject.id = nil
|
|
|
|
expect(subject.nobody?).to be true
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns false when id is not nil' do
|
|
|
|
subject.id = 123
|
|
|
|
expect(subject.nobody?).to be false
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|