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
92d9f432
Commit
92d9f432
authored
Mar 07, 2022
by
Felipe Artur
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Allow to list related epics on REST API
Add REST API endpoint to list related epics. Changelog: added EE: true
parent
9ec1872a
Changes
8
Hide whitespace changes
Inline
Side-by-side
Showing
8 changed files
with
193 additions
and
3 deletions
+193
-3
ee/app/models/ee/epic.rb
ee/app/models/ee/epic.rb
+1
-0
ee/lib/api/entities/related_epic.rb
ee/lib/api/entities/related_epic.rb
+12
-0
ee/lib/api/helpers/epics_helpers.rb
ee/lib/api/helpers/epics_helpers.rb
+8
-3
ee/lib/api/related_epic_links.rb
ee/lib/api/related_epic_links.rb
+50
-0
ee/lib/ee/api/api.rb
ee/lib/ee/api/api.rb
+1
-0
ee/spec/fixtures/api/schemas/public_api/v4/related_epic_link.json
...fixtures/api/schemas/public_api/v4/related_epic_link.json
+22
-0
ee/spec/fixtures/api/schemas/public_api/v4/related_epic_links.json
...ixtures/api/schemas/public_api/v4/related_epic_links.json
+9
-0
ee/spec/requests/api/related_epic_links_spec.rb
ee/spec/requests/api/related_epic_links_spec.rb
+90
-0
No files found.
ee/app/models/ee/epic.rb
View file @
92d9f432
...
@@ -86,6 +86,7 @@ module EE
...
@@ -86,6 +86,7 @@ module EE
scope
:from_id
,
->
(
epic_id
)
{
where
(
'epics.id >= ?'
,
epic_id
)
}
scope
:from_id
,
->
(
epic_id
)
{
where
(
'epics.id >= ?'
,
epic_id
)
}
scope
:with_web_entity_associations
,
->
{
preload
(
:author
,
group:
[
:ip_restrictions
,
:route
])
}
scope
:with_web_entity_associations
,
->
{
preload
(
:author
,
group:
[
:ip_restrictions
,
:route
])
}
scope
:with_api_entity_associations
,
->
{
preload
(
:author
,
:labels
)
}
scope
:within_timeframe
,
->
(
start_date
,
end_date
)
do
scope
:within_timeframe
,
->
(
start_date
,
end_date
)
do
where
(
'start_date is not NULL or end_date is not NULL'
)
where
(
'start_date is not NULL or end_date is not NULL'
)
...
...
ee/lib/api/entities/related_epic.rb
0 → 100644
View file @
92d9f432
# frozen_string_literal: true
module
API
module
Entities
class
RelatedEpic
<
EE
::
API
::
Entities
::
Epic
expose
:related_epic_link_id
expose
:epic_link_type
,
as: :link_type
expose
:related_epic_link_created_at
,
as: :link_created_at
expose
:related_epic_link_updated_at
,
as: :link_updated_at
end
end
end
ee/lib/api/helpers/epics_helpers.rb
View file @
92d9f432
...
@@ -11,6 +11,10 @@ module API
...
@@ -11,6 +11,10 @@ module API
forbidden!
unless
user_group
.
licensed_feature_available?
(
:subepics
)
forbidden!
unless
user_group
.
licensed_feature_available?
(
:subepics
)
end
end
def
authorize_related_epics_feature!
forbidden!
unless
user_group
.
licensed_feature_available?
(
:related_epics
)
end
def
authorize_can_read!
def
authorize_can_read!
authorize!
(
:read_epic
,
epic
)
authorize!
(
:read_epic
,
epic
)
end
end
...
@@ -56,11 +60,12 @@ module API
...
@@ -56,11 +60,12 @@ module API
end
end
# rubocop: enable CodeReuse/ActiveRecord
# rubocop: enable CodeReuse/ActiveRecord
def
epic_options
def
epic_options
(
entity:
EE
::
API
::
Entities
::
Epic
,
issuable_metadata:
nil
)
{
{
with:
EE
::
API
::
Entities
::
Epic
,
with:
entity
,
user:
current_user
,
user:
current_user
,
group:
user_group
group:
user_group
,
issuable_metadata:
issuable_metadata
}
}
end
end
end
end
...
...
ee/lib/api/related_epic_links.rb
0 → 100644
View file @
92d9f432
# frozen_string_literal: true
# These API endpoints are used to manage the relationship between one epic with other epic
# (similar to issue links). Note that this relation is different from existing
# API::EpicLinks (which is used for parent-child epic hierarchy).
module
API
class
RelatedEpicLinks
<
::
API
::
Base
include
PaginationParams
feature_category
:portfolio_management
helpers
do
def
authorize_related_epics_feature_flag!
not_found!
unless
Feature
.
enabled?
(
:related_epics_widget
,
user_group
,
default_enabled: :yaml
)
end
end
helpers
::
API
::
Helpers
::
EpicsHelpers
before
do
authenticate_non_get!
authorize_related_epics_feature_flag!
authorize_related_epics_feature!
end
params
do
requires
:id
,
type:
String
,
desc:
'The ID of a group'
requires
:epic_iid
,
type:
Integer
,
desc:
'The internal ID of a group epic'
end
resource
:groups
,
requirements:
API
::
NAMESPACE_OR_PROJECT_REQUIREMENTS
do
desc
'Get related epics'
do
success
Entities
::
RelatedEpic
end
get
':id/epics/:epic_iid/related_epics'
do
authorize_can_read!
preload_for_collection
=
[
group:
[
:saml_provider
,
:route
]]
related_epics
=
epic
.
related_epics
(
current_user
,
preload:
preload_for_collection
)
do
|
epics
|
epics
.
with_api_entity_associations
end
epics_metadata
=
Gitlab
::
IssuableMetadata
.
new
(
current_user
,
related_epics
).
data
presenter_options
=
epic_options
(
entity:
Entities
::
RelatedEpic
,
issuable_metadata:
epics_metadata
)
present
related_epics
,
presenter_options
end
end
end
end
ee/lib/ee/api/api.rb
View file @
92d9f432
...
@@ -19,6 +19,7 @@ module EE
...
@@ -19,6 +19,7 @@ module EE
mount
::
API
::
EpicIssues
mount
::
API
::
EpicIssues
mount
::
API
::
EpicLinks
mount
::
API
::
EpicLinks
mount
::
API
::
Epics
mount
::
API
::
Epics
mount
::
API
::
RelatedEpicLinks
mount
::
API
::
ElasticsearchIndexedNamespaces
mount
::
API
::
ElasticsearchIndexedNamespaces
mount
::
API
::
Experiments
mount
::
API
::
Experiments
mount
::
API
::
GeoReplication
mount
::
API
::
GeoReplication
...
...
ee/spec/fixtures/api/schemas/public_api/v4/related_epic_link.json
0 → 100644
View file @
92d9f432
{
"type"
:
"object"
,
"properties"
:
{
"source_epic"
:
{
"allOf"
:
[
{
"$ref"
:
"../../../../../../spec/fixtures/api/schemas/public_api/v4/epic.json"
}
]
},
"target_epic"
:
{
"allOf"
:
[
{
"$ref"
:
"../../../../../../spec/fixtures/api/schemas/public_api/v4/epic.json"
}
]
},
"link_type"
:
{
"type"
:
"string"
,
"enum"
:
[
"relates_to"
,
"blocks"
,
"is_blocked_by"
]
},
"link_created_at"
:
{
"type"
:
"string"
},
"link_updated_at"
:
{
"type"
:
"string"
}
},
"required"
:
[
"source_epic"
,
"target_epic"
,
"link_type"
]
}
ee/spec/fixtures/api/schemas/public_api/v4/related_epic_links.json
0 → 100644
View file @
92d9f432
{
"type"
:
"array"
,
"items"
:
{
"type"
:
"object"
,
"properties"
:
{
"$ref"
:
"./related_epic_link.json"
}
}
}
ee/spec/requests/api/related_epic_links_spec.rb
0 → 100644
View file @
92d9f432
# frozen_string_literal: true
require
'spec_helper'
RSpec
.
describe
API
::
RelatedEpicLinks
do
include
ExternalAuthorizationServiceHelpers
let_it_be
(
:user
)
{
create
(
:user
)
}
let_it_be
(
:group
)
{
create
(
:group
,
:private
)
}
let_it_be
(
:epic
)
{
create
(
:epic
,
group:
group
)
}
before
do
stub_licensed_features
(
epics:
true
,
related_epics:
true
)
end
shared_examples
'a not available endpoint'
do
subject
{
perform_request
(
user
)
}
context
'when epics feature is not available'
do
before
do
stub_licensed_features
(
epics:
false
,
related_epics:
true
)
end
it
{
is_expected
.
to
eq
(
403
)
}
end
context
'when related_epics feature is not available '
do
before
do
stub_licensed_features
(
epics:
true
,
related_epics:
false
)
end
it
{
is_expected
.
to
eq
(
403
)
}
end
context
'when related epics widget feature flag is disabled'
do
before
do
stub_licensed_features
(
epics:
true
,
related_epics:
true
)
stub_feature_flags
(
related_epics_widget:
false
)
end
it
{
is_expected
.
to
eq
(
404
)
}
end
end
describe
'GET /related_epics'
do
def
perform_request
(
user
=
nil
,
params
=
{})
get
api
(
"/groups/
#{
group
.
id
}
/epics/
#{
epic
.
iid
}
/related_epics"
,
user
),
params:
params
end
context
'when user cannot read epics'
do
it
'returns 404'
do
perform_request
expect
(
response
).
to
have_gitlab_http_status
(
:not_found
)
end
end
context
'when user can read epics'
do
let_it_be
(
:group_2
)
{
create
(
:group
)
}
let_it_be
(
:related_epic_link_1
)
{
create
(
:related_epic_link
,
source:
epic
,
target:
create
(
:epic
,
group:
group
))
}
let_it_be
(
:related_epic_link_2
)
{
create
(
:related_epic_link
,
source:
epic
,
target:
create
(
:epic
,
group:
group_2
))
}
before
do
group
.
add_guest
(
user
)
end
it_behaves_like
'a not available endpoint'
it
'returns related epics'
do
perform_request
(
user
)
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
expect
(
json_response
).
to
be_an
Array
expect
(
json_response
.
length
).
to
eq
(
2
)
expect
(
response
).
to
match_response_schema
(
'public_api/v4/related_epic_links'
,
dir:
'ee'
)
end
it
'returns multiple links without N + 1'
do
perform_request
(
user
)
control_count
=
ActiveRecord
::
QueryRecorder
.
new
{
perform_request
(
user
)
}.
count
create
(
:related_epic_link
,
source:
epic
,
target:
create
(
:epic
,
group:
group
))
expect
{
perform_request
(
user
)
}.
not_to
exceed_query_limit
(
control_count
)
expect
(
response
).
to
have_gitlab_http_status
(
:ok
)
end
end
end
end
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