# -*- encoding: utf-8 -*- require 'cucumber/initializer' require 'cucumber/core/gherkin/parser' require 'cucumber/core/gherkin/writer' module Cucumber module Core module Gherkin describe Parser do let(:receiver) { double } let(:parser) { Parser.new(receiver) } let(:visitor) { double } def parse parser.document(source) end context "for invalid gherkin" do let(:source) { Gherkin::Document.new(path, 'not gherkin') } let(:path) { 'path_to/the.feature' } it "raises an error" do expect { parse }.to raise_error(ParseError) do |error| expect( error.message ).to match(/not gherkin/) expect( error.message ).to match(/#{path}/) end end end include Writer def self.source(&block) let(:source) { gherkin(&block) } end def feature result = nil allow( receiver ).to receive(:feature) { |feature| result = feature } parse result end context "when the Gherkin has a language header" do source do feature(language: 'ja', keyword: '機能') end it "sets the language from the Gherkin" do expect( feature.language.iso_code ).to eq 'ja' end end context "a Scenario with a DocString" do source do feature do scenario do step do doc_string("content") end end end end it "parses doc strings without error" do allow( visitor ).to receive(:feature).and_yield(visitor) allow( visitor ).to receive(:scenario).and_yield(visitor) allow( visitor ).to receive(:step).and_yield(visitor) location = double expected = Ast::DocString.new("content", "", location) expect( visitor ).to receive(:doc_string).with(expected) feature.describe_to(visitor) end end context "a Scenario with a DataTable" do source do feature do scenario do step do table do row "name", "surname" row "rob", "westgeest" end end end end end it "parses the DataTable" do visitor = double allow( visitor ).to receive(:feature).and_yield(visitor) allow( visitor ).to receive(:scenario).and_yield(visitor) allow( visitor ).to receive(:step).and_yield(visitor) expected = Ast::DataTable.new([['name', 'surname'], ['rob', 'westgeest']], Ast::Location.new('foo.feature', 23)) expect( visitor ).to receive(:data_table).with(expected) feature.describe_to(visitor) end end context "a Scenario with a Comment" do source do feature do comment 'wow' scenario end end it "parses the comment into the AST" do visitor = double allow( visitor ).to receive(:feature).and_yield(visitor) expect( visitor ).to receive(:scenario) do |scenario| expect( scenario.comments.join ).to eq "# wow" end feature.describe_to(visitor) end end context "a Scenario Outline" do source do feature do scenario_outline 'outline name' do step 'passing ' examples do row 'arg' row '1' row '2' end examples do row 'arg' row 'a' end end end end it "creates a scenario outline node" do allow( visitor ).to receive(:feature).and_yield(visitor) expect( visitor ).to receive(:scenario_outline) do |outline| expect( outline.name ).to eq 'outline name' end feature.describe_to(visitor) end it "creates a step node for each step of the scenario outline" do allow( visitor ).to receive(:feature).and_yield(visitor) allow( visitor ).to receive(:scenario_outline).and_yield(visitor) allow( visitor ).to receive(:examples_table) expect( visitor ).to receive(:outline_step) do |step| expect( step.name ).to eq 'passing ' end feature.describe_to(visitor) end it "creates an examples table node for each examples table" do allow( visitor ).to receive(:feature).and_yield(visitor) allow( visitor ).to receive(:scenario_outline).and_yield(visitor) allow( visitor ).to receive(:outline_step) expect( visitor ).to receive(:examples_table).exactly(2).times.and_yield(visitor) expect( visitor ).to receive(:examples_table_row) do |row| expect( row.number ).to eq 1 expect( row.values ).to eq ['1'] end.once.ordered expect( visitor ).to receive(:examples_table_row) do |row| expect( row.number ).to eq 2 expect( row.values ).to eq ['2'] end.once.ordered expect( visitor ).to receive(:examples_table_row) do |row| expect( row.number ).to eq 1 expect( row.values ).to eq ['a'] end.once.ordered feature.describe_to(visitor) end end context "a Scenario Outline with no Examples" do source do feature do scenario_outline do step 'passing ' end end end it "throws an error" do expect { feature.describe_to(double.as_null_object) }.to raise_error(ParseError) end end end end end end