-
-
Notifications
You must be signed in to change notification settings - Fork 81
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add new
Performance/StringBytesize
cop
- Loading branch information
1 parent
1ae01bc
commit 15a940b
Showing
4 changed files
with
116 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
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,48 @@ | ||
# 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 | ||
(send (send !nil? :bytes) {:size :length :count}) | ||
MATCHER | ||
|
||
def on_send(node) | ||
string_bytes_method?(node) do | ||
add_offense(node) do |corrector| | ||
corrector.replace(node, replacement(node)) | ||
end | ||
end | ||
end | ||
|
||
def replacement(node) | ||
receiver = node.receiver.receiver.source | ||
|
||
"#{receiver}.bytesize" | ||
end | ||
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,61 @@ | ||
# 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 'does not register an offenses without array size method' do | ||
expect_no_offenses(<<~RUBY) | ||
string.bytes | ||
RUBY | ||
end | ||
|
||
it 'does not register an offenses with `bytes` without explicit receiver' do | ||
expect_no_offenses(<<~RUBY) | ||
bytes.size | ||
RUBY | ||
end | ||
end |