summaryrefslogtreecommitdiff
path: root/spec/models/concerns/chronic_duration_attribute_spec.rb
blob: 8847623f705b953fecb1d313f1297a0a4d9d28ff (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
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
require 'spec_helper'

shared_examples 'ChronicDurationAttribute reader' do
  it 'contains dynamically created reader method' do
    expect(subject.class).to be_public_method_defined(virtual_field)
  end

  it 'outputs chronic duration formatted value' do
    subject.send("#{source_field}=", 120)

    expect(subject.send(virtual_field)).to eq('2m')
  end

  context 'when value is set to nil' do
    it 'outputs nil' do
      subject.send("#{source_field}=", nil)

      expect(subject.send(virtual_field)).to be_nil
    end
  end
end

shared_examples 'ChronicDurationAttribute writer' do
  it 'contains dynamically created writer method' do
    expect(subject.class).to be_public_method_defined("#{virtual_field}=")
  end

  before do
    subject.send("#{virtual_field}=", '10m')
  end

  it 'parses chronic duration input' do
    expect(subject.send(source_field)).to eq(600)
  end

  it 'passes validation' do
    expect(subject.valid?).to be_truthy
  end

  context 'when negative input is used' do
    before do
      subject.send("#{source_field}=", 3600)
    end

    it "doesn't raise exception" do
      expect { subject.send("#{virtual_field}=", '-10m') }.not_to raise_error(ChronicDuration::DurationParseError)
    end

    it "doesn't change value" do
      expect { subject.send("#{virtual_field}=", '-10m') }.not_to change { subject.send(source_field) }
    end

    it "doesn't pass validation" do
      subject.send("#{virtual_field}=", '-10m')

      expect(subject.valid?).to be_falsey
      expect(subject.errors&.messages).to include(virtual_field => ['is not a correct duration'])
    end
  end

  context 'when empty input is used' do
    before do
      subject.send("#{virtual_field}=", '')
    end

    it 'writes default value' do
      expect(subject.send(source_field)).to eq(default_value)
    end

    it 'passes validation' do
      expect(subject.valid?).to be_truthy
    end
  end

  context 'when nil input is used' do
    before do
      subject.send("#{virtual_field}=", nil)
    end

    it 'writes default value' do
      expect(subject.send(source_field)).to eq(default_value)
    end

    it 'passes validation' do
      expect(subject.valid?).to be_truthy
    end

    it "doesn't raise exception" do
      expect { subject.send("#{virtual_field}=", nil) }.not_to raise_error(NoMethodError)
    end
  end
end

describe 'ChronicDurationAttribute' do
  context 'when default value is not set' do
    let(:source_field) {:maximum_timeout}
    let(:virtual_field) {:maximum_timeout_human_readable}
    let(:default_value) { nil }

    subject { create(:ci_runner) }

    it_behaves_like 'ChronicDurationAttribute reader'
    it_behaves_like 'ChronicDurationAttribute writer'
  end

  context 'when default value is set' do
    let(:source_field) {:build_timeout}
    let(:virtual_field) {:build_timeout_human_readable}
    let(:default_value) { 3600 }

    subject { create(:project) }

    it_behaves_like 'ChronicDurationAttribute reader'
    it_behaves_like 'ChronicDurationAttribute writer'
  end
end

describe 'ChronicDurationAttribute - reader' do
  let(:source_field) {:timeout}
  let(:virtual_field) {:timeout_human_readable}

  subject { create(:ci_build).ensure_metadata }

  it "doesn't contain dynamically created writer method" do
    expect(subject.class).not_to be_public_method_defined("#{virtual_field}=")
  end

  it_behaves_like 'ChronicDurationAttribute reader'
end