summaryrefslogtreecommitdiff
path: root/spec/lib/gitlab/ci/pipeline/expression/parser_spec.rb
blob: a02c247925e989e21a61f28fb926cc978190de9e (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
# frozen_string_literal: true

require 'spec_helper'

RSpec.describe Gitlab::Ci::Pipeline::Expression::Parser do
  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 }

      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 using parenthesis' do
      subject { described_class.seed('(($VAR1 == "a" || $VAR2 == "b") && $VAR3 == "c" || $VAR4 == "d") && $VAR5 == "e"').tree }

      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
      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
  end
end