Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
G
gitlab-ce
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
1
Merge Requests
1
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
gitlab-ce
Commits
f1aa62e3
Commit
f1aa62e3
authored
Jan 11, 2019
by
GitLab Bot
Browse files
Options
Browse Files
Download
Plain Diff
Automatic merge of gitlab-org/gitlab-ce master
parents
3f14d2f4
7a10ef6e
Changes
5
Hide whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
152 additions
and
30 deletions
+152
-30
app/finders/issuable_finder.rb
app/finders/issuable_finder.rb
+17
-3
app/models/milestone.rb
app/models/milestone.rb
+26
-13
changelogs/unreleased/53431-fix-upcoming-milestone-filter-for-groups.yml
...leased/53431-fix-upcoming-milestone-filter-for-groups.yml
+5
-0
spec/finders/issues_finder_spec.rb
spec/finders/issues_finder_spec.rb
+12
-7
spec/models/milestone_spec.rb
spec/models/milestone_spec.rb
+92
-7
No files found.
app/finders/issuable_finder.rb
View file @
f1aa62e3
...
...
@@ -149,6 +149,18 @@ class IssuableFinder
end
end
def
related_groups
if
project?
&&
project
&&
project
.
group
&&
Ability
.
allowed?
(
current_user
,
:read_group
,
project
.
group
)
project
.
group
.
self_and_ancestors
elsif
group
[
group
]
elsif
current_user
Gitlab
::
ObjectHierarchy
.
new
(
current_user
.
authorized_groups
,
current_user
.
groups
).
all_objects
else
[]
end
end
def
project?
params
[
:project_id
].
present?
end
...
...
@@ -163,8 +175,10 @@ class IssuableFinder
end
# rubocop: disable CodeReuse/ActiveRecord
def
projects
(
items
=
nil
)
return
@projects
=
project
if
project?
def
projects
return
@projects
if
defined?
(
@projects
)
return
@projects
=
[
project
]
if
project?
projects
=
if
current_user
&&
params
[
:authorized_only
].
presence
&&
!
current_user_related?
...
...
@@ -459,7 +473,7 @@ class IssuableFinder
elsif
filter_by_any_milestone?
items
=
items
.
any_milestone
elsif
filter_by_upcoming_milestone?
upcoming_ids
=
Milestone
.
upcoming_ids
_by_projects
(
projects
(
items
)
)
upcoming_ids
=
Milestone
.
upcoming_ids
(
projects
,
related_groups
)
items
=
items
.
left_joins_milestones
.
where
(
milestone_id:
upcoming_ids
)
elsif
filter_by_started_milestone?
items
=
items
.
left_joins_milestones
.
where
(
'milestones.start_date <= NOW()'
)
...
...
app/models/milestone.rb
View file @
f1aa62e3
...
...
@@ -40,12 +40,14 @@ class Milestone < ActiveRecord::Base
scope
:closed
,
->
{
with_state
(
:closed
)
}
scope
:for_projects
,
->
{
where
(
group:
nil
).
includes
(
:project
)
}
scope
:for_projects_and_groups
,
->
(
project_ids
,
group_ids
)
do
conditions
=
[]
conditions
<<
arel_table
[
:project_id
].
in
(
project_ids
)
if
project_ids
&
.
compact
&
.
any?
conditions
<<
arel_table
[
:group_id
].
in
(
group_ids
)
if
group_ids
&
.
compact
&
.
any?
scope
:for_projects_and_groups
,
->
(
projects
,
groups
)
do
projects
=
projects
.
compact
if
projects
.
is_a?
Array
projects
=
[]
if
projects
.
nil?
where
(
conditions
.
reduce
(
:or
))
groups
=
groups
.
compact
if
groups
.
is_a?
Array
groups
=
[]
if
groups
.
nil?
where
(
project:
projects
).
or
(
where
(
group:
groups
))
end
scope
:order_by_name_asc
,
->
{
order
(
Arel
::
Nodes
::
Ascending
.
new
(
arel_table
[
:title
].
lower
))
}
...
...
@@ -135,18 +137,29 @@ class Milestone < ActiveRecord::Base
@link_reference_pattern
||=
super
(
"milestones"
,
/(?<milestone>\d+)/
)
end
def
self
.
upcoming_ids_by_projects
(
projects
)
rel
=
unscoped
.
of_projects
(
projects
).
active
.
where
(
'due_date > ?'
,
Time
.
now
)
def
self
.
upcoming_ids
(
projects
,
groups
)
rel
=
unscoped
.
for_projects_and_groups
(
projects
,
groups
)
.
active
.
where
(
'milestones.due_date > NOW()'
)
if
Gitlab
::
Database
.
postgresql?
rel
.
order
(
:project_id
,
:
due_date
).
select
(
'DISTINCT ON (project
_id) id'
)
rel
.
order
(
:project_id
,
:
group_id
,
:due_date
).
select
(
'DISTINCT ON (project_id, group
_id) id'
)
else
# We need to use MySQL's NULL-safe comparison operator `<=>` here
# because one of `project_id` or `group_id` is always NULL
join_clause
=
<<~
HEREDOC
LEFT OUTER JOIN milestones earlier_milestones
ON milestones.project_id <=> earlier_milestones.project_id
AND milestones.group_id <=> earlier_milestones.group_id
AND milestones.due_date > earlier_milestones.due_date
AND earlier_milestones.due_date > NOW()
AND earlier_milestones.state = 'active'
HEREDOC
rel
.
group
(
:project_id
,
:due_date
,
:id
)
.
having
(
'due_date = MIN(due_date)'
)
.
pluck
(
:id
,
:project_id
,
:due_date
)
.
uniq
(
&
:second
)
.
map
(
&
:first
)
.
joins
(
join_clause
)
.
where
(
'earlier_milestones.id IS NULL'
)
.
select
(
:id
)
end
end
...
...
changelogs/unreleased/53431-fix-upcoming-milestone-filter-for-groups.yml
0 → 100644
View file @
f1aa62e3
---
title
:
Fix upcoming milestones filter not including group milestones
merge_request
:
23098
author
:
Heinrich Lee Yu
type
:
fixed
spec/finders/issues_finder_spec.rb
View file @
f1aa62e3
...
...
@@ -220,9 +220,13 @@ describe IssuesFinder do
context
'filtering by upcoming milestone'
do
let
(
:params
)
{
{
milestone_title:
Milestone
::
Upcoming
.
name
}
}
let!
(
:group
)
{
create
(
:group
,
:public
)
}
let!
(
:group_member
)
{
create
(
:group_member
,
group:
group
,
user:
user
)
}
let
(
:project_no_upcoming_milestones
)
{
create
(
:project
,
:public
)
}
let
(
:project_next_1_1
)
{
create
(
:project
,
:public
)
}
let
(
:project_next_8_8
)
{
create
(
:project
,
:public
)
}
let
(
:project_in_group
)
{
create
(
:project
,
:public
,
namespace:
group
)
}
let
(
:yesterday
)
{
Date
.
today
-
1
.
day
}
let
(
:tomorrow
)
{
Date
.
today
+
1
.
day
}
...
...
@@ -233,21 +237,22 @@ describe IssuesFinder do
[
create
(
:milestone
,
:closed
,
project:
project_no_upcoming_milestones
),
create
(
:milestone
,
project:
project_next_1_1
,
title:
'1.1'
,
due_date:
two_days_from_now
),
create
(
:milestone
,
project:
project_next_1_1
,
title:
'8.8'
,
due_date:
ten_days_from_now
),
create
(
:milestone
,
project:
project_next_8_8
,
title:
'1.1'
,
due_date:
yesterday
),
create
(
:milestone
,
project:
project_next_8_8
,
title:
'8.8'
,
due_date:
tomorrow
)
create
(
:milestone
,
project:
project_next_1_1
,
title:
'8.9'
,
due_date:
ten_days_from_now
),
create
(
:milestone
,
project:
project_next_8_8
,
title:
'1.2'
,
due_date:
yesterday
),
create
(
:milestone
,
project:
project_next_8_8
,
title:
'8.8'
,
due_date:
tomorrow
),
create
(
:milestone
,
group:
group
,
title:
'9.9'
,
due_date:
tomorrow
)
]
end
before
do
milestones
.
each
do
|
milestone
|
create
(
:issue
,
project:
milestone
.
project
,
milestone:
milestone
,
author:
user
,
assignees:
[
user
])
create
(
:issue
,
project:
milestone
.
project
||
project_in_group
,
milestone:
milestone
,
author:
user
,
assignees:
[
user
])
end
end
it
'returns issues in the upcoming milestone for each project'
do
expect
(
issues
.
map
{
|
issue
|
issue
.
milestone
.
title
}).
to
contain_exactly
(
'1.1'
,
'8.8'
)
expect
(
issues
.
map
{
|
issue
|
issue
.
milestone
.
due_date
}).
to
contain_exactly
(
tomorrow
,
two_days_from_now
)
it
'returns issues in the upcoming milestone for each project
or group
'
do
expect
(
issues
.
map
{
|
issue
|
issue
.
milestone
.
title
}).
to
contain_exactly
(
'1.1'
,
'8.8'
,
'9.9'
)
expect
(
issues
.
map
{
|
issue
|
issue
.
milestone
.
due_date
}).
to
contain_exactly
(
tomorrow
,
two_days_from_now
,
tomorrow
)
end
end
...
...
spec/models/milestone_spec.rb
View file @
f1aa62e3
...
...
@@ -242,7 +242,88 @@ describe Milestone do
end
end
describe
'.upcoming_ids_by_projects'
do
describe
'#for_projects_and_groups'
do
let
(
:project
)
{
create
(
:project
)
}
let
(
:project_other
)
{
create
(
:project
)
}
let
(
:group
)
{
create
(
:group
)
}
let
(
:group_other
)
{
create
(
:group
)
}
before
do
create
(
:milestone
,
project:
project
)
create
(
:milestone
,
project:
project_other
)
create
(
:milestone
,
group:
group
)
create
(
:milestone
,
group:
group_other
)
end
subject
{
described_class
.
for_projects_and_groups
(
projects
,
groups
)
}
shared_examples
'filters by projects and groups'
do
it
'returns milestones filtered by project'
do
milestones
=
described_class
.
for_projects_and_groups
(
projects
,
[])
expect
(
milestones
.
count
).
to
eq
(
1
)
expect
(
milestones
.
first
.
project_id
).
to
eq
(
project
.
id
)
end
it
'returns milestones filtered by group'
do
milestones
=
described_class
.
for_projects_and_groups
([],
groups
)
expect
(
milestones
.
count
).
to
eq
(
1
)
expect
(
milestones
.
first
.
group_id
).
to
eq
(
group
.
id
)
end
it
'returns milestones filtered by both project and group'
do
milestones
=
described_class
.
for_projects_and_groups
(
projects
,
groups
)
expect
(
milestones
.
count
).
to
eq
(
2
)
expect
(
milestones
).
to
contain_exactly
(
project
.
milestones
.
first
,
group
.
milestones
.
first
)
end
end
context
'ids as params'
do
let
(
:projects
)
{
[
project
.
id
]
}
let
(
:groups
)
{
[
group
.
id
]
}
it_behaves_like
'filters by projects and groups'
end
context
'relations as params'
do
let
(
:projects
)
{
Project
.
where
(
id:
project
.
id
)
}
let
(
:groups
)
{
Group
.
where
(
id:
group
.
id
)
}
it_behaves_like
'filters by projects and groups'
end
context
'objects as params'
do
let
(
:projects
)
{
[
project
]
}
let
(
:groups
)
{
[
group
]
}
it_behaves_like
'filters by projects and groups'
end
it
'returns no records if projects and groups are nil'
do
milestones
=
described_class
.
for_projects_and_groups
(
nil
,
nil
)
expect
(
milestones
).
to
be_empty
end
end
describe
'.upcoming_ids'
do
let
(
:group_1
)
{
create
(
:group
)
}
let
(
:group_2
)
{
create
(
:group
)
}
let
(
:group_3
)
{
create
(
:group
)
}
let
(
:groups
)
{
[
group_1
,
group_2
,
group_3
]
}
let!
(
:past_milestone_group_1
)
{
create
(
:milestone
,
group:
group_1
,
due_date:
Time
.
now
-
1
.
day
)
}
let!
(
:current_milestone_group_1
)
{
create
(
:milestone
,
group:
group_1
,
due_date:
Time
.
now
+
1
.
day
)
}
let!
(
:future_milestone_group_1
)
{
create
(
:milestone
,
group:
group_1
,
due_date:
Time
.
now
+
2
.
days
)
}
let!
(
:past_milestone_group_2
)
{
create
(
:milestone
,
group:
group_2
,
due_date:
Time
.
now
-
1
.
day
)
}
let!
(
:closed_milestone_group_2
)
{
create
(
:milestone
,
:closed
,
group:
group_2
,
due_date:
Time
.
now
+
1
.
day
)
}
let!
(
:current_milestone_group_2
)
{
create
(
:milestone
,
group:
group_2
,
due_date:
Time
.
now
+
2
.
days
)
}
let!
(
:past_milestone_group_3
)
{
create
(
:milestone
,
group:
group_3
,
due_date:
Time
.
now
-
1
.
day
)
}
let
(
:project_1
)
{
create
(
:project
)
}
let
(
:project_2
)
{
create
(
:project
)
}
let
(
:project_3
)
{
create
(
:project
)
}
...
...
@@ -258,16 +339,20 @@ describe Milestone do
let!
(
:past_milestone_project_3
)
{
create
(
:milestone
,
project:
project_3
,
due_date:
Time
.
now
-
1
.
day
)
}
# The call to `#try` is because this returns a relation with a Postgres DB,
# and an array of IDs with a MySQL DB.
let
(
:milestone_ids
)
{
described_class
.
upcoming_ids_by_projects
(
projects
).
map
{
|
id
|
id
.
try
(
:id
)
||
id
}
}
let
(
:milestone_ids
)
{
described_class
.
upcoming_ids
(
projects
,
groups
).
map
(
&
:id
)
}
it
'returns the next upcoming open milestone ID for each project'
do
expect
(
milestone_ids
).
to
contain_exactly
(
current_milestone_project_1
.
id
,
current_milestone_project_2
.
id
)
it
'returns the next upcoming open milestone ID for each project and group'
do
expect
(
milestone_ids
).
to
contain_exactly
(
current_milestone_project_1
.
id
,
current_milestone_project_2
.
id
,
current_milestone_group_1
.
id
,
current_milestone_group_2
.
id
)
end
context
'when the projects have no open upcoming milestones'
do
context
'when the projects
and groups
have no open upcoming milestones'
do
let
(
:projects
)
{
[
project_3
]
}
let
(
:groups
)
{
[
group_3
]
}
it
'returns no results'
do
expect
(
milestone_ids
).
to
be_empty
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment