cucumber-tag-expressions 5.0.6 → 6.1.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/lib/cucumber/tag_expressions/expressions.rb +10 -8
- data/lib/cucumber/tag_expressions/parser.rb +48 -63
- data/lib/cucumber/tag_expressions.rb +2 -0
- data/spec/errors_spec.rb +4 -4
- data/spec/evaluations_spec.rb +3 -3
- data/spec/parsing_spec.rb +5 -6
- data/spec/spec_helper.rb +4 -0
- metadata +23 -13
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: c0f706cda9810598cd7d38933b862a0e50e2870be4525012e64e4b0b6596b3b3
|
4
|
+
data.tar.gz: 40b6af7268af69b1af1406d567cf0e536bd00d6d28dd3a6693a29c822b4e4876
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 838196e211cb93599402f1ae4f928ca91f039a2a3ae5cdf7b1245e1ad8dc5609edf5ca1ce6f6e56d722a642c37a3afa6a7d8c89ce8a0baf3de8f2ee551631287
|
7
|
+
data.tar.gz: 3ab907243a1fa9a2ab3330829650f6fdf8769dcc532b7fddb04bc70bf8acaaa51e5fa74426840975d1bfcf15fd32084a2fcccab0f950d374dd10775457d5d665
|
@@ -1,3 +1,5 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
module Cucumber
|
2
4
|
module TagExpressions
|
3
5
|
# Literal expression node
|
@@ -12,10 +14,10 @@ module Cucumber
|
|
12
14
|
|
13
15
|
def to_s
|
14
16
|
@value
|
15
|
-
.gsub(/\\/,
|
16
|
-
.gsub(/\(/,
|
17
|
-
.gsub(/\)/,
|
18
|
-
.gsub(/\s/,
|
17
|
+
.gsub(/\\/, '\\\\\\\\')
|
18
|
+
.gsub(/\(/, '\\(')
|
19
|
+
.gsub(/\)/, '\\)')
|
20
|
+
.gsub(/\s/, '\\ ')
|
19
21
|
end
|
20
22
|
end
|
21
23
|
|
@@ -30,8 +32,8 @@ module Cucumber
|
|
30
32
|
end
|
31
33
|
|
32
34
|
def to_s
|
33
|
-
if @expression.is_a?(And)|| @expression.is_a?(Or)
|
34
|
-
# -- HINT: Binary
|
35
|
+
if @expression.is_a?(And) || @expression.is_a?(Or)
|
36
|
+
# -- HINT: Binary operations already provide "( ... )"
|
35
37
|
"not #{@expression}"
|
36
38
|
else
|
37
39
|
"not ( #{@expression} )"
|
@@ -72,12 +74,12 @@ module Cucumber
|
|
72
74
|
end
|
73
75
|
|
74
76
|
class True
|
75
|
-
def evaluate(
|
77
|
+
def evaluate(_variables)
|
76
78
|
true
|
77
79
|
end
|
78
80
|
|
79
81
|
def to_s
|
80
|
-
|
82
|
+
'true'
|
81
83
|
end
|
82
84
|
end
|
83
85
|
end
|
@@ -1,38 +1,39 @@
|
|
1
|
-
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'cucumber/tag_expressions/expressions'
|
2
4
|
|
3
5
|
module Cucumber
|
4
6
|
module TagExpressions
|
5
|
-
# Ruby tag expression parser
|
6
7
|
class Parser
|
7
8
|
def initialize
|
8
9
|
@expressions = []
|
9
10
|
@operators = []
|
10
11
|
|
11
12
|
@operator_types = {
|
12
|
-
'or'
|
13
|
+
'or' => { type: :binary_operator, precedence: 0, assoc: :left },
|
13
14
|
'and' => { type: :binary_operator, precedence: 1, assoc: :left },
|
14
15
|
'not' => { type: :unary_operator, precedence: 2, assoc: :right },
|
15
|
-
')'
|
16
|
-
'('
|
16
|
+
')' => { type: :close_paren, precedence: -1 },
|
17
|
+
'(' => { type: :open_paren, precedence: 1 }
|
17
18
|
}
|
18
19
|
end
|
19
20
|
|
20
21
|
def parse(infix_expression)
|
21
22
|
expected_token_type = :operand
|
22
|
-
|
23
23
|
tokens = tokenize(infix_expression)
|
24
24
|
return True.new if tokens.empty?
|
25
25
|
|
26
26
|
tokens.each do |token|
|
27
27
|
if @operator_types[token]
|
28
|
-
expected_token_type = send("handle_#{@operator_types
|
28
|
+
expected_token_type = send("handle_#{@operator_types.dig(token, :type)}", infix_expression, token, expected_token_type)
|
29
29
|
else
|
30
30
|
expected_token_type = handle_literal(infix_expression, token, expected_token_type)
|
31
31
|
end
|
32
32
|
end
|
33
33
|
|
34
34
|
while @operators.any?
|
35
|
-
raise %
|
35
|
+
raise %{Tag expression "#{infix_expression}" could not be parsed because of syntax error: Unmatched (.} if @operators.last == '('
|
36
|
+
|
36
37
|
push_expression(pop(@operators))
|
37
38
|
end
|
38
39
|
expression = pop(@expressions)
|
@@ -41,77 +42,60 @@ module Cucumber
|
|
41
42
|
|
42
43
|
private
|
43
44
|
|
44
|
-
############################################################################
|
45
|
-
# Helpers
|
46
|
-
#
|
47
45
|
def assoc_of(token, value)
|
48
|
-
@operator_types
|
46
|
+
@operator_types.dig(token, :assoc) == value
|
49
47
|
end
|
50
48
|
|
51
49
|
def lower_precedence?(operation)
|
52
|
-
(assoc_of(operation, :left) &&
|
53
|
-
|
54
|
-
(assoc_of(operation, :right) &&
|
55
|
-
precedence(operation) < precedence(@operators.last))
|
50
|
+
(assoc_of(operation, :left) && precedence(operation) <= precedence(@operators.last)) ||
|
51
|
+
(assoc_of(operation, :right) && precedence(operation) < precedence(@operators.last))
|
56
52
|
end
|
57
53
|
|
58
54
|
def operator?(token)
|
59
|
-
@operator_types
|
60
|
-
@operator_types[token][:type] == :binary_operator
|
55
|
+
[:unary_operator, :binary_operator].include?(@operator_types.dig(token, :type))
|
61
56
|
end
|
62
57
|
|
63
58
|
def precedence(token)
|
64
|
-
@operator_types
|
59
|
+
@operator_types.dig(token, :precedence)
|
65
60
|
end
|
66
61
|
|
67
62
|
def tokenize(infix_expression)
|
68
63
|
tokens = []
|
69
64
|
escaped = false
|
70
|
-
token =
|
71
|
-
infix_expression.chars.each do |
|
65
|
+
token = +''
|
66
|
+
infix_expression.chars.each do |char|
|
72
67
|
if escaped
|
73
|
-
|
74
|
-
|
75
|
-
escaped = false
|
76
|
-
else
|
77
|
-
raise %Q{Tag expression "#{infix_expression}" could not be parsed because of syntax error: Illegal escape before "#{ch}".}
|
68
|
+
unless char == '(' || char == ')' || char == '\\' || whitespace?(char)
|
69
|
+
raise %(Tag expression "#{infix_expression}" could not be parsed because of syntax error: Illegal escape before "#{char}".)
|
78
70
|
end
|
79
|
-
|
71
|
+
|
72
|
+
token += char
|
73
|
+
escaped = false
|
74
|
+
elsif char == '\\'
|
80
75
|
escaped = true
|
81
|
-
elsif
|
82
|
-
if token.length
|
76
|
+
elsif char == '(' || char == ')' || whitespace?(char)
|
77
|
+
if token.length.positive?
|
83
78
|
tokens.push(token)
|
84
|
-
token =
|
85
|
-
end
|
86
|
-
if !ch.match(/\s/)
|
87
|
-
tokens.push(ch)
|
79
|
+
token = +''
|
88
80
|
end
|
81
|
+
tokens.push(char) unless whitespace?(char)
|
89
82
|
else
|
90
|
-
token +=
|
83
|
+
token += char
|
91
84
|
end
|
92
85
|
end
|
93
|
-
if token.length
|
94
|
-
tokens.push(token)
|
95
|
-
end
|
86
|
+
tokens.push(token) if token.length.positive?
|
96
87
|
tokens
|
97
88
|
end
|
98
89
|
|
99
90
|
def push_expression(token)
|
100
91
|
case token
|
101
|
-
when 'and'
|
102
|
-
|
103
|
-
when '
|
104
|
-
|
105
|
-
when 'not'
|
106
|
-
@expressions.push(Not.new(pop(@expressions)))
|
107
|
-
else
|
108
|
-
@expressions.push(Literal.new(token))
|
92
|
+
when 'and'; @expressions.push(And.new(*pop(@expressions, 2)))
|
93
|
+
when 'or'; @expressions.push(Or.new(*pop(@expressions, 2)))
|
94
|
+
when 'not'; @expressions.push(Not.new(pop(@expressions)))
|
95
|
+
else @expressions.push(Literal.new(token))
|
109
96
|
end
|
110
97
|
end
|
111
98
|
|
112
|
-
############################################################################
|
113
|
-
# Handlers
|
114
|
-
#
|
115
99
|
def handle_unary_operator(infix_expression, token, expected_token_type)
|
116
100
|
check(infix_expression, expected_token_type, :operand)
|
117
101
|
@operators.push(token)
|
@@ -120,10 +104,7 @@ module Cucumber
|
|
120
104
|
|
121
105
|
def handle_binary_operator(infix_expression, token, expected_token_type)
|
122
106
|
check(infix_expression, expected_token_type, :operator)
|
123
|
-
while @operators.any? && operator?(@operators.last) &&
|
124
|
-
lower_precedence?(token)
|
125
|
-
push_expression(pop(@operators))
|
126
|
-
end
|
107
|
+
push_expression(pop(@operators)) while @operators.any? && operator?(@operators.last) && lower_precedence?(token)
|
127
108
|
@operators.push(token)
|
128
109
|
:operand
|
129
110
|
end
|
@@ -136,10 +117,9 @@ module Cucumber
|
|
136
117
|
|
137
118
|
def handle_close_paren(infix_expression, _token, expected_token_type)
|
138
119
|
check(infix_expression, expected_token_type, :operator)
|
139
|
-
while @operators.any? && @operators.last != '('
|
140
|
-
|
141
|
-
|
142
|
-
raise %Q{Tag expression "#{infix_expression}" could not be parsed because of syntax error: Unmatched ).} if @operators.empty?
|
120
|
+
push_expression(pop(@operators)) while @operators.any? && @operators.last != '('
|
121
|
+
raise %{Tag expression "#{infix_expression}" could not be parsed because of syntax error: Unmatched ).} if @operators.empty?
|
122
|
+
|
143
123
|
pop(@operators) if @operators.last == '('
|
144
124
|
:operator
|
145
125
|
end
|
@@ -151,15 +131,20 @@ module Cucumber
|
|
151
131
|
end
|
152
132
|
|
153
133
|
def check(infix_expression, expected_token_type, token_type)
|
154
|
-
if expected_token_type
|
155
|
-
|
156
|
-
|
134
|
+
return if expected_token_type == token_type
|
135
|
+
|
136
|
+
raise %(Tag expression "#{infix_expression}" could not be parsed because of syntax error: Expected #{expected_token_type}.)
|
137
|
+
end
|
138
|
+
|
139
|
+
def pop(array, amount = 1)
|
140
|
+
result = array.pop(amount)
|
141
|
+
raise('Empty stack') if result.length != amount
|
142
|
+
|
143
|
+
amount == 1 ? result.first : result
|
157
144
|
end
|
158
145
|
|
159
|
-
def
|
160
|
-
|
161
|
-
raise('Empty stack') if result.size != n
|
162
|
-
n == 1 ? result.first : result
|
146
|
+
def whitespace?(char)
|
147
|
+
char.match(/\s/)
|
163
148
|
end
|
164
149
|
end
|
165
150
|
end
|
data/spec/errors_spec.rb
CHANGED
@@ -1,12 +1,12 @@
|
|
1
|
-
|
2
|
-
require 'yaml'
|
1
|
+
# frozen_string_literal: true
|
3
2
|
|
4
3
|
tests = YAML.load_file('../testdata/errors.yml')
|
5
4
|
|
6
5
|
describe 'Errors' do
|
7
6
|
tests.each do |test|
|
8
|
-
|
9
|
-
|
7
|
+
let(:parser) { Cucumber::TagExpressions::Parser.new }
|
8
|
+
|
9
|
+
it "fails to parse '#{test['expression']}' with '#{test['error']}'" do
|
10
10
|
expect { parser.parse(test['expression']) }.to raise_error(test['error'])
|
11
11
|
end
|
12
12
|
end
|
data/spec/evaluations_spec.rb
CHANGED
@@ -1,14 +1,14 @@
|
|
1
|
-
|
2
|
-
require 'yaml'
|
1
|
+
# frozen_string_literal: true
|
3
2
|
|
4
3
|
evaluations = YAML.load_file('../testdata/evaluations.yml')
|
5
4
|
|
6
5
|
describe 'Evaluations' do
|
7
6
|
evaluations.each do |evaluation|
|
8
7
|
context evaluation['expression'] do
|
8
|
+
let(:parser) { Cucumber::TagExpressions::Parser.new }
|
9
|
+
|
9
10
|
evaluation['tests'].each do |test|
|
10
11
|
it "evaluates [#{test['variables'].join(', ')}] to #{test['result']}" do
|
11
|
-
parser = Cucumber::TagExpressions::Parser.new
|
12
12
|
expect(parser.parse(evaluation['expression']).evaluate(test['variables'])).to eq(test['result'])
|
13
13
|
end
|
14
14
|
end
|
data/spec/parsing_spec.rb
CHANGED
@@ -1,14 +1,13 @@
|
|
1
|
-
|
2
|
-
require 'yaml'
|
1
|
+
# frozen_string_literal: true
|
3
2
|
|
4
3
|
tests = YAML.load_file('../testdata/parsing.yml')
|
5
4
|
|
6
5
|
describe 'Parsing' do
|
6
|
+
let(:parser) { Cucumber::TagExpressions::Parser.new }
|
7
|
+
|
7
8
|
tests.each do |test|
|
8
|
-
it
|
9
|
-
parser
|
10
|
-
expression = parser.parse(test['expression'])
|
11
|
-
expect(expression.to_s).to eq(test['formatted'])
|
9
|
+
it "parses '#{test['expression']}' into '#{test['formatted']}'" do
|
10
|
+
expect(parser.parse(test['expression']).to_s).to eq(test['formatted'])
|
12
11
|
end
|
13
12
|
end
|
14
13
|
end
|
data/spec/spec_helper.rb
ADDED
metadata
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: cucumber-tag-expressions
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version:
|
4
|
+
version: 6.1.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Andrea Nodari
|
@@ -9,7 +9,7 @@ authors:
|
|
9
9
|
autorequire:
|
10
10
|
bindir: bin
|
11
11
|
cert_chain: []
|
12
|
-
date:
|
12
|
+
date: 2024-01-10 00:00:00.000000000 Z
|
13
13
|
dependencies:
|
14
14
|
- !ruby/object:Gem::Dependency
|
15
15
|
name: rake
|
@@ -37,20 +37,28 @@ dependencies:
|
|
37
37
|
requirements:
|
38
38
|
- - "~>"
|
39
39
|
- !ruby/object:Gem::Version
|
40
|
-
version: '3.
|
41
|
-
- - ">="
|
42
|
-
- !ruby/object:Gem::Version
|
43
|
-
version: 3.10.0
|
40
|
+
version: '3.11'
|
44
41
|
type: :development
|
45
42
|
prerelease: false
|
46
43
|
version_requirements: !ruby/object:Gem::Requirement
|
47
44
|
requirements:
|
48
45
|
- - "~>"
|
49
46
|
- !ruby/object:Gem::Version
|
50
|
-
version: '3.
|
51
|
-
|
47
|
+
version: '3.11'
|
48
|
+
- !ruby/object:Gem::Dependency
|
49
|
+
name: rubocop
|
50
|
+
requirement: !ruby/object:Gem::Requirement
|
51
|
+
requirements:
|
52
|
+
- - "~>"
|
53
|
+
- !ruby/object:Gem::Version
|
54
|
+
version: 0.79.0
|
55
|
+
type: :development
|
56
|
+
prerelease: false
|
57
|
+
version_requirements: !ruby/object:Gem::Requirement
|
58
|
+
requirements:
|
59
|
+
- - "~>"
|
52
60
|
- !ruby/object:Gem::Version
|
53
|
-
version:
|
61
|
+
version: 0.79.0
|
54
62
|
description: Cucumber tag expressions for ruby
|
55
63
|
email: cukes@googlegroups.com
|
56
64
|
executables: []
|
@@ -64,6 +72,7 @@ files:
|
|
64
72
|
- spec/errors_spec.rb
|
65
73
|
- spec/evaluations_spec.rb
|
66
74
|
- spec/parsing_spec.rb
|
75
|
+
- spec/spec_helper.rb
|
67
76
|
homepage: https://cucumber.io/docs/cucumber/api/#tag-expressions
|
68
77
|
licenses:
|
69
78
|
- MIT
|
@@ -82,18 +91,19 @@ required_ruby_version: !ruby/object:Gem::Requirement
|
|
82
91
|
requirements:
|
83
92
|
- - ">="
|
84
93
|
- !ruby/object:Gem::Version
|
85
|
-
version:
|
94
|
+
version: '2.3'
|
86
95
|
required_rubygems_version: !ruby/object:Gem::Requirement
|
87
96
|
requirements:
|
88
97
|
- - ">="
|
89
98
|
- !ruby/object:Gem::Version
|
90
|
-
version:
|
99
|
+
version: 3.0.8
|
91
100
|
requirements: []
|
92
|
-
rubygems_version: 3.
|
101
|
+
rubygems_version: 3.4.10
|
93
102
|
signing_key:
|
94
103
|
specification_version: 4
|
95
|
-
summary: cucumber-tag-expressions-
|
104
|
+
summary: cucumber-tag-expressions-6.1.0
|
96
105
|
test_files:
|
97
106
|
- spec/errors_spec.rb
|
98
107
|
- spec/evaluations_spec.rb
|
99
108
|
- spec/parsing_spec.rb
|
109
|
+
- spec/spec_helper.rb
|