summaryrefslogtreecommitdiff
path: root/spec/lib/gitlab/ci/pipeline/expression/parser_spec.rb
blob: 3394a75ac0aaaac31b41725dcec4ce7d5c274ab9 (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
130
131
132
133
134
135
# frozen_string_literal: true

require 'spec_helper'

RSpec.describe Gitlab::Ci::Pipeline::Expression::Parser do
  before do
    stub_feature_flags(ci_if_parenthesis_enabled: true)
  end

  describe '#tree' do
    context 'validates simple operators' do
      using RSpec::Parameterized::TableSyntax

      where(:expression, :result_tree) do
        '$VAR1 == "123"' | 'equals($VAR1, "123")'
        '$VAR1 == "123" == $VAR2' | 'equals(equals($VAR1, "123"), $VAR2)'
        '$VAR' | '$VAR'
        '"some value"' | '"some value"'
        'null' | 'null'
        '$VAR1 || $VAR2 && $VAR3' | 'or($VAR1, and($VAR2, $VAR3))'
        '$VAR1 && $VAR2 || $VAR3' | 'or(and($VAR1, $VAR2), $VAR3)'
        '$VAR1 && $VAR2 || $VAR3 && $VAR4' | 'or(and($VAR1, $VAR2), and($VAR3, $VAR4))'
        '$VAR1 && ($VAR2 || $VAR3) && $VAR4' | 'and(and($VAR1, or($VAR2, $VAR3)), $VAR4)'
      end

      with_them do
        it { expect(described_class.seed(expression).tree.inspect).to eq(result_tree) }
      end
    end

    context 'when combining && and OR operators' do
      subject { described_class.seed('$VAR1 == "a" || $VAR2 == "b" && $VAR3 == "c" || $VAR4 == "d" && $VAR5 == "e"').tree }

      context 'when parenthesis engine is enabled' do
        before do
          stub_feature_flags(ci_if_parenthesis_enabled: true)
        end

        it 'returns operations in a correct order' do
          expect(subject.inspect)
            .to eq('or(or(equals($VAR1, "a"), and(equals($VAR2, "b"), equals($VAR3, "c"))), and(equals($VAR4, "d"), equals($VAR5, "e")))')
        end
      end

      context 'when parenthesis engine is disabled (legacy)' do
        before do
          stub_feature_flags(ci_if_parenthesis_enabled: false)
        end

        it 'returns operations in a invalid order' do
          expect(subject.inspect)
            .to eq('or(equals($VAR1, "a"), and(equals($VAR2, "b"), or(equals($VAR3, "c"), and(equals($VAR4, "d"), equals($VAR5, "e")))))')
        end
      end
    end

    context 'when using parenthesis' do
      subject { described_class.seed('(($VAR1 == "a" || $VAR2 == "b") && $VAR3 == "c" || $VAR4 == "d") && $VAR5 == "e"').tree }

      before do
        stub_feature_flags(ci_if_parenthesis_enabled: true)
      end

      it 'returns operations in a correct order' do
        expect(subject.inspect)
          .to eq('and(or(and(or(equals($VAR1, "a"), equals($VAR2, "b")), equals($VAR3, "c")), equals($VAR4, "d")), equals($VAR5, "e"))')
      end
    end

    context 'when expression is empty' do
      it 'raises a parsing error' do
        expect { described_class.seed('').tree }
          .to raise_error Gitlab::Ci::Pipeline::Expression::Parser::ParseError
      end
    end

    context 'when two value tokens have no operator' do
      it 'raises a parsing error' do
        expect { described_class.seed('$VAR "text"').tree }
          .to raise_error Gitlab::Ci::Pipeline::Expression::Parser::ParseError
      end
    end

    context 'when an operator has no left side' do
      it 'raises an OperatorError' do
        expect { described_class.seed('== "123"').tree }
            .to raise_error Gitlab::Ci::Pipeline::Expression::Lexeme::Operator::OperatorError
      end
    end

    context 'when an operator has no right side' do
      it 'raises an OperatorError' do
        expect { described_class.seed('$VAR ==').tree }
            .to raise_error Gitlab::Ci::Pipeline::Expression::Lexeme::Operator::OperatorError
      end
    end

    context 'when parenthesis are unmatched' do
      context 'when parenthesis engine is enabled' do
        before do
          stub_feature_flags(ci_if_parenthesis_enabled: true)
        end

        where(:expression) do
          [
            '$VAR == (',
            '$VAR2 == ("aa"',
            '$VAR2 == ("aa"))',
            '$VAR2 == "aa")',
            '(($VAR2 == "aa")',
            '($VAR2 == "aa"))'
          ]
        end

        with_them do
          it 'raises a ParseError' do
            expect { described_class.seed(expression).tree }
              .to raise_error Gitlab::Ci::Pipeline::Expression::Parser::ParseError
          end
        end
      end

      context 'when parenthesis engine is disabled' do
        before do
          stub_feature_flags(ci_if_parenthesis_enabled: false)
        end

        it 'raises an SyntaxError' do
          expect { described_class.seed('$VAR == (').tree }
            .to raise_error Gitlab::Ci::Pipeline::Expression::Lexer::SyntaxError
        end
      end
    end
  end
end