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
552cbbe9
Commit
552cbbe9
authored
Oct 28, 2020
by
Tristan Read
Committed by
Andreas Brandl
Oct 28, 2020
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Add SLA sort for issues
Update docs
parent
ba9eb61c
Changes
13
Hide whitespace changes
Inline
Side-by-side
Showing
13 changed files
with
141 additions
and
21 deletions
+141
-21
app/assets/javascripts/incidents/components/incidents_list.vue
...ssets/javascripts/incidents/components/incidents_list.vue
+17
-5
app/models/concerns/issue_available_features.rb
app/models/concerns/issue_available_features.rb
+12
-5
doc/api/graphql/reference/gitlab_schema.graphql
doc/api/graphql/reference/gitlab_schema.graphql
+10
-0
doc/api/graphql/reference/gitlab_schema.json
doc/api/graphql/reference/gitlab_schema.json
+12
-0
doc/api/graphql/reference/index.md
doc/api/graphql/reference/index.md
+2
-0
ee/app/graphql/ee/types/issue_sort_enum.rb
ee/app/graphql/ee/types/issue_sort_enum.rb
+2
-0
ee/app/models/concerns/ee/issue_available_features.rb
ee/app/models/concerns/ee/issue_available_features.rb
+8
-6
ee/app/models/ee/issue.rb
ee/app/models/ee/issue.rb
+4
-0
ee/changelogs/unreleased/tr-sort-sla-column.yml
ee/changelogs/unreleased/tr-sort-sla-column.yml
+5
-0
ee/spec/graphql/ee/resolvers/issues_resolver_spec.rb
ee/spec/graphql/ee/resolvers/issues_resolver_spec.rb
+18
-0
ee/spec/graphql/ee/types/issue_sort_enum_spec.rb
ee/spec/graphql/ee/types/issue_sort_enum_spec.rb
+1
-1
ee/spec/models/issue_spec.rb
ee/spec/models/issue_spec.rb
+44
-0
spec/frontend/incidents/components/incidents_list_spec.js
spec/frontend/incidents/components/incidents_list_spec.js
+6
-4
No files found.
app/assets/javascripts/incidents/components/incidents_list.vue
View file @
552cbbe9
...
@@ -69,9 +69,12 @@ export default {
...
@@ -69,9 +69,12 @@ export default {
{
{
key
:
'
incidentSla
'
,
key
:
'
incidentSla
'
,
label
:
s__
(
'
IncidentManagement|Time to SLA
'
),
label
:
s__
(
'
IncidentManagement|Time to SLA
'
),
thClass
:
`gl-
pointer-events-none gl-
text-right gl-w-eighth`
,
thClass
:
`gl-text-right gl-w-eighth`
,
tdClass
:
`
${
tdClass
}
gl-text-right`
,
tdClass
:
`
${
tdClass
}
gl-text-right`
,
thAttr
:
TH_INCIDENT_SLA_TEST_ID
,
thAttr
:
TH_INCIDENT_SLA_TEST_ID
,
sortKey
:
'
SLA_DUE_AT
'
,
sortable
:
true
,
sortDirection
:
'
asc
'
,
},
},
{
{
key
:
'
assignees
'
,
key
:
'
assignees
'
,
...
@@ -253,13 +256,22 @@ export default {
...
@@ -253,13 +256,22 @@ export default {
this
.
redirecting
=
true
;
this
.
redirecting
=
true
;
},
},
fetchSortedData
({
sortBy
,
sortDesc
})
{
fetchSortedData
({
sortBy
,
sortDesc
})
{
let
sortKey
;
// In bootstrap-vue v2.17.0, sortKey becomes natively supported and we can eliminate this function
const
field
=
this
.
availableFields
.
find
(({
key
})
=>
key
===
sortBy
);
const
sortingDirection
=
sortDesc
?
'
DESC
'
:
'
ASC
'
;
const
sortingDirection
=
sortDesc
?
'
DESC
'
:
'
ASC
'
;
const
sortingColumn
=
convertToSnakeCase
(
sortBy
)
.
replace
(
/_.*/
,
''
)
// Use `sortKey` if provided, otherwise fall back to existing algorithm
.
toUpperCase
();
if
(
field
?.
sortKey
)
{
sortKey
=
field
.
sortKey
;
}
else
{
sortKey
=
convertToSnakeCase
(
sortBy
)
.
replace
(
/_.*/
,
''
)
.
toUpperCase
();
}
this
.
pagination
=
initialPaginationState
;
this
.
pagination
=
initialPaginationState
;
this
.
sort
=
`
${
sort
ingColumn
}
_
${
sortingDirection
}
`
;
this
.
sort
=
`
${
sort
Key
}
_
${
sortingDirection
}
`
;
},
},
getSeverity
(
severity
)
{
getSeverity
(
severity
)
{
return
INCIDENT_SEVERITY
[
severity
];
return
INCIDENT_SEVERITY
[
severity
];
...
...
app/models/concerns/issue_available_features.rb
View file @
552cbbe9
...
@@ -6,18 +6,25 @@
...
@@ -6,18 +6,25 @@
module
IssueAvailableFeatures
module
IssueAvailableFeatures
extend
ActiveSupport
::
Concern
extend
ActiveSupport
::
Concern
# EE only features are listed on EE::IssueAvailableFeatures
class_methods
do
def
available_features_for_issue_types
# EE only features are listed on EE::IssueAvailableFeatures
{}.
with_indifferent_access
def
available_features_for_issue_types
{}.
with_indifferent_access
end
end
included
do
scope
:with_feature
,
->
(
feature
)
{
where
(
issue_type:
available_features_for_issue_types
[
feature
])
}
end
end
def
issue_type_supports?
(
feature
)
def
issue_type_supports?
(
feature
)
unless
available_features_for_issue_types
.
has_key?
(
feature
)
unless
self
.
class
.
available_features_for_issue_types
.
has_key?
(
feature
)
raise
ArgumentError
,
'invalid feature'
raise
ArgumentError
,
'invalid feature'
end
end
available_features_for_issue_types
[
feature
].
include?
(
issue_type
)
self
.
class
.
available_features_for_issue_types
[
feature
].
include?
(
issue_type
)
end
end
end
end
IssueAvailableFeatures
.
prepend_if_ee
(
'EE::IssueAvailableFeatures'
)
IssueAvailableFeatures
.
prepend_if_ee
(
'EE::IssueAvailableFeatures'
)
IssueAvailableFeatures
::
ClassMethods
.
prepend_if_ee
(
'EE::IssueAvailableFeatures::ClassMethods'
)
doc/api/graphql/reference/gitlab_schema.graphql
View file @
552cbbe9
...
@@ -10197,6 +10197,16 @@ enum IssueSort {
...
@@ -10197,6 +10197,16 @@ enum IssueSort {
"""
"""
SEVERITY_DESC
SEVERITY_DESC
"""
Issues
with
earliest
SLA
due
time
shown
first
"""
SLA_DUE_AT_ASC
"""
Issues
with
latest
SLA
due
time
shown
first
"""
SLA_DUE_AT_DESC
"""
"""
Updated
at
ascending
order
Updated
at
ascending
order
"""
"""
...
...
doc/api/graphql/reference/gitlab_schema.json
View file @
552cbbe9
...
@@ -27836,6 +27836,18 @@
...
@@ -27836,6 +27836,18 @@
"description": "Published issues shown first",
"description": "Published issues shown first",
"isDeprecated": false,
"isDeprecated": false,
"deprecationReason": null
"deprecationReason": null
},
{
"name": "SLA_DUE_AT_ASC",
"description": "Issues with earliest SLA due time shown first",
"isDeprecated": false,
"deprecationReason": null
},
{
"name": "SLA_DUE_AT_DESC",
"description": "Issues with latest SLA due time shown first",
"isDeprecated": false,
"deprecationReason": null
}
}
],
],
"possibleTypes": null
"possibleTypes": null
doc/api/graphql/reference/index.md
View file @
552cbbe9
...
@@ -3485,6 +3485,8 @@ Values for sorting issues.
...
@@ -3485,6 +3485,8 @@ Values for sorting issues.
|
`RELATIVE_POSITION_ASC`
| Relative position by ascending order |
|
`RELATIVE_POSITION_ASC`
| Relative position by ascending order |
|
`SEVERITY_ASC`
| Severity from less critical to more critical |
|
`SEVERITY_ASC`
| Severity from less critical to more critical |
|
`SEVERITY_DESC`
| Severity from more critical to less critical |
|
`SEVERITY_DESC`
| Severity from more critical to less critical |
|
`SLA_DUE_AT_ASC`
| Issues with earliest SLA due time shown first |
|
`SLA_DUE_AT_DESC`
| Issues with latest SLA due time shown first |
|
`UPDATED_ASC`
| Updated at ascending order |
|
`UPDATED_ASC`
| Updated at ascending order |
|
`UPDATED_DESC`
| Updated at descending order |
|
`UPDATED_DESC`
| Updated at descending order |
|
`WEIGHT_ASC`
| Weight by ascending order |
|
`WEIGHT_ASC`
| Weight by ascending order |
...
...
ee/app/graphql/ee/types/issue_sort_enum.rb
View file @
552cbbe9
...
@@ -10,6 +10,8 @@ module EE
...
@@ -10,6 +10,8 @@ module EE
value
'WEIGHT_DESC'
,
'Weight by descending order'
,
value:
'weight_desc'
value
'WEIGHT_DESC'
,
'Weight by descending order'
,
value:
'weight_desc'
value
'PUBLISHED_ASC'
,
'Published issues shown last'
,
value: :published_asc
value
'PUBLISHED_ASC'
,
'Published issues shown last'
,
value: :published_asc
value
'PUBLISHED_DESC'
,
'Published issues shown first'
,
value: :published_desc
value
'PUBLISHED_DESC'
,
'Published issues shown first'
,
value: :published_desc
value
'SLA_DUE_AT_ASC'
,
'Issues with earliest SLA due time shown first'
,
value: :sla_due_at_asc
value
'SLA_DUE_AT_DESC'
,
'Issues with latest SLA due time shown first'
,
value: :sla_due_at_desc
end
end
end
end
end
end
...
...
ee/app/models/concerns/ee/issue_available_features.rb
View file @
552cbbe9
...
@@ -2,13 +2,15 @@
...
@@ -2,13 +2,15 @@
module
EE
module
EE
module
IssueAvailableFeatures
module
IssueAvailableFeatures
include
::
Gitlab
::
Utils
::
StrongMemoize
extend
ActiveSupport
::
Concern
extend
::
Gitlab
::
Utils
::
Override
override
:available_features_for_issue_types
class_methods
do
def
available_features_for_issue_types
include
::
Gitlab
::
Utils
::
StrongMemoize
strong_memoize
(
:available_features_for_issue_types
)
do
super
.
merge
(
epics:
%w(issue)
)
def
available_features_for_issue_types
strong_memoize
(
:available_features_for_issue_types
)
do
super
.
merge
(
epics:
%w(issue)
,
sla:
%w(incident)
)
end
end
end
end
end
end
end
...
...
ee/app/models/ee/issue.rb
View file @
552cbbe9
...
@@ -23,6 +23,8 @@ module EE
...
@@ -23,6 +23,8 @@ module EE
scope
:order_weight_asc
,
->
{
reorder
::
Gitlab
::
Database
.
nulls_last_order
(
'weight'
)
}
scope
:order_weight_asc
,
->
{
reorder
::
Gitlab
::
Database
.
nulls_last_order
(
'weight'
)
}
scope
:order_status_page_published_first
,
->
{
includes
(
:status_page_published_incident
).
order
(
'status_page_published_incidents.id ASC NULLS LAST'
)
}
scope
:order_status_page_published_first
,
->
{
includes
(
:status_page_published_incident
).
order
(
'status_page_published_incidents.id ASC NULLS LAST'
)
}
scope
:order_status_page_published_last
,
->
{
includes
(
:status_page_published_incident
).
order
(
'status_page_published_incidents.id ASC NULLS FIRST'
)
}
scope
:order_status_page_published_last
,
->
{
includes
(
:status_page_published_incident
).
order
(
'status_page_published_incidents.id ASC NULLS FIRST'
)
}
scope
:order_sla_due_at_asc
,
->
{
includes
(
:issuable_sla
).
order
(
'issuable_slas.due_at ASC NULLS LAST'
)
}
scope
:order_sla_due_at_desc
,
->
{
includes
(
:issuable_sla
).
order
(
'issuable_slas.due_at DESC NULLS LAST'
)
}
scope
:no_epic
,
->
{
left_outer_joins
(
:epic_issue
).
where
(
epic_issues:
{
epic_id:
nil
})
}
scope
:no_epic
,
->
{
left_outer_joins
(
:epic_issue
).
where
(
epic_issues:
{
epic_id:
nil
})
}
scope
:any_epic
,
->
{
joins
(
:epic_issue
)
}
scope
:any_epic
,
->
{
joins
(
:epic_issue
)
}
scope
:in_epics
,
->
(
epics
)
{
joins
(
:epic_issue
).
where
(
epic_issues:
{
epic_id:
epics
})
}
scope
:in_epics
,
->
(
epics
)
{
joins
(
:epic_issue
).
where
(
epic_issues:
{
epic_id:
epics
})
}
...
@@ -190,6 +192,8 @@ module EE
...
@@ -190,6 +192,8 @@ module EE
when
'weight_desc'
then
order_weight_desc
.
with_order_id_desc
when
'weight_desc'
then
order_weight_desc
.
with_order_id_desc
when
'published_asc'
then
order_status_page_published_last
.
with_order_id_desc
when
'published_asc'
then
order_status_page_published_last
.
with_order_id_desc
when
'published_desc'
then
order_status_page_published_first
.
with_order_id_desc
when
'published_desc'
then
order_status_page_published_first
.
with_order_id_desc
when
'sla_due_at_asc'
then
with_feature
(
:sla
).
order_sla_due_at_asc
.
with_order_id_desc
when
'sla_due_at_desc'
then
with_feature
(
:sla
).
order_sla_due_at_desc
.
with_order_id_desc
else
else
super
super
end
end
...
...
ee/changelogs/unreleased/tr-sort-sla-column.yml
0 → 100644
View file @
552cbbe9
---
title
:
Allow sorting of the Incident SLA column
merge_request
:
45344
author
:
type
:
added
ee/spec/graphql/ee/resolvers/issues_resolver_spec.rb
View file @
552cbbe9
...
@@ -43,6 +43,24 @@ RSpec.describe Resolvers::IssuesResolver do
...
@@ -43,6 +43,24 @@ RSpec.describe Resolvers::IssuesResolver do
expect
(
resolve_issues
(
sort: :published_desc
)).
to
eq
[
published
,
not_published
]
expect
(
resolve_issues
(
sort: :published_desc
)).
to
eq
[
published
,
not_published
]
end
end
end
end
context
'when sorting by sla due at'
do
let_it_be
(
:sla_due_first
)
{
create
(
:incident
,
project:
project
)
}
let_it_be
(
:sla_due_last
)
{
create
(
:incident
,
project:
project
)
}
before_all
do
create
(
:issuable_sla
,
:exceeded
,
issue:
sla_due_first
)
create
(
:issuable_sla
,
issue:
sla_due_last
)
end
it
'sorts issues ascending'
do
expect
(
resolve_issues
(
sort: :sla_due_at_asc
)).
to
eq
[
sla_due_first
,
sla_due_last
]
end
it
'sorts issues descending'
do
expect
(
resolve_issues
(
sort: :sla_due_at_desc
)).
to
eq
[
sla_due_last
,
sla_due_first
]
end
end
end
end
describe
'filtering by iteration'
do
describe
'filtering by iteration'
do
...
...
ee/spec/graphql/ee/types/issue_sort_enum_spec.rb
View file @
552cbbe9
...
@@ -8,6 +8,6 @@ RSpec.describe GitlabSchema.types['IssueSort'] do
...
@@ -8,6 +8,6 @@ RSpec.describe GitlabSchema.types['IssueSort'] do
it_behaves_like
'common sort values'
it_behaves_like
'common sort values'
it
'exposes all the existing EE issue sort values'
do
it
'exposes all the existing EE issue sort values'
do
expect
(
described_class
.
values
.
keys
).
to
include
(
*
%w[WEIGHT_ASC WEIGHT_DESC PUBLISHED_ASC PUBLISHED_DESC]
)
expect
(
described_class
.
values
.
keys
).
to
include
(
*
%w[WEIGHT_ASC WEIGHT_DESC PUBLISHED_ASC PUBLISHED_DESC
SLA_DUE_AT_ASC SLA_DUE_AT_DESC
]
)
end
end
end
end
ee/spec/models/issue_spec.rb
View file @
552cbbe9
...
@@ -100,6 +100,26 @@ RSpec.describe Issue do
...
@@ -100,6 +100,26 @@ RSpec.describe Issue do
end
end
end
end
describe
'.with_feature'
do
let_it_be
(
:project
)
{
create
(
:project
)
}
let_it_be
(
:issue
)
{
create
(
:issue
,
project:
project
)
}
let_it_be
(
:incident
)
{
create
(
:incident
,
project:
project
)
}
let_it_be
(
:test_case
)
{
create
(
:quality_test_case
,
project:
project
)
}
it
'gives issues that support the given feature'
,
:aggregate_failures
do
expect
(
described_class
.
with_feature
(
'epics'
))
.
to
contain_exactly
(
issue
)
expect
(
described_class
.
with_feature
(
'sla'
))
.
to
contain_exactly
(
incident
)
end
it
'returns an empty collection when given an unknown feature'
do
expect
(
described_class
.
with_feature
(
'something-unknown'
))
.
to
be_empty
end
end
context
'epics'
do
context
'epics'
do
let_it_be
(
:epic1
)
{
create
(
:epic
)
}
let_it_be
(
:epic1
)
{
create
(
:epic
)
}
let_it_be
(
:epic2
)
{
create
(
:epic
)
}
let_it_be
(
:epic2
)
{
create
(
:epic
)
}
...
@@ -207,6 +227,30 @@ RSpec.describe Issue do
...
@@ -207,6 +227,30 @@ RSpec.describe Issue do
it
{
is_expected
.
to
eq
([
not_published
,
published
])
}
it
{
is_expected
.
to
eq
([
not_published
,
published
])
}
end
end
end
end
context
'sla due at'
do
let_it_be
(
:project
)
{
create
(
:project
)
}
let_it_be
(
:sla_due_first
)
{
create
(
:issue
,
project:
project
)
}
let_it_be
(
:sla_due_last
)
{
create
(
:issue
,
project:
project
)
}
let_it_be
(
:no_sla
)
{
create
(
:issue
,
project:
project
)
}
before_all
do
create
(
:issuable_sla
,
:exceeded
,
issue:
sla_due_first
)
create
(
:issuable_sla
,
issue:
sla_due_last
)
end
describe
'.order_sla_due_at_asc'
do
subject
{
described_class
.
order_sla_due_at_asc
}
it
{
is_expected
.
to
eq
([
sla_due_first
,
sla_due_last
,
no_sla
])
}
end
describe
'.order_sla_due_at_desc'
do
subject
{
described_class
.
order_sla_due_at_desc
}
it
{
is_expected
.
to
eq
([
sla_due_last
,
sla_due_first
,
no_sla
])
}
end
end
end
end
describe
'validations'
do
describe
'validations'
do
...
...
spec/frontend/incidents/components/incidents_list_spec.js
View file @
552cbbe9
...
@@ -10,6 +10,7 @@ import {
...
@@ -10,6 +10,7 @@ import {
TH_CREATED_AT_TEST_ID
,
TH_CREATED_AT_TEST_ID
,
TH_SEVERITY_TEST_ID
,
TH_SEVERITY_TEST_ID
,
TH_PUBLISHED_TEST_ID
,
TH_PUBLISHED_TEST_ID
,
TH_INCIDENT_SLA_TEST_ID
,
trackIncidentCreateNewOptions
,
trackIncidentCreateNewOptions
,
trackIncidentListViewsOptions
,
trackIncidentListViewsOptions
,
}
from
'
~/incidents/constants
'
;
}
from
'
~/incidents/constants
'
;
...
@@ -277,10 +278,11 @@ describe('Incidents List', () => {
...
@@ -277,10 +278,11 @@ describe('Incidents List', () => {
const
noneSort
=
'
none
'
;
const
noneSort
=
'
none
'
;
it
.
each
`
it
.
each
`
selector | initialSort | firstSort | nextSort
selector | initialSort | firstSort | nextSort
${
TH_CREATED_AT_TEST_ID
}
|
${
descSort
}
|
${
ascSort
}
|
${
descSort
}
${
TH_CREATED_AT_TEST_ID
}
|
${
descSort
}
|
${
ascSort
}
|
${
descSort
}
${
TH_SEVERITY_TEST_ID
}
|
${
noneSort
}
|
${
descSort
}
|
${
ascSort
}
${
TH_SEVERITY_TEST_ID
}
|
${
noneSort
}
|
${
descSort
}
|
${
ascSort
}
${
TH_PUBLISHED_TEST_ID
}
|
${
noneSort
}
|
${
descSort
}
|
${
ascSort
}
${
TH_PUBLISHED_TEST_ID
}
|
${
noneSort
}
|
${
descSort
}
|
${
ascSort
}
${
TH_INCIDENT_SLA_TEST_ID
}
|
${
noneSort
}
|
${
ascSort
}
|
${
descSort
}
`
(
'
updates sort with new direction
'
,
async
({
selector
,
initialSort
,
firstSort
,
nextSort
})
=>
{
`
(
'
updates sort with new direction
'
,
async
({
selector
,
initialSort
,
firstSort
,
nextSort
})
=>
{
const
[[
attr
,
value
]]
=
Object
.
entries
(
selector
);
const
[[
attr
,
value
]]
=
Object
.
entries
(
selector
);
const
columnHeader
=
()
=>
wrapper
.
find
(
`[
${
attr
}
="
${
value
}
"]`
);
const
columnHeader
=
()
=>
wrapper
.
find
(
`[
${
attr
}
="
${
value
}
"]`
);
...
...
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