module Cucumber
module Ast
# Walks the AST, executing steps and notifying listeners
class TreeWalker
attr_accessor :options #:nodoc:
attr_reader :step_mother #:nodoc:
def initialize(step_mother, listeners = [], options = {}, io = STDOUT)
@step_mother, @listeners, @options, @io = step_mother, listeners, options, io
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)
broadcast(keyword, step_match, multiline_arg, status, exception, source_indent, background) do
visit_step_name(keyword, step_match, status, source_indent, background)
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) #:nodoc:
broadcast(keyword, step_match, status, source_indent, background)
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_py_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 +announcement+. This method can be called from within StepDefinitions.
def announce(announcement)
broadcast(announcement)
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)
broadcast(file, mime_type)
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
end
def send_to_all(message, *args)
@listeners.each do |listener|
if listener.respond_to?(message)
if legacy_listener?(listener)
warn_legacy(listener)
legacy_invoke(listener, message, *args)
else
listener.__send__(message, *args)
end
end
end
end
def legacy_listener?(listener)
listener.respond_to?(:feature_name) &&
(listener.method(:feature_name) rescue false) &&
listener.method(:feature_name).arity == 1
end
def warn_legacy(listener)
@warned_listeners ||= []
unless @warned_listeners.index(listener)
warn("#{listener.class} is using a deprecated formatter API. Starting with Cucumber 0.7.0 the signatures\n" +
"that have changed are:\n" +
" feature_name(keyword, name) # Two arguments. The keyword argument will not contain a colon.\n" +
" scenario_name(keyword, name, file_colon_line, source_indent) # The keyword argument will not contain a colon.\n" +
" examples_name(keyword, name) # The keyword argument will not contain a colon.\n"
)
end
@warned_listeners << listener
end
def legacy_invoke(listener, message, *args)
case message.to_sym
when :feature_name
listener.feature_name("#{args[0]}: #{args[1]}")
when :scenario_name, :examples_name
args_with_colon = args.dup
args_with_colon[0] += ':'
listener.__send__(message, *args_with_colon)
else
listener.__send__(message, *args)
end
end
def extract_method_name_from(call_stack)
call_stack[0].match(/in `(.*)'/).captures[0]
end
end
end
end