module Cucumber
module Ast
# Walks the AST, executing steps and notifying listeners
class TreeWalker
attr_accessor :configuration #:nodoc:
attr_reader :runtime #:nodoc:
def initialize(runtime, listeners = [], configuration = Cucumber::Configuration.default)
@runtime, @listeners, @configuration = runtime, listeners, configuration
end
def execute(scenario, skip_hooks)
runtime.with_hooks(scenario, skip_hooks) do
scenario.skip_invoke! if scenario.failed?
visit_steps(scenario.steps)
end
end
def visit_features(features)
broadcast(features) do
features.accept(self)
end
end
def visit_feature(feature)
broadcast(feature) do
feature.accept(self)
end
end
def visit_comment(comment)
broadcast(comment) do
comment.accept(self)
end
end
def visit_comment_line(comment_line)
broadcast(comment_line)
end
def visit_tags(tags)
broadcast(tags) do
tags.accept(self)
end
end
def visit_tag_name(tag_name)
broadcast(tag_name)
end
def visit_feature_name(keyword, name)
broadcast(keyword, name)
end
# +feature_element+ is either Scenario or ScenarioOutline
def visit_feature_element(feature_element)
broadcast(feature_element) do
feature_element.accept(self)
end
end
def visit_background(background)
broadcast(background) do
background.accept(self)
end
end
def visit_background_name(keyword, name, file_colon_line, source_indent)
broadcast(keyword, name, file_colon_line, source_indent)
end
def visit_examples_array(examples_array)
broadcast(examples_array) do
examples_array.accept(self)
end
end
def visit_examples(examples)
broadcast(examples) do
examples.accept(self)
end
end
def visit_examples_name(keyword, name)
broadcast(keyword, name)
end
def visit_outline_table(outline_table)
broadcast(outline_table) do
outline_table.accept(self)
end
end
def visit_scenario_name(keyword, name, file_colon_line, source_indent)
broadcast(keyword, name, file_colon_line, source_indent)
end
def visit_steps(steps)
broadcast(steps) do
steps.accept(self)
end
end
def visit_step(step)
broadcast(step) do
step.accept(self)
end
end
def visit_step_result(keyword, step_match, multiline_arg, status, exception, source_indent, background, file_colon_line)
broadcast(keyword, step_match, multiline_arg, status, exception, source_indent, background, file_colon_line) do
visit_step_name(keyword, step_match, status, source_indent, background, file_colon_line)
visit_multiline_arg(multiline_arg) if multiline_arg
visit_exception(exception, status) if exception
end
end
def visit_step_name(keyword, step_match, status, source_indent, background, file_colon_line) #:nodoc:
broadcast(keyword, step_match, status, source_indent, background, file_colon_line)
end
def visit_multiline_arg(multiline_arg) #:nodoc:
broadcast(multiline_arg) do
multiline_arg.accept(self)
end
end
def visit_exception(exception, status) #:nodoc:
broadcast(exception, status)
end
def visit_doc_string(string)
broadcast(string)
end
def visit_table_row(table_row)
broadcast(table_row) do
table_row.accept(self)
end
end
def visit_table_cell(table_cell)
broadcast(table_cell) do
table_cell.accept(self)
end
end
def visit_table_cell_value(value, status)
broadcast(value, status)
end
# Print +messages+. This method can be called from within StepDefinitions.
def puts(*messages)
broadcast(*messages)
end
# Embed +file+ of +mime_type+ in the formatter. This method can be called from within StepDefinitions.
# For most formatters this is a no-op.
def embed(file, mime_type, label)
broadcast(file, mime_type, label)
end
private
def broadcast(*args, &block)
message = extract_method_name_from(caller)
message.gsub!('visit_', '')
if block_given?
send_to_all("before_#{message}", *args)
yield if block_given?
send_to_all("after_#{message}", *args)
else
send_to_all(message, *args)
end
self
end
def send_to_all(message, *args)
@listeners.each do |listener|
if listener.respond_to?(message)
listener.__send__(message, *args)
end
end
end
def extract_method_name_from(call_stack)
call_stack[0].match(/in `(.*)'/).captures[0]
end
end
end
end