# frozen_string_literal: true require 'rails_helper' RSpec.describe UserRole do 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 describe 'position' do subject { Fabricate.build :user_role } let(:excess) { 2**32 } let(:limit) { 2**31 } it { is_expected.to_not allow_values(-excess, excess).for(:position) } it { is_expected.to allow_values(-limit, limit).for(:position) } end 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 describe '#can?' do subject { Fabricate :user_role } context 'with a single flag' do it 'returns true if any of them are present' do subject.permissions = described_class::FLAGS[:manage_reports] 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 subject.permissions = described_class::FLAGS[:manage_users] 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 subject.permissions = described_class::FLAGS[:invite_users] | described_class::FLAGS[:view_dashboard] | described_class::FLAGS[:manage_reports] 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 let(:input) { nil } before do subject.permissions_as_keys = input end context 'with a single value' do let(:input) { %w(manage_users) } it 'sets permission flags' do expect(subject.permissions).to eq described_class::FLAGS[:manage_users] end end context 'with multiple values' do let(:input) { %w(manage_users manage_reports) } it 'sets permission flags' do expect(subject.permissions).to eq described_class::FLAGS[:manage_users] | described_class::FLAGS[:manage_reports] end end context 'with an unknown value' do let(:input) { %w(foo) } it 'does not set permission flags' do expect(subject.permissions).to eq described_class::Flags::NONE end end end describe '#computed_permissions' do subject { Fabricate :user_role } context 'when the role is nobody' do subject { described_class.nobody } it 'returns none' do expect(subject.computed_permissions).to eq described_class::Flags::NONE end end context 'when the role is everyone' do subject { described_class.everyone } it 'returns permissions' do expect(subject.computed_permissions).to eq subject.permissions end end context 'when role has the administrator flag' do before do subject.permissions = described_class::FLAGS[:administrator] end it 'returns all permissions' do expect(subject.computed_permissions).to eq described_class::Flags::ALL end end it 'returns permissions combined with the everyone role' do expect(subject.computed_permissions).to eq described_class.everyone.permissions end end describe '.everyone' do subject { described_class.everyone } it 'returns a role' do expect(subject).to be_a(described_class) end it 'is identified as the everyone role' do expect(subject.everyone?).to be true end it 'has default permissions' do expect(subject.permissions).to eq described_class::FLAGS[:invite_users] end it 'has negative position' do expect(subject.position).to eq(described_class::NOBODY_POSITION) end end describe '.nobody' do subject { described_class.nobody } it 'returns a role' do expect(subject).to be_a(described_class) end it 'is identified as the nobody role' do expect(subject.nobody?).to be true end it 'has no permissions' do expect(subject.permissions).to eq described_class::Flags::NONE end it 'has negative position' do expect(subject.position).to eq(described_class::NOBODY_POSITION) end end describe '#everyone?' do it 'returns true when id matches the everyone id' do subject.id = described_class::EVERYONE_ROLE_ID expect(subject.everyone?).to be true end it 'returns false when id does not match the everyone id' do 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