summaryrefslogtreecommitdiff
path: root/spec/experiments/members/invite_email_experiment_spec.rb
blob: a9a269347e07503b2786771090c892bd9278aa83 (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
104
105
106
107
108
109
110
111
112
113
# frozen_string_literal: true

require 'spec_helper'

RSpec.describe Members::InviteEmailExperiment, :clean_gitlab_redis_shared_state do
  subject(:invite_email) { experiment('members/invite_email', **context) }

  let(:context) { { actor: double('Member', created_by: double('User', avatar_url: '_avatar_url_')) } }

  before do
    allow(invite_email).to receive(:enabled?).and_return(true)
  end

  describe "exclusions", :experiment do
    it "excludes when created by is nil" do
      expect(experiment('members/invite_email')).to exclude(actor: double(created_by: nil))
    end

    it "excludes when avatar_url is nil" do
      member_without_avatar_url = double('Member', created_by: double('User', avatar_url: nil))

      expect(experiment('members/invite_email')).to exclude(actor: member_without_avatar_url)
    end
  end

  describe "variant resolution" do
    it "proves out round robin in variant selection", :aggregate_failures do
      instance_1 = described_class.new('members/invite_email', **context)
      allow(instance_1).to receive(:enabled?).and_return(true)
      instance_2 = described_class.new('members/invite_email', **context)
      allow(instance_2).to receive(:enabled?).and_return(true)
      instance_3 = described_class.new('members/invite_email', **context)
      allow(instance_3).to receive(:enabled?).and_return(true)

      instance_1.try { }

      expect(instance_1.variant.name).to eq('permission_info')

      instance_2.try { }

      expect(instance_2.variant.name).to eq('control')

      instance_3.try { }

      expect(instance_3.variant.name).to eq('avatar')
    end
  end

  describe Members::RoundRobin do
    subject(:round_robin) { Members::RoundRobin.new('_key_', %i[variant1 variant2]) }

    describe "execute" do
      context "when there are 2 variants" do
        it "proves out round robin in selection", :aggregate_failures do
          expect(round_robin.execute).to eq :variant2
          expect(round_robin.execute).to eq :variant1
          expect(round_robin.execute).to eq :variant2
        end
      end

      context "when there are more than 2 variants" do
        subject(:round_robin) { Members::RoundRobin.new('_key_', %i[variant1 variant2 variant3]) }

        it "proves out round robin in selection", :aggregate_failures do
          expect(round_robin.execute).to eq :variant2
          expect(round_robin.execute).to eq :variant3
          expect(round_robin.execute).to eq :variant1

          expect(round_robin.execute).to eq :variant2
          expect(round_robin.execute).to eq :variant3
          expect(round_robin.execute).to eq :variant1
        end
      end

      context "when writing to cache fails" do
        subject(:round_robin) { Members::RoundRobin.new('_key_', []) }

        it "raises an error and logs" do
          allow(Gitlab::Redis::SharedState).to receive(:with).and_raise(Members::RoundRobin::CacheError)
          expect(Gitlab::AppLogger).to receive(:warn)

          expect { round_robin.execute }.to raise_error(Members::RoundRobin::CacheError)
        end
      end
    end

    describe "#counter_expires_in" do
      it 'displays the expiration time in seconds' do
        round_robin.execute

        expect(round_robin.counter_expires_in).to be_between(0, described_class::COUNTER_EXPIRE_TIME)
      end
    end

    describe '#value' do
      it 'get the count' do
        expect(round_robin.counter_value).to eq(0)

        round_robin.execute

        expect(round_robin.counter_value).to eq(1)
      end
    end

    describe '#reset!' do
      it 'resets the count down to zero' do
        3.times { round_robin.execute }

        expect { round_robin.reset! }.to change { round_robin.counter_value }.from(3).to(0)
      end
    end
  end
end