-
Notifications
You must be signed in to change notification settings - Fork 1
Home
Testing instructions here first to see how it'll look before posting to parallel_tests wiki
Rails file cache -- by eLobato
Concurrent access to cache files by different process can easily make your tests fail. Make each process have its own cache file store to avoid this problem.
Add to config/environments/test.rb
:
config.cache_store = :file_store, Rails.root.join("tmp", "cache", "paralleltests#{ENV['TEST_ENV_NUMBER']}")
Rails Sprockets file cache -- by jordan-brough
Sprockets uses a default cache path of <rails-root>/tmp/cache/assets/<environment>
. Concurrent usage of the same cache directory can cause sporadic problems like "ActionView::Template::Error: end of file reached".
Add to config/environments/test.rb
:
if ENV['TEST_ENV_NUMBER']
assets_cache_path = Rails.root.join("tmp/cache/assets/paralleltests#{ENV['TEST_ENV_NUMBER']}")
Rails.application.assets.cache = Sprockets::Cache::FileStore.new(assets_cache_path)
end
With sphinx -- by ujh
At file config/sphinx.yml in the test section:
test:
mysql41: <%= 9313 + ENV['TEST_ENV_NUMBER'].to_i %>
indices_location: <%= File.join(Rails.root, "db", "sphinx", "test#{ENV['TEST_ENV_NUMBER']}") %>
configuration_file: <%= File.join(Rails.root, "config", "test#{ENV['TEST_ENV_NUMBER']}.sphinx.conf")%>
log: <%= File.join(Rails.root, "log", "test#{ENV['TEST_ENV_NUMBER']}.searchd.log") %>
query_log: <%= File.join(Rails.root, "log", "test#{ENV['TEST_ENV_NUMBER']}.searchd.query.log") %>
binlog_path: <%= File.join(Rails.root, "tmp", "binlog", "test#{ENV['TEST_ENV_NUMBER']}") %>
pid_file: <%= File.join(Rails.root, "tmp", "pids", "test#{ENV['TEST_ENV_NUMBER']}.sphinx.pid") %>
If solution above doesn't work, then you may try another approach via pat's recommendation:
test:
mysql41: <%= ENV['TEST_ENV_NUMBER'].to_i + 9307 %>
pid_file: <%= File.join(Rails.root, "tmp", "searchd.#{ENV['TEST_ENV_NUMBER']}.pid") %>
indices_location: <%= File.join(Rails.root, "db", "sphinx", "#{ENV['TEST_ENV_NUMBER']}") %>
configuration_file: <%= File.join(Rails.root, "config", "test.#{ENV['TEST_ENV_NUMBER']}.sphinx.conf") %>
binlog_path: <%= File.join(Rails.root, "db", "sphinx", "#{ENV['TEST_ENV_NUMBER']}", "binlog") %>
With capybara(~>0.4.0)+selenium -- by rgo
Capybara.server_port = 9887 + ENV['TEST_ENV_NUMBER'].to_i
With capybara(=0.3.9)/Rails 2.3 -- by xunker
Add to features/support/env.rb:
if ENV['TEST_ENV_NUMBER']
class Capybara::Server
def find_available_port
@port = 9887 + ENV['TEST_ENV_NUMBER'].to_i
@port += 1 while is_port_open?(@port) and not is_running_on_port?(@port)
end
end
end
With rspec_junit_formatter -- by jgarber
I've had better results with rspec_junit_formatter than with ci_reporter. Parallelizing it is easy!
Add this to .rspec_parallel
:
--format RspecJunitFormatter
--out tmp/rspec<%= ENV['TEST_ENV_NUMBER'] %>.xml
Then configure Jenkins to publish JUnit test result report with tmp/rspec*.xml
for "Test report XMLs".
@danielheath : I've had corrupted xml files with larger numbers of concurrent processes - not sure this is reliable.
With ci_reporter for rspec -- by morganchristiansson
export CI_REPORTS=results
Add spec/parallel_specs.opts with the contents:
--format progress
--require ci/reporter/rake/rspec_loader
--format CI::Reporter::RSpec:/dev/null
Our project has the following in test/test_helper.rb
if ENV["CI_REPORTS"] == "results"
require "ci/reporter/rake/test_unit_loader"
end
Run the tasks like this:
rake "parallel:features[,,--format progress --format junit --out ${CI_REPORTS} --no-profile -r features]"
Or without rake like this:
bundle exec $(bundle show parallel_tests)/bin/parallel_test --type features -o '--format progress --format junit --out ${CI_REPORTS} --no-profile -r features'
Add following to RAILS_ROOT/.rspec_parallel
.
--format progress
--require ci/reporter/rake/rspec_loader
--format CI::Reporter::RSpec
Run rake parallel:spec
, then rspec generates reports in spec/reports
.
For more information on how to configure ci_reporter check under advanced usage on http://caldersphere.rubyforge.org/ci_reporter/
With ci_reporter for test_unit -- by phoet
See this issue 29 for more information:
# add the ci_reporter to create reports for test-runs, since parallel_tests is not invoked through rake
puts "running on #{Socket.gethostname}"
if /buildserver/ =~ Socket.gethostname
require 'ci/reporter/test_unit'
module Test
module Unit
module UI
module Console
class TestRunner
def create_mediator(suite)
# swap in ci_reporter custom mediator
return CI::Reporter::TestUnit.new(suite)
end
end
end
end
end
end
end
With DatabaseCleaner for RSpec -- by sciprog
See issue 66 for more information.
Do not use the truncation strategy in DatabaseCleaner, in your RSpec config.
This strategy seems to cause a bottleneck which will negate any gain made
through parallelization of your tests. If possible, use the transaction strategy
over the truncation strategy.
*Note: This issue does not seem to exist in relation to features, only to specs.
If you have put your features into subdirectories you may have problems running them in parallel as it will not find your step_definitions. To work around this I put all my features into the features directory.
You have to require features/ folder for cucumber in order to load step definitions.
rake "parallel:features[4, '', '-rfeatures/']"
If you want it to work with rake parallel:features
add -rfeatures/
to the end of std_opts
in config/cucumber.yml
rake parallel:features[,, --retry 1]
- 1 digit used for the number of times you want to retry Rerun file no longer required
If you do not have enough scenarios to warrant a separate instance of external service as described in sphinx section above, create a "mutex" in your env.rb file
Before("@solr") do
#we do not want solr tests to run in parallel, so let's simulate a mutex
while File.exists?("tmp/cucumber_solr")
sleep(0.2)
end
File.open("tmp/cucumber_solr", "w") {}
Sunspot.session = $original_sunspot_session
Sunspot.remove_all!
# or do other things
end
After("@solr") do
File.delete("tmp/cucumber_solr")
end
When running with cucumber + capybara + selenium-webdriver (w firefox), this error may be encountered due to firing of too many firefox instances altogether. To fix this issue, add the following to features/support/env.rb:
unless (env_no = ENV['TEST_ENV_NUMBER'].to_i).zero?
# As described in the readme
Capybara.server_port = 8888 + env_no
# Enforces a sleep time, i need to multiply by 10 to achieve consistent results on
# my 8 cores vm, may work for less though.
sleep env_no * 10
end
With action_mailer_cache_delivery (~> 0.3.2) -- by p0deje
You may get unexpected errors like EOFError
. If so, make sure cache files differ for processes. Change you config/environment/test.rb
config.action_mailer.cache_settings = { :location => "#{Rails.root}/tmp/cache/action_mailer_cache_delivery#{ENV['TEST_ENV_NUMBER']}.cache" }
It's best to use with sunspot-rails-tester. Then you don't need to run solr with rake task. For this you will only have to update config/sunspot.yml you can make it like this:
test:
solr:
hostname: localhost
port: <%= 8981 + ENV['TEST_ENV_NUMBER'].to_i %>
log_level: WARNING
data_path: <%= File.join(::Rails.root, 'solr', 'data', ::Rails.env, ENV['TEST_ENV_NUMBER'].to_i.to_s) %>
It's best to use with sunspot-rails-tester. Then you don't need to run solr with rake task. For this you will only have to update config/sunspot.yml you can make it like this:
test:
solr:
port: <%= 8981 + ENV['TEST_ENV_NUMBER'].to_i %>
solr_home: <%= File.join(::Rails.root, 'solr', 'data', ::Rails.env, ENV['TEST_ENV_NUMBER'].to_i.to_s) %>
With TeamCity -- by aaronjensen
TeamCity has its own logger so you'll need to use the --serialize-stdout
flag when you run anything in parallel. You'll need to use a custom rake task or script and call parallel_rspec
/parallel_test
/parallel_cucumber
directly.
You will need this rake task so that TeamCity will calculate the number of specs correctly. However what it does not pick up is duplicated test names within a spec. TeamCity processes this as a duplicate and does not add it to the total spec count.
namespace :teamcity do
task parallel_rspec: :environment do
sh('parallel_rspec spec --serialize-stdout')
end
end
Spork helps minimize the rails load time per-core, only performing it once. On an 8-virtual core rMBP this saved 20 seconds, running 1000 specs that used to take 65 seconds in just under 15 seconds.
The problem is spork seems not to use the separate database instances configured by rake parallel:create so we get a slew of DB lock issues. However, if we configure rspec to use in-memory databases, this problem goes away. This post shows how, but basically just add
setup_sqlite_db = lambda do
ActiveRecord::Base.establish_connection(adapter: 'sqlite3', database: ':memory:')
load "#{Rails.root.to_s}/db/schema.rb" # use db agnostic schema by default
end
silence_stream(STDOUT, &setup_sqlite_db)
to the bottom of your spec_helper.rb, and voila!
parallel_tests, simplecov, and json do not seem to play well together. Caused a number of errors in the form:
/.rvm/gems/ruby-1.9.3-p484@mercury/gems/json-1.8.1/lib/json/common.rb:155:in `parse': 795: unexpected token at 'null, (MultiJson::LoadError)
Until a better workaround is found, remove simplecov from your project.
Note: PARALLEL_TEST_GROUPS is an environment variable which has the number of test groups in use. It is unset if running without parallel_test.
#!/bin/bash
# custom_parallel_script.sh
# Usage: parallel_test --exec ./custom_parallel_script.sh
FILES=$(find . -type f -name \*.rb)
# workaround for the fact that TEST_ENV_NUMBER is '' for the 1st group - default to 1 if unset
LOCAL_TEST_ENV_NUMBER=${TEST_ENV_NUMBER:-1}
i=0
for f in $FILES; do
if [[ -z "$PARALLEL_TEST_GROUPS" || $(($i % $PARALLEL_TEST_GROUPS)) -eq $LOCAL_TEST_ENV_NUMBER ]]; then
echo $f # real action here
fi
((i++))
done
Tear down after all processes are done (RSpec) -- by demental
Say your specs rely on building a js app before suite, and build files must be deleted after suite is finished. before(:suite)
will run after the first process is done, so all the other tests running after that are likely to fail.
As a workaround, you can create a task and redefine rake parallel:spec
require_relative '../../spec/support/test_tools.rb'
namespace :parallel do
desc "Run parallel spec Clear artifact after parallel testing"
task :clean_spec => [:spec, :clean]
desc "Remove artifacts after parallel tests"
task :clean do
TestTools.remove_build_files
end
end
# In spec_helper.rb
config.after(:suite) do
TestTools.remove_build_files unless ENV.key? 'TEST_ENV_NUMBER'
end
# spec/support/test_tools.rb
module TestTools
def self.remove_build_files
# Doing the cleanup here...
FileUtils.rm_rf(Rails.root.join('tmp','test_build'))
end
end
If running regular rspec command, cleanup will be done after(:suite) as usual. Otherwise, it will be run after rake task parallel:spec is finished. Credits goes to seuros for this workaround.
With searchkick -- by emaxi
Set index name based in TEST_ENV_NUMBER for each model using searchkick, example app/model/product.rb
:
class Product < ActiveRecord::Base
searchkick index_name: "products#{ENV['TEST_ENV_NUMBER']}"
end
You can also override the .env
via an initializer:
if Rails.env.test?
Searchkick.env = "test#{ENV['TEST_ENV_NUMBER']}"
end
In 2.2.1 or more you can use Searchkick.index_suffix
Searchkick.index_suffix = ENV["TEST_ENV_NUMBER"]
https://github.com/ankane/searchkick#parallel-tests
With poltergeist -- by emaxi
Use different port for each process spec/spec_helper.rb
:
Capybara.register_driver :poltergeist do |app|
options = {
port: 51674 + ENV['TEST_ENV_NUMBER'].to_i
}
Capybara::Poltergeist::Driver.new(app, options)
end
At file config/mongoid.yml in the test section:
Mongoid >= 5.x
test:
sessions:
default:
database: app_name_test<%= ENV['TEST_ENV_NUMBER'] %>
hosts:
- localhost:27017
Mongoid <= 4.x
test:
client:
default:
database: app_name_test<%= ENV['TEST_ENV_NUMBER'] %>
hosts:
- localhost:27017
With headless -- by sauliusgrigaitis
Headless.new(display: 100, reuse: true, destroy_at_exit: false).start
With SIMPLECOV simplecov -- by a grateful user
To print the simplecov report at the end only, not for each thread.
spec_helper.rb
if ENV['COVERAGE'] == 'true'
require 'simplecov'
require 'simplecov-console'
SimpleCov.formatter = SimpleCov::Formatter::Console
SimpleCov.start 'rails'
if ENV['TEST_ENV_NUMBER'] # parallel specs
SimpleCov.at_exit do
result = SimpleCov.result
result.format! if ParallelTests.number_of_running_processes <= 1
end
end
end
RSpec.configure do |config|
...
gems:
group :test do
gem 'simplecov'
gem 'simplecov-console'
More tips to reduce noise in the output:
spec_helper.rb or rails_helper.rb
RSpec.configure do |config|
...
# mute noise for parallel tests
config.silence_filter_announcements = true if ENV['TEST_ENV_NUMBER']
!! Add your own experience / gotchas !!