Commit 8326e84e authored by Stan Hu's avatar Stan Hu

Merge branch 'ab-routable-nplus1' into 'master'

Preload routes information in Routable

See merge request gitlab-org/gitlab-ce!32352
parents 264b0878 53801b12
...@@ -14,7 +14,11 @@ class Admin::GroupsController < Admin::ApplicationController ...@@ -14,7 +14,11 @@ class Admin::GroupsController < Admin::ApplicationController
# rubocop: disable CodeReuse/ActiveRecord # rubocop: disable CodeReuse/ActiveRecord
def show def show
@group = Group.with_statistics.joins(:route).group('routes.path').find_by_full_path(params[:id]) # Group.with_statistics doesn't behave nicely when including other relations.
# Group.find_by_full_path includes the routes relation to avoid a common N+1
# (at the expense of this action: there are two queries here to find and retrieve
# the Group with statistics).
@group = Group.with_statistics.find(group&.id)
@members = present_members( @members = present_members(
@group.members.order("access_level DESC").page(params[:members_page])) @group.members.order("access_level DESC").page(params[:members_page]))
@requesters = present_members( @requesters = present_members(
......
...@@ -38,7 +38,7 @@ module Routable ...@@ -38,7 +38,7 @@ module Routable
if Feature.enabled?(:routable_two_step_lookup) if Feature.enabled?(:routable_two_step_lookup)
# Case sensitive match first (it's cheaper and the usual case) # Case sensitive match first (it's cheaper and the usual case)
# If we didn't have an exact match, we perform a case insensitive search # If we didn't have an exact match, we perform a case insensitive search
found = joins(:route).find_by(routes: { path: path }) || where_full_path_in([path]).take found = includes(:route).find_by(routes: { path: path }) || where_full_path_in([path]).take
else else
order_sql = Arel.sql("(CASE WHEN routes.path = #{connection.quote(path)} THEN 0 ELSE 1 END)") order_sql = Arel.sql("(CASE WHEN routes.path = #{connection.quote(path)} THEN 0 ELSE 1 END)")
found = where_full_path_in([path]).reorder(order_sql).take found = where_full_path_in([path]).reorder(order_sql).take
...@@ -67,7 +67,7 @@ module Routable ...@@ -67,7 +67,7 @@ module Routable
"(LOWER(routes.path) = LOWER(#{connection.quote(path)}))" "(LOWER(routes.path) = LOWER(#{connection.quote(path)}))"
end end
joins(:route).where(wheres.join(' OR ')) includes(:route).where(wheres.join(' OR ')).references(:routes)
end end
# Temporary instrumentation of method calls # Temporary instrumentation of method calls
......
---
title: Preload routes information to fix N+1 issue
merge_request: 32352
author:
type: performance
...@@ -66,6 +66,13 @@ describe Group, 'Routable' do ...@@ -66,6 +66,13 @@ describe Group, 'Routable' do
it { expect(described_class.find_by_full_path(group.to_param.upcase)).to eq(group) } it { expect(described_class.find_by_full_path(group.to_param.upcase)).to eq(group) }
it { expect(described_class.find_by_full_path(nested_group.to_param)).to eq(nested_group) } it { expect(described_class.find_by_full_path(nested_group.to_param)).to eq(nested_group) }
it { expect(described_class.find_by_full_path('unknown')).to eq(nil) } it { expect(described_class.find_by_full_path('unknown')).to eq(nil) }
it 'includes route information when loading a record' do
path = group.to_param
control_count = ActiveRecord::QueryRecorder.new { described_class.find_by_full_path(path) }.count
expect { described_class.find_by_full_path(path).route }.not_to exceed_all_query_limit(control_count)
end
end end
context 'with redirect routes' do context 'with redirect routes' do
......
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