mrspec
Runs Minitest tests using RSpec's runner. Also runs RSpec's tests, so if you want to use them side-by-side, this will do it.
Pronounciation
"Mister Spec"
Examples
Run specs and tests in tandem
It matches test/*_test.rb
, test/test_*.rb
, spec/*_spec.rb
.
The RSpec group description is the class name without Test
prefix/suffix.
The example name is the method name without the test_
prefix,
and with underscores switched to spaces.
It finds test classes and test methods by asking Minitest what it is tracking.
# file: spec/a_spec.rb
RSpec.describe 'An RSpec test' do
it('does rspec things') { }
end
# file: test/b_test.rb
class AMinitestTest < Minitest::Test
def test_it_does_minitesty_things
end
end
# file: test/test_c.rb
class TestSomethingElse < Minitest::Test
def some_helper_method # won't show up
end
def test_this_also_does_minitesty_things
end
end
# Added because Minitest::Runnable knows about it
class AnotherTestWithNeitherThePrefixNorTheSuffix < Minitest::Test
# This causes Minitest to consider it a test, so we consider it one
def self.runnables
['is_this_a_test_yes_it_is']
end
def is_this_a_test_yes_it_is
end
end
# Ignored b/c Minitest::Runnable doesn't know about it
class NotATest
def test_whatevz
end
end
# file: test/d_spec.rb
require 'minitest/spec'
describe 'I am a minitest spec' do
it 'does minitesty things' do
assert_includes self.class.ancestors, Minitest::Spec
end
end
Failures, Errors, Skips
It understands Minitest skips and errors/failures.
# file: various_errors.rb
class VariousErrors < Minitest::Test
def test_this_passes() assert true end
def test_they_arent_equal() assert_equal 1, 2 end
def test_is_not_included() assert_includes %w[a b c], 'd' end
def test_skipped_becauseā¦_reasons() skip end
end
I used --format progress
here, because there's enough errors that my default documentation
formatter makes it spammy >.<
Fail Fast and filtering
The --fail-fast
flag is a favourite of mine. It continues running tests until it sees a failure, then it stops.
We can also use tags to filter which tests to run. Mrspec adds RSpec metadata to Minitest classes and tests, the metadata behaves as a tag.
The best thing about tags is they're easy to add, and they continue to apply to the same test, when it moves around (line numbers change), They stay correct even if I rename it! I won't have to tweak my command-line invocation until I've got the test passing!
# file: test.rb
# Here, we tag the `NoFailures` class with `them_passing_tests`
# And the `TwoFailures` class with `them_failing_tests`
class NoFailures < Minitest::Test
classmeta them_passing_tests: true
def test_1() end
def test_2() end
end
class TwoFailures < Minitest::Test
classmeta them_failing_tests: true
# I like short tagnames, b/c usually my use is transient.
# I just keep them around until they're fixed.
meta f1: true
def test_3
raise 'first failure'
end
meta f2: true
def test_4
raise 'second failure'
end
end
Default configuration
You can place a file named .rspec
in your home directory with command-line arguments in it.
These will be used as defaults when you run mrspec
or rspec
.
Here is mine.
Why?
The default way to run minitest tests is with Rake. And if you have multiple suites, that can be nice, or if you already use Rake, then it's not adding a new dependency. But here are some frustrations I have with it as a test runner:
- I don't want to add a dependency on Rake, unless I need it.
- The
Rake::TestTask
is difficult to make sense of. - The Rake tasks ultimately just shell out (RSpec's, Minitest's). So I don't see what they offer over invoking the program directly (usually I know what I want to pass the program, and I am trying to figure out how to configure the test task to do that). The overhead of running additional processes can also be high: think how long Rails takes to start up, now imagine paying that twice every time you want to run your tests!
- It makes it difficult to dynamically alter my test invocation.
With Minitest, you can pass
-n test_something
and it will only run the test namedtest_something
, but now I have to edit code to make that happen.
Furthermore, if someone doesn't know about the test task, or it seems formidable, as it often does to new students (I'm a teacher), then they won't use it. They instead run files one at a time. When I go to run the tests, they don't have a way to run all of them. This overhead, in turn, disinclines them to run the tests, such that they may be failing and not realize it.
Anyway, all of this is to say that Minitest needs a runner. I hear Rails is working on one, but I don't know when that'll be available, or if it will be written in a way that it can be used outside of Rails.
But the RSpec runner is very nice, it has a lot of features that I use frequently. Someone suggested running Minitest with the RSpec runner (see attribution section), and I thought that was an interesting idea that could have value if it worked. ...so, here we are.
Nuances
Changes the default pattern to look for any files suffixed with _test.rb
or _spec.rb
, or prefixed with test_
(RSpec, by itself, only looks for suffixes of _spec.rb
).
Changes the default search directories to be test
and spec
(RSpec, by itself, only looks in spec
).
Adds test
to the $LOAD_PATH
, if it exists.
(RSpec, by itself, only adds spec
).
Turns off monkey patching, so you cannot use RSPec's toplevel describe, or should
.
There are 2 reasons for this:
- It conflicts with
Minitest::Spec
's definition ofKernel#describe
(here). And must be preemptively turned off, because after-the-fact disabling causes it to be undefined on bothmain
andModule
(here), which means that even if you don't use it, it will still interfere withMinitest::Spec
(removing methods allows method lookup to find superclass definitions, but undefining them ends method lookup.) - You should avoid things like this, in general. Monkey patching is a bad plan, all around, just use the namespaced methods, or create your own methods to wrap the assertion syntax.
Only expected to support Rubies in the .travis.yml.
Running the tests
$ bundle
$ bundle exec bin/mrspec
$ bundle exec cucumber
Why are all the tests written in Cucumber? Well... mostly just b/c I initially wrote this as a script for my dotfiles, which I mostly test with Cucumber and Haiti, as they are usually heavily oriented towards integration, and often not written in Ruby.
What about the test
directory?
I decided to describe all the behaviour that can be unit tested,
but haven't taken the time to implement most of them yet,
because I don't currently have any features I'm trying to add.
As I maintain this, though, I'll begin implementing them,
as it will be easier in the end :)
Run mrpsec as the Rake Test Task
Add this to your Rakefile
in your Rails App Root Dir
belowRails.application.load_tasks
# ----- To run minitest is the bit that I added -----
# I know it's ridiculous, but there isn't a better way, it's what RSpec does, too:
# https://github.com/rspec/rspec-rails/blob/682a12067eab233c646057f984692e3b70749f32/lib/rspec/rails/tasks/rspec.rake#L2-L4
tasks = Rake.application.instance_variable_get('@tasks')
tasks['test'].clear_actions if tasks['test']
tasks['spec'].clear_actions if tasks['spec']
mrspec = Proc.new do
sh 'mrspec', '--fail-fast'
end
task :test, &mrspec
task :spec, &mrspec
Attribution
Idea from e2, proposed here. Iniitial code was based off of this gist
MIT License
Copyright (c) 2015 Josh Cheek
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.