-
-
Notifications
You must be signed in to change notification settings - Fork 83
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
1 parent
1ae01bc
commit d3637be
Showing
5 changed files
with
142 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
* [#474](https://github.com/rubocop/rubocop-performance/pull/474): Add new `Performance/StringBytesize` cop. ([@viralpraxis][]) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,45 @@ | ||
# frozen_string_literal: true | ||
|
||
module RuboCop | ||
module Cop | ||
module Performance | ||
# Checks for calls to `#bytes` counting method and suggests using `bytesize` instead. | ||
# The `bytesize` method is more efficient and directly returns the size in bytes, | ||
# avoiding the intermediate array allocation that `bytes.size` incurs. | ||
# | ||
# @safety | ||
# This cop is unsafe because it assumes that the receiver | ||
# responds to `#bytesize` method. | ||
# | ||
# @example | ||
# # bad | ||
# string_var.bytes.count | ||
# "foobar".bytes.size | ||
# | ||
# # good | ||
# string_var.bytesize | ||
# "foobar".bytesize | ||
class StringBytesize < Base | ||
extend AutoCorrector | ||
|
||
MSG = 'Use `String#bytesize` instead of calculating the size of the bytes array.' | ||
RESTRICT_ON_SEND = %i[size length count].freeze | ||
|
||
def_node_matcher :string_bytes_method?, <<~MATCHER | ||
(call (call !{nil? int} :bytes) {:size :length :count}) | ||
MATCHER | ||
|
||
def on_send(node) | ||
string_bytes_method?(node) do | ||
range = node.receiver.loc.selector.begin.join(node.source_range.end) | ||
|
||
add_offense(range) do |corrector| | ||
corrector.replace(range, 'bytesize') | ||
end | ||
end | ||
end | ||
alias on_csend on_send | ||
end | ||
end | ||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,89 @@ | ||
# frozen_string_literal: true | ||
|
||
RSpec.describe RuboCop::Cop::Performance::StringBytesize, :config do | ||
let(:msg) { 'Use `String#bytesize` instead of calculating the size of the bytes array.' } | ||
|
||
it 'registers an offense with `size` method' do | ||
expect_offense(<<~RUBY) | ||
string.bytes.size | ||
^^^^^^^^^^ #{msg} | ||
RUBY | ||
|
||
expect_correction(<<~RUBY) | ||
string.bytesize | ||
RUBY | ||
end | ||
|
||
it 'registers an offense with `length` method' do | ||
expect_offense(<<~RUBY) | ||
string.bytes.length | ||
^^^^^^^^^^^^ #{msg} | ||
RUBY | ||
|
||
expect_correction(<<~RUBY) | ||
string.bytesize | ||
RUBY | ||
end | ||
|
||
it 'registers an offense with `count` method' do | ||
expect_offense(<<~RUBY) | ||
string.bytes.count | ||
^^^^^^^^^^^ #{msg} | ||
RUBY | ||
|
||
expect_correction(<<~RUBY) | ||
string.bytesize | ||
RUBY | ||
end | ||
|
||
it 'registers an offense with string literal' do | ||
expect_offense(<<~RUBY) | ||
"foobar".bytes.count | ||
^^^^^^^^^^^ #{msg} | ||
RUBY | ||
|
||
expect_correction(<<~RUBY) | ||
"foobar".bytesize | ||
RUBY | ||
end | ||
|
||
it 'registers an offense and autocorrects with safe navigation' do | ||
expect_offense(<<~RUBY) | ||
string&.bytes&.count | ||
^^^^^^^^^^^^ #{msg} | ||
RUBY | ||
|
||
expect_correction(<<~RUBY) | ||
string&.bytesize | ||
RUBY | ||
end | ||
|
||
it 'registers an offense and autocorrects with partial safe navigation' do | ||
expect_offense(<<~RUBY) | ||
string&.bytes.count | ||
^^^^^^^^^^^ #{msg} | ||
RUBY | ||
|
||
expect_correction(<<~RUBY) | ||
string&.bytesize | ||
RUBY | ||
end | ||
|
||
it 'does not register an offense without array size method' do | ||
expect_no_offenses(<<~RUBY) | ||
string.bytes | ||
RUBY | ||
end | ||
|
||
it 'does not register an offense with `bytes` without explicit receiver' do | ||
expect_no_offenses(<<~RUBY) | ||
bytes.size | ||
RUBY | ||
end | ||
|
||
it 'does not register an offense when the receiver is of type `int`' do | ||
expect_no_offenses(<<~RUBY) | ||
3.bytes.size | ||
RUBY | ||
end | ||
end |