Commit 141a7cd2 authored by Steve Abrams's avatar Steve Abrams

Merge branch 'iterations-finder-reduce-queries' into 'master'

Reduce number of queries for finding iterations

See merge request gitlab-org/gitlab!62303
parents 7af59b1d 733d5df3
...@@ -126,10 +126,13 @@ module EE ...@@ -126,10 +126,13 @@ module EE
end end
def iterations_finder_params def iterations_finder_params
IterationsFinder.params_for_parent(params.parent, include_ancestors: true).merge!( {
parent: params.parent,
include_ancestors: true,
state: 'opened', state: 'opened',
start_date: Date.today, start_date: Date.today,
end_date: Date.today) end_date: Date.today
}
end end
end end
end end
...@@ -3,8 +3,8 @@ ...@@ -3,8 +3,8 @@
# Search for iterations # Search for iterations
# #
# params - Hash # params - Hash
# project_ids: Array of project ids or single project id or ActiveRecord relation. # parent - The group in which to look-up iterations.
# group_ids: Array of group ids or single group id or ActiveRecord relation. # include_ancestors - whether to look-up iterations in group ancestors.
# order - Orders by field default due date asc. # order - Orders by field default due date asc.
# title - Filter by title. # title - Filter by title.
# state - Filters by state. # state - Filters by state.
...@@ -15,39 +15,16 @@ class IterationsFinder ...@@ -15,39 +15,16 @@ class IterationsFinder
attr_reader :params, :current_user attr_reader :params, :current_user
class << self
def params_for_parent(parent, include_ancestors: false)
case parent
when Group
if include_ancestors
{ group_ids: parent.self_and_ancestors.select(:id) }
else
{ group_ids: parent.id }
end
when Project
if include_ancestors && parent.parent_id.present?
{ group_ids: parent.parent.self_and_ancestors.select(:id), project_ids: parent.id }
else
{ project_ids: parent.id }
end
else
raise ArgumentError, 'Invalid parent class. Only Project and Group are supported.'
end
end
end
def initialize(current_user, params = {}) def initialize(current_user, params = {})
@params = params @params = params
@current_user = current_user @current_user = current_user
end end
def execute def execute
filter_permissions
items = Iteration.all items = Iteration.all
items = by_id(items) items = by_id(items)
items = by_iid(items) items = by_iid(items)
items = by_groups_and_projects(items) items = by_groups(items)
items = by_title(items) items = by_title(items)
items = by_search_title(items) items = by_search_title(items)
items = by_state(items) items = by_state(items)
...@@ -59,33 +36,10 @@ class IterationsFinder ...@@ -59,33 +36,10 @@ class IterationsFinder
private private
def filter_permissions def by_groups(items)
filter_allowed_projects return Iteration.none unless Ability.allowed?(current_user, :read_iteration, params[:parent])
filter_allowed_groups
# Only allow either one project_id or one group_id when filtering by `iid`
if params[:iid] && params.slice(:project_ids, :group_ids).keys.count > 1
raise ArgumentError, 'You can specify only one scope if you use iid filter'
end
end
def filter_allowed_projects
return unless params[:project_ids].present?
projects = Project.id_in(params[:project_ids])
params[:project_ids] = Project.projects_user_can(projects, current_user, :read_iteration)
end
def filter_allowed_groups
return unless params[:group_ids].present?
groups = Group.id_in(params[:group_ids]) items.of_groups(groups)
params[:group_ids] = Group.groups_user_can(groups, current_user, :read_iteration)
end
def by_groups_and_projects(items)
items.for_projects_and_groups(params[:project_ids], params[:group_ids])
end end
def by_id(items) def by_id(items)
...@@ -128,4 +82,19 @@ class IterationsFinder ...@@ -128,4 +82,19 @@ class IterationsFinder
items.reorder(order_statement).order(:title) items.reorder(order_statement).order(:title)
end end
# rubocop: enable CodeReuse/ActiveRecord # rubocop: enable CodeReuse/ActiveRecord
def groups
parent = params[:parent]
group = case parent
when Group
parent
when Project
parent.parent
else
raise ArgumentError, 'Invalid parent class. Only Project and Group are supported.'
end
params[:include_ancestors] ? group.self_and_ancestors : group
end
end end
...@@ -67,7 +67,7 @@ module Mutations ...@@ -67,7 +67,7 @@ module Mutations
private private
def find_object(parent:, id:) def find_object(parent:, id:)
params = IterationsFinder.params_for_parent(parent).merge!(id: id) params = { parent: parent, id: id }
IterationsFinder.new(context[:current_user], params).execute.first IterationsFinder.new(context[:current_user], params).execute.first
end end
......
...@@ -51,7 +51,9 @@ module Resolvers ...@@ -51,7 +51,9 @@ module Resolvers
private private
def iterations_finder_params(args) def iterations_finder_params(args)
IterationsFinder.params_for_parent(parent, include_ancestors: args[:include_ancestors]).merge!( {
parent: parent,
include_ancestors: args[:include_ancestors],
id: args[:id], id: args[:id],
iid: args[:iid], iid: args[:iid],
iteration_cadence_ids: args[:iteration_cadence_ids], iteration_cadence_ids: args[:iteration_cadence_ids],
...@@ -59,7 +61,7 @@ module Resolvers ...@@ -59,7 +61,7 @@ module Resolvers
start_date: args.dig(:timeframe, :start) || args[:start_date], start_date: args.dig(:timeframe, :start) || args[:start_date],
end_date: args.dig(:timeframe, :end) || args[:end_date], end_date: args.dig(:timeframe, :end) || args[:end_date],
search_title: args[:title] search_title: args[:title]
) }
end end
def parent def parent
......
...@@ -53,7 +53,7 @@ module EE ...@@ -53,7 +53,7 @@ module EE
end end
def iterations_finder_params def iterations_finder_params
IterationsFinder.params_for_parent(parent, include_ancestors: true).merge(state: 'all') { parent: parent, include_ancestors: true, state: 'all' }
end end
end end
end end
......
...@@ -83,7 +83,7 @@ module EE ...@@ -83,7 +83,7 @@ module EE
end end
def find_iteration(board) def find_iteration(board)
parent_params = ::IterationsFinder.params_for_parent(board.resource_parent, include_ancestors: true) parent_params = { parent: board.resource_parent, include_ancestors: true }
::IterationsFinder.new(current_user, parent_params).find_by(id: params['iteration_id']) # rubocop: disable CodeReuse/ActiveRecord ::IterationsFinder.new(current_user, parent_params).find_by(id: params['iteration_id']) # rubocop: disable CodeReuse/ActiveRecord
end end
......
...@@ -23,10 +23,12 @@ module API ...@@ -23,10 +23,12 @@ module API
end end
def iterations_finder_params(parent) def iterations_finder_params(parent)
IterationsFinder.params_for_parent(parent, include_ancestors: params[:include_ancestors]).merge!( {
parent: parent,
include_ancestors: params[:include_ancestors],
state: params[:state], state: params[:state],
search_title: params[:search] search_title: params[:search]
) }
end end
end end
......
...@@ -131,7 +131,7 @@ module EE ...@@ -131,7 +131,7 @@ module EE
end end
def find_iterations(project, params = {}) def find_iterations(project, params = {})
parent_params = ::IterationsFinder.params_for_parent(project, include_ancestors: true) parent_params = { parent: project, include_ancestors: true }
::IterationsFinder.new(current_user, params.merge(parent_params)).execute ::IterationsFinder.new(current_user, params.merge(parent_params)).execute
end end
......
This diff is collapsed.
...@@ -13,7 +13,7 @@ RSpec.describe Resolvers::IterationsResolver do ...@@ -13,7 +13,7 @@ RSpec.describe Resolvers::IterationsResolver do
id: nil, id: nil,
iid: nil, iid: nil,
iteration_cadence_ids: nil, iteration_cadence_ids: nil,
group_ids: nil, parent: nil,
state: nil, state: nil,
start_date: nil, start_date: nil,
end_date: nil, end_date: nil,
...@@ -43,7 +43,7 @@ RSpec.describe Resolvers::IterationsResolver do ...@@ -43,7 +43,7 @@ RSpec.describe Resolvers::IterationsResolver do
context 'without parameters' do context 'without parameters' do
it 'calls IterationsFinder to retrieve all iterations' do it 'calls IterationsFinder to retrieve all iterations' do
params = params_list.merge(group_ids: Group.where(id: group.id).select(:id), state: 'all') params = params_list.merge(parent: group, include_ancestors: true, state: 'all')
expect(IterationsFinder).to receive(:new).with(current_user, params).and_call_original expect(IterationsFinder).to receive(:new).with(current_user, params).and_call_original
...@@ -60,7 +60,7 @@ RSpec.describe Resolvers::IterationsResolver do ...@@ -60,7 +60,7 @@ RSpec.describe Resolvers::IterationsResolver do
iid = 2 iid = 2
iteration_cadence_ids = ['5'] iteration_cadence_ids = ['5']
params = params_list.merge(id: id, iid: iid, iteration_cadence_ids: iteration_cadence_ids, group_ids: group.id, state: 'closed', start_date: start_date, end_date: end_date, search_title: search) params = params_list.merge(id: id, iid: iid, iteration_cadence_ids: iteration_cadence_ids, parent: group, include_ancestors: nil, state: 'closed', start_date: start_date, end_date: end_date, search_title: search)
expect(IterationsFinder).to receive(:new).with(current_user, params).and_call_original expect(IterationsFinder).to receive(:new).with(current_user, params).and_call_original
...@@ -75,7 +75,7 @@ RSpec.describe Resolvers::IterationsResolver do ...@@ -75,7 +75,7 @@ RSpec.describe Resolvers::IterationsResolver do
iid = 2 iid = 2
iteration_cadence_ids = ['5'] iteration_cadence_ids = ['5']
params = params_list.merge(id: id, iid: iid, iteration_cadence_ids: iteration_cadence_ids, group_ids: group.id, state: 'closed', start_date: start_date, end_date: end_date, search_title: search) params = params_list.merge(id: id, iid: iid, iteration_cadence_ids: iteration_cadence_ids, parent: group, include_ancestors: nil, state: 'closed', start_date: start_date, end_date: end_date, search_title: search)
expect(IterationsFinder).to receive(:new).with(current_user, params).and_call_original expect(IterationsFinder).to receive(:new).with(current_user, params).and_call_original
...@@ -85,7 +85,7 @@ RSpec.describe Resolvers::IterationsResolver do ...@@ -85,7 +85,7 @@ RSpec.describe Resolvers::IterationsResolver do
it 'accepts a raw model id for backward compatibility' do it 'accepts a raw model id for backward compatibility' do
id = 1 id = 1
iid = 2 iid = 2
params = params_list.merge(id: id, iid: iid, group_ids: group.id, state: 'all') params = params_list.merge(id: id, iid: iid, parent: group, include_ancestors: nil, state: 'all')
expect(IterationsFinder).to receive(:new).with(current_user, params).and_call_original expect(IterationsFinder).to receive(:new).with(current_user, params).and_call_original
...@@ -97,7 +97,7 @@ RSpec.describe Resolvers::IterationsResolver do ...@@ -97,7 +97,7 @@ RSpec.describe Resolvers::IterationsResolver do
let_it_be(:subgroup) { create(:group, :private, parent: group) } let_it_be(:subgroup) { create(:group, :private, parent: group) }
it 'defaults to include_ancestors' do it 'defaults to include_ancestors' do
params = params_list.merge(group_ids: subgroup.self_and_ancestors.select(:id), state: 'all') params = params_list.merge(parent: subgroup, include_ancestors: true, state: 'all')
expect(IterationsFinder).to receive(:new).with(current_user, params).and_call_original expect(IterationsFinder).to receive(:new).with(current_user, params).and_call_original
...@@ -105,7 +105,7 @@ RSpec.describe Resolvers::IterationsResolver do ...@@ -105,7 +105,7 @@ RSpec.describe Resolvers::IterationsResolver do
end end
it 'does not default to include_ancestors if IID is supplied' do it 'does not default to include_ancestors if IID is supplied' do
params = params_list.merge(iid: 1, group_ids: subgroup.id, state: 'all') params = params_list.merge(iid: 1, parent: subgroup, include_ancestors: false, state: 'all')
expect(IterationsFinder).to receive(:new).with(current_user, params).and_call_original expect(IterationsFinder).to receive(:new).with(current_user, params).and_call_original
...@@ -113,7 +113,7 @@ RSpec.describe Resolvers::IterationsResolver do ...@@ -113,7 +113,7 @@ RSpec.describe Resolvers::IterationsResolver do
end end
it 'accepts include_ancestors false' do it 'accepts include_ancestors false' do
params = params_list.merge(group_ids: subgroup.id, state: 'all') params = params_list.merge(parent: subgroup, include_ancestors: false, state: 'all')
expect(IterationsFinder).to receive(:new).with(current_user, params).and_call_original expect(IterationsFinder).to receive(:new).with(current_user, params).and_call_original
......
...@@ -107,16 +107,6 @@ RSpec.describe 'Querying an Iteration' do ...@@ -107,16 +107,6 @@ RSpec.describe 'Querying an Iteration' do
let(:expected_web_url) { /#{expected_web_path}$/ } let(:expected_web_url) { /#{expected_web_path}$/ }
end end
end end
describe 'project-owned iteration' do
it_behaves_like 'scoped path' do
let(:queried_iteration) { project_iteration }
let(:expected_scope_path) { project_iteration_path(project, project_iteration.id) }
let(:expected_scope_url) { /#{expected_scope_path}$/ }
let(:expected_web_path) { project_iteration_path(project, project_iteration.id) }
let(:expected_web_url) { /#{expected_web_path}$/ }
end
end
end end
context 'inside a group context' do context 'inside a group context' do
......
...@@ -78,11 +78,11 @@ RSpec.describe API::Iterations do ...@@ -78,11 +78,11 @@ RSpec.describe API::Iterations do
it_behaves_like 'iterations list' it_behaves_like 'iterations list'
it 'excludes ancestor iterations when include_ancestors is set to false' do it 'return direct parent group iterations when include_ancestors is set to false' do
get api(api_path, user), params: { include_ancestors: false } get api(api_path, user), params: { include_ancestors: false }
expect(response).to have_gitlab_http_status(:ok) expect(response).to have_gitlab_http_status(:ok)
expect(json_response.size).to eq(0) expect(json_response.map { |i| i['id'] }).to contain_exactly(iteration.id, closed_iteration.id)
end end
end end
end end
...@@ -34,12 +34,24 @@ RSpec.describe Boards::CreateService, services: true do ...@@ -34,12 +34,24 @@ RSpec.describe Boards::CreateService, services: true do
end end
end end
context 'when setting a timebox' do
let(:user) { create(:user) }
before do
parent.add_reporter(user)
end
it_behaves_like 'setting a milestone scope' do it_behaves_like 'setting a milestone scope' do
subject { described_class.new(parent, double, milestone_id: milestone.id).execute.payload } before do
parent.add_reporter(user)
end
subject { described_class.new(parent, user, milestone_id: milestone.id).execute.payload }
end end
it_behaves_like 'setting an iteration scope' do it_behaves_like 'setting an iteration scope' do
subject { described_class.new(parent, nil, iteration_id: iteration.id).execute.payload } subject { described_class.new(parent, user, iteration_id: iteration.id).execute.payload }
end
end end
end end
end end
...@@ -34,6 +34,10 @@ RSpec.describe Boards::UpdateService, services: true do ...@@ -34,6 +34,10 @@ RSpec.describe Boards::UpdateService, services: true do
hide_backlog_list: true, hide_closed_list: true } hide_backlog_list: true, hide_closed_list: true }
end end
before do
project.add_reporter(user)
end
context 'with group board' do context 'with group board' do
let!(:board) { create(:board, group: group, name: 'Backend') } let!(:board) { create(:board, group: group, name: 'Backend') }
...@@ -46,11 +50,18 @@ RSpec.describe Boards::UpdateService, services: true do ...@@ -46,11 +50,18 @@ RSpec.describe Boards::UpdateService, services: true do
it_behaves_like 'board update service' it_behaves_like 'board update service'
end end
context 'when setting a timebox' do
let(:user) { create(:user) }
before do
parent.add_reporter(user)
end
it_behaves_like 'setting a milestone scope' do it_behaves_like 'setting a milestone scope' do
subject { board.reload } subject { board.reload }
before do before do
described_class.new(parent, double, milestone_id: milestone.id).execute(board) described_class.new(parent, user, milestone_id: milestone.id).execute(board)
end end
end end
...@@ -58,7 +69,8 @@ RSpec.describe Boards::UpdateService, services: true do ...@@ -58,7 +69,8 @@ RSpec.describe Boards::UpdateService, services: true do
subject { board.reload } subject { board.reload }
before do before do
described_class.new(parent, nil, iteration_id: iteration.id).execute(board) described_class.new(parent, user, iteration_id: iteration.id).execute(board)
end
end end
end end
end end
......
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