Commit 6190421d authored by Qingyu Zhao's avatar Qingyu Zhao Committed by Dmytro Zaporozhets

Add rubocop cop for .keys.first and .values.first

`.keys.first` and `.values.first` load all keys/values into memory.
This cop use `.each_key.first` and `.each_value.first` to reduce memory
usage and execution time.
parent 9bf12de4
...@@ -42,7 +42,7 @@ module Projects ...@@ -42,7 +42,7 @@ module Projects
# Deletes the dummy image # Deletes the dummy image
# All created tag digests are the same since they all have the same dummy image. # All created tag digests are the same since they all have the same dummy image.
# a single delete is sufficient to remove all tags with it # a single delete is sufficient to remove all tags with it
if deleted_tags.any? && container_repository.delete_tag_by_digest(deleted_tags.values.first) if deleted_tags.any? && container_repository.delete_tag_by_digest(deleted_tags.each_value.first)
success(deleted: deleted_tags.keys) success(deleted: deleted_tags.keys)
else else
error('could not delete tags') error('could not delete tags')
......
...@@ -282,7 +282,7 @@ class License < ApplicationRecord ...@@ -282,7 +282,7 @@ class License < ApplicationRecord
end end
def data_filename def data_filename
company_name = self.licensee["Company"] || self.licensee.values.first company_name = self.licensee["Company"] || self.licensee.each_value.first
clean_company_name = company_name.gsub(/[^A-Za-z0-9]/, "") clean_company_name = company_name.gsub(/[^A-Za-z0-9]/, "")
"#{clean_company_name}.gitlab-license" "#{clean_company_name}.gitlab-license"
end end
......
...@@ -45,7 +45,7 @@ module Packages ...@@ -45,7 +45,7 @@ module Packages
def version def version
strong_memoize(:version) do strong_memoize(:version) do
params[:versions].keys.first params[:versions].each_key.first
end end
end end
...@@ -58,7 +58,7 @@ module Packages ...@@ -58,7 +58,7 @@ module Packages
end end
def dist_tag def dist_tag
params['dist-tags'].keys.first params['dist-tags'].each_key.first
end end
def package_file_name def package_file_name
......
...@@ -31,7 +31,7 @@ module Audit ...@@ -31,7 +31,7 @@ module Audit
def action_text def action_text
action = @details.slice(*ACTIONS) action = @details.slice(*ACTIONS)
case action.keys.first case action.each_key.first
when :add when :add
"Added #{target_name}#{@details[:as] ? " as #{@details[:as]}" : ''}" "Added #{target_name}#{@details[:as] ? " as #{@details[:as]}" : ''}"
when :remove when :remove
...@@ -103,7 +103,7 @@ module Audit ...@@ -103,7 +103,7 @@ module Audit
end end
def detail_value def detail_value
@details.values.first @details.each_value.first
end end
def oauth_label def oauth_label
......
...@@ -26,7 +26,7 @@ module Gitlab ...@@ -26,7 +26,7 @@ module Gitlab
attr_reader :group, :params, :finder attr_reader :group, :params, :finder
def finder_class def finder_class
FINDER_CLASSES.fetch(params[:subject], FINDER_CLASSES.keys.first) FINDER_CLASSES.fetch(params[:subject], FINDER_CLASSES.each_key.first)
end end
def format_result(result) def format_result(result)
......
...@@ -58,7 +58,7 @@ describe "Admin uploads license" do ...@@ -58,7 +58,7 @@ describe "Admin uploads license" do
attach_and_upload(path) attach_and_upload(path)
expect(page).to have_content("The license was successfully uploaded and is now active.") expect(page).to have_content("The license was successfully uploaded and is now active.")
.and have_content(license.licensee.values.first) .and have_content(license.licensee.each_value.first)
end end
end end
......
...@@ -57,7 +57,7 @@ describe "Admin views license" do ...@@ -57,7 +57,7 @@ describe "Admin views license" do
end end
it "shows only expiration duration" do it "shows only expiration duration" do
expect(page).to have_content(license.licensee.values.first) expect(page).to have_content(license.licensee.each_value.first)
page.within(".js-license-info-panel") do page.within(".js-license-info-panel") do
expect(page).not_to have_content("Expires: Free trial will expire in") expect(page).not_to have_content("Expires: Free trial will expire in")
...@@ -83,7 +83,7 @@ describe "Admin views license" do ...@@ -83,7 +83,7 @@ describe "Admin views license" do
license_history = page.find("#license_history") license_history = page.find("#license_history")
License.previous.each do |license| License.previous.each do |license|
expect(license_history).to have_content(license.licensee.values.first) expect(license_history).to have_content(license.licensee.each_value.first)
end end
end end
end end
......
...@@ -15,7 +15,7 @@ describe Packages::CreateDependencyService do ...@@ -15,7 +15,7 @@ describe Packages::CreateDependencyService do
.gsub('1.0.1', version)) .gsub('1.0.1', version))
.with_indifferent_access .with_indifferent_access
end end
let(:package_version) { params[:versions].keys.first } let(:package_version) { params[:versions].each_key.first }
let(:dependencies) { params[:versions][package_version] } let(:dependencies) { params[:versions][package_version] }
let(:package) { create(:npm_package) } let(:package) { create(:npm_package) }
let(:dependency_names) { package.dependency_links.flat_map(&:dependency).map(&:name).sort } let(:dependency_names) { package.dependency_links.flat_map(&:dependency).map(&:name).sort }
......
...@@ -9,7 +9,7 @@ describe Geo::DesignRepositoryShardSyncWorker, :geo, :geo_fdw, :clean_gitlab_red ...@@ -9,7 +9,7 @@ describe Geo::DesignRepositoryShardSyncWorker, :geo, :geo_fdw, :clean_gitlab_red
let!(:primary) { create(:geo_node, :primary) } let!(:primary) { create(:geo_node, :primary) }
let!(:secondary) { create(:geo_node) } let!(:secondary) { create(:geo_node) }
let(:shard_name) { Gitlab.config.repositories.storages.keys.first } let(:shard_name) { Gitlab.config.repositories.storages.each_key.first }
before do before do
stub_current_geo_node(secondary) stub_current_geo_node(secondary)
......
...@@ -9,7 +9,7 @@ describe Geo::RepositoryShardSyncWorker, :geo, :geo_fdw, :clean_gitlab_redis_cac ...@@ -9,7 +9,7 @@ describe Geo::RepositoryShardSyncWorker, :geo, :geo_fdw, :clean_gitlab_redis_cac
let!(:primary) { create(:geo_node, :primary) } let!(:primary) { create(:geo_node, :primary) }
let!(:secondary) { create(:geo_node) } let!(:secondary) { create(:geo_node) }
let(:shard_name) { Gitlab.config.repositories.storages.keys.first } let(:shard_name) { Gitlab.config.repositories.storages.each_key.first }
before do before do
stub_current_geo_node(secondary) stub_current_geo_node(secondary)
......
...@@ -7,7 +7,7 @@ describe Geo::RepositoryVerification::Primary::ShardWorker, :clean_gitlab_redis_ ...@@ -7,7 +7,7 @@ describe Geo::RepositoryVerification::Primary::ShardWorker, :clean_gitlab_redis_
include ExclusiveLeaseHelpers include ExclusiveLeaseHelpers
let!(:primary) { create(:geo_node, :primary) } let!(:primary) { create(:geo_node, :primary) }
let(:shard_name) { Gitlab.config.repositories.storages.keys.first } let(:shard_name) { Gitlab.config.repositories.storages.each_key.first }
let(:primary_singleworker) { Geo::RepositoryVerification::Primary::SingleWorker } let(:primary_singleworker) { Geo::RepositoryVerification::Primary::SingleWorker }
before do before do
......
...@@ -7,7 +7,7 @@ describe Geo::RepositoryVerification::Secondary::ShardWorker, :geo, :geo_fdw, :r ...@@ -7,7 +7,7 @@ describe Geo::RepositoryVerification::Secondary::ShardWorker, :geo, :geo_fdw, :r
include ExclusiveLeaseHelpers include ExclusiveLeaseHelpers
let!(:secondary) { create(:geo_node) } let!(:secondary) { create(:geo_node) }
let(:shard_name) { Gitlab.config.repositories.storages.keys.first } let(:shard_name) { Gitlab.config.repositories.storages.each_key.first }
let(:secondary_single_worker) { Geo::RepositoryVerification::Secondary::SingleWorker } let(:secondary_single_worker) { Geo::RepositoryVerification::Secondary::SingleWorker }
let!(:project) { create(:project) } let!(:project) { create(:project) }
......
...@@ -7,7 +7,7 @@ describe Geo::Secondary::RepositoryBackfillWorker, :geo, :geo_fdw, :clean_gitlab ...@@ -7,7 +7,7 @@ describe Geo::Secondary::RepositoryBackfillWorker, :geo, :geo_fdw, :clean_gitlab
let(:primary) { create(:geo_node, :primary) } let(:primary) { create(:geo_node, :primary) }
let(:secondary) { create(:geo_node, repos_max_capacity: 5) } let(:secondary) { create(:geo_node, repos_max_capacity: 5) }
let(:shard_name) { Gitlab.config.repositories.storages.keys.first } let(:shard_name) { Gitlab.config.repositories.storages.each_key.first }
before do before do
stub_current_geo_node(secondary) stub_current_geo_node(secondary)
......
module RuboCop
module Cop
module Gitlab
class KeysFirstAndValuesFirst < RuboCop::Cop::Cop
FIRST_PATTERN = /\Afirst\z/.freeze
def message(used_method)
<<~MSG
Don't use `.keys.first` and `.values.first`.
Instead use `.each_key.first` and `.each_value.first` (or `.first.first` and `first.second`)
This will reduce memory usage and execution time.
MSG
end
def on_send(node)
if find_on_keys_or_values?(node)
add_offense(node, location: :selector, message: message(node.method_name))
end
end
def autocorrect(node)
lambda do |corrector|
replace_with = if node.descendants.first.method_name == :values
'.each_value'
elsif node.descendants.first.method_name == :keys
'.each_key'
else
throw("Expect '.values.first' or '.keys.first', but get #{node.descendants.first.method_name}.first")
end
upto_including_keys_or_values = node.descendants.first.source_range
before_keys_or_values = node.descendants[1].source_range
range_to_replace = node.source_range
.with(begin_pos: before_keys_or_values.end_pos,
end_pos: upto_including_keys_or_values.end_pos)
corrector.replace(range_to_replace, replace_with)
end
end
def find_on_keys_or_values?(node)
chained_on_node = node.descendants.first
node.method_name.to_s =~ FIRST_PATTERN &&
chained_on_node.is_a?(RuboCop::AST::SendNode) &&
[:keys, :values].include?(chained_on_node.method_name) &&
node.descendants[1]
end
def method_name_for_node(node)
children[1].to_s
end
end
end
end
end
...@@ -5,6 +5,7 @@ require_relative 'cop/gitlab/httparty' ...@@ -5,6 +5,7 @@ require_relative 'cop/gitlab/httparty'
require_relative 'cop/gitlab/finder_with_find_by' require_relative 'cop/gitlab/finder_with_find_by'
require_relative 'cop/gitlab/union' require_relative 'cop/gitlab/union'
require_relative 'cop/gitlab/rails_logger' require_relative 'cop/gitlab/rails_logger'
require_relative 'cop/gitlab/keys-first-and-values-first'
require_relative 'cop/include_sidekiq_worker' require_relative 'cop/include_sidekiq_worker'
require_relative 'cop/safe_params' require_relative 'cop/safe_params'
require_relative 'cop/active_record_association_reload' require_relative 'cop/active_record_association_reload'
......
...@@ -13,7 +13,7 @@ describe UserCalloutsController do ...@@ -13,7 +13,7 @@ describe UserCalloutsController do
subject { post :create, params: { feature_name: feature_name }, format: :json } subject { post :create, params: { feature_name: feature_name }, format: :json }
context 'with valid feature name' do context 'with valid feature name' do
let(:feature_name) { UserCallout.feature_names.keys.first } let(:feature_name) { UserCallout.feature_names.each_key.first }
context 'when callout entry does not exist' do context 'when callout entry does not exist' do
it 'creates a callout entry with dismissed state' do it 'creates a callout entry with dismissed state' do
...@@ -28,7 +28,7 @@ describe UserCalloutsController do ...@@ -28,7 +28,7 @@ describe UserCalloutsController do
end end
context 'when callout entry already exists' do context 'when callout entry already exists' do
let!(:callout) { create(:user_callout, feature_name: UserCallout.feature_names.keys.first, user: user) } let!(:callout) { create(:user_callout, feature_name: UserCallout.feature_names.each_key.first, user: user) }
it 'returns success' do it 'returns success' do
subject subject
......
...@@ -9,7 +9,7 @@ describe Gitlab::ImportExport::HashUtil do ...@@ -9,7 +9,7 @@ describe Gitlab::ImportExport::HashUtil do
describe '.deep_symbolize_array!' do describe '.deep_symbolize_array!' do
it 'symbolizes keys' do it 'symbolizes keys' do
expect { described_class.deep_symbolize_array!(stringified_array) }.to change { expect { described_class.deep_symbolize_array!(stringified_array) }.to change {
stringified_array.first.keys.first stringified_array.first.each_key.first
}.from('test').to(:test) }.from('test').to(:test)
end end
end end
...@@ -17,13 +17,13 @@ describe Gitlab::ImportExport::HashUtil do ...@@ -17,13 +17,13 @@ describe Gitlab::ImportExport::HashUtil do
describe '.deep_symbolize_array_with_date!' do describe '.deep_symbolize_array_with_date!' do
it 'symbolizes keys' do it 'symbolizes keys' do
expect { described_class.deep_symbolize_array_with_date!(stringified_array_with_date) }.to change { expect { described_class.deep_symbolize_array_with_date!(stringified_array_with_date) }.to change {
stringified_array_with_date.first.keys.first stringified_array_with_date.first.each_key.first
}.from('test_date').to(:test_date) }.from('test_date').to(:test_date)
end end
it 'transforms date strings into Time objects' do it 'transforms date strings into Time objects' do
expect { described_class.deep_symbolize_array_with_date!(stringified_array_with_date) }.to change { expect { described_class.deep_symbolize_array_with_date!(stringified_array_with_date) }.to change {
stringified_array_with_date.first.values.first.class stringified_array_with_date.first.each_value.first.class
}.from(String).to(ActiveSupport::TimeWithZone) }.from(String).to(ActiveSupport::TimeWithZone)
end end
end end
......
...@@ -397,7 +397,7 @@ describe Gitlab::Shell do ...@@ -397,7 +397,7 @@ describe Gitlab::Shell do
describe 'namespace actions' do describe 'namespace actions' do
subject { described_class.new } subject { described_class.new }
let(:storage) { Gitlab.config.repositories.storages.keys.first } let(:storage) { Gitlab.config.repositories.storages.each_key.first }
describe '#add_namespace' do describe '#add_namespace' do
it 'creates a namespace' do it 'creates a namespace' do
......
...@@ -28,7 +28,7 @@ describe RedisCacheable do ...@@ -28,7 +28,7 @@ describe RedisCacheable do
end end
describe '#cached_attribute' do describe '#cached_attribute' do
subject { instance.cached_attribute(payload.keys.first) } subject { instance.cached_attribute(payload.each_key.first) }
it 'gets the cache attribute' do it 'gets the cache attribute' do
Gitlab::Redis::SharedState.with do |redis| Gitlab::Redis::SharedState.with do |redis|
...@@ -36,7 +36,7 @@ describe RedisCacheable do ...@@ -36,7 +36,7 @@ describe RedisCacheable do
.and_return(payload.to_json) .and_return(payload.to_json)
end end
expect(subject).to eq(payload.values.first) expect(subject).to eq(payload.each_value.first)
end end
end end
......
...@@ -372,7 +372,7 @@ describe Repository do ...@@ -372,7 +372,7 @@ describe Repository do
context 'when some commits are not found ' do context 'when some commits are not found ' do
let(:oids) do let(:oids) do
['deadbeef'] + TestEnv::BRANCH_SHA.values.first(10) ['deadbeef'] + TestEnv::BRANCH_SHA.each_value.first(10)
end end
it 'returns only found commits' do it 'returns only found commits' do
......
...@@ -4160,7 +4160,7 @@ describe User, :do_not_mock_admin_mode do ...@@ -4160,7 +4160,7 @@ describe User, :do_not_mock_admin_mode do
describe '#dismissed_callout?' do describe '#dismissed_callout?' do
subject(:user) { create(:user) } subject(:user) { create(:user) }
let(:feature_name) { UserCallout.feature_names.keys.first } let(:feature_name) { UserCallout.feature_names.each_key.first }
context 'when no callout dismissal record exists' do context 'when no callout dismissal record exists' do
it 'returns false when no ignore_dismissal_earlier_than provided' do it 'returns false when no ignore_dismissal_earlier_than provided' do
......
...@@ -4,7 +4,7 @@ require 'spec_helper' ...@@ -4,7 +4,7 @@ require 'spec_helper'
describe TestSuiteEntity do describe TestSuiteEntity do
let(:pipeline) { create(:ci_pipeline, :with_test_reports) } let(:pipeline) { create(:ci_pipeline, :with_test_reports) }
let(:entity) { described_class.new(pipeline.test_reports.test_suites.values.first) } let(:entity) { described_class.new(pipeline.test_reports.test_suites.each_value.first) }
describe '#as_json' do describe '#as_json' do
subject(:as_json) { entity.as_json } subject(:as_json) { entity.as_json }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment