brew vendor-gems: commit updates.

This commit is contained in:
BrewTestBot 2021-06-21 18:05:23 +00:00
parent 346d2087f1
commit c1e35b7f8b
No known key found for this signature in database
GPG Key ID: 82D7D104050B0F0F
107 changed files with 473 additions and 73 deletions

View File

@ -81,7 +81,7 @@ $:.unshift "#{path}/../#{ruby_engine}/#{ruby_version}/gems/ruby-progressbar-1.11
$:.unshift "#{path}/../#{ruby_engine}/#{ruby_version}/gems/unicode-display_width-2.0.0/lib" $:.unshift "#{path}/../#{ruby_engine}/#{ruby_version}/gems/unicode-display_width-2.0.0/lib"
$:.unshift "#{path}/../#{ruby_engine}/#{ruby_version}/gems/rubocop-1.17.0/lib" $:.unshift "#{path}/../#{ruby_engine}/#{ruby_version}/gems/rubocop-1.17.0/lib"
$:.unshift "#{path}/../#{ruby_engine}/#{ruby_version}/gems/rubocop-performance-1.11.3/lib" $:.unshift "#{path}/../#{ruby_engine}/#{ruby_version}/gems/rubocop-performance-1.11.3/lib"
$:.unshift "#{path}/../#{ruby_engine}/#{ruby_version}/gems/rubocop-rails-2.10.1/lib" $:.unshift "#{path}/../#{ruby_engine}/#{ruby_version}/gems/rubocop-rails-2.11.0/lib"
$:.unshift "#{path}/../#{ruby_engine}/#{ruby_version}/gems/rubocop-rspec-2.4.0/lib" $:.unshift "#{path}/../#{ruby_engine}/#{ruby_version}/gems/rubocop-rspec-2.4.0/lib"
$:.unshift "#{path}/../#{ruby_engine}/#{ruby_version}/gems/rubocop-sorbet-0.6.2/lib" $:.unshift "#{path}/../#{ruby_engine}/#{ruby_version}/gems/rubocop-sorbet-0.6.2/lib"
$:.unshift "#{path}/../#{ruby_engine}/#{ruby_version}/gems/ruby-macho-2.5.1/lib" $:.unshift "#{path}/../#{ruby_engine}/#{ruby_version}/gems/ruby-macho-2.5.1/lib"

View File

@ -83,6 +83,16 @@ Rails/ActiveSupportAliases:
Enabled: true Enabled: true
VersionAdded: '0.48' VersionAdded: '0.48'
Rails/AddColumnIndex:
Description: >-
Rails migrations don't make use of a given `index` key, but also
doesn't given an error when it's used, so it makes it seem like an
index might be used.
Enabled: pending
VersionAdded: '2.11'
Include:
- db/migrate/*.rb
Rails/AfterCommitOverride: Rails/AfterCommitOverride:
Description: >- Description: >-
This cop enforces that there is only one call to `after_commit` This cop enforces that there is only one call to `after_commit`
@ -193,7 +203,7 @@ Rails/Date:
such as Date.today, Date.current etc. such as Date.today, Date.current etc.
Enabled: true Enabled: true
VersionAdded: '0.30' VersionAdded: '0.30'
VersionChanged: '0.33' VersionChanged: '2.11'
# The value `strict` disallows usage of `Date.today`, `Date.current`, # The value `strict` disallows usage of `Date.today`, `Date.current`,
# `Date#to_time` etc. # `Date#to_time` etc.
# The value `flexible` allows usage of `Date.current`, `Date.yesterday`, etc # The value `flexible` allows usage of `Date.current`, `Date.yesterday`, etc
@ -203,6 +213,7 @@ Rails/Date:
SupportedStyles: SupportedStyles:
- strict - strict
- flexible - flexible
AllowToTime: true
Rails/DefaultScope: Rails/DefaultScope:
Description: 'Avoid use of `default_scope`.' Description: 'Avoid use of `default_scope`.'
@ -240,6 +251,12 @@ Rails/DynamicFindBy:
AllowedReceivers: AllowedReceivers:
- Gem::Specification - Gem::Specification
Rails/EagerEvaluationLogMessage:
Description: 'Checks that blocks are used for interpolated strings passed to `Rails.logger.debug`.'
Reference: 'https://guides.rubyonrails.org/debugging_rails_applications.html#impact-of-logs-on-performance'
Enabled: pending
VersionAdded: '2.11'
Rails/EnumHash: Rails/EnumHash:
Description: 'Prefer hash syntax over array syntax when defining enums.' Description: 'Prefer hash syntax over array syntax when defining enums.'
StyleGuide: 'https://rails.rubystyle.guide#enums' StyleGuide: 'https://rails.rubystyle.guide#enums'
@ -262,8 +279,10 @@ Rails/EnvironmentComparison:
Rails/EnvironmentVariableAccess: Rails/EnvironmentVariableAccess:
Description: 'Do not access `ENV` directly after initialization.' Description: 'Do not access `ENV` directly after initialization.'
Enabled: pending # TODO: Set to `pending` status in RuboCop Rails 2 series when migration doc will be written.
Enabled: false
VersionAdded: '2.10' VersionAdded: '2.10'
VersionChanged: '2.11'
Include: Include:
- app/**/*.rb - app/**/*.rb
- lib/**/*.rb - lib/**/*.rb
@ -286,6 +305,11 @@ Rails/Exit:
Exclude: Exclude:
- lib/**/*.rake - lib/**/*.rake
Rails/ExpandedDateRange:
Description: 'Checks for expanded date range.'
Enabled: pending
VersionAdded: '2.11'
Rails/FilePath: Rails/FilePath:
Description: 'Use `Rails.root.join` for file path joining.' Description: 'Use `Rails.root.join` for file path joining.'
Enabled: true Enabled: true
@ -301,6 +325,8 @@ Rails/FindBy:
StyleGuide: 'https://rails.rubystyle.guide#find_by' StyleGuide: 'https://rails.rubystyle.guide#find_by'
Enabled: true Enabled: true
VersionAdded: '0.30' VersionAdded: '0.30'
VersionChanged: '2.11'
IgnoreWhereFirst: true
Include: Include:
- app/models/**/*.rb - app/models/**/*.rb
@ -362,11 +388,20 @@ Rails/HttpStatus:
Description: 'Enforces use of symbolic or numeric value to define HTTP status.' Description: 'Enforces use of symbolic or numeric value to define HTTP status.'
Enabled: true Enabled: true
VersionAdded: '0.54' VersionAdded: '0.54'
VersionChanged: '2.11'
EnforcedStyle: symbolic EnforcedStyle: symbolic
SupportedStyles: SupportedStyles:
- numeric - numeric
- symbolic - symbolic
Rails/I18nLocaleAssignment:
Description: 'Prefer the usage of `I18n.with_locale` instead of manually updating `I18n.locale` value.'
Enabled: 'pending'
VersionAdded: '2.11'
Include:
- spec/**/*.rb
- test/**/*.rb
Rails/IgnoredSkipActionFilterOption: Rails/IgnoredSkipActionFilterOption:
Description: 'Checks that `if` and `only` (or `except`) are not used together as options of `skip_*` action filter.' Description: 'Checks that `if` and `only` (or `except`) are not used together as options of `skip_*` action filter.'
Reference: 'https://api.rubyonrails.org/classes/AbstractController/Callbacks/ClassMethods.html#method-i-_normalize_callback_options' Reference: 'https://api.rubyonrails.org/classes/AbstractController/Callbacks/ClassMethods.html#method-i-_normalize_callback_options'
@ -784,6 +819,13 @@ Rails/UnknownEnv:
- test - test
- production - production
Rails/UnusedIgnoredColumns:
Description: 'Remove a column that does not exist from `ignored_columns`.'
Enabled: pending
VersionAdded: '2.11'
Include:
- app/models/**/*.rb
Rails/Validation: Rails/Validation:
Description: 'Use validates :attribute, hash of validations.' Description: 'Use validates :attribute, hash of validations.'
Enabled: true Enabled: true

View File

@ -0,0 +1,64 @@
# frozen_string_literal: true
module RuboCop
module Cop
module Rails
# This cop checks for migrations using `add_column` that have an `index`
# key. `add_column` does not accept `index`, but also does not raise an
# error for extra keys, so it is possible to mistakenly add the key without
# realizing it will not actually add an index.
#
# @example
# # bad (will not add an index)
# add_column :table, :column, :integer, index: true
#
# # good
# add_column :table, :column, :integer
# add_index :table, :column
#
class AddColumnIndex < Base
extend AutoCorrector
include RangeHelp
MSG = '`add_column` does not accept an `index` key, use `add_index` instead.'
RESTRICT_ON_SEND = %i[add_column].freeze
# @!method add_column_with_index(node)
def_node_matcher :add_column_with_index, <<~PATTERN
(
send nil? :add_column $_table $_column
<(hash <$(pair {(sym :index) (str "index")} $_) ...>) ...>
)
PATTERN
def on_send(node)
table, column, pair, value = add_column_with_index(node)
return unless pair
add_offense(pair) do |corrector|
corrector.remove(index_range(pair))
add_index = "add_index #{table.source}, #{column.source}"
add_index_opts = ''
if value.hash_type?
hash = value.loc.expression.adjust(begin_pos: 1, end_pos: -1).source.strip
add_index_opts = ", #{hash}"
end
corrector.insert_after(node, "\n#{add_index}#{add_index_opts}")
end
end
private
def index_range(pair_node)
range_with_surrounding_comma(
range_with_surrounding_space(range: pair_node.loc.expression, side: :left),
:left
)
end
end
end
end
end

View File

@ -12,20 +12,21 @@ module RuboCop
# The cop also reports warnings when you are using `to_time` method, # The cop also reports warnings when you are using `to_time` method,
# because it doesn't know about Rails time zone either. # because it doesn't know about Rails time zone either.
# #
# Two styles are supported for this cop. When EnforcedStyle is 'strict' # Two styles are supported for this cop. When `EnforcedStyle` is 'strict'
# then the Date methods `today`, `current`, `yesterday`, and `tomorrow` # then the Date methods `today`, `current`, `yesterday`, and `tomorrow`
# are prohibited and the usage of both `to_time` # are prohibited and the usage of both `to_time`
# and 'to_time_in_current_zone' are reported as warning. # and 'to_time_in_current_zone' are reported as warning.
# #
# When EnforcedStyle is 'flexible' then only `Date.today` is prohibited # When `EnforcedStyle` is `flexible` then only `Date.today` is prohibited.
# and only `to_time` is reported as warning. #
# And you can set a warning for `to_time` with `AllowToTime: false`.
# `AllowToTime` is `true` by default to prevent false positive on `DateTime` object.
# #
# @example EnforcedStyle: strict # @example EnforcedStyle: strict
# # bad # # bad
# Date.current # Date.current
# Date.yesterday # Date.yesterday
# Date.today # Date.today
# date.to_time
# #
# # good # # good
# Time.zone.today # Time.zone.today
@ -34,7 +35,6 @@ module RuboCop
# @example EnforcedStyle: flexible (default) # @example EnforcedStyle: flexible (default)
# # bad # # bad
# Date.today # Date.today
# date.to_time
# #
# # good # # good
# Time.zone.today # Time.zone.today
@ -43,6 +43,13 @@ module RuboCop
# Date.yesterday # Date.yesterday
# date.in_time_zone # date.in_time_zone
# #
# @example AllowToTime: true (default)
# # good
# date.to_time
#
# @example AllowToTime: false
# # bad
# date.to_time
class Date < Base class Date < Base
include ConfigurableEnforcedStyle include ConfigurableEnforcedStyle
@ -73,7 +80,7 @@ module RuboCop
def on_send(node) def on_send(node)
return unless node.receiver && bad_methods.include?(node.method_name) return unless node.receiver && bad_methods.include?(node.method_name)
return if allow_to_time? && node.method?(:to_time)
return if safe_chain?(node) || safe_to_time?(node) return if safe_chain?(node) || safe_to_time?(node)
check_deprecated_methods(node) check_deprecated_methods(node)
@ -139,6 +146,10 @@ module RuboCop
end end
end end
def allow_to_time?
cop_config.fetch('AllowToTime', true)
end
def good_days def good_days
style == :strict ? [] : %i[current yesterday tomorrow] style == :strict ? [] : %i[current yesterday tomorrow]
end end

View File

@ -37,6 +37,7 @@ module RuboCop
MSG = 'Use `%<static_name>s` instead of dynamic `%<method>s`.' MSG = 'Use `%<static_name>s` instead of dynamic `%<method>s`.'
METHOD_PATTERN = /^find_by_(.+?)(!)?$/.freeze METHOD_PATTERN = /^find_by_(.+?)(!)?$/.freeze
IGNORED_ARGUMENT_TYPES = %i[hash splat].freeze
def on_send(node) def on_send(node)
return if node.receiver.nil? && !inherit_active_record_base?(node) || allowed_invocation?(node) return if node.receiver.nil? && !inherit_active_record_base?(node) || allowed_invocation?(node)
@ -44,7 +45,7 @@ module RuboCop
method_name = node.method_name method_name = node.method_name
static_name = static_method_name(method_name) static_name = static_method_name(method_name)
return unless static_name return unless static_name
return if node.arguments.any?(&:splat_type?) return if node.arguments.any? { |argument| IGNORED_ARGUMENT_TYPES.include?(argument.type) }
message = format(MSG, static_name: static_name, method: method_name) message = format(MSG, static_name: static_name, method: method_name)
add_offense(node, message: message) do |corrector| add_offense(node, message: message) do |corrector|

View File

@ -0,0 +1,78 @@
# frozen_string_literal: true
module RuboCop
module Cop
module Rails
# This cop checks that blocks are used for interpolated strings passed to
# `Rails.logger.debug`.
#
# By default, Rails production environments use the `:info` log level.
# At the `:info` log level, `Rails.logger.debug` statements do not result
# in log output. However, Ruby must eagerly evaluate interpolated string
# arguments passed as method arguments. Passing a block to
# `Rails.logger.debug` prevents costly evaluation of interpolated strings
# when no output would be produced anyway.
#
# @example
# #bad
# Rails.logger.debug "The time is #{Time.zone.now}."
#
# #good
# Rails.logger.debug { "The time is #{Time.zone.now}." }
#
class EagerEvaluationLogMessage < Base
extend AutoCorrector
MSG = 'Pass a block to `Rails.logger.debug`.'
RESTRICT_ON_SEND = %i[debug].freeze
def_node_matcher :interpolated_string_passed_to_debug, <<~PATTERN
(send
(send
(const {cbase nil?} :Rails)
:logger
)
:debug
$(dstr ...)
)
PATTERN
def on_send(node)
return if node.parent&.block_type?
interpolated_string_passed_to_debug(node) do |arguments|
message = format(MSG)
range = replacement_range(node)
replacement = replacement_source(node, arguments)
add_offense(range, message: message) do |corrector|
corrector.replace(range, replacement)
end
end
end
private
def replacement_range(node)
stop = node.loc.expression.end
start = node.loc.selector.end
if node.parenthesized_call?
stop.with(begin_pos: start.begin_pos)
else
stop.with(begin_pos: start.begin_pos + 1)
end
end
def replacement_source(node, arguments)
if node.parenthesized_call?
" { #{arguments.source} }"
else
"{ #{arguments.source} }"
end
end
end
end
end
end

View File

@ -0,0 +1,86 @@
# frozen_string_literal: true
module RuboCop
module Cop
module Rails
# This cop checks for expanded date range. It only compatible `..` range is targeted.
# Incompatible `...` range is ignored.
#
# @example
# # bad
# date.beginning_of_day..date.end_of_day
# date.beginning_of_week..date.end_of_week
# date.beginning_of_month..date.end_of_month
# date.beginning_of_quarter..date.end_of_quarter
# date.beginning_of_year..date.end_of_year
#
# # good
# date.all_day
# date.all_week
# date.all_month
# date.all_quarter
# date.all_year
#
class ExpandedDateRange < Base
extend AutoCorrector
extend TargetRailsVersion
MSG = 'Use `%<preferred_method>s` instead.'
minimum_target_rails_version 5.1
def_node_matcher :expanded_date_range, <<~PATTERN
(irange
(send
$_ {:beginning_of_day :beginning_of_week :beginning_of_month :beginning_of_quarter :beginning_of_year})
(send
$_ {:end_of_day :end_of_week :end_of_month :end_of_quarter :end_of_year}))
PATTERN
PREFERRED_METHODS = {
beginning_of_day: 'all_day',
beginning_of_week: 'all_week',
beginning_of_month: 'all_month',
beginning_of_quarter: 'all_quarter',
beginning_of_year: 'all_year'
}.freeze
MAPPED_DATE_RANGE_METHODS = {
beginning_of_day: :end_of_day,
beginning_of_week: :end_of_week,
beginning_of_month: :end_of_month,
beginning_of_quarter: :end_of_quarter,
beginning_of_year: :end_of_year
}.freeze
def on_irange(node)
return unless expanded_date_range(node)
begin_node = node.begin
end_node = node.end
return unless same_receiver?(begin_node, end_node)
beginning_method = begin_node.method_name
end_method = end_node.method_name
return unless use_mapped_methods?(beginning_method, end_method)
preferred_method = "#{begin_node.receiver.source}.#{PREFERRED_METHODS[beginning_method]}"
add_offense(node, message: format(MSG, preferred_method: preferred_method)) do |corrector|
corrector.replace(node, preferred_method)
end
end
private
def same_receiver?(begin_node, end_node)
begin_node.receiver.source == end_node.receiver.source
end
def use_mapped_methods?(beginning_method, end_method)
MAPPED_DATE_RANGE_METHODS[beginning_method] == end_method
end
end
end
end
end

View File

@ -3,16 +3,27 @@
module RuboCop module RuboCop
module Cop module Cop
module Rails module Rails
# This cop is used to identify usages of `where.first` and # This cop is used to identify usages of `where.take` and change them to use `find_by` instead.
# change them to use `find_by` instead. #
# And `where(...).first` can return different results from `find_by`.
# (They order records differently, so the "first" record can be different.)
#
# If you also want to detect `where.first`, you can set `IgnoreWhereFirst` to false.
# #
# @example # @example
# # bad # # bad
# User.where(name: 'Bruce').first
# User.where(name: 'Bruce').take # User.where(name: 'Bruce').take
# #
# # good # # good
# User.find_by(name: 'Bruce') # User.find_by(name: 'Bruce')
#
# @example IgnoreWhereFirst: true (default)
# # good
# User.where(name: 'Bruce').first
#
# @example IgnoreWhereFirst: false
# # bad
# User.where(name: 'Bruce').first
class FindBy < Base class FindBy < Base
include RangeHelp include RangeHelp
extend AutoCorrector extend AutoCorrector
@ -20,12 +31,8 @@ module RuboCop
MSG = 'Use `find_by` instead of `where.%<method>s`.' MSG = 'Use `find_by` instead of `where.%<method>s`.'
RESTRICT_ON_SEND = %i[first take].freeze RESTRICT_ON_SEND = %i[first take].freeze
def_node_matcher :where_first?, <<~PATTERN
(send ({send csend} _ :where ...) {:first :take})
PATTERN
def on_send(node) def on_send(node)
return unless where_first?(node) return if ignore_where_first? && node.method?(:first)
range = range_between(node.receiver.loc.selector.begin_pos, node.loc.selector.end_pos) range = range_between(node.receiver.loc.selector.begin_pos, node.loc.selector.end_pos)
@ -38,9 +45,6 @@ module RuboCop
private private
def autocorrect(corrector, node) def autocorrect(corrector, node)
# Don't autocorrect where(...).first, because it can return different
# results from find_by. (They order records differently, so the
# 'first' record can be different.)
return if node.method?(:first) return if node.method?(:first)
where_loc = node.receiver.loc.selector where_loc = node.receiver.loc.selector
@ -49,6 +53,10 @@ module RuboCop
corrector.replace(where_loc, 'find_by') corrector.replace(where_loc, 'find_by')
corrector.replace(first_loc, '') corrector.replace(first_loc, '')
end end
def ignore_where_first?
cop_config.fetch('IgnoreWhereFirst', true)
end
end end
end end
end end

View File

@ -11,12 +11,14 @@ module RuboCop
# render json: { foo: 'bar' }, status: 200 # render json: { foo: 'bar' }, status: 200
# render plain: 'foo/bar', status: 304 # render plain: 'foo/bar', status: 304
# redirect_to root_url, status: 301 # redirect_to root_url, status: 301
# head 200
# #
# # good # # good
# render :foo, status: :ok # render :foo, status: :ok
# render json: { foo: 'bar' }, status: :ok # render json: { foo: 'bar' }, status: :ok
# render plain: 'foo/bar', status: :not_modified # render plain: 'foo/bar', status: :not_modified
# redirect_to root_url, status: :moved_permanently # redirect_to root_url, status: :moved_permanently
# head :ok
# #
# @example EnforcedStyle: numeric # @example EnforcedStyle: numeric
# # bad # # bad
@ -24,23 +26,26 @@ module RuboCop
# render json: { foo: 'bar' }, status: :not_found # render json: { foo: 'bar' }, status: :not_found
# render plain: 'foo/bar', status: :not_modified # render plain: 'foo/bar', status: :not_modified
# redirect_to root_url, status: :moved_permanently # redirect_to root_url, status: :moved_permanently
# head :ok
# #
# # good # # good
# render :foo, status: 200 # render :foo, status: 200
# render json: { foo: 'bar' }, status: 404 # render json: { foo: 'bar' }, status: 404
# render plain: 'foo/bar', status: 304 # render plain: 'foo/bar', status: 304
# redirect_to root_url, status: 301 # redirect_to root_url, status: 301
# head 200
# #
class HttpStatus < Base class HttpStatus < Base
include ConfigurableEnforcedStyle include ConfigurableEnforcedStyle
extend AutoCorrector extend AutoCorrector
RESTRICT_ON_SEND = %i[render redirect_to].freeze RESTRICT_ON_SEND = %i[render redirect_to head].freeze
def_node_matcher :http_status, <<~PATTERN def_node_matcher :http_status, <<~PATTERN
{ {
(send nil? {:render :redirect_to} _ $hash) (send nil? {:render :redirect_to} _ $hash)
(send nil? {:render :redirect_to} $hash) (send nil? {:render :redirect_to} $hash)
(send nil? :head ${int sym} ...)
} }
PATTERN PATTERN
@ -49,8 +54,12 @@ module RuboCop
PATTERN PATTERN
def on_send(node) def on_send(node)
http_status(node) do |hash_node| http_status(node) do |hash_node_or_status_code|
status = status_code(hash_node) status = if hash_node_or_status_code.hash_type?
status_code(hash_node_or_status_code)
else
hash_node_or_status_code
end
return unless status return unless status
checker = checker_class.new(status) checker = checker_class.new(status)

View File

@ -0,0 +1,37 @@
# frozen_string_literal: true
module RuboCop
module Cop
module Rails
# This cop checks for the use of `I18n.locale=` method.
#
# The `locale` attribute persists for the rest of the Ruby runtime, potentially causing
# unexpected behavior at a later time.
# Using `I18n.with_locale` ensures the code passed in the block is the only place `I18n.locale` is affected.
# It eliminates the possibility of a `locale` sticking around longer than intended.
#
# @example
# # bad
# I18n.locale = :fr
#
# # good
# I18n.with_locale(:fr) do
# end
#
class I18nLocaleAssignment < Base
MSG = 'Use `I18n.with_locale` with block instead of `I18n.locale=`.'
RESTRICT_ON_SEND = %i[locale=].freeze
def_node_matcher :i18n_locale_assignment?, <<~PATTERN
(send (const {nil? cbase} :I18n) :locale= ...)
PATTERN
def on_send(node)
return unless i18n_locale_assignment?(node)
add_offense(node)
end
end
end
end
end

View File

@ -33,19 +33,15 @@ module RuboCop
MSG = 'Do not assign `%<method_name>s` to constants as it ' \ MSG = 'Do not assign `%<method_name>s` to constants as it ' \
'will be evaluated only once.' 'will be evaluated only once.'
RELATIVE_DATE_METHODS = %i[since from_now after ago until before yesterday tomorrow].freeze RELATIVE_DATE_METHODS = %i[since from_now after ago until before yesterday tomorrow].to_set.freeze
def on_casgn(node) def on_casgn(node)
return if node.children[2]&.block_type? nested_relative_date(node) do |method_name|
node.each_descendant(:send) do |send_node|
relative_date?(send_node) do |method_name|
add_offense(node, message: message(method_name)) do |corrector| add_offense(node, message: message(method_name)) do |corrector|
autocorrect(corrector, node) autocorrect(corrector, node)
end end
end end
end end
end
def on_masgn(node) def on_masgn(node)
lhs, rhs = *node lhs, rhs = *node
@ -55,7 +51,7 @@ module RuboCop
lhs.children.zip(rhs.children).each do |(name, value)| lhs.children.zip(rhs.children).each do |(name, value)|
next unless name.casgn_type? next unless name.casgn_type?
relative_date?(value) do |method_name| nested_relative_date(value) do |method_name|
add_offense(offense_range(name, value), message: message(method_name)) do |corrector| add_offense(offense_range(name, value), message: message(method_name)) do |corrector|
autocorrect(corrector, node) autocorrect(corrector, node)
end end
@ -64,7 +60,7 @@ module RuboCop
end end
def on_or_asgn(node) def on_or_asgn(node)
relative_date_or_assignment?(node) do |method_name| relative_date_or_assignment(node) do |method_name|
add_offense(node, message: format(MSG, method_name: method_name)) add_offense(node, message: format(MSG, method_name: method_name))
end end
end end
@ -93,20 +89,22 @@ module RuboCop
range_between(name.loc.expression.begin_pos, value.loc.expression.end_pos) range_between(name.loc.expression.begin_pos, value.loc.expression.end_pos)
end end
def relative_date_method?(method_name) def nested_relative_date(node, &callback)
RELATIVE_DATE_METHODS.include?(method_name) return if node.block_type?
node.each_child_node do |child|
nested_relative_date(child, &callback)
end end
def_node_matcher :relative_date_or_assignment?, <<~PATTERN relative_date(node, &callback)
(:or_asgn (casgn _ _) (send _ $#relative_date_method?)) end
def_node_matcher :relative_date_or_assignment, <<~PATTERN
(:or_asgn (casgn _ _) (send _ $RELATIVE_DATE_METHODS))
PATTERN PATTERN
def_node_matcher :relative_date?, <<~PATTERN def_node_matcher :relative_date, <<~PATTERN
{ (send _ $RELATIVE_DATE_METHODS)
({erange irange} _ (send _ $#relative_date_method?))
({erange irange} (send _ $#relative_date_method?) _)
(send _ $#relative_date_method?)
}
PATTERN PATTERN
end end
end end

View File

@ -50,7 +50,7 @@ module RuboCop
(class (class
(const nil? _) (const nil? _)
(send (send
(const (const nil? :ActiveRecord) :Migration) (const (const {nil? cbase} :ActiveRecord) :Migration)
:[] :[]
(float _)) (float _))
_) _)

View File

@ -8,41 +8,33 @@ module RuboCop
# Built on top of Ruby on Rails style guide (https://rails.rubystyle.guide#time) # Built on top of Ruby on Rails style guide (https://rails.rubystyle.guide#time)
# and the article http://danilenko.org/2012/7/6/rails_timezones/ # and the article http://danilenko.org/2012/7/6/rails_timezones/
# #
# Two styles are supported for this cop. When EnforcedStyle is 'strict' # Two styles are supported for this cop. When `EnforcedStyle` is 'strict'
# then only use of Time.zone is allowed. # then only use of `Time.zone` is allowed.
# #
# When EnforcedStyle is 'flexible' then it's also allowed # When EnforcedStyle is 'flexible' then it's also allowed
# to use Time.in_time_zone. # to use `Time#in_time_zone`.
#
# @example
# # bad
# Time.now
# Time.parse('2015-03-02T19:05:37')
#
# # good
# Time.current
# Time.zone.now
# Time.zone.parse('2015-03-02T19:05:37')
# Time.zone.parse('2015-03-02T19:05:37Z') # Respect ISO 8601 format with timezone specifier.
# #
# @example EnforcedStyle: strict # @example EnforcedStyle: strict
# # `strict` means that `Time` should be used with `zone`. # # `strict` means that `Time` should be used with `zone`.
# #
# # bad # # bad
# Time.now
# Time.parse('2015-03-02T19:05:37')
#
# # bad
# Time.current
# Time.at(timestamp).in_time_zone # Time.at(timestamp).in_time_zone
# #
# # good
# Time.zone.now
# Time.zone.parse('2015-03-02T19:05:37')
# Time.zone.parse('2015-03-02T19:05:37Z') # Respect ISO 8601 format with timezone specifier.
#
# @example EnforcedStyle: flexible (default) # @example EnforcedStyle: flexible (default)
# # `flexible` allows usage of `in_time_zone` instead of `zone`. # # `flexible` allows usage of `in_time_zone` instead of `zone`.
# #
# # bad
# Time.now
# Time.parse('2015-03-02T19:05:37')
#
# # good # # good
# Time.zone.now
# Time.zone.parse('2015-03-02T19:05:37')
#
# # good
# Time.current
# Time.at(timestamp).in_time_zone # Time.at(timestamp).in_time_zone
class TimeZone < Base class TimeZone < Base
include ConfigurableEnforcedStyle include ConfigurableEnforcedStyle
@ -59,7 +51,7 @@ module RuboCop
GOOD_METHODS = %i[zone zone_default find_zone find_zone!].freeze GOOD_METHODS = %i[zone zone_default find_zone find_zone!].freeze
DANGEROUS_METHODS = %i[now local new parse at current].freeze DANGEROUS_METHODS = %i[now local new parse at].freeze
ACCEPTED_METHODS = %i[in_time_zone utc getlocal xmlschema iso8601 ACCEPTED_METHODS = %i[in_time_zone utc getlocal xmlschema iso8601
jisx0301 rfc3339 httpdate to_i to_f].freeze jisx0301 rfc3339 httpdate to_i to_f].freeze

View File

@ -6,8 +6,8 @@ module RuboCop
# This cop checks for the use of `Time.zone=` method. # This cop checks for the use of `Time.zone=` method.
# #
# The `zone` attribute persists for the rest of the Ruby runtime, potentially causing # The `zone` attribute persists for the rest of the Ruby runtime, potentially causing
# unexpected behaviour at a later time. # unexpected behavior at a later time.
# Using `Time.use_zone` ensures the code passed in block is the only place Time.zone is affected. # Using `Time.use_zone` ensures the code passed in the block is the only place Time.zone is affected.
# It eliminates the possibility of a `zone` sticking around longer than intended. # It eliminates the possibility of a `zone` sticking around longer than intended.
# #
# @example # @example
@ -19,7 +19,7 @@ module RuboCop
# end # end
# #
class TimeZoneAssignment < Base class TimeZoneAssignment < Base
MSG = 'Use `Time.use_zone` with blocks instead of `Time.zone=`.' MSG = 'Use `Time.use_zone` with block instead of `Time.zone=`.'
RESTRICT_ON_SEND = %i[zone=].freeze RESTRICT_ON_SEND = %i[zone=].freeze
def_node_matcher :time_zone_assignement?, <<~PATTERN def_node_matcher :time_zone_assignement?, <<~PATTERN

View File

@ -0,0 +1,69 @@
# frozen_string_literal: true
module RuboCop
module Cop
module Rails
# This cop suggests you remove a column that does not exist in the schema from `ignored_columns`.
# `ignored_columns` is necessary to drop a column from RDBMS, but you don't need it after the migration
# to drop the column. You avoid forgetting to remove `ignored_columns` by this cop.
#
# @example
# # bad
# class User < ApplicationRecord
# self.ignored_columns = [:already_removed_column]
# end
#
# # good
# class User < ApplicationRecord
# self.ignored_columns = [:still_existing_column]
# end
#
class UnusedIgnoredColumns < Base
include ActiveRecordHelper
MSG = 'Remove `%<column_name>s` from `ignored_columns` because the column does not exist.'
RESTRICT_ON_SEND = %i[ignored_columns=].freeze
def_node_matcher :ignored_columns, <<~PATTERN
(send self :ignored_columns= $array)
PATTERN
def_node_matcher :column_name, <<~PATTERN
({str sym} $_)
PATTERN
def on_send(node)
return unless (columns = ignored_columns(node))
return unless schema
table = table(node)
return unless table
columns.children.each do |column_node|
check_column_existence(column_node, table)
end
end
private
def check_column_existence(column_node, table)
column_name = column_name(column_node)
return unless column_name
return if table.with_column?(name: column_name.to_s)
message = format(MSG, column_name: column_name)
add_offense(column_node, message: message)
end
def class_node(node)
node.each_ancestor.find(&:class_type?)
end
def table(node)
klass = class_node(node)
schema.table_by(name: table_name(klass))
end
end
end
end
end

Some files were not shown because too many files have changed in this diff Show More