gherkin 2.2.5-x86-mswin32
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/.gitattributes +2 -0
- data/.gitignore +11 -0
- data/.mailmap +2 -0
- data/.rspec +1 -0
- data/.rvmrc +1 -0
- data/Gemfile +5 -0
- data/History.txt +306 -0
- data/LICENSE +20 -0
- data/README.rdoc +59 -0
- data/Rakefile +16 -0
- data/VERSION +1 -0
- data/bin/gherkin +5 -0
- data/build_native_gems.sh +8 -0
- data/cucumber.yml +3 -0
- data/features/escaped_pipes.feature +8 -0
- data/features/feature_parser.feature +237 -0
- data/features/json_formatter.feature +278 -0
- data/features/json_parser.feature +318 -0
- data/features/native_lexer.feature +19 -0
- data/features/parser_with_native_lexer.feature +205 -0
- data/features/pretty_formatter.feature +15 -0
- data/features/step_definitions/eyeball_steps.rb +3 -0
- data/features/step_definitions/gherkin_steps.rb +29 -0
- data/features/step_definitions/json_formatter_steps.rb +28 -0
- data/features/step_definitions/json_parser_steps.rb +20 -0
- data/features/step_definitions/pretty_formatter_steps.rb +82 -0
- data/features/steps_parser.feature +46 -0
- data/features/support/env.rb +38 -0
- data/gherkin.gemspec +59 -0
- data/ikvm/.gitignore +3 -0
- data/java/.gitignore +2 -0
- data/java/src/main/java/gherkin/lexer/i18n/.gitignore +1 -0
- data/java/src/main/resources/gherkin/.gitignore +1 -0
- data/lib/.gitignore +4 -0
- data/lib/gherkin.rb +2 -0
- data/lib/gherkin/c_lexer.rb +17 -0
- data/lib/gherkin/cli/main.rb +33 -0
- data/lib/gherkin/formatter/argument.rb +28 -0
- data/lib/gherkin/formatter/colors.rb +119 -0
- data/lib/gherkin/formatter/escaping.rb +15 -0
- data/lib/gherkin/formatter/filter_formatter.rb +136 -0
- data/lib/gherkin/formatter/json_formatter.rb +72 -0
- data/lib/gherkin/formatter/line_filter.rb +26 -0
- data/lib/gherkin/formatter/model.rb +231 -0
- data/lib/gherkin/formatter/monochrome_format.rb +9 -0
- data/lib/gherkin/formatter/pretty_formatter.rb +174 -0
- data/lib/gherkin/formatter/regexp_filter.rb +21 -0
- data/lib/gherkin/formatter/tag_count_formatter.rb +47 -0
- data/lib/gherkin/formatter/tag_filter.rb +19 -0
- data/lib/gherkin/i18n.rb +180 -0
- data/lib/gherkin/i18n.yml +601 -0
- data/lib/gherkin/json_parser.rb +88 -0
- data/lib/gherkin/lexer/i18n_lexer.rb +47 -0
- data/lib/gherkin/listener/event.rb +45 -0
- data/lib/gherkin/listener/formatter_listener.rb +113 -0
- data/lib/gherkin/native.rb +7 -0
- data/lib/gherkin/native/ikvm.rb +55 -0
- data/lib/gherkin/native/java.rb +55 -0
- data/lib/gherkin/native/null.rb +9 -0
- data/lib/gherkin/parser/meta.txt +5 -0
- data/lib/gherkin/parser/parser.rb +164 -0
- data/lib/gherkin/parser/root.txt +11 -0
- data/lib/gherkin/parser/steps.txt +4 -0
- data/lib/gherkin/rb_lexer.rb +8 -0
- data/lib/gherkin/rb_lexer/.gitignore +1 -0
- data/lib/gherkin/rb_lexer/README.rdoc +8 -0
- data/lib/gherkin/rubify.rb +24 -0
- data/lib/gherkin/tag_expression.rb +62 -0
- data/lib/gherkin/tools.rb +8 -0
- data/lib/gherkin/tools/files.rb +34 -0
- data/lib/gherkin/tools/reformat.rb +20 -0
- data/lib/gherkin/tools/stats.rb +20 -0
- data/lib/gherkin/tools/stats_listener.rb +60 -0
- data/lib/gherkin/version.rb +3 -0
- data/ragel/i18n/.gitignore +1 -0
- data/ragel/lexer.c.rl.erb +459 -0
- data/ragel/lexer.java.rl.erb +224 -0
- data/ragel/lexer.rb.rl.erb +179 -0
- data/ragel/lexer_common.rl.erb +50 -0
- data/spec/gherkin/c_lexer_spec.rb +21 -0
- data/spec/gherkin/fixtures/1.feature +8 -0
- data/spec/gherkin/fixtures/comments_in_table.feature +9 -0
- data/spec/gherkin/fixtures/complex.feature +45 -0
- data/spec/gherkin/fixtures/complex.json +143 -0
- data/spec/gherkin/fixtures/complex_for_filtering.feature +60 -0
- data/spec/gherkin/fixtures/complex_with_tags.feature +61 -0
- data/spec/gherkin/fixtures/dos_line_endings.feature +45 -0
- data/spec/gherkin/fixtures/hantu_pisang.feature +35 -0
- data/spec/gherkin/fixtures/i18n_fr.feature +14 -0
- data/spec/gherkin/fixtures/i18n_no.feature +7 -0
- data/spec/gherkin/fixtures/i18n_zh-CN.feature +9 -0
- data/spec/gherkin/fixtures/scenario_outline_with_tags.feature +13 -0
- data/spec/gherkin/fixtures/scenario_without_steps.feature +5 -0
- data/spec/gherkin/fixtures/simple_with_comments.feature +7 -0
- data/spec/gherkin/fixtures/simple_with_tags.feature +11 -0
- data/spec/gherkin/fixtures/with_bom.feature +3 -0
- data/spec/gherkin/formatter/argument_spec.rb +28 -0
- data/spec/gherkin/formatter/colors_spec.rb +18 -0
- data/spec/gherkin/formatter/filter_formatter_spec.rb +165 -0
- data/spec/gherkin/formatter/model_spec.rb +15 -0
- data/spec/gherkin/formatter/pretty_formatter_spec.rb +140 -0
- data/spec/gherkin/formatter/spaces.feature +9 -0
- data/spec/gherkin/formatter/tabs.feature +9 -0
- data/spec/gherkin/formatter/tag_count_formatter_spec.rb +30 -0
- data/spec/gherkin/i18n_spec.rb +149 -0
- data/spec/gherkin/java_lexer_spec.rb +20 -0
- data/spec/gherkin/json.rb +5 -0
- data/spec/gherkin/json_parser_spec.rb +67 -0
- data/spec/gherkin/lexer/i18n_lexer_spec.rb +43 -0
- data/spec/gherkin/output_stream_string_io.rb +24 -0
- data/spec/gherkin/parser/parser_spec.rb +16 -0
- data/spec/gherkin/rb_lexer_spec.rb +19 -0
- data/spec/gherkin/sexp_recorder.rb +56 -0
- data/spec/gherkin/shared/lexer_group.rb +592 -0
- data/spec/gherkin/shared/py_string_group.rb +153 -0
- data/spec/gherkin/shared/row_group.rb +120 -0
- data/spec/gherkin/shared/tags_group.rb +54 -0
- data/spec/gherkin/tag_expression_spec.rb +137 -0
- data/spec/spec_helper.rb +68 -0
- data/tasks/bench.rake +184 -0
- data/tasks/bench/feature_builder.rb +49 -0
- data/tasks/bench/generated/.gitignore +1 -0
- data/tasks/bench/null_listener.rb +4 -0
- data/tasks/compile.rake +102 -0
- data/tasks/cucumber.rake +18 -0
- data/tasks/gems.rake +42 -0
- data/tasks/ikvm.rake +54 -0
- data/tasks/ragel_task.rb +70 -0
- data/tasks/rdoc.rake +9 -0
- data/tasks/release.rake +30 -0
- data/tasks/rspec.rake +8 -0
- metadata +447 -0
@@ -0,0 +1,88 @@
|
|
1
|
+
require 'json'
|
2
|
+
require 'gherkin/formatter/model'
|
3
|
+
require 'gherkin/native'
|
4
|
+
|
5
|
+
module Gherkin
|
6
|
+
class JSONParser
|
7
|
+
native_impl('gherkin')
|
8
|
+
|
9
|
+
def initialize(formatter)
|
10
|
+
@formatter = formatter
|
11
|
+
end
|
12
|
+
|
13
|
+
# Parse a gherkin object +o+, which can either be a JSON String,
|
14
|
+
# or a Hash (from a parsed JSON String).
|
15
|
+
def parse(o, feature_uri='unknown.json', line_offset=0)
|
16
|
+
o = JSON.parse(o) if String === o
|
17
|
+
@formatter.uri(feature_uri)
|
18
|
+
|
19
|
+
Formatter::Model::Feature.new(comments(o), tags(o), keyword(o), name(o), description(o), line(o)).replay(@formatter)
|
20
|
+
(o["elements"] || []).each do |feature_element|
|
21
|
+
feature_element(feature_element).replay(@formatter)
|
22
|
+
(feature_element["steps"] || []).each do |step|
|
23
|
+
step(step).replay(@formatter)
|
24
|
+
end
|
25
|
+
(feature_element["examples"] || []).each do |eo|
|
26
|
+
Formatter::Model::Examples.new(comments(eo), tags(eo), keyword(eo), name(eo), description(eo), line(eo), rows(eo['rows'])).replay(@formatter)
|
27
|
+
end
|
28
|
+
end
|
29
|
+
|
30
|
+
@formatter.eof
|
31
|
+
end
|
32
|
+
|
33
|
+
def feature_element(o)
|
34
|
+
case o['type']
|
35
|
+
when 'background'
|
36
|
+
Formatter::Model::Background.new(comments(o), keyword(o), name(o), description(o), line(o))
|
37
|
+
when 'scenario'
|
38
|
+
Formatter::Model::Scenario.new(comments(o), tags(o), keyword(o), name(o), description(o), line(o))
|
39
|
+
when 'scenario_outline'
|
40
|
+
Formatter::Model::ScenarioOutline.new(comments(o), tags(o), keyword(o), name(o), description(o), line(o))
|
41
|
+
end
|
42
|
+
end
|
43
|
+
|
44
|
+
def step(o)
|
45
|
+
multiline_arg = nil
|
46
|
+
if(ma = o['multiline_arg'])
|
47
|
+
if(ma['type'] == 'table')
|
48
|
+
multiline_arg = rows(ma['value'])
|
49
|
+
else
|
50
|
+
multiline_arg = Formatter::Model::PyString.new(ma['value'], ma['line'])
|
51
|
+
end
|
52
|
+
end
|
53
|
+
Formatter::Model::Step.new(comments(o), keyword(o), name(o), line(o), multiline_arg)
|
54
|
+
end
|
55
|
+
|
56
|
+
def rows(o)
|
57
|
+
o.map{|row| Formatter::Model::Row.new(comments(row), row['cells'], row['line'])}
|
58
|
+
end
|
59
|
+
|
60
|
+
def comments(o)
|
61
|
+
(o['comments'] || []).map do |comment|
|
62
|
+
Formatter::Model::Comment.new(comment['value'], comment['line'])
|
63
|
+
end
|
64
|
+
end
|
65
|
+
|
66
|
+
def tags(o)
|
67
|
+
(o['tags'] || []).map do |tag|
|
68
|
+
Formatter::Model::Tag.new(tag['name'], tag['line'])
|
69
|
+
end
|
70
|
+
end
|
71
|
+
|
72
|
+
def keyword(o)
|
73
|
+
o['keyword']
|
74
|
+
end
|
75
|
+
|
76
|
+
def name(o)
|
77
|
+
o['name']
|
78
|
+
end
|
79
|
+
|
80
|
+
def description(o)
|
81
|
+
o['description']
|
82
|
+
end
|
83
|
+
|
84
|
+
def line(o)
|
85
|
+
o['line']
|
86
|
+
end
|
87
|
+
end
|
88
|
+
end
|
@@ -0,0 +1,47 @@
|
|
1
|
+
require 'gherkin/i18n'
|
2
|
+
require 'gherkin/native'
|
3
|
+
|
4
|
+
module Gherkin
|
5
|
+
module Lexer
|
6
|
+
I18nLexerNotFound = Class.new(LoadError)
|
7
|
+
LexingError = Class.new(StandardError)
|
8
|
+
|
9
|
+
# The main entry point to lexing Gherkin source.
|
10
|
+
class I18nLexer
|
11
|
+
native_impl('gherkin')
|
12
|
+
|
13
|
+
COMMENT_OR_EMPTY_LINE_PATTERN = /^\s*#|^\s*$/
|
14
|
+
LANGUAGE_PATTERN = /^\s*#\s*language\s*:\s*([a-zA-Z\-]+)/ #:nodoc:
|
15
|
+
attr_reader :i18n_language
|
16
|
+
|
17
|
+
def initialize(listener, force_ruby=false)
|
18
|
+
@listener = listener
|
19
|
+
@force_ruby = force_ruby
|
20
|
+
end
|
21
|
+
|
22
|
+
def scan(source)
|
23
|
+
create_delegate(source).scan(source)
|
24
|
+
end
|
25
|
+
|
26
|
+
private
|
27
|
+
|
28
|
+
def create_delegate(source)
|
29
|
+
@i18n_language = lang(source)
|
30
|
+
@i18n_language.lexer(@listener, @force_ruby)
|
31
|
+
end
|
32
|
+
|
33
|
+
def lang(source)
|
34
|
+
key = 'en'
|
35
|
+
source.split(/\n/).each do |line|
|
36
|
+
break unless COMMENT_OR_EMPTY_LINE_PATTERN =~ line
|
37
|
+
if LANGUAGE_PATTERN =~ line
|
38
|
+
key = $1
|
39
|
+
break
|
40
|
+
end
|
41
|
+
end
|
42
|
+
I18n.get(key)
|
43
|
+
end
|
44
|
+
|
45
|
+
end
|
46
|
+
end
|
47
|
+
end
|
@@ -0,0 +1,45 @@
|
|
1
|
+
module Gherkin
|
2
|
+
module Listener
|
3
|
+
class Event < Array
|
4
|
+
def event
|
5
|
+
self[0]
|
6
|
+
end
|
7
|
+
|
8
|
+
def keyword
|
9
|
+
self[1]
|
10
|
+
end
|
11
|
+
|
12
|
+
def line_match?(lines)
|
13
|
+
lines.include?(line)
|
14
|
+
end
|
15
|
+
|
16
|
+
def name_match?(name_regexen)
|
17
|
+
return false unless [:feature, :background, :scenario, :scenario_outline, :examples].include?(event)
|
18
|
+
name_regexen.detect{|name_regex| name =~ name_regex}
|
19
|
+
end
|
20
|
+
|
21
|
+
def replay(listener)
|
22
|
+
begin
|
23
|
+
listener.__send__(event, *args)
|
24
|
+
rescue ArgumentError => e
|
25
|
+
e.message << "\nListener: #{listener.class}, args: #{args.inspect}"
|
26
|
+
raise e
|
27
|
+
end
|
28
|
+
end
|
29
|
+
|
30
|
+
private
|
31
|
+
|
32
|
+
def name
|
33
|
+
self[2]
|
34
|
+
end
|
35
|
+
|
36
|
+
def line
|
37
|
+
self[-1]
|
38
|
+
end
|
39
|
+
|
40
|
+
def args
|
41
|
+
self[1..-1]
|
42
|
+
end
|
43
|
+
end
|
44
|
+
end
|
45
|
+
end
|
@@ -0,0 +1,113 @@
|
|
1
|
+
require 'gherkin/native'
|
2
|
+
require 'gherkin/formatter/model'
|
3
|
+
|
4
|
+
module Gherkin
|
5
|
+
module Listener
|
6
|
+
# Adapter from the "raw" Gherkin <tt>Listener</tt> API
|
7
|
+
# to the slightly more high-level <tt>Formatter</tt> API,
|
8
|
+
# which is easier to implement (less state to keep track of).
|
9
|
+
class FormatterListener
|
10
|
+
native_impl('gherkin')
|
11
|
+
|
12
|
+
def initialize(formatter)
|
13
|
+
@formatter = formatter
|
14
|
+
@comments = []
|
15
|
+
@tags = []
|
16
|
+
@table = nil
|
17
|
+
end
|
18
|
+
|
19
|
+
def comment(value, line)
|
20
|
+
@comments << Formatter::Model::Comment.new(value, line)
|
21
|
+
end
|
22
|
+
|
23
|
+
def tag(name, line)
|
24
|
+
@tags << Formatter::Model::Tag.new(name, line)
|
25
|
+
end
|
26
|
+
|
27
|
+
def feature(keyword, name, description, line)
|
28
|
+
@formatter.feature(Formatter::Model::Feature.new(grab_comments!, grab_tags!, keyword, name, description, line))
|
29
|
+
end
|
30
|
+
|
31
|
+
def background(keyword, name, description, line)
|
32
|
+
@formatter.background(Formatter::Model::Background.new(grab_comments!, keyword, name, description, line))
|
33
|
+
end
|
34
|
+
|
35
|
+
def scenario(keyword, name, description, line)
|
36
|
+
replay_step_or_examples
|
37
|
+
@formatter.scenario(Formatter::Model::Scenario.new(grab_comments!, grab_tags!, keyword, name, description, line))
|
38
|
+
end
|
39
|
+
|
40
|
+
def scenario_outline(keyword, name, description, line)
|
41
|
+
replay_step_or_examples
|
42
|
+
@formatter.scenario_outline(Formatter::Model::ScenarioOutline.new(grab_comments!, grab_tags!, keyword, name, description, line))
|
43
|
+
end
|
44
|
+
|
45
|
+
def examples(keyword, name, description, line)
|
46
|
+
replay_step_or_examples
|
47
|
+
@examples_statement = Formatter::Model::Examples.new(grab_comments!, grab_tags!, keyword, name, description, line)
|
48
|
+
end
|
49
|
+
|
50
|
+
def step(keyword, name, line)
|
51
|
+
replay_step_or_examples
|
52
|
+
@step_statement = Formatter::Model::Step.new(grab_comments!, keyword, name, line)
|
53
|
+
end
|
54
|
+
|
55
|
+
def row(cells, line)
|
56
|
+
@table ||= []
|
57
|
+
@table << Formatter::Model::Row.new(grab_comments!, cells, line)
|
58
|
+
end
|
59
|
+
|
60
|
+
def py_string(string, line)
|
61
|
+
@py_string = Formatter::Model::PyString.new(string, line)
|
62
|
+
end
|
63
|
+
|
64
|
+
def eof
|
65
|
+
replay_step_or_examples
|
66
|
+
@formatter.eof
|
67
|
+
end
|
68
|
+
|
69
|
+
def syntax_error(state, ev, legal_events, uri, line)
|
70
|
+
@formatter.syntax_error(state, ev, legal_events, uri, line)
|
71
|
+
end
|
72
|
+
|
73
|
+
private
|
74
|
+
|
75
|
+
def grab_comments!
|
76
|
+
comments = @comments
|
77
|
+
@comments = []
|
78
|
+
comments
|
79
|
+
end
|
80
|
+
|
81
|
+
def grab_tags!
|
82
|
+
tags = @tags
|
83
|
+
@tags = []
|
84
|
+
tags
|
85
|
+
end
|
86
|
+
|
87
|
+
def grab_rows!
|
88
|
+
table = @table
|
89
|
+
@table = nil
|
90
|
+
table
|
91
|
+
end
|
92
|
+
|
93
|
+
def grab_py_string!
|
94
|
+
py_string = @py_string
|
95
|
+
@py_string = nil
|
96
|
+
py_string
|
97
|
+
end
|
98
|
+
|
99
|
+
def replay_step_or_examples
|
100
|
+
if(@step_statement)
|
101
|
+
@step_statement.multiline_arg = grab_py_string! || grab_rows!
|
102
|
+
@formatter.step(@step_statement)
|
103
|
+
@step_statement = nil
|
104
|
+
end
|
105
|
+
if(@examples_statement)
|
106
|
+
@examples_statement.rows = grab_rows!
|
107
|
+
@formatter.examples(@examples_statement)
|
108
|
+
@examples_statement = nil
|
109
|
+
end
|
110
|
+
end
|
111
|
+
end
|
112
|
+
end
|
113
|
+
end
|
@@ -0,0 +1,55 @@
|
|
1
|
+
class Class
|
2
|
+
|
3
|
+
def implements(java_class_name)
|
4
|
+
m = java_class_name.split('.').inject(Object) do |mod, name|
|
5
|
+
mod = mod.const_get(name)
|
6
|
+
end
|
7
|
+
include m
|
8
|
+
end
|
9
|
+
|
10
|
+
# Causes a .NET class to be instantiated instead of the Ruby class when
|
11
|
+
# running on IronRuby. This is used to test both pure .NET and pure Ruby classes
|
12
|
+
# from the same Ruby based test suite. The .NET Class must have a package name
|
13
|
+
# that corresponds with the Ruby class.
|
14
|
+
def native_impl(lib)
|
15
|
+
begin
|
16
|
+
load_assembly(lib)
|
17
|
+
rescue LoadError => e
|
18
|
+
e.message << "\nTry this: SET MONO_PATH=#{File.expand_path(File.dirname(__FILE__) + '/../..')} (or export MONO_PATH=...)"
|
19
|
+
raise e
|
20
|
+
end
|
21
|
+
|
22
|
+
class << self
|
23
|
+
def ikvmify(arg)
|
24
|
+
if Array === arg
|
25
|
+
arg.map{|a| ikvmify(a)}
|
26
|
+
else
|
27
|
+
case(arg)
|
28
|
+
when Regexp
|
29
|
+
Object.const_get('java').const_get('util').const_get('regex').const_get('Pattern').compile(arg.source)
|
30
|
+
else
|
31
|
+
arg
|
32
|
+
end
|
33
|
+
end
|
34
|
+
end
|
35
|
+
|
36
|
+
def new(*args)
|
37
|
+
ikvm_class.new(*ikvmify(args))
|
38
|
+
end
|
39
|
+
|
40
|
+
def ===(object)
|
41
|
+
super || object.java_kind_of?(java_class)
|
42
|
+
end
|
43
|
+
|
44
|
+
def ikvm_class
|
45
|
+
names = self.name.split('::')
|
46
|
+
namespace = Object
|
47
|
+
names[0..-2].each do |module_name|
|
48
|
+
namespace = namespace.const_get(module_name.downcase)
|
49
|
+
end
|
50
|
+
|
51
|
+
namespace.const_get(names[-1])
|
52
|
+
end
|
53
|
+
end
|
54
|
+
end
|
55
|
+
end
|
@@ -0,0 +1,55 @@
|
|
1
|
+
class Class
|
2
|
+
|
3
|
+
def implements(java_class_name)
|
4
|
+
# no-op
|
5
|
+
end
|
6
|
+
|
7
|
+
# Causes a Java class to be instantiated instead of the Ruby class when
|
8
|
+
# running on JRuby. This is used to test both pure Java and pure Ruby classes
|
9
|
+
# from the same Ruby based test suite. The Java Class must have a package name
|
10
|
+
# that corresponds with the Ruby class.
|
11
|
+
def native_impl(lib)
|
12
|
+
require "#{lib}.jar"
|
13
|
+
|
14
|
+
class << self
|
15
|
+
def javaify(arg)
|
16
|
+
if Array === arg
|
17
|
+
arg.map{|a| javaify(a)}
|
18
|
+
else
|
19
|
+
case(arg)
|
20
|
+
when Regexp
|
21
|
+
java.util.regex.Pattern.compile(arg.source)
|
22
|
+
else
|
23
|
+
arg
|
24
|
+
end
|
25
|
+
end
|
26
|
+
end
|
27
|
+
|
28
|
+
def new(*args)
|
29
|
+
begin
|
30
|
+
java_class.new(*javaify(args))
|
31
|
+
rescue ArgumentError => e
|
32
|
+
e.message << "\n#{java_class.name}"
|
33
|
+
raise e
|
34
|
+
rescue NameError => e
|
35
|
+
e.message << "\n args: #{args.inspect}"
|
36
|
+
raise e
|
37
|
+
end
|
38
|
+
end
|
39
|
+
|
40
|
+
def ===(object)
|
41
|
+
super || object.java_kind_of?(java_class)
|
42
|
+
end
|
43
|
+
|
44
|
+
def java_class
|
45
|
+
names = self.name.split('::')
|
46
|
+
package = Java
|
47
|
+
names[0..-2].each do |module_name|
|
48
|
+
package = package.__send__(module_name.downcase)
|
49
|
+
end
|
50
|
+
|
51
|
+
package.__send__(names[-1])
|
52
|
+
end
|
53
|
+
end
|
54
|
+
end
|
55
|
+
end
|
@@ -0,0 +1,5 @@
|
|
1
|
+
| | feature | background | scenario | scenario_outline | examples | step | row | py_string | eof | comment | tag |
|
2
|
+
| meta | E | E | E | E | E | E | E | E | eof | comment | tag |
|
3
|
+
| comment | pop() | pop() | pop() | pop() | pop() | pop() | pop() | pop() | eof | pop() | tag |
|
4
|
+
| tag | pop() | E | pop() | pop() | pop() | E | E | E | E | E | tag |
|
5
|
+
| eof | E | E | E | E | E | E | E | E | E | E | E |
|
@@ -0,0 +1,164 @@
|
|
1
|
+
require 'gherkin/i18n'
|
2
|
+
require 'gherkin/lexer/i18n_lexer'
|
3
|
+
require 'gherkin/native'
|
4
|
+
require 'gherkin/listener/formatter_listener'
|
5
|
+
|
6
|
+
module Gherkin
|
7
|
+
module Parser
|
8
|
+
class ParseError < StandardError
|
9
|
+
def initialize(state, new_state, expected_states, uri, line)
|
10
|
+
super("Parse error at #{uri}:#{line}. Found #{new_state} when expecting one of: #{expected_states.join(', ')}. (Current state: #{state}).")
|
11
|
+
end
|
12
|
+
end
|
13
|
+
|
14
|
+
class Parser
|
15
|
+
native_impl('gherkin')
|
16
|
+
|
17
|
+
# Initialize the parser. +machine_name+ refers to a state machine table.
|
18
|
+
def initialize(formatter, raise_on_error=true, machine_name='root', force_ruby=false)
|
19
|
+
@formatter = formatter
|
20
|
+
@listener = Listener::FormatterListener.new(@formatter)
|
21
|
+
@raise_on_error = raise_on_error
|
22
|
+
@machine_name = machine_name
|
23
|
+
@machines = []
|
24
|
+
push_machine(@machine_name)
|
25
|
+
@lexer = Gherkin::Lexer::I18nLexer.new(self, force_ruby)
|
26
|
+
end
|
27
|
+
|
28
|
+
def parse(gherkin, feature_uri, line_offset)
|
29
|
+
@formatter.uri(feature_uri)
|
30
|
+
@line_offset = line_offset
|
31
|
+
@lexer.scan(gherkin)
|
32
|
+
end
|
33
|
+
|
34
|
+
def i18n_language
|
35
|
+
@lexer.i18n_language
|
36
|
+
end
|
37
|
+
|
38
|
+
def errors
|
39
|
+
@lexer.errors
|
40
|
+
end
|
41
|
+
|
42
|
+
# Doesn't yet fall back to super
|
43
|
+
def method_missing(method, *args)
|
44
|
+
# TODO: Catch exception and call super
|
45
|
+
event(method.to_s, args[-1])
|
46
|
+
@listener.__send__(method, *args)
|
47
|
+
if method == :eof
|
48
|
+
pop_machine
|
49
|
+
push_machine(@machine_name)
|
50
|
+
end
|
51
|
+
end
|
52
|
+
|
53
|
+
def event(ev, line)
|
54
|
+
l = line ? @line_offset+line : nil
|
55
|
+
machine.event(ev, l) do |state, legal_events|
|
56
|
+
if @raise_on_error
|
57
|
+
raise ParseError.new(state, ev, legal_events, @feature_uri, l)
|
58
|
+
else
|
59
|
+
# Only used for testing
|
60
|
+
@listener.syntax_error(state, ev, legal_events, @feature_uri, l)
|
61
|
+
end
|
62
|
+
end
|
63
|
+
end
|
64
|
+
|
65
|
+
def push_machine(name)
|
66
|
+
@machines.push(Machine.new(self, name))
|
67
|
+
end
|
68
|
+
|
69
|
+
def pop_machine
|
70
|
+
@machines.pop
|
71
|
+
end
|
72
|
+
|
73
|
+
def machine
|
74
|
+
@machines[-1]
|
75
|
+
end
|
76
|
+
|
77
|
+
def expected
|
78
|
+
machine.expected
|
79
|
+
end
|
80
|
+
|
81
|
+
def force_state(state)
|
82
|
+
machine.instance_variable_set('@state', state)
|
83
|
+
end
|
84
|
+
|
85
|
+
class Machine
|
86
|
+
def initialize(parser, name)
|
87
|
+
@parser = parser
|
88
|
+
@name = name
|
89
|
+
@transition_map = transition_map(name)
|
90
|
+
@state = name
|
91
|
+
end
|
92
|
+
|
93
|
+
def event(ev, line)
|
94
|
+
states = @transition_map[@state]
|
95
|
+
raise "Unknown state: #{@state.inspect} for machine #{@name}" if states.nil?
|
96
|
+
new_state = states[ev]
|
97
|
+
case new_state
|
98
|
+
when "E"
|
99
|
+
yield @state, expected
|
100
|
+
when /push\((.+)\)/
|
101
|
+
@parser.push_machine($1)
|
102
|
+
@parser.event(ev, line)
|
103
|
+
when "pop()"
|
104
|
+
@parser.pop_machine()
|
105
|
+
@parser.event(ev, line)
|
106
|
+
else
|
107
|
+
raise "Unknown transition: #{ev.inspect} among #{states.inspect} for machine #{@name}" if new_state.nil?
|
108
|
+
@state = new_state
|
109
|
+
end
|
110
|
+
end
|
111
|
+
|
112
|
+
def expected
|
113
|
+
allowed = @transition_map[@state].find_all { |_, action| action != "E" }
|
114
|
+
allowed.collect { |state| state[0] }.sort - ['eof']
|
115
|
+
end
|
116
|
+
|
117
|
+
private
|
118
|
+
|
119
|
+
@@transition_maps = {}
|
120
|
+
|
121
|
+
def transition_map(name)
|
122
|
+
@@transition_maps[name] ||= build_transition_map(name)
|
123
|
+
end
|
124
|
+
|
125
|
+
def build_transition_map(name)
|
126
|
+
table = transition_table(name)
|
127
|
+
events = table.shift[1..-1]
|
128
|
+
table.inject({}) do |machine, actions|
|
129
|
+
state = actions.shift
|
130
|
+
machine[state] = Hash[*events.zip(actions).flatten]
|
131
|
+
machine
|
132
|
+
end
|
133
|
+
end
|
134
|
+
|
135
|
+
def transition_table(name)
|
136
|
+
state_machine_reader = StateMachineReader.new
|
137
|
+
lexer = Gherkin::I18n.new('en').lexer(state_machine_reader)
|
138
|
+
machine = File.dirname(__FILE__) + "/#{name}.txt"
|
139
|
+
lexer.scan(File.read(machine))
|
140
|
+
state_machine_reader.rows
|
141
|
+
end
|
142
|
+
|
143
|
+
class StateMachineReader
|
144
|
+
attr_reader :rows
|
145
|
+
|
146
|
+
def initialize
|
147
|
+
@rows = []
|
148
|
+
end
|
149
|
+
|
150
|
+
def uri(uri)
|
151
|
+
end
|
152
|
+
|
153
|
+
def row(row, line_number)
|
154
|
+
@rows << row
|
155
|
+
end
|
156
|
+
|
157
|
+
def eof
|
158
|
+
end
|
159
|
+
end
|
160
|
+
|
161
|
+
end
|
162
|
+
end
|
163
|
+
end
|
164
|
+
end
|