module Cucumber module Ast # Walks the AST, executing steps and notifying listeners class TreeWalker attr_accessor :configuration #:nodoc: attr_reader :step_mother #:nodoc: def initialize(step_mother, listeners = [], configuration = Cucumber::Configuration.default) @step_mother, @listeners, @configuration = step_mother, listeners, configuration 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