Skip to content

Commit

Permalink
Add Rails/SafeNavigationWithBlank cop
Browse files Browse the repository at this point in the history
  • Loading branch information
gyfis committed Nov 11, 2019
1 parent 9a6058f commit 0a4e377
Show file tree
Hide file tree
Showing 9 changed files with 171 additions and 18 deletions.
4 changes: 4 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -6,13 +6,15 @@

* [#123](https://github.com/rubocop-hq/rubocop-rails/pull/123): Add new `Rails/ApplicationController` and `Rails/ApplicationMailer` cops. ([@eugeneius][])
* [#130](https://github.com/rubocop-hq/rubocop-rails/pull/130): Add new `Rails/RakeEnvironment` cop. ([@pocke][])
* [#132](https://github.com/rubocop-hq/rubocop-rails/pull/132): Add new `Rails/SafeNavigationWithBlank` cop. ([@gyfis][])

### Bug fixes

* [#120](https://github.com/rubocop-hq/rubocop-rails/issues/120): Fix message for `Rails/SaveBang` when the save is in the body of a conditional. ([@jas14][])
* [#131](https://github.com/rubocop-hq/rubocop-rails/pull/131): Fix an incorrect autocorrect for `Rails/Presence` when using `[]` method. ([@forresty][])
* [#142](https://github.com/rubocop-hq/rubocop-rails/pull/142): Fix an incorrect autocorrect for `Rails/EnumHash` when using nested constants. ([@koic][])
* [#136](https://github.com/rubocop-hq/rubocop-rails/pull/136): Fix a false positive for `Rails/ReversibleMigration` when using `change_default` with `:from` and `:to` options. ([@sinsoku][])
* [#144](https://github.com/rubocop-hq/rubocop-rails/issues/144): Fix a false positive for `Rails/ReversibleMigration` when using `change_table_comment` or `change_column_comment` with a `:from` and `:to` hash. ([@DNA][])

## 2.3.2 (2019-09-01)

Expand Down Expand Up @@ -97,3 +99,5 @@
[@forresty]: https://github.com/forresty
[@sinsoku]: https://github.com/sinsoku
[@pocke]: https://github.com/pocke
[@gyfis]: https:/github.com/gyfis
[@DNA]: https://github.com/DNA
12 changes: 12 additions & 0 deletions config/default.yml
Original file line number Diff line number Diff line change
Expand Up @@ -414,6 +414,18 @@ Rails/SafeNavigation:
# implement the intended method. `try` will not raise an exception for this.
ConvertTry: false

Rails/SafeNavigationWithBlank:
Description: 'Avoid `foo&.blank?` in conditionals.'
Enabled: true
VersionAdded: '2.4'
# While the safe navigation operator is generally a good idea, when
# checking `foo&.blank?` in a conditional, `foo` being `nil` will actually
# do the opposite of what the author intends.
#
# foo&.blank? #=> nil
# foo.blank? #=> true
SafeAutoCorrect: false

Rails/SaveBang:
Description: 'Identifies possible cases where Active Record save! or related should be used.'
StyleGuide: 'https://rails.rubystyle.guide#save-bang'
Expand Down
31 changes: 13 additions & 18 deletions lib/rubocop/cop/rails/reversible_migration.rb
Original file line number Diff line number Diff line change
Expand Up @@ -129,17 +129,13 @@ class ReversibleMigration < Cop
MSG = '%<action>s is not reversible.'

def_node_matcher :irreversible_schema_statement_call, <<-PATTERN
(send nil? ${:change_table_comment :execute :remove_belongs_to} ...)
(send nil? ${:execute :remove_belongs_to} ...)
PATTERN

def_node_matcher :drop_table_call, <<-PATTERN
(send nil? :drop_table ...)
PATTERN

def_node_matcher :change_column_default_call, <<-PATTERN
(send nil? :change_column_default {[(sym _) (sym _)] (splat _)} $...)
PATTERN

def_node_matcher :remove_column_call, <<-PATTERN
(send nil? :remove_column $...)
PATTERN
Expand All @@ -158,7 +154,7 @@ def on_send(node)

check_irreversible_schema_statement_node(node)
check_drop_table_node(node)
check_change_column_default_node(node)
check_reversible_hash_node(node)
check_remove_column_node(node)
check_remove_foreign_key_node(node)
end
Expand Down Expand Up @@ -190,17 +186,15 @@ def check_drop_table_node(node)
end
end

def check_change_column_default_node(node)
change_column_default_call(node) do |args|
unless all_hash_key?(args.last, :from, :to)
add_offense(
node,
message: format(
MSG, action: 'change_column_default(without :from and :to)'
)
)
end
end
def check_reversible_hash_node(node)
return if reversible_change_table_call?(node)

add_offense(
node,
message: format(
MSG, action: "#{node.method_name}(without :from and :to)"
)
)
end

def check_remove_column_node(node)
Expand Down Expand Up @@ -253,7 +247,8 @@ def reversible_change_table_call?(node)
case node.method_name
when :change, :remove
false
when :change_default
when :change_default, :change_column_default, :change_table_comment,
:change_column_comment
all_hash_key?(node.arguments.last, :from, :to)
else
true
Expand Down
48 changes: 48 additions & 0 deletions lib/rubocop/cop/rails/safe_nagivation_with_blank.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
# frozen_string_literal: true

module RuboCop
module Cop
module Rails
# This cop checks to make sure safe navigation isn't used with `blank?` in
# a conditional.
#
# While the safe navigation operator is generally a good idea, when
# checking `foo&.blank?` in a conditional, `foo` being `nil` will actually
# do the opposite of what the author intends.
#
# @example
# # bad
# do_something if foo&.blank?
# do_something unless foo&.blank?
#
# # good
# do_something if foo.blank?
# do_something unless foo.blank?
#
class SafeNavigationWithBlank < Cop
MSG =
'Avoid calling `blank?` with the safe navigation operator ' \
'in conditionals.'

def_node_matcher :safe_navigation_blank_in_conditional?, <<-PATTERN
(if $(csend ... :blank?) ...)
PATTERN

def on_if(node)
return unless safe_navigation_blank_in_conditional?(node)

add_offense(node)
end

def autocorrect(node)
lambda do |corrector|
corrector.replace(
safe_navigation_blank_in_conditional?(node).location.dot,
'.'
)
end
end
end
end
end
end
1 change: 1 addition & 0 deletions lib/rubocop/cop/rails_cops.rb
Original file line number Diff line number Diff line change
Expand Up @@ -51,6 +51,7 @@
require_relative 'rails/request_referer'
require_relative 'rails/reversible_migration'
require_relative 'rails/safe_navigation'
require_relative 'rails/safe_nagivation_with_blank'
require_relative 'rails/save_bang'
require_relative 'rails/scope_args'
require_relative 'rails/skips_model_validations'
Expand Down
1 change: 1 addition & 0 deletions manual/cops.md
Original file line number Diff line number Diff line change
Expand Up @@ -50,6 +50,7 @@
* [Rails/RequestReferer](cops_rails.md#railsrequestreferer)
* [Rails/ReversibleMigration](cops_rails.md#railsreversiblemigration)
* [Rails/SafeNavigation](cops_rails.md#railssafenavigation)
* [Rails/SafeNavigationWithBlank](cops_rails.md#railssafenavigationwithblank)
* [Rails/SaveBang](cops_rails.md#railssavebang)
* [Rails/ScopeArgs](cops_rails.md#railsscopeargs)
* [Rails/SkipsModelValidations](cops_rails.md#railsskipsmodelvalidations)
Expand Down
25 changes: 25 additions & 0 deletions manual/cops_rails.md
Original file line number Diff line number Diff line change
Expand Up @@ -2144,6 +2144,31 @@ Name | Default value | Configurable values
--- | --- | ---
ConvertTry | `false` | Boolean

## Rails/SafeNavigationWithBlank

Enabled by default | Safe | Supports autocorrection | VersionAdded | VersionChanged
--- | --- | --- | --- | ---
Enabled | Yes | Yes (Unsafe) | 2.4 | -

This cop checks to make sure safe navigation isn't used with `blank?` in
a conditional.

While the safe navigation operator is generally a good idea, when
checking `foo&.blank?` in a conditional, `foo` being `nil` will actually
do the opposite of what the author intends.

### Examples

```ruby
# bad
do_something if foo&.blank?
do_something unless foo&.blank?

# good
do_something if foo.blank?
do_something unless foo.blank?
```

## Rails/SaveBang

Enabled by default | Safe | Supports autocorrection | VersionAdded | VersionChanged
Expand Down
24 changes: 24 additions & 0 deletions spec/rubocop/cop/rails/reversible_migration_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -126,6 +126,30 @@ def change
RUBY
end

context 'change_table_comment' do
it_behaves_like 'accepts',
'change_table_comment(with :from and :to)', <<-RUBY
change_table_comment(:posts, from: nil, to: "draft")
RUBY

it_behaves_like 'offense',
'change_table_comment(without :from and :to)', <<-RUBY
change_table_comment(:suppliers, 'new')
RUBY
end

context 'change_column_comment' do
it_behaves_like 'accepts',
'change_column_comment(with :from and :to)', <<-RUBY
change_column_comment(:posts, :state, from: nil, to: "draft")
RUBY

it_behaves_like 'offense',
'change_column_comment(without :from and :to)', <<-RUBY
change_column_comment(:suppliers, :qualification, 'new')
RUBY
end

context 'remove_column' do
it_behaves_like 'accepts', 'remove_column(with type)', <<-RUBY
remove_column(:suppliers, :qualification, :string)
Expand Down
43 changes: 43 additions & 0 deletions spec/rubocop/cop/rails/safe_navigation_with_blank_spec.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
# frozen_string_literal: true

RSpec.describe RuboCop::Cop::Rails::SafeNavigationWithBlank do
subject(:cop) { described_class.new }

context 'in a conditional' do
it 'registers an offense on a single conditional' do
expect_offense(<<~RUBY)
do_something unless foo&.blank?
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ Avoid calling `blank?` with the safe navigation operator in conditionals.
RUBY

expect_correction(<<~RUBY)
do_something unless foo.blank?
RUBY
end

it 'registers an offense on chained conditionals' do
expect_offense(<<~RUBY)
do_something unless foo&.bar&.blank?
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ Avoid calling `blank?` with the safe navigation operator in conditionals.
RUBY

expect_correction(<<~RUBY)
do_something unless foo&.bar.blank?
RUBY
end

it 'does not register an offense on `.blank?`' do
expect_no_offenses(<<~RUBY)
return if foo.blank?
RUBY
end
end

context 'outside a conditional' do
it 'registers no offense' do
expect_no_offenses(<<~RUBY)
bar = foo&.blank?
RUBY
end
end
end

0 comments on commit 0a4e377

Please sign in to comment.