summaryrefslogtreecommitdiff
path: root/spec/models/concerns/featurable_spec.rb
blob: bf104fe1b3000d86fcb090b644a1ff9419f001d4 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
# frozen_string_literal: true

require 'spec_helper'

RSpec.describe Featurable do
  let!(:klass) do
    Class.new(ApplicationRecord) do
      include Featurable

      self.table_name = 'project_features'

      set_available_features %i(feature1 feature2 feature3)

      def feature1_access_level
        Featurable::DISABLED
      end

      def feature2_access_level
        Featurable::ENABLED
      end

      def feature3_access_level
        Featurable::PRIVATE
      end
    end
  end

  subject { klass.new }

  describe '.set_available_features' do
    it { expect(klass.available_features).to match_array [:feature1, :feature2, :feature3] }
  end

  describe '#*_enabled?' do
    it { expect(subject.feature1_enabled?).to be_falsey }
    it { expect(subject.feature2_enabled?).to be_truthy }
  end

  describe '.quoted_access_level_column' do
    it 'returns the table name and quoted column name for a feature' do
      expect(klass.quoted_access_level_column(:feature1)).to eq('"project_features"."feature1_access_level"')
    end
  end

  describe '.access_level_attribute' do
    it { expect(klass.access_level_attribute(:feature1)).to eq :feature1_access_level }

    it 'raises error for unspecified feature' do
      expect { klass.access_level_attribute(:unknown) }
        .to raise_error(ArgumentError, /invalid feature: unknown/)
    end
  end

  describe '#access_level' do
    it 'returns access level' do
      expect(subject.access_level(:feature1)).to eq(subject.feature1_access_level)
    end
  end

  describe '#feature_available?' do
    context 'when features are disabled' do
      it 'returns false' do
        expect(subject.feature_available?(:feature1)).to eq(false)
      end
    end

    context 'when features are enabled only for team members' do
      let_it_be(:user) { create(:user) }

      before do
        expect(subject).to receive(:member?).and_call_original
      end

      context 'when user is not present' do
        it 'returns false' do
          expect(subject.feature_available?(:feature3)).to eq(false)
        end
      end

      context 'when user can read all resources' do
        it 'returns true' do
          allow(user).to receive(:can_read_all_resources?).and_return(true)

          expect(subject.feature_available?(:feature3, user)).to eq(true)
        end
      end

      context 'when user cannot read all resources' do
        it 'raises NotImplementedError exception' do
          expect(subject).to receive(:resource_member?).and_call_original

          expect { subject.feature_available?(:feature3, user) }.to raise_error(NotImplementedError)
        end
      end
    end

    context 'when feature is enabled for everyone' do
      it 'returns true' do
        expect(subject.feature_available?(:feature2)).to eq(true)
      end
    end
  end
end