Commit 8b7b4cf0 authored by Mike Greiling's avatar Mike Greiling

Merge branch '10904-insights-include-projects' into 'master'

Limit group insights query in a subset of projects

Closes #10904

See merge request gitlab-org/gitlab!15930
parents 3e2a830d 6f256428
...@@ -51,14 +51,21 @@ module Routable ...@@ -51,14 +51,21 @@ module Routable
# Klass.where_full_path_in(%w{gitlab-org/gitlab-foss gitlab-org/gitlab}) # Klass.where_full_path_in(%w{gitlab-org/gitlab-foss gitlab-org/gitlab})
# #
# Returns an ActiveRecord::Relation. # Returns an ActiveRecord::Relation.
def where_full_path_in(paths) def where_full_path_in(paths, use_includes: true)
return none if paths.empty? return none if paths.empty?
wheres = paths.map do |path| wheres = paths.map do |path|
"(LOWER(routes.path) = LOWER(#{connection.quote(path)}))" "(LOWER(routes.path) = LOWER(#{connection.quote(path)}))"
end end
includes(:route).where(wheres.join(' OR ')).references(:routes) route =
if use_includes
includes(:route).references(:routes)
else
joins(:route)
end
route.where(wheres.join(' OR '))
end end
end end
......
---
title: Add projects.only option to Insights
merge_request: 15930
author:
type: added
...@@ -58,9 +58,9 @@ For example, here's a single definition for Insights that will display one page ...@@ -58,9 +58,9 @@ For example, here's a single definition for Insights that will display one page
```yaml ```yaml
bugsCharts: bugsCharts:
title: 'Charts for Bugs' title: "Charts for bugs"
charts: charts:
- title: Monthly Bugs Created (bar) - title: "Monthly bugs created"
type: bar type: bar
query: query:
issuable_type: issue issuable_type: issue
...@@ -76,7 +76,7 @@ Each chart definition is made up of a hash composed of key-value pairs. ...@@ -76,7 +76,7 @@ Each chart definition is made up of a hash composed of key-value pairs.
For example, here's single chart definition: For example, here's single chart definition:
```yaml ```yaml
- title: Monthly Bugs Created (bar) - title: "Monthly bugs created"
type: bar type: bar
query: query:
issuable_type: issue issuable_type: issue
...@@ -111,7 +111,7 @@ For example: ...@@ -111,7 +111,7 @@ For example:
```yaml ```yaml
monthlyBugsCreated: monthlyBugsCreated:
title: Monthly Bugs Created (bar) title: "Monthly bugs created"
``` ```
### `type` ### `type`
...@@ -122,7 +122,7 @@ For example: ...@@ -122,7 +122,7 @@ For example:
```yaml ```yaml
monthlyBugsCreated: monthlyBugsCreated:
title: Monthly Bugs Created (bar) title: "Monthly bugs created"
type: bar type: bar
``` ```
...@@ -145,7 +145,7 @@ Example: ...@@ -145,7 +145,7 @@ Example:
```yaml ```yaml
monthlyBugsCreated: monthlyBugsCreated:
title: Monthly Bugs Created (bar) title: "Monthly bugs created"
type: bar type: bar
query: query:
issuable_type: issue issuable_type: issue
...@@ -174,7 +174,7 @@ Supported values are: ...@@ -174,7 +174,7 @@ Supported values are:
Filter by the state of the queried "issuable". Filter by the state of the queried "issuable".
If you omit it, the `opened` state filter will be applied. By default, the `opened` state filter will be applied.
Supported values are: Supported values are:
...@@ -188,14 +188,14 @@ Supported values are: ...@@ -188,14 +188,14 @@ Supported values are:
Filter by labels applied to the queried "issuable". Filter by labels applied to the queried "issuable".
If you omit it, no labels filter will be applied. All the defined labels must be By default, no labels filter will be applied. All the defined labels must be
applied to the "issuable" in order for it to be selected. applied to the "issuable" in order for it to be selected.
Example: Example:
```yaml ```yaml
monthlyBugsCreated: monthlyBugsCreated:
title: Monthly regressions Created (bar) title: "Monthly regressions created"
type: bar type: bar
query: query:
issuable_type: issue issuable_type: issue
...@@ -209,14 +209,14 @@ monthlyBugsCreated: ...@@ -209,14 +209,14 @@ monthlyBugsCreated:
Group "issuable" by the configured labels. Group "issuable" by the configured labels.
If you omit it, no grouping will be done. When using this keyword, you need to By default, no grouping will be done. When using this keyword, you need to
set `type` to either `line` or `stacked-bar`. set `type` to either `line` or `stacked-bar`.
Example: Example:
```yaml ```yaml
weeklyBugsBySeverity: weeklyBugsBySeverity:
title: Weekly Bugs By Severity (stacked bar) title: "Weekly bugs by severity"
type: stacked-bar type: stacked-bar
query: query:
issuable_type: issue issuable_type: issue
...@@ -248,7 +248,7 @@ The unit is related to the `query.group_by` you defined. For instance if you ...@@ -248,7 +248,7 @@ The unit is related to the `query.group_by` you defined. For instance if you
defined `query.group_by: 'day'` then `query.period_limit: 365` would mean defined `query.group_by: 'day'` then `query.period_limit: 365` would mean
"Gather and display data for the last 365 days". "Gather and display data for the last 365 days".
If you omit it, default values will be applied depending on the `query.group_by` By default, default values will be applied depending on the `query.group_by`
you defined. you defined.
| `query.group_by` | Default value | | `query.group_by` | Default value |
...@@ -257,14 +257,63 @@ you defined. ...@@ -257,14 +257,63 @@ you defined.
| `week` | 4 | | `week` | 4 |
| `month` | 12 | | `month` | 12 |
### `projects`
> [Introduced](https://gitlab.com/gitlab-org/gitlab/issues/10904) in [GitLab Ultimate](https://about.gitlab.com/pricing/) 12.4.
You can limit where the "issuables" can be queried from:
- If `.gitlab/insights.yml` is used for a [group's insights](../../group/insights/index.md#configure-your-insights), with `projects`, you can limit the projects to be queried. By default, all projects under the group will be used.
- If `.gitlab/insights.yml` is used for a project's insights, specifying any other projects will yield no results. By default, the project itself will be used.
#### `projects.only`
The `projects.only` option specifies the projects which the "issuables"
should be queried from.
Projects listed here will be ignored when:
- They don't exist.
- The current user doesn't have sufficient permissions to read them.
- They are outside of the group.
In the following `insights.yml` example, we specify the projects
the queries will be used on. This example is useful when setting
a group's insights:
```yaml
monthlyBugsCreated:
title: "Monthly bugs created"
type: bar
query:
issuable_type: issue
issuable_state: opened
filter_labels:
- bug
projects:
only:
- 3 # You can use the project ID
- groupA/projectA # Or full project path
- groupA/subgroupB/projectC # Projects in subgroups can be included
- groupB/project # Projects outside the group will be ignored
```
## Complete example ## Complete example
```yaml ```yaml
.projectsOnly: &projectsOnly
projects:
only:
- 3
- groupA/projectA
- groupA/subgroupB/projectC
bugsCharts: bugsCharts:
title: 'Charts for Bugs' title: "Charts for bugs"
charts: charts:
- title: Monthly Bugs Created (bar) - title: "Monthly bugs created"
type: bar type: bar
<<: *projectsOnly
query: query:
issuable_type: issue issuable_type: issue
issuable_state: opened issuable_state: opened
...@@ -272,8 +321,10 @@ bugsCharts: ...@@ -272,8 +321,10 @@ bugsCharts:
- bug - bug
group_by: month group_by: month
period_limit: 24 period_limit: 24
- title: Weekly Bugs By Severity (stacked bar)
- title: "Weekly bugs by severity"
type: stacked-bar type: stacked-bar
<<: *projectsOnly
query: query:
issuable_type: issue issuable_type: issue
issuable_state: opened issuable_state: opened
...@@ -286,8 +337,10 @@ bugsCharts: ...@@ -286,8 +337,10 @@ bugsCharts:
- S4 - S4
group_by: week group_by: week
period_limit: 104 period_limit: 104
- title: Monthly Bugs By Team (line)
- title: "Monthly bugs by team"
type: line type: line
<<: *projectsOnly
query: query:
issuable_type: merge_request issuable_type: merge_request
issuable_state: opened issuable_state: opened
......
...@@ -30,10 +30,7 @@ export const receiveChartDataError = ({ commit }, { chart, error }) => ...@@ -30,10 +30,7 @@ export const receiveChartDataError = ({ commit }, { chart, error }) =>
export const fetchChartData = ({ dispatch }, { endpoint, chart }) => export const fetchChartData = ({ dispatch }, { endpoint, chart }) =>
axios axios
.post(endpoint, { .post(endpoint, chart)
query: chart.query,
chart_type: chart.type,
})
.then(({ data }) => dispatch('receiveChartDataSuccess', { chart, data })) .then(({ data }) => dispatch('receiveChartDataSuccess', { chart, data }))
.catch(error => { .catch(error => {
let message = `${__('There was an error gathering the chart data')}`; let message = `${__('There was an error gathering the chart data')}`;
......
...@@ -39,8 +39,8 @@ module InsightsActions ...@@ -39,8 +39,8 @@ module InsightsActions
Gitlab::Insights::Validators::ParamsValidator.new(params).validate! Gitlab::Insights::Validators::ParamsValidator.new(params).validate!
end end
def chart_type_param def type_param
@chart_type_param ||= params[:chart_type] @type_param ||= params[:type]
end end
def query_param def query_param
...@@ -51,6 +51,10 @@ module InsightsActions ...@@ -51,6 +51,10 @@ module InsightsActions
@period_param ||= query_param[:group_by] @period_param ||= query_param[:group_by]
end end
def projects_param
@projects_param ||= params[:projects] || {}
end
def collection_labels_param def collection_labels_param
@collection_labels_param ||= query_param[:collection_labels] @collection_labels_param ||= query_param[:collection_labels]
end end
...@@ -62,7 +66,7 @@ module InsightsActions ...@@ -62,7 +66,7 @@ module InsightsActions
end end
def reduce(issuables:, period_limit: nil) def reduce(issuables:, period_limit: nil)
case chart_type_param case type_param
when 'stacked-bar', 'line' when 'stacked-bar', 'line'
Gitlab::Insights::Reducers::LabelCountPerPeriodReducer.reduce(issuables, period: period_param, period_limit: period_limit, labels: collection_labels_param) Gitlab::Insights::Reducers::LabelCountPerPeriodReducer.reduce(issuables, period: period_param, period_limit: period_limit, labels: collection_labels_param)
when 'bar', 'pie' when 'bar', 'pie'
...@@ -77,11 +81,12 @@ module InsightsActions ...@@ -77,11 +81,12 @@ module InsightsActions
def finder def finder
@finder ||= @finder ||=
Gitlab::Insights::Finders::IssuableFinder Gitlab::Insights::Finders::IssuableFinder
.new(insights_entity, current_user, query_param) .new(insights_entity, current_user,
query: query_param, projects: projects_param)
end end
def serializer def serializer
case chart_type_param case type_param
when 'stacked-bar' when 'stacked-bar'
Gitlab::Insights::Serializers::Chartjs::MultiSeriesSerializer Gitlab::Insights::Serializers::Chartjs::MultiSeriesSerializer
when 'bar', 'pie' when 'bar', 'pie'
......
...@@ -4,6 +4,8 @@ module Gitlab ...@@ -4,6 +4,8 @@ module Gitlab
module Insights module Insights
module Finders module Finders
class IssuableFinder class IssuableFinder
include Gitlab::Utils::StrongMemoize
IssuableFinderError = Class.new(StandardError) IssuableFinderError = Class.new(StandardError)
InvalidIssuableTypeError = Class.new(IssuableFinderError) InvalidIssuableTypeError = Class.new(IssuableFinderError)
InvalidGroupByError = Class.new(IssuableFinderError) InvalidGroupByError = Class.new(IssuableFinderError)
...@@ -20,10 +22,11 @@ module Gitlab ...@@ -20,10 +22,11 @@ module Gitlab
months: { default: 12 } months: { default: 12 }
}.with_indifferent_access.freeze }.with_indifferent_access.freeze
def initialize(entity, current_user, opts) def initialize(entity, current_user, query: {}, projects: {})
@entity = entity @entity = entity
@current_user = current_user @current_user = current_user
@opts = opts @query = query
@projects = projects
end end
# Returns an Active Record relation of issuables. # Returns an Active Record relation of issuables.
...@@ -31,18 +34,18 @@ module Gitlab ...@@ -31,18 +34,18 @@ module Gitlab
relation = finder relation = finder
.new(current_user, finder_args) .new(current_user, finder_args)
.execute .execute
relation = relation.preload(:labels) if opts.key?(:collection_labels) # rubocop:disable CodeReuse/ActiveRecord relation = relation.preload(:labels) if query.key?(:collection_labels) # rubocop:disable CodeReuse/ActiveRecord
relation relation
end end
def period_limit def period_limit
@period_limit ||= @period_limit ||=
if opts.key?(:period_limit) if query.key?(:period_limit)
begin begin
Integer(opts[:period_limit]) Integer(query[:period_limit])
rescue ArgumentError rescue ArgumentError
raise InvalidPeriodLimitError, "Invalid `:period_limit` option: `#{opts[:period_limit]}`. Expected an integer!" raise InvalidPeriodLimitError, "Invalid `:period_limit` option: `#{query[:period_limit]}`. Expected an integer!"
end end
else else
PERIODS.dig(period, :default) PERIODS.dig(period, :default)
...@@ -51,49 +54,54 @@ module Gitlab ...@@ -51,49 +54,54 @@ module Gitlab
private private
attr_reader :entity, :current_user, :opts attr_reader :entity, :current_user, :query, :projects
def finder def finder
issuable_type = opts[:issuable_type]&.to_sym issuable_type = query[:issuable_type]&.to_sym
FINDERS[issuable_type] || FINDERS[issuable_type] ||
raise(InvalidIssuableTypeError, "Invalid `:issuable_type` option: `#{opts[:issuable_type]}`. Allowed values are #{FINDERS.keys}!") raise(InvalidIssuableTypeError, "Invalid `:issuable_type` option: `#{query[:issuable_type]}`. Allowed values are #{FINDERS.keys}!")
end end
def finder_args def finder_args
{ {
include_subgroups: true, include_subgroups: true,
state: opts[:issuable_state] || 'opened', state: query[:issuable_state] || 'opened',
label_name: opts[:filter_labels], label_name: query[:filter_labels],
sort: 'created_asc', sort: 'created_asc',
created_after: created_after_argument created_after: created_after_argument,
}.merge(entity_key => entity.id) projects: finder_projects
}.merge(entity_arg)
end end
def entity_key def entity_arg
case entity case entity
when ::Project when ::Project
:project_id if finder_projects
{} # We just rely on projects argument
else
{ project_id: entity.id }
end
when ::Namespace when ::Namespace
:group_id { group_id: entity.id }
else else
raise InvalidEntityError, "Entity class `#{entity.class}` is not supported. Supported classes are Project and Namespace!" raise InvalidEntityError, "Entity class `#{entity.class}` is not supported. Supported classes are Project and Namespace!"
end end
end end
def created_after_argument def created_after_argument
return unless opts.key?(:group_by) return unless query.key?(:group_by)
Time.zone.now.advance(period => -period_limit) Time.zone.now.advance(period => -period_limit)
end end
def period def period
@period ||= @period ||=
if opts.key?(:group_by) if query.key?(:group_by)
period = opts[:group_by].to_s.pluralize.to_sym period = query[:group_by].to_s.pluralize.to_sym
unless PERIODS.key?(period) unless PERIODS.key?(period)
raise InvalidGroupByError, "Invalid `:group_by` option: `#{opts[:group_by]}`. Allowed values are #{PERIODS.keys}!" raise InvalidGroupByError, "Invalid `:group_by` option: `#{query[:group_by]}`. Allowed values are #{PERIODS.keys}!"
end end
period period
...@@ -101,6 +109,43 @@ module Gitlab ...@@ -101,6 +109,43 @@ module Gitlab
:days :days
end end
end end
def finder_projects
strong_memoize(:finder_projects) do
if projects.empty?
nil
elsif finder_projects_options[:ids] && finder_projects_options[:paths]
Project.from_union([finder_projects_ids, finder_projects_paths])
elsif finder_projects_options[:ids]
finder_projects_ids
elsif finder_projects_options[:paths]
finder_projects_paths
end
end
end
def finder_projects_ids
Project.id_in(finder_projects_options[:ids]).select(:id)
end
def finder_projects_paths
Project.where_full_path_in(
finder_projects_options[:paths], use_includes: false
).select(:id)
end
def finder_projects_options
@finder_projects_options ||= projects[:only]&.group_by do |item|
case item
when Integer
:ids
when String
:paths
else
:unknown
end
end || {}
end
end end
end end
end end
......
...@@ -5,17 +5,28 @@ module Gitlab ...@@ -5,17 +5,28 @@ module Gitlab
module Validators module Validators
class ParamsValidator class ParamsValidator
ParamsValidatorError = Class.new(StandardError) ParamsValidatorError = Class.new(StandardError)
InvalidChartTypeError = Class.new(ParamsValidatorError) InvalidTypeError = Class.new(ParamsValidatorError)
InvalidProjectsError = Class.new(ParamsValidatorError)
SUPPORTER_CHART_TYPES = %w[bar line stacked-bar pie].freeze SUPPORTER_TYPES = %w[bar line stacked-bar pie].freeze
def initialize(params) def initialize(params)
@params = params @params = params
end end
def validate! def validate!
unless SUPPORTER_CHART_TYPES.include?(params[:chart_type]) unless SUPPORTER_TYPES.include?(params[:type])
raise InvalidChartTypeError, "Invalid `:chart_type`: `#{params[:chart_type]}`. Allowed values are #{SUPPORTER_CHART_TYPES}!" raise InvalidTypeError, "Invalid `:type`: `#{params[:type]}`. Allowed values are #{SUPPORTER_TYPES}!"
end
if params[:projects]
unless params[:projects].is_a?(Hash) || params[:projects].is_a?(ActionController::Parameters)
raise InvalidProjectsError, "Invalid `:projects`: `#{params[:projects]}`. It should be a hash."
end
unless params.dig(:projects, :only).is_a?(Array)
raise InvalidProjectsError, "Invalid `:projects`.`only`: `#{params.dig(:projects, :only)}`. It should be an array."
end
end end
end end
......
...@@ -8,7 +8,8 @@ describe Groups::InsightsController do ...@@ -8,7 +8,8 @@ describe Groups::InsightsController do
set(:project) { create(:project, :private) } set(:project) { create(:project, :private) }
set(:insight) { create(:insight, group: parent_group, project: project) } set(:insight) { create(:insight, group: parent_group, project: project) }
set(:user) { create(:user) } set(:user) { create(:user) }
let(:query_params) { { chart_type: 'bar', query: { issuable_type: 'issue', collection_labels: ['bug'] } } } let(:query_params) { { type: 'bar', query: { issuable_type: 'issue', collection_labels: ['bug'] }, projects: projects_params } }
let(:projects_params) { { only: [project.id, project.full_path] } }
before do before do
stub_licensed_features(insights: true) stub_licensed_features(insights: true)
......
...@@ -174,12 +174,7 @@ describe('Insights store actions', () => { ...@@ -174,12 +174,7 @@ describe('Insights store actions', () => {
describe('successful request', () => { describe('successful request', () => {
beforeEach(() => { beforeEach(() => {
mock mock.onPost(`${gl.TEST_HOST}/query`, chart).reply(200, chartData);
.onPost(`${gl.TEST_HOST}/query`, {
query: chart.query,
chart_type: chart.type,
})
.reply(200, chartData);
}); });
it('calls receiveChartDataSuccess with chart data', done => { it('calls receiveChartDataSuccess with chart data', done => {
...@@ -202,12 +197,7 @@ describe('Insights store actions', () => { ...@@ -202,12 +197,7 @@ describe('Insights store actions', () => {
describe('failed request', () => { describe('failed request', () => {
beforeEach(() => { beforeEach(() => {
mock mock.onPost(`${gl.TEST_HOST}/query`, chart).reply(500);
.onPost(`${gl.TEST_HOST}/query`, {
query: chart.query,
chart_type: chart.type,
})
.reply(500);
}); });
it('calls receiveChartDataError with error message', done => { it('calls receiveChartDataError with error message', done => {
......
...@@ -7,7 +7,7 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do ...@@ -7,7 +7,7 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do
Timecop.freeze(Time.utc(2019, 3, 5)) { example.run } Timecop.freeze(Time.utc(2019, 3, 5)) { example.run }
end end
let(:base_opts) do let(:base_query) do
{ {
state: 'opened', state: 'opened',
group_by: 'months' group_by: 'months'
...@@ -15,28 +15,36 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do ...@@ -15,28 +15,36 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do
end end
describe '#find' do describe '#find' do
def find(entity, opts) def find(entity, query:, projects: {})
described_class.new(entity, nil, opts).find described_class.new(entity, nil, query: query, projects: projects).find
end end
it 'raises an error for an invalid :issuable_type option' do it 'raises an error for an invalid :issuable_type option' do
expect { find(build(:project), issuable_type: 'foo') }.to raise_error(described_class::InvalidIssuableTypeError, "Invalid `:issuable_type` option: `foo`. Allowed values are #{described_class::FINDERS.keys}!") expect do
find(build(:project), query: { issuable_type: 'foo' })
end.to raise_error(described_class::InvalidIssuableTypeError, "Invalid `:issuable_type` option: `foo`. Allowed values are #{described_class::FINDERS.keys}!")
end end
it 'raises an error for an invalid entity object' do it 'raises an error for an invalid entity object' do
expect { find(build(:user), issuable_type: 'issue') }.to raise_error(described_class::InvalidEntityError, 'Entity class `User` is not supported. Supported classes are Project and Namespace!') expect do
find(build(:user), query: { issuable_type: 'issue' })
end.to raise_error(described_class::InvalidEntityError, 'Entity class `User` is not supported. Supported classes are Project and Namespace!')
end end
it 'raises an error for an invalid :group_by option' do it 'raises an error for an invalid :group_by option' do
expect { find(build(:project), issuable_type: 'issue', group_by: 'foo') }.to raise_error(described_class::InvalidGroupByError, "Invalid `:group_by` option: `foo`. Allowed values are #{described_class::PERIODS.keys}!") expect do
find(build(:project), query: { issuable_type: 'issue', group_by: 'foo' })
end.to raise_error(described_class::InvalidGroupByError, "Invalid `:group_by` option: `foo`. Allowed values are #{described_class::PERIODS.keys}!")
end end
it 'defaults to the "days" period if no :group_by is given' do it 'defaults to the "days" period if no :group_by is given' do
expect(described_class.new(build(:project), nil, issuable_type: 'issue').__send__(:period)).to eq(:days) expect(described_class.new(build(:project), nil, query: { issuable_type: 'issue' }).__send__(:period)).to eq(:days)
end end
it 'raises an error for an invalid :period_limit option' do it 'raises an error for an invalid :period_limit option' do
expect { find(build(:project), issuable_type: 'issue', group_by: 'months', period_limit: 'many') }.to raise_error(described_class::InvalidPeriodLimitError, "Invalid `:period_limit` option: `many`. Expected an integer!") expect do
find(build(:project), query: { issuable_type: 'issue', group_by: 'months', period_limit: 'many' })
end.to raise_error(described_class::InvalidPeriodLimitError, "Invalid `:period_limit` option: `many`. Expected an integer!")
end end
shared_examples_for "insights issuable finder" do shared_examples_for "insights issuable finder" do
...@@ -46,31 +54,34 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do ...@@ -46,31 +54,34 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do
let(:label_create) { create(label_type, label_entity_association_key => entity, name: 'Create') } let(:label_create) { create(label_type, label_entity_association_key => entity, name: 'Create') }
let(:label_quality) { create(label_type, label_entity_association_key => entity, name: 'Quality') } let(:label_quality) { create(label_type, label_entity_association_key => entity, name: 'Quality') }
let(:extra_issuable_attrs) { [{}, {}, {}, {}, {}, {}] } let(:extra_issuable_attrs) { [{}, {}, {}, {}, {}, {}] }
let!(:issuable0) { create(:"labeled_#{issuable_type}", :opened, created_at: Time.utc(2018, 2, 1), project_association_key => project, **extra_issuable_attrs[0]) } let!(:issuable0) { create(:"labeled_#{issuable_type}", :opened, created_at: Time.utc(2018, 1, 1), project_association_key => project, **extra_issuable_attrs[0]) }
let!(:issuable1) { create(:"labeled_#{issuable_type}", :opened, created_at: Time.utc(2018, 2, 1), labels: [label_bug, label_manage], project_association_key => project, **extra_issuable_attrs[1]) } let!(:issuable1) { create(:"labeled_#{issuable_type}", :opened, created_at: Time.utc(2018, 2, 1), labels: [label_bug, label_manage], project_association_key => project, **extra_issuable_attrs[1]) }
let!(:issuable2) { create(:"labeled_#{issuable_type}", :opened, created_at: Time.utc(2019, 2, 6), labels: [label_bug, label_plan], project_association_key => project, **extra_issuable_attrs[2]) } let!(:issuable2) { create(:"labeled_#{issuable_type}", :opened, created_at: Time.utc(2019, 2, 6), labels: [label_bug, label_plan], project_association_key => project, **extra_issuable_attrs[2]) }
let!(:issuable3) { create(:"labeled_#{issuable_type}", :opened, created_at: Time.utc(2019, 2, 20), labels: [label_bug, label_create], project_association_key => project, **extra_issuable_attrs[3]) } let!(:issuable3) { create(:"labeled_#{issuable_type}", :opened, created_at: Time.utc(2019, 2, 20), labels: [label_bug, label_create], project_association_key => project, **extra_issuable_attrs[3]) }
let!(:issuable4) { create(:"labeled_#{issuable_type}", :opened, created_at: Time.utc(2019, 3, 5), labels: [label_bug, label_quality], project_association_key => project, **extra_issuable_attrs[4]) } let!(:issuable4) { create(:"labeled_#{issuable_type}", :opened, created_at: Time.utc(2019, 3, 5), labels: [label_bug, label_quality], project_association_key => project, **extra_issuable_attrs[4]) }
let(:opts) do let(:query) do
base_opts.merge( base_query.merge(
issuable_type: issuable_type, issuable_type: issuable_type,
filter_labels: [label_bug.title], filter_labels: [label_bug.title],
collection_labels: [label_manage.title, label_plan.title, label_create.title]) collection_labels: [label_manage.title, label_plan.title, label_create.title])
end end
let(:projects) { {} }
subject { find(entity, opts) } subject { find(entity, query: query, projects: projects) }
it 'avoids N + 1 queries' do it 'avoids N + 1 queries' do
control_queries = ActiveRecord::QueryRecorder.new { subject.map { |issuable| issuable.labels.map(&:title) } } control_queries = ActiveRecord::QueryRecorder.new { subject.map { |issuable| issuable.labels.map(&:title) } }
create(:"labeled_#{issuable_type}", :opened, created_at: Time.utc(2019, 3, 5), labels: [label_bug], project_association_key => project, **extra_issuable_attrs[5]) create(:"labeled_#{issuable_type}", :opened, created_at: Time.utc(2019, 3, 5), labels: [label_bug], project_association_key => project, **extra_issuable_attrs[5])
expect { find(entity, opts).map { |issuable| issuable.labels.map(&:title) } }.not_to exceed_query_limit(control_queries) expect do
find(entity, query: query).map { |issuable| issuable.labels.map(&:title) }
end.not_to exceed_query_limit(control_queries)
end end
context ':period_limit option' do context ':period_limit query' do
context 'with group_by: "day"' do context 'with group_by: "day"' do
before do before do
opts.merge!(group_by: 'day') query.merge!(group_by: 'day')
end end
it 'returns issuable created after 30 days ago' do it 'returns issuable created after 30 days ago' do
...@@ -80,7 +91,7 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do ...@@ -80,7 +91,7 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do
context 'with group_by: "day", period_limit: 1' do context 'with group_by: "day", period_limit: 1' do
before do before do
opts.merge!(group_by: 'day', period_limit: 1) query.merge!(group_by: 'day', period_limit: 1)
end end
it 'returns issuable created after one day ago' do it 'returns issuable created after one day ago' do
...@@ -90,7 +101,7 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do ...@@ -90,7 +101,7 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do
context 'with group_by: "week"' do context 'with group_by: "week"' do
before do before do
opts.merge!(group_by: 'week') query.merge!(group_by: 'week')
end end
it 'returns issuable created after 12 weeks ago' do it 'returns issuable created after 12 weeks ago' do
...@@ -100,7 +111,7 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do ...@@ -100,7 +111,7 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do
context 'with group_by: "week", period_limit: 1' do context 'with group_by: "week", period_limit: 1' do
before do before do
opts.merge!(group_by: 'week', period_limit: 1) query.merge!(group_by: 'week', period_limit: 1)
end end
it 'returns issuable created after one week ago' do it 'returns issuable created after one week ago' do
...@@ -110,7 +121,7 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do ...@@ -110,7 +121,7 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do
context 'with group_by: "month"' do context 'with group_by: "month"' do
before do before do
opts.merge!(group_by: 'month') query.merge!(group_by: 'month')
end end
it 'returns issuable created after 12 months ago' do it 'returns issuable created after 12 months ago' do
...@@ -120,7 +131,7 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do ...@@ -120,7 +131,7 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do
context 'with group_by: "month", period_limit: 1' do context 'with group_by: "month", period_limit: 1' do
before do before do
opts.merge!(group_by: 'month', period_limit: 1) query.merge!(group_by: 'month', period_limit: 1)
end end
it 'returns issuable created after one month ago' do it 'returns issuable created after one month ago' do
...@@ -128,6 +139,84 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do ...@@ -128,6 +139,84 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do
end end
end end
end end
context ':projects option' do
let(:query) do
{ issuable_type: issuable_type }
end
before do
# For merge requests we need to update both projects
attributes =
Hash[
[
[:project, other_project],
[project_association_key, other_project]
].uniq
]
issuable0.update!(attributes)
issuable1.update!(attributes)
end
context 'when `projects.only` are specified by one id' do
let(:projects) { { only: [project.id] } }
it 'returns issuables for that project' do
expect(subject.to_a).to eq([issuable2, issuable3, issuable4])
end
end
context 'when `projects.only` are specified by two ids' do
let(:projects) { { only: [project.id, other_project.id] } }
it 'returns issuables for all projects' do
expect(subject.to_a)
.to eq([issuable0, issuable1, issuable2, issuable3, issuable4])
end
end
context 'when `projects.only` are specified by bad id' do
let(:projects) { { only: [0] } }
it 'returns nothing' do
expect(subject.to_a).to be_empty
end
end
context 'when `projects.only` are specified by bad id and good id' do
let(:projects) { { only: [0, other_project.id] } }
it 'returns issuables for good project' do
expect(subject.to_a).to eq([issuable0, issuable1])
end
end
context 'when `projects.only` are specified by one project full path' do
let(:projects) { { only: [project.full_path] } }
it 'returns issuables for that project' do
expect(subject.to_a).to eq([issuable2, issuable3, issuable4])
end
end
context 'when `projects.only` are specified by project full path and id' do
let(:projects) { { only: [project.id, other_project.full_path] } }
it 'returns issuables for all projects' do
expect(subject.to_a)
.to eq([issuable0, issuable1, issuable2, issuable3, issuable4])
end
end
context 'when `projects.only` are specified by bad project path' do
let(:projects) { { only: [project.full_path.reverse] } }
it 'returns nothing' do
expect(subject.to_a).to be_empty
end
end
end
end end
shared_examples_for 'group tests' do shared_examples_for 'group tests' do
...@@ -163,17 +252,20 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do ...@@ -163,17 +252,20 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do
context 'for a group' do context 'for a group' do
include_examples 'group tests' do include_examples 'group tests' do
let(:project) { create(:project, :public, group: entity) } let(:project) { create(:project, :public, group: entity) }
let(:other_project) { create(:project, :public, group: entity) }
end end
end end
context 'for a group with subgroups' do context 'for a group with subgroups' do
include_examples 'group tests' do include_examples 'group tests' do
let(:project) { create(:project, :public, group: create(:group, parent: entity)) } let(:project) { create(:project, :public, group: create(:group, parent: entity)) }
let(:other_project) { create(:project, :public, group: entity) }
end end
end end
context 'for a project' do context 'for a project' do
let(:project) { create(:project, :public) } let(:project) { create(:project, :public) }
let(:other_project) { create(:project, :public) }
let(:entity) { project } let(:entity) { project }
let(:label_type) { :label } let(:label_type) { :label }
let(:label_entity_association_key) { :project } let(:label_entity_association_key) { :project }
...@@ -205,11 +297,11 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do ...@@ -205,11 +297,11 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do
end end
describe '#period_limit' do describe '#period_limit' do
subject { described_class.new(create(:project, :public), nil, opts).period_limit } subject { described_class.new(create(:project, :public), nil, query: query).period_limit }
describe 'default values' do describe 'default values' do
context 'with group_by: "day"' do context 'with group_by: "day"' do
let(:opts) { base_opts.merge!(group_by: 'day') } let(:query) { base_query.merge!(group_by: 'day') }
it 'returns 30' do it 'returns 30' do
expect(subject).to eq(30) expect(subject).to eq(30)
...@@ -217,7 +309,7 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do ...@@ -217,7 +309,7 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do
end end
context 'with group_by: "week"' do context 'with group_by: "week"' do
let(:opts) { base_opts.merge!(group_by: 'week') } let(:query) { base_query.merge!(group_by: 'week') }
it 'returns 12' do it 'returns 12' do
expect(subject).to eq(12) expect(subject).to eq(12)
...@@ -225,7 +317,7 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do ...@@ -225,7 +317,7 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do
end end
context 'with group_by: "month"' do context 'with group_by: "month"' do
let(:opts) { base_opts.merge!(group_by: 'month') } let(:query) { base_query.merge!(group_by: 'month') }
it 'returns 12' do it 'returns 12' do
expect(subject).to eq(12) expect(subject).to eq(12)
...@@ -235,7 +327,7 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do ...@@ -235,7 +327,7 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do
describe 'custom values' do describe 'custom values' do
context 'with period_limit: 42' do context 'with period_limit: 42' do
let(:opts) { base_opts.merge!(period_limit: 42) } let(:query) { base_query.merge!(period_limit: 42) }
it 'returns 42' do it 'returns 42' do
expect(subject).to eq(42) expect(subject).to eq(42)
...@@ -243,7 +335,7 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do ...@@ -243,7 +335,7 @@ RSpec.describe Gitlab::Insights::Finders::IssuableFinder do
end end
context 'with an invalid period_limit' do context 'with an invalid period_limit' do
let(:opts) { base_opts.merge!(period_limit: 'many') } let(:query) { base_query.merge!(period_limit: 'many') }
it 'raises an error' do it 'raises an error' do
expect { subject }.to raise_error(described_class::InvalidPeriodLimitError, "Invalid `:period_limit` option: `many`. Expected an integer!") expect { subject }.to raise_error(described_class::InvalidPeriodLimitError, "Invalid `:period_limit` option: `many`. Expected an integer!")
......
...@@ -5,15 +5,15 @@ require 'spec_helper' ...@@ -5,15 +5,15 @@ require 'spec_helper'
RSpec.describe Gitlab::Insights::Reducers::CountPerLabelReducer do RSpec.describe Gitlab::Insights::Reducers::CountPerLabelReducer do
include_context 'Insights reducers context' include_context 'Insights reducers context'
def find_issuables(project, opts) def find_issuables(project, query)
Gitlab::Insights::Finders::IssuableFinder.new(project, nil, opts).find Gitlab::Insights::Finders::IssuableFinder.new(project, nil, query: query).find
end end
def reduce(issuable_relation, labels) def reduce(issuable_relation, labels)
described_class.reduce(issuable_relation, labels: labels) described_class.reduce(issuable_relation, labels: labels)
end end
let(:opts) do let(:query) do
{ {
state: 'opened', state: 'opened',
issuable_type: 'issue', issuable_type: 'issue',
...@@ -23,9 +23,9 @@ RSpec.describe Gitlab::Insights::Reducers::CountPerLabelReducer do ...@@ -23,9 +23,9 @@ RSpec.describe Gitlab::Insights::Reducers::CountPerLabelReducer do
period_limit: 5 period_limit: 5
} }
end end
let(:issuable_relation) { find_issuables(project, opts) } let(:issuable_relation) { find_issuables(project, query) }
subject { reduce(issuable_relation, opts[:collection_labels]) } subject { reduce(issuable_relation, query[:collection_labels]) }
let(:expected) do let(:expected) do
{ {
...@@ -47,6 +47,6 @@ RSpec.describe Gitlab::Insights::Reducers::CountPerLabelReducer do ...@@ -47,6 +47,6 @@ RSpec.describe Gitlab::Insights::Reducers::CountPerLabelReducer do
control_queries = ActiveRecord::QueryRecorder.new { subject } control_queries = ActiveRecord::QueryRecorder.new { subject }
create(:labeled_issue, :opened, labels: [label_bug], project: project) create(:labeled_issue, :opened, labels: [label_bug], project: project)
expect { reduce(find_issuables(project, opts), opts[:collection_labels]) }.not_to exceed_query_limit(control_queries) expect { reduce(find_issuables(project, query), query[:collection_labels]) }.not_to exceed_query_limit(control_queries)
end end
end end
...@@ -5,15 +5,15 @@ require 'spec_helper' ...@@ -5,15 +5,15 @@ require 'spec_helper'
RSpec.describe Gitlab::Insights::Reducers::CountPerPeriodReducer do RSpec.describe Gitlab::Insights::Reducers::CountPerPeriodReducer do
include_context 'Insights reducers context' include_context 'Insights reducers context'
def find_issuables(project, opts) def find_issuables(project, query)
Gitlab::Insights::Finders::IssuableFinder.new(project, nil, opts).find Gitlab::Insights::Finders::IssuableFinder.new(project, nil, query: query).find
end end
def reduce(issuable_relation, period, period_limit = 5, period_field = :created_at) def reduce(issuable_relation, period, period_limit = 5, period_field = :created_at)
described_class.reduce(issuable_relation, period: period, period_limit: period_limit, period_field: period_field) described_class.reduce(issuable_relation, period: period, period_limit: period_limit, period_field: period_field)
end end
let(:opts) do let(:query) do
{ {
state: 'opened', state: 'opened',
issuable_type: 'issue', issuable_type: 'issue',
...@@ -22,9 +22,9 @@ RSpec.describe Gitlab::Insights::Reducers::CountPerPeriodReducer do ...@@ -22,9 +22,9 @@ RSpec.describe Gitlab::Insights::Reducers::CountPerPeriodReducer do
period_limit: 5 period_limit: 5
} }
end end
let(:issuable_relation) { find_issuables(project, opts) } let(:issuable_relation) { find_issuables(project, query) }
subject { reduce(issuable_relation, opts[:group_by]) } subject { reduce(issuable_relation, query[:group_by]) }
let(:expected) do let(:expected) do
{ {
...@@ -56,6 +56,6 @@ RSpec.describe Gitlab::Insights::Reducers::CountPerPeriodReducer do ...@@ -56,6 +56,6 @@ RSpec.describe Gitlab::Insights::Reducers::CountPerPeriodReducer do
control_queries = ActiveRecord::QueryRecorder.new { subject } control_queries = ActiveRecord::QueryRecorder.new { subject }
create(:labeled_issue, :opened, created_at: Time.utc(2019, 2, 5), labels: [label_bug], project: project) create(:labeled_issue, :opened, created_at: Time.utc(2019, 2, 5), labels: [label_bug], project: project)
expect { reduce(find_issuables(project, opts), opts[:group_by]) }.not_to exceed_query_limit(control_queries) expect { reduce(find_issuables(project, query), query[:group_by]) }.not_to exceed_query_limit(control_queries)
end end
end end
...@@ -5,15 +5,15 @@ require 'spec_helper' ...@@ -5,15 +5,15 @@ require 'spec_helper'
RSpec.describe Gitlab::Insights::Reducers::LabelCountPerPeriodReducer do RSpec.describe Gitlab::Insights::Reducers::LabelCountPerPeriodReducer do
include_context 'Insights reducers context' include_context 'Insights reducers context'
def find_issuables(project, opts) def find_issuables(project, query)
Gitlab::Insights::Finders::IssuableFinder.new(project, nil, opts).find Gitlab::Insights::Finders::IssuableFinder.new(project, nil, query: query).find
end end
def reduce(issuable_relation, period, labels) def reduce(issuable_relation, period, labels)
described_class.reduce(issuable_relation, period: period, period_limit: 5, labels: labels) described_class.reduce(issuable_relation, period: period, period_limit: 5, labels: labels)
end end
let(:opts) do let(:query) do
{ {
state: 'opened', state: 'opened',
issuable_type: 'issue', issuable_type: 'issue',
...@@ -23,9 +23,9 @@ RSpec.describe Gitlab::Insights::Reducers::LabelCountPerPeriodReducer do ...@@ -23,9 +23,9 @@ RSpec.describe Gitlab::Insights::Reducers::LabelCountPerPeriodReducer do
period_limit: 5 period_limit: 5
} }
end end
let(:issuable_relation) { find_issuables(project, opts) } let(:issuable_relation) { find_issuables(project, query) }
subject { reduce(issuable_relation, opts[:group_by], opts[:collection_labels]) } subject { reduce(issuable_relation, query[:group_by], query[:collection_labels]) }
let(:expected) do let(:expected) do
{ {
...@@ -65,6 +65,6 @@ RSpec.describe Gitlab::Insights::Reducers::LabelCountPerPeriodReducer do ...@@ -65,6 +65,6 @@ RSpec.describe Gitlab::Insights::Reducers::LabelCountPerPeriodReducer do
control_queries = ActiveRecord::QueryRecorder.new { subject } control_queries = ActiveRecord::QueryRecorder.new { subject }
create(:labeled_issue, :opened, created_at: Time.utc(2019, 2, 5), labels: [label_bug], project: project) create(:labeled_issue, :opened, created_at: Time.utc(2019, 2, 5), labels: [label_bug], project: project)
expect { reduce(find_issuables(project, opts), opts[:group_by], opts[:collection_labels]) }.not_to exceed_query_limit(control_queries) expect { reduce(find_issuables(project, query), query[:group_by], query[:collection_labels]) }.not_to exceed_query_limit(control_queries)
end end
end end
# frozen_string_literal: true # frozen_string_literal: true
require 'spec_helper' require 'fast_spec_helper'
RSpec.describe Gitlab::Insights::Validators::ParamsValidator do RSpec.describe Gitlab::Insights::Validators::ParamsValidator do
subject { described_class.new(params).validate! } subject { described_class.new(params).validate! }
describe ':chart_type' do describe ':type' do
described_class::SUPPORTER_CHART_TYPES.each do |chart_type| described_class::SUPPORTER_TYPES.each do |type|
context "with chart_type: '#{chart_type}'" do context "with type: '#{type}'" do
let(:params) do let(:params) do
{ chart_type: chart_type } { type: type }
end end
it 'does not raise an error' do it 'does not raise an error' do
...@@ -18,13 +18,47 @@ RSpec.describe Gitlab::Insights::Validators::ParamsValidator do ...@@ -18,13 +18,47 @@ RSpec.describe Gitlab::Insights::Validators::ParamsValidator do
end end
end end
context 'with an invalid :chart_type' do context 'with an invalid :type' do
let(:params) do let(:params) do
{ chart_type: 'unknown' } { type: 'unknown' }
end end
it 'raises an error' do it 'raises an error' do
expect { subject }.to raise_error(described_class::InvalidChartTypeError, "Invalid `:chart_type`: `unknown`. Allowed values are #{described_class::SUPPORTER_CHART_TYPES}!") expect { subject }.to raise_error(described_class::InvalidTypeError, "Invalid `:type`: `unknown`. Allowed values are #{described_class::SUPPORTER_TYPES}!")
end
end
end
describe ':projects' do
let(:base_params) { { type: described_class::SUPPORTER_TYPES.first } }
context 'when projects is an array' do
let(:params) do
base_params.merge(projects: [])
end
it 'raises an error' do
expect { subject }.to raise_error(described_class::InvalidProjectsError, "Invalid `:projects`: `[]`. It should be a hash.")
end
end
context 'when projects is a hash, having `only` with an integer' do
let(:params) do
base_params.merge(projects: { only: 1 })
end
it 'raises an error' do
expect { subject }.to raise_error(described_class::InvalidProjectsError, "Invalid `:projects`.`only`: `1`. It should be an array.")
end
end
context 'when projects is a hash, having `only` with an array' do
let(:params) do
base_params.merge(projects: { only: [] })
end
it 'does not raise an error' do
expect { subject }.not_to raise_error
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