Upgrade to Version 2.x
Configuration File Update
In version 2.x:
-
RSpec/InvalidPredicateMatcher
cop is removed -
CustomIncludeMethods
configuration option forRSpec/EmptyExampleGroup
is removed -
cop departments are nested for cops with a department that doesn’t match the extension name (
Capybara
,FactoryBot
,Rails
) -
AllCops/RSpec/Patterns
/AllCops/FactoryBot/Patterns
options are removed -
Calling
super
from#on_new_investigation
defined in a cop is mandatory now -
In specs, do not define
cop
Adjust the configuration of RSpec/EmptyExampleGroup
# .rubocop.yml
# Before
RSpec/EmptyExampleGroup:
CustomIncludeMethods:
- include_tests
# After
RSpec:
Language:
Includes:
Examples:
- include_tests
Add a top-level RSpec
department
RuboCop extensions had cops with clashing names and departments, e.g. both rspec-rails
and rubocop-rspec
had Rails::HttpStatus
cops.
To avoid issues, e.g. inability to disable just one of the cops, each extension now has its own uber-department.
Expectedly, RuboCop RSpec’s uber-department name is RSpec
.
Changes are only applied to cops that don’t already have the department set to RSpec
, i.e. Capybara
, FactoryBot
and Rails
.
# .rubocop.yml
# Before
Capybara/CurrentPathExpectation:
Enabled: false
FactoryBot/AttributeDefinedStatically:
Enabled: false
# remains the same
RSpec/EmptyExampleGroup:
Enabled: false
# After
RSpec/Capybara/CurrentPathExpectation:
Enabled: false
RSpec/FactoryBot/AttributeDefinedStatically:
Enabled: false
# remains the same
RSpec/EmptyExampleGroup:
Enabled: false
Use the RuboCop standard Include
option to filter inspected files
Patterns
was a RuboCop RSpec-specific option, and RuboCop has a standard replacement.
# .rubocop.yml
# Before
AllCops:
RSpec/FactoryBot:
Patterns:
- spec/factories/**/*.rb
- property/factories/**/*.rb
# After
RSpec/FactoryBot:
Include:
- spec/factories/**/*.rb
- property/factories/**/*.rb
Please keep in mind that Include’s merge mode is set to override the default settings, so if you intend to add a path while keeping the default paths, you should include the default `Include paths in your configuration.
|
Custom Cop Update Guide
Due to significant API changes, custom cops may break. Here is the summary of the changes:
-
The base class for cops is now
RuboCop::Cop::RSpec::Base
instead ofRuboCop::Cop::RSpec::Cop
. -
The module
RuboCop::Cop::RSpec::TopLevelDescribe
is replaced with a more genericRuboCop::Cop::RSpec::TopLevelGroup
. -
RuboCop::RSpec::Language
has been completely rewritten to support dynamic RSpec DSL aliases and negated matchers to fully support third-party libraries such as RSpec Rails, Pundit, Action Policy and many others. -
RuboCop RSpec updated the dependency of RuboCop to 1.0+.
Below are the necessary steps to update custom cops to work with rubocop-rspec
version 2.x.
Change the Parent Class
Change the parent class of the custom cops from RuboCop::Cop::RSpec::Cop
to RuboCop::Cop::RSpec::Base
.
# Before
module RuboCop
module Cop
module RSpec
class FightPowerty < Cop
# After
module RuboCop
module Cop
module RSpec
class FightPowerty < Base
Replace TopLevelDescribe
TopLevelDescribe
was incomplete, had poor performance and did not distinguish between example groups and shared example groups.
TopLevelGroup
provides a similar interface, but instead of a single on_top_level_describe
hook there are two, on_top_level_example_group
and on_top_level_group
.
There’s no need yet for on_top_level_shared_group
for RuboCop core cops, but if your custom cop needs such a hook, please feel free to send a pull request.
Additionally, single_top_level_describe?
is removed with no direct replacement.
You may use top_level_groups
query method instead, e.g. top_level_groups.one?
.
Change the Language
Module Usages
To allow for lazy initialization, and for loading of the language configuration after the class are loaded, a function call feature of RuboCop AST is used.
The RuboCop::RSpec::Language
is completely different now.
Hooks::ALL
and alike, and their accompanying helpers work differently.
# Before
def_node_matcher :shared_context,
SharedGroups::CONTEXT.block_pattern
# After
def_node_matcher :shared_context,
block_pattern('#SharedGroups.context')
# Before
def_node_search :examples?,
(Includes::EXAMPLES + Examples::ALL).send_pattern
# After
def_node_search :examples?,
send_pattern('{#Includes.examples #Examples.all}')
# Before
def_node_search :find_rspec_blocks,
ExampleGroups::ALL.block_pattern
# After
def_node_search :find_rspec_blocks,
block_pattern('#ExampleGroups.all')
If you were calling Language elements directly, you have to make the same adjustments:
# Before
node&.sym_type? && Hooks::Scopes::ALL.include?(node.value)
# After
node&.sym_type? && Language::HookScopes.all(node.value)
You may see a common pattern in the change. There is a small exception, though:
# Before
ExampleGroups::GROUPS
# After
ExampleGroups.regular
# Before
Examples::EXAMPLES
# After
Examples.regular
Always call super
from on_new_investigation
in your cops
on_new_investigation
is now used for internal purposes, and not calling super
from your cop involves a risk of configuration not being properly loaded, and dynamic RSpec DSL matchers won’t work.
You don’t have to define on_new_investigation in your cops unless you need to.
|
module RuboCop
module Cop
module RSpec
class MultipleMemoizedHelpers < Base
def on_new_investigation
super # Always call `super`
@example_group_memoized_helpers = {}
end
end
end
end
end
Use :config
RSpec metadata in cop specs
:config
metadata should be added to the top-level example group of your cop spec.
Doing otherwise will not pass configuration to the cop, and dynamic RSpec DSL matchers might not work.
# Before
RSpec.describe 'MyMightyCop' do
let(:cop) { described_class.new }
# ...
end
# After
RSpec.describe 'MyMightyCop', :config do
# `cop` is defined for you by RuboCop's shared context that is included
# to example groups with :config metadata
# ...
end
Conform with RuboCop API Changes
The parent project, RuboCop, has API changes. While they won’t result in cop breakages, it is recommended to update cops to use new API’s. Follow the RuboCop v1 update guide to adjust custom cops’ use of RuboCop’s auto-correction API.