Skip to content

Instantly share code, notes, and snippets.

@dblock
Created May 4, 2012 19:52
Show Gist options
  • Star 9 You must be signed in to star a gist
  • Fork 2 You must be signed in to fork a gist
  • Save dblock/2597305 to your computer and use it in GitHub Desktop.
Save dblock/2597305 to your computer and use it in GitHub Desktop.
RSpec tests broken up into suites with retry.
# inspired by https://github.com/rspec/rspec-core/pull/596
require 'rspec/core/formatters/base_formatter'
module RSpec
module Core
module Formatters
class FailuresFormatter < BaseFormatter
def dump_failures
return if failed_examples.empty?
f = File.new("rspec.failures", "w+")
failed_examples.each do |example|
f.puts retry_command(example)
end
f.close
end
def retry_command(example)
example_name = example.full_description.gsub("\"", "\\\"")
"-e \"#{example_name}\""
end
end
end
end
end
#
# Organize RSpec tests in suites w/ retry. Requires rspec-core 2.11.0.
#
# (c) 2012 Daniel Doubrovkine, Art.sy
# MIT License
#
require 'rspec/core/rake_task'
SPEC_SUITES = [
{ :id => :initializers, :title => 'initializers', :pattern => "spec/initializers/**/*_spec.rb" },
{ :id => :models, :title => 'model tests', :pattern => "spec/models/**/*_spec.rb" },
{ :id => :acceptance, :title => 'acceptance tests', :pattern => "spec/acceptance/**/*_spec.rb" },
{ :id => :controllers, :title => 'controller tests', :pattern => "spec/controllers/**/*_spec.rb" },
{ :id => :views, :title => 'view tests', :pattern => "spec/views/**/*_spec.rb" },
{ :id => :mailers, :title => 'mailer tests', :pattern => "spec/mailers/*_spec.rb" },
]
namespace :spec do
namespace :suite do
SPEC_SUITES.each do |suite|
desc "Run all specs in #{suite[:title]} spec suite"
task "#{suite[:id]}" do
rspec_failures = File.join(Rails.root, 'rspec.failures')
FileUtils.rm_f rspec_failures
Rake::Task["spec:suite:#{suite[:id]}:run"].execute
unless $?.success?
puts "[#{Time.now}] Failed, retrying #{File.read(rspec_failures).split(/\n+/).count} failure(s) in spec:suite:#{suite[:id]} ..."
Rake::Task["spec:suite:#{suite[:id]}:retry"].execute
end
end
RSpec::Core::RakeTask.new("#{suite[:id]}:run") do |t|
t.pattern = suite[:pattern]
t.verbose = false
t.fail_on_error = false
t.spec_opts = [
"--require", "#{Rails.root}/spec/support/formatters/failures_formatter.rb",
"--format", "RSpec::Core::Formatters::FailuresFormatter",
File.read(File.join(Rails.root, ".rspec")).split(/\n+/).map { |l| l.shellsplit }
].flatten
end
RSpec::Core::RakeTask.new("#{suite[:id]}:retry") do |t|
t.pattern = suite[:pattern]
t.verbose = false
t.fail_on_error = false
t.spec_opts = [
"-O", File.join(Rails.root, 'rspec.failures'),
File.read(File.join(Rails.root, '.rspec')).split(/\n+/).map { |l| l.shellsplit }
].flatten
end
end
desc "Run all spec suites"
task :all => :environment do
failed_suites = []
SPEC_SUITES..each do |suite|
puts "Running spec:suite:#{suite[:id]} ..."
Rake::Task["spec:suite:#{suite[:id]}"].execute
failed_suites << suite unless $?.success?
end
raise "Spec suite failed" unless failed_suites.empty?
end
end
end
@voodoorai2000
Copy link

For really flaky situations, you can use a loop, instead of the unless statement, to rerun failing specs multiple times.

tries = 0
while not $?.success? and tries < 3
puts "[#{Time.now}] Failed, retrying #{File.read(rspec_failures).split(/\n+/).count} failure(s) in spec:suite:#{suite[:id]} ..."
Rake::Task["spec:suite:#{suite[:id]}:retry"].execute
tries += 1
end

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment