module RSpecStepwise class ApatheticReporter < ::RSpec::Core::Reporter def notify(*args) #noop end end class WholeListExample < RSpec::Core::Example def initialize(example_group_class, descriptions, metadata) super @reporter = ApatheticReporter.new build_example_block end def start(reporter) super(@reporter) end def finish(reporter) super(@reporter) end def build_example_block #variables of concern: reporter, instance @example_block = proc do begin self.class.filtered_examples.inject(true) do |success, example| break if RSpec.wants_to_quit example.extend StepExample unless success example.metadata[:pending] = true example.metadata[:execution_result][:pending_message] = "Previous step failed" end succeeded = with_indelible_ivars do example.run(self, reporter) end RSpec.wants_to_quit = true if self.class.fail_fast? && !succeeded success && succeeded end end end end end module StepExample def run_before_each @example_group_class.run_before_step(self) rescue Object => ex puts "\n#{__FILE__}:#{__LINE__} => #{[ex, ex.backtrace].pretty_inspect}" end def run_after_each @example_group_class.run_after_step(self) end def with_around_hooks yield end end module ClassMethods #This is hacky and needs a more general solution #Something like cloning the current conf and having RSpec::Stepwise::config ? def suspend_transactional_fixtures if self.respond_to? :use_transactional_fixtures begin old_val = self.use_transactional_fixtures self.use_transactional_fixtures = false yield ensure self.use_transactional_fixtures = old_val end else yield end end def build_before_hook(options, &block) if defined? RSpec::Core::Hooks::BeforeHookExtension block.extend(RSpec::Core::Hooks::BeforeHookExtension).with(options) else RSpec::Core::Hooks::BeforeHook.new(block, options) end end def build_after_hook(options, &block) if defined? RSpec::Core::Hooks::AfterHookExtension block.extend(RSpec::Core::Hooks::AfterHookExtension).with(options) else RSpec::Core::Hooks::AfterHook.new(block, options) end end def before(*args, &block) if args.first == :step args.shift options = build_metadata_hash_from(args) return ((hooks[:before][:step] ||= []) << build_before_hook(options, &block)) end if args.first == :each puts "before blocks declared for steps are always treated as :all scope" end super end def after(*args, &block) if args.first == :step args.shift options = build_metadata_hash_from(args) hooks[:after][:step] ||= [] return (hooks[:after][:step].unshift build_after_hook(options, &block)) end if args.first == :each puts "after blocks declared for steps are always treated as :all scope" end super end def around(*args, &block) if args.first == :each puts "around :each blocks declared for steps are treated as :all scope" end super end def example_synonym(named, desc=nil, *args, &block) unless desc.nil? desc = [named, desc].join(" ") end it(desc, *args, &block) end def when(*args, &block); example_synonym("when", *args, &block); end def then(*args, &block); example_synonym("then", *args, &block); end def next(*args, &block); example_synonym("next", *args, &block); end def step(*args, &block); example_synonym("step", *args, &block); end def run_step(example, hook, &sorting) groups = if respond_to?(:parent_groups) parent_groups else ancestors end if block_given? groups = yield groups end RSpec::Core::Hooks::HookCollection.new(groups.map {|a| a.hooks[hook][:step]}.flatten.compact).for(example).run end def run_before_step(example) run_step(example, :before) end def run_after_step(example) run_step(example, :after) do |groups| groups.reverse end end def perform_steps(name, *args, &customization_block) shared_block = nil if world.respond_to? :shared_example_groups shared_block = world.shared_example_groups[name] else shared_block = shared_example_groups[name] end raise "Could not find shared example group named #{name.inspect}" unless shared_block module_eval_with_args(*args, &shared_block) module_eval(&customization_block) if customization_block end def run_examples(reporter) whole_list_example = WholeListExample.new(self, "step list", {}) instance = new set_ivars(instance, before_all_ivars) instance.example = whole_list_example instance.reporter = reporter result = suspend_transactional_fixtures do whole_list_example.run(instance, reporter) end unless whole_list_example.exception.nil? RSpec.wants_to_quit = true if fail_fast? fail_filtered_examples(whole_list_example.exception, reporter) end result end end attr_accessor :reporter def with_indelible_ivars old_value, @ivars_indelible = @ivars_indelible, true result = yield @ivars_indelible = old_value result rescue Object @ivars_indelible = old_value raise end def instance_variable_set(name, value) if !@ivars_indelible super end end def self.included(base) base.extend(ClassMethods) end end