forked from rubocop/rubocop-rails
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbelongs_to.rb
90 lines (83 loc) · 3.25 KB
/
belongs_to.rb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
# frozen_string_literal: true
module RuboCop
module Cop
module Rails
# Looks for belongs_to associations where we control whether the
# association is required via the deprecated `required` option instead.
#
# Since Rails 5, belongs_to associations are required by default and this
# can be controlled through the use of `optional: true`.
#
# From the release notes:
#
# belongs_to will now trigger a validation error by default if the
# association is not present. You can turn this off on a
# per-association basis with optional: true. Also deprecate required
# option in favor of optional for belongs_to. (Pull Request)
#
# In the case that the developer is doing `required: false`, we
# definitely want to autocorrect to `optional: true`.
#
# However, without knowing whether they've set overridden the default
# value of `config.active_record.belongs_to_required_by_default`, we
# can't say whether it's safe to remove `required: true` or whether we
# should replace it with `optional: false` (or, similarly, remove a
# superfluous `optional: false`). Therefore, in the cases we're using
# `required: true`, we'll simply invert it to `optional: false` and the
# user can remove depending on their defaults.
#
# @example
# # bad
# class Post < ApplicationRecord
# belongs_to :blog, required: false
# end
#
# # good
# class Post < ApplicationRecord
# belongs_to :blog, optional: true
# end
#
# # bad
# class Post < ApplicationRecord
# belongs_to :blog, required: true
# end
#
# # good
# class Post < ApplicationRecord
# belongs_to :blog, optional: false
# end
class BelongsTo < Base
extend AutoCorrector
extend TargetRailsVersion
minimum_target_rails_version 5.0
SUPERFLOUS_REQUIRE_FALSE_MSG =
'You specified `required: false`, in Rails > 5.0 the required ' \
'option is deprecated and you want to use `optional: true`.'
SUPERFLOUS_REQUIRE_TRUE_MSG =
'You specified `required: true`, in Rails > 5.0 the required ' \
'option is deprecated and you want to use `optional: false`. ' \
'In most configurations, this is the default and you can omit ' \
'this option altogether'
RESTRICT_ON_SEND = %i[belongs_to].freeze
def_node_matcher :match_belongs_to_with_options, <<~PATTERN
(send _ :belongs_to ...
(hash <$(pair (sym :required) ${true false}) ...>)
)
PATTERN
def on_send(node)
match_belongs_to_with_options(node) do |option_node, option_value|
message, replacement =
if option_value.true_type?
[SUPERFLOUS_REQUIRE_TRUE_MSG, 'optional: false']
elsif option_value.false_type?
[SUPERFLOUS_REQUIRE_FALSE_MSG, 'optional: true']
end
add_offense(node.loc.selector, message: message) do |corrector|
corrector.replace(option_node, replacement)
end
end
end
end
end
end
end