aslakhellesoy-gherkin 0.0.1

Sign up to get free protection for your applications and to get access to all the features.
@@ -0,0 +1,23 @@
1
+ #Comment on line 1
2
+ @tag1 @tag2
3
+ #Comment on line 3
4
+ Feature: Feature Text
5
+ In order to test multiline forms
6
+ As a ragel writer
7
+ I need to check for complex combinations
8
+
9
+ #Comment on line 9
10
+
11
+ #Comment on line 11
12
+
13
+ @tag3 @tag4
14
+ Scenario: Reading a Scenario
15
+ Given there is a step
16
+ But not another step
17
+
18
+ @tag3
19
+ Scenario: Reading a second scenario
20
+ #Comment on line 20
21
+ Given a third step
22
+ #Comment on line 22
23
+ Then I am happy
@@ -0,0 +1,6 @@
1
+ Egenskap: i18n support
2
+
3
+ Scenario: Parsing many languages
4
+ Gitt Gherkin supports many languages
5
+ Når Norwegian keywords are parsed
6
+ Så they should be recognized
@@ -0,0 +1,3 @@
1
+ Feature: Feature Text
2
+ Scenario: Reading a Scenario
3
+ Given there is a step
@@ -0,0 +1,7 @@
1
+ # Here is a comment
2
+ Feature: Feature Text
3
+ # Here is another # comment
4
+ Scenario: Reading a Scenario
5
+ # Here is a third comment
6
+ Given there is a step
7
+ # Here is a fourth comment
@@ -0,0 +1,5 @@
1
+ @tag1 @tag2
2
+ Feature: Feature Text
3
+ @tag3 @tag4
4
+ Scenario: Reading a Scenario
5
+ Given there is a step
@@ -0,0 +1,28 @@
1
+ #encoding: utf-8
2
+ require File.expand_path(File.dirname(__FILE__) + '/../spec_helper')
3
+
4
+ module Gherkin
5
+ module Parser
6
+ describe "i18n parsing" do
7
+ before do
8
+ @listener = Gherkin::SexpRecorder.new
9
+ @parser = Parser['no'].new(@listener)
10
+ end
11
+
12
+ def scan_file(file)
13
+ @parser.scan(File.new(File.dirname(__FILE__) + "/gherkin_parser/" + file).read)
14
+ end
15
+
16
+ it "should recognize keywords in the language of the parser" do
17
+ scan_file("i18n_no.feature")
18
+ @listener.to_sexp.should == [
19
+ [:feature, "Egenskap", "i18n support", 1],
20
+ [:scenario, "Scenario", "Parsing many languages", 3],
21
+ [:step, "Gitt", "Gherkin supports many languages", 4],
22
+ [:step, "Når", "Norwegian keywords are parsed", 5],
23
+ [:step, "Så", "they should be recognized", 6]
24
+ ]
25
+ end
26
+ end
27
+ end
28
+ end
@@ -0,0 +1,98 @@
1
+ # encoding: utf-8
2
+ require File.expand_path(File.dirname(__FILE__) + '/../spec_helper')
3
+
4
+ module Gherkin
5
+ module Parser
6
+ describe "parsing multiline step arguments (pystrings)" do
7
+
8
+ def ps(content)
9
+ '"""%s"""' % ("\n" + content + "\n")
10
+ end
11
+
12
+ def indent(s, n)
13
+ if n >= 0
14
+ s.gsub(/^/, ' ' * n)
15
+ else
16
+ s.gsub(/^ {0,#{-n}}/, "")
17
+ end
18
+ end
19
+
20
+ before do
21
+ @listener = mock('listener')
22
+ @parser = Misc.new(@listener)
23
+ end
24
+
25
+ it "should provide the amount of indentation to the listener"
26
+
27
+ it "should parse a simple pystring" do
28
+ @listener.should_receive(:pystring).with("I am a pystring")
29
+ @parser.scan ps("I am a pystring")
30
+ end
31
+
32
+ it "should parse an empty pystring" do
33
+ @listener.should_receive(:pystring).with("")
34
+ @parser.scan ps("")
35
+ end
36
+
37
+ it "should parse a string containing only newlines" do
38
+ @listener.should_receive(:pystring).with("\n\n")
39
+ @parser.scan ps("\n\n")
40
+ end
41
+
42
+ it "should parse a content separated by two newlines" do
43
+ @listener.should_receive(:pystring).with("A\n\nB")
44
+ @parser.scan ps("A\n\nB")
45
+ end
46
+
47
+ it "should parse a multiline string" do
48
+ @listener.should_receive(:pystring).with("A\nB\nC\nD")
49
+ @parser.scan ps("A\nB\nC\nD")
50
+ end
51
+
52
+ it "should ignore unescaped quotes inside the string delimeters" do
53
+ @listener.should_receive(:pystring).with("What does \"this\" mean?")
54
+ @parser.scan ps('What does "this" mean?')
55
+ end
56
+
57
+ it "should remove whitespace up to the column of the opening quote" do
58
+ @listener.should_receive(:pystring).with("I have been indented for reasons of style")
59
+ @parser.scan indent(ps('I have been indented for reasons of style'), 4)
60
+ end
61
+
62
+ it "should preserve whitespace after the column of the opening quote" do
63
+ @listener.should_receive(:pystring).with(" I have been indented to preserve whitespace")
64
+ @parser.scan indent(ps(' I have been indented to preserve whitespace'), 4)
65
+ end
66
+
67
+ it "should preserve tabs within the content" do
68
+ @listener.should_receive(:pystring).with("I have\tsome tabs\nInside\t\tthe content")
69
+ @parser.scan ps("I have\tsome tabs\nInside\t\tthe content")
70
+ end
71
+
72
+ it "should handle complex pystrings" do
73
+ pystring = %{
74
+ # Feature comment
75
+ @one
76
+ Feature: Sample
77
+
78
+ @two @three
79
+ Scenario: Missing
80
+ Given missing
81
+
82
+ 1 scenario (1 passed)
83
+ 1 step (1 passed)
84
+
85
+ }
86
+
87
+ @listener.should_receive(:pystring).with(pystring)
88
+ @parser.scan ps(pystring)
89
+ end
90
+
91
+ it "should set indentation to zero if the content begins before the start delimeter" do
92
+ pystring = " \"\"\"\nContent\n\"\"\""
93
+ @listener.should_receive(:pystring).with("Content")
94
+ @parser.scan(pystring)
95
+ end
96
+ end
97
+ end
98
+ end
@@ -0,0 +1,15 @@
1
+ module Gherkin
2
+ class SexpRecorder
3
+ def initialize
4
+ @sexps = []
5
+ end
6
+
7
+ def method_missing(m, *args)
8
+ @sexps << [m] + args
9
+ end
10
+
11
+ def to_sexp
12
+ @sexps
13
+ end
14
+ end
15
+ end
@@ -0,0 +1,83 @@
1
+ # encoding: utf-8
2
+ require File.expand_path(File.dirname(__FILE__) + '/../spec_helper')
3
+
4
+ module Gherkin
5
+ module Parser
6
+ describe Table do
7
+ before do
8
+ @listener = Gherkin::SexpRecorder.new
9
+ @table = Table.new(@listener)
10
+ end
11
+
12
+ tables = {
13
+ "|a|b|\n" => [%w{a b}],
14
+ "|a|b|c|\n" => [%w{a b c}],
15
+ "|c|d|\n|e|f|\n" => [%w{c d}, %w{e f}]
16
+ }
17
+
18
+ tables.each do |text, expected|
19
+ it "should parse #{text}" do
20
+ @listener.should_receive(:table).with(expected)
21
+ @table.scan(text)
22
+ end
23
+ end
24
+
25
+ it "should parse a multicharacter cell content" do
26
+ @listener.should_receive(:table).with([%w{foo bar}])
27
+ @table.scan("| foo | bar |\n")
28
+ end
29
+
30
+ it "should parse cells with spaces within the content" do
31
+ @listener.should_receive(:table).with([["Dill pickle", "Valencia orange"], ["Ruby red grapefruit", "Tire iron"]])
32
+ @table.scan("| Dill pickle | Valencia orange |\n| Ruby red grapefruit | Tire iron |\n")
33
+ end
34
+
35
+ it "should parse a 1x2 table with newline" do
36
+ @listener.should_receive(:table).with([%w{1 2}])
37
+ @table.scan("| 1 | 2 |\n")
38
+ end
39
+
40
+ it "should allow utf-8" do
41
+ @listener.should_receive(:table).with([%w{ůﻚ 2}])
42
+ @table.scan(" | ůﻚ | 2 | \n")
43
+ end
44
+
45
+ it "should parse a 2x2 table" do
46
+ @listener.should_receive(:table).with([%w{1 2}, %w{3 4}])
47
+ @table.scan("| 1 | 2 |\n| 3 | 4 |\n")
48
+ end
49
+
50
+ it "should parse a 2x2 table with several newlines" do
51
+ @listener.should_receive(:table).with([%w{1 2}, %w{3 4}])
52
+ @table.scan("| 1 | 2 |\n| 3 | 4 |\n\n\n")
53
+ end
54
+
55
+ it "should parse a 2x2 table with empty cells" do
56
+ @listener.should_receive(:table).with([['1', nil], [nil, '4']])
57
+ @table.scan("| 1 | |\n|| 4 |\n")
58
+ end
59
+
60
+ it "should parse a 1x2 table without newline" do
61
+ @listener.should_receive(:table).with([%w{1 2}])
62
+ @table.scan("| 1 | 2 |")
63
+ end
64
+
65
+ it "should parse a 1x2 table without spaces and newline" do
66
+ @listener.should_receive(:table).with([%w{1 2}])
67
+ @table.scan("|1|2|")
68
+ end
69
+
70
+ it "should not parse a 2x2 table that isn't closed" do
71
+ @table.scan("| 1 | |\n|| 4 ")
72
+ @listener.to_sexp.should == [
73
+ [:table, [['1', nil]]]
74
+ ]
75
+ end
76
+
77
+ it "should parse a table with tab spacing" do
78
+ @listener.should_receive(:table).with([["abc", "123"]])
79
+ @table.scan("|\tabc\t|\t123\t\t\t|\n")
80
+ end
81
+ end
82
+ end
83
+ end
@@ -0,0 +1,57 @@
1
+ #encoding: utf-8
2
+ require File.expand_path(File.dirname(__FILE__) + '/../spec_helper')
3
+
4
+ module Gherkin
5
+ module Parser
6
+ describe "parsing" do
7
+ describe "tags" do
8
+ before do
9
+ @listener = mock('listener')
10
+ @feature = Parser['en'].new(@listener)
11
+ end
12
+
13
+ it "should parse a single tag" do
14
+ @listener.should_receive(:tag).with("dog", 1)
15
+ @feature.scan("@dog\n")
16
+ end
17
+
18
+ it "should parse multiple tags" do
19
+ @listener.should_receive(:tag).twice
20
+ @feature.scan("@dog @cat\n")
21
+ end
22
+
23
+ it "should parse UTF-8 tags" do
24
+ @listener.should_receive(:tag).with("シナリオテンプレート", 1)
25
+ @feature.scan("@シナリオテンプレート\n")
26
+ end
27
+
28
+ it "should parse mixed tags" do
29
+ @listener.should_receive(:tag).with("wip", 1).ordered
30
+ @listener.should_receive(:tag).with("Значения", 1).ordered
31
+ @feature.scan("@wip @Значения\n")
32
+ end
33
+
34
+ it "should parse wacky identifiers" do
35
+ @listener.should_receive(:tag).exactly(4).times
36
+ @feature.scan("@BJ-x98.77 @BJ-z12.33 @O_o" "@#not_a_comment\n")
37
+ end
38
+
39
+ # TODO: Ask on ML for opinions about this one
40
+ it "should parse tags without spaces between them?" do
41
+ @listener.should_receive(:tag).twice
42
+ @feature.scan("@one@two\n")
43
+ end
44
+
45
+ it "should not parse tags beginning with two @@ signs" do
46
+ @listener.should_not_receive(:tag)
47
+ @feature.scan("@@test\n")
48
+ end
49
+
50
+ it "should not parse a lone @ sign" do
51
+ @listener.should_not_receive(:tag)
52
+ @feature.scan("@\n")
53
+ end
54
+ end
55
+ end
56
+ end
57
+ end
@@ -0,0 +1,10 @@
1
+ $LOAD_PATH.unshift(File.dirname(__FILE__))
2
+ $LOAD_PATH.unshift(File.join(File.dirname(__FILE__), '..', 'lib'))
3
+ require 'gherkin'
4
+ require 'gherkin/sexp_recorder'
5
+ require 'spec'
6
+ require 'spec/autorun'
7
+
8
+ Spec::Runner.configure do |config|
9
+
10
+ end
data/tasks/ext.rake ADDED
@@ -0,0 +1,35 @@
1
+ CLEAN.include %w(**/*.{o,bundle,jar,so,obj,pdb,lib,def,exp,log} ext/*/Makefile ext/*/*.c ext/*/conftest.dSYM)
2
+ WIN = (RUBY_PLATFORM =~ /mswin|cygwin/)
3
+
4
+ def ext_task(name)
5
+ ext_dir = "ext/#{name}"
6
+ ext_bundle = "#{ext_dir}/#{name}.#{Config::CONFIG['DLEXT']}"
7
+ ext_files = FileList[
8
+ "#{ext_dir}/*.c",
9
+ "#{ext_dir}/*.h",
10
+ "#{ext_dir}/*.rl",
11
+ "#{ext_dir}/extconf.rb",
12
+ "#{ext_dir}/Makefile",
13
+ "lib"
14
+ ]
15
+
16
+ task "compile:#{name}" => ["#{ext_dir}/Makefile", ext_bundle]
17
+ task :compile => ['ragel:c', "compile:#{name}"]
18
+
19
+ file "#{ext_dir}/Makefile" => ["#{ext_dir}/extconf.rb"] do
20
+ cd(ext_dir) { ruby "extconf.rb" }
21
+ end
22
+
23
+ file ext_bundle => ext_files do
24
+ cd ext_dir do
25
+ sh(WIN ? 'nmake' : 'make')
26
+ end
27
+ cp ext_bundle, 'lib/'
28
+ end
29
+ end
30
+
31
+ ext_task :gherkin
32
+
33
+ desc "Compile the C extensions"
34
+ task :compile
35
+ task :package => :compile
data/tasks/ragel.rake ADDED
@@ -0,0 +1,81 @@
1
+ class RagelCompiler
2
+ def initialize
3
+ require 'yaml'
4
+ require 'erb'
5
+
6
+ @impl = ERB.new(IO.read(File.dirname(__FILE__) + '/../ragel/feature.rb.rl.erb'))
7
+ @common = ERB.new(IO.read(File.dirname(__FILE__) + '/../ragel/feature_common.rl.erb'))
8
+ @langs = YAML.load_file(File.dirname(__FILE__) + '/../lib/gherkin/i18n.yml')
9
+ end
10
+
11
+ def compile_all
12
+ @langs.keys.each do |lang|
13
+ compile(lang)
14
+ end
15
+ end
16
+
17
+ def compile(lang)
18
+ i18n = @langs['en'].merge(@langs[lang])
19
+ common_file = File.dirname(__FILE__) + "/../ragel/feature_common.#{lang}.rl"
20
+ impl_file = File.dirname(__FILE__) + "/../ragel/feature_#{lang}.rb.rl"
21
+
22
+ common = @common.result(binding)
23
+ impl = @impl.result(binding)
24
+
25
+ write common, common_file
26
+ write impl, impl_file
27
+
28
+ sh "ragel -R #{impl_file} -o lib/gherkin/parser/feature_#{lang}.rb"
29
+
30
+ FileUtils.rm([impl_file, common_file])
31
+ end
32
+
33
+ def write(content, filename)
34
+ File.open(filename, "wb") do |file|
35
+ file.write(content)
36
+ end
37
+ end
38
+ end
39
+
40
+ namespace :ragel do
41
+ desc "Generate Ruby from the Ragel rule files"
42
+ task :rb => :i18n_en do
43
+ Dir["ragel/*.rb.rl"].each do |rl|
44
+ basename = File.basename(rl[0..-4])
45
+ sh "ragel -R #{rl} -o lib/gherkin/parser/#{basename}"
46
+ end
47
+ end
48
+
49
+ desc "Generate C from the Ragel rule files"
50
+ task :c do
51
+ Dir["ragel/*.c.rl"].each do |rl|
52
+ basename = File.basename(rl[0..-3])
53
+ sh "ragel -C #{rl} -o ext/gherkin/#{basename}"
54
+ end
55
+ end
56
+
57
+ desc "Generate all Ruby i18n parsers"
58
+ task :i18n do
59
+ RagelCompiler.new.compile_all
60
+ end
61
+
62
+ desc "Generate Ruby English language parser"
63
+ task :i18n_en do
64
+ RagelCompiler.new.compile('en')
65
+ end
66
+
67
+ desc "Generate a dot file of the Ragel state machine"
68
+ task :dot do
69
+ Dir["ragel/*.rb.rl"].each do |path|
70
+ sh "ragel -V #{path} -o #{File.basename(path, '.rl')}.dot"
71
+ end
72
+ end
73
+
74
+ desc "Generate a png diagram of the Ragel state machine"
75
+ task :png => :dot do
76
+ Dir["*dot"].each do |path|
77
+ sh "dot -Tpng #{path} > #{File.basename(path, '.dot')}.png"
78
+ end
79
+ end
80
+ end
81
+