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
954f7c73
Commit
954f7c73
authored
Jan 25, 2019
by
Mark Chao
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Add API endpoints for approval rules
Accepts nested attribute for rules
parent
ccd8cb79
Changes
20
Show whitespace changes
Inline
Side-by-side
Showing
20 changed files
with
1081 additions
and
60 deletions
+1081
-60
ee/app/controllers/ee/projects/merge_requests/application_controller.rb
...lers/ee/projects/merge_requests/application_controller.rb
+1
-0
ee/app/models/ee/merge_request.rb
ee/app/models/ee/merge_request.rb
+2
-0
ee/app/services/ee/merge_requests/refresh_service.rb
ee/app/services/ee/merge_requests/refresh_service.rb
+16
-10
ee/app/services/ee/projects/update_service.rb
ee/app/services/ee/projects/update_service.rb
+1
-0
ee/lib/api/helpers/approval_helpers.rb
ee/lib/api/helpers/approval_helpers.rb
+15
-0
ee/lib/api/merge_request_approvals.rb
ee/lib/api/merge_request_approvals.rb
+22
-10
ee/lib/api/project_approval_rules.rb
ee/lib/api/project_approval_rules.rb
+91
-0
ee/lib/api/project_approvals.rb
ee/lib/api/project_approvals.rb
+6
-6
ee/lib/ee/api/endpoints.rb
ee/lib/ee/api/endpoints.rb
+1
-0
ee/lib/ee/api/entities.rb
ee/lib/ee/api/entities.rb
+108
-0
ee/lib/ee/api/merge_requests.rb
ee/lib/ee/api/merge_requests.rb
+4
-0
ee/spec/fixtures/api/schemas/public_api/v4/project_approval_rule.json
...ures/api/schemas/public_api/v4/project_approval_rule.json
+30
-0
ee/spec/fixtures/api/schemas/public_api/v4/project_approval_rules.json
...res/api/schemas/public_api/v4/project_approval_rules.json
+10
-0
ee/spec/presenters/merge_request_presenter_spec.rb
ee/spec/presenters/merge_request_presenter_spec.rb
+41
-0
ee/spec/requests/api/merge_request_approvals_spec.rb
ee/spec/requests/api/merge_request_approvals_spec.rb
+447
-0
ee/spec/requests/api/merge_requests_spec.rb
ee/spec/requests/api/merge_requests_spec.rb
+26
-0
ee/spec/requests/api/project_approval_rules_spec.rb
ee/spec/requests/api/project_approval_rules_spec.rb
+218
-0
ee/spec/services/ee/merge_requests/refresh_service_spec.rb
ee/spec/services/ee/merge_requests/refresh_service_spec.rb
+30
-0
ee/spec/services/projects/update_service_spec.rb
ee/spec/services/projects/update_service_spec.rb
+12
-0
lib/api/entities.rb
lib/api/entities.rb
+0
-34
No files found.
ee/app/controllers/ee/projects/merge_requests/application_controller.rb
View file @
954f7c73
...
@@ -14,6 +14,7 @@ module EE
...
@@ -14,6 +14,7 @@ module EE
def
merge_request_params_attributes
def
merge_request_params_attributes
attrs
=
super
.
push
(
attrs
=
super
.
push
(
{
approval_rules_attributes:
[
:id
,
:name
,
{
user_ids:
[]
},
{
group_ids:
[]
},
:approvals_required
,
:approval_project_rule_id
,
:_destroy
]
},
:approvals_before_merge
,
:approvals_before_merge
,
:approver_group_ids
,
:approver_group_ids
,
:approver_ids
:approver_ids
...
...
ee/app/models/ee/merge_request.rb
View file @
954f7c73
...
@@ -28,6 +28,8 @@ module EE
...
@@ -28,6 +28,8 @@ module EE
delegate
:merge_requests_author_approval?
,
to: :target_project
,
allow_nil:
true
delegate
:merge_requests_author_approval?
,
to: :target_project
,
allow_nil:
true
participant
:participant_approvers
participant
:participant_approvers
accepts_nested_attributes_for
:approval_rules
,
allow_destroy:
true
end
end
override
:mergeable?
override
:mergeable?
...
...
ee/app/services/ee/merge_requests/refresh_service.rb
View file @
954f7c73
...
@@ -40,6 +40,11 @@ module EE
...
@@ -40,6 +40,11 @@ module EE
def
update_approvers
def
update_approvers
return
yield
unless
project
.
feature_available?
(
:code_owners
)
return
yield
unless
project
.
feature_available?
(
:code_owners
)
if
::
Feature
.
enabled?
(
:approval_rule
)
results
=
yield
merge_requests_for_source_branch
.
each
(
&
:sync_code_owners_with_approvers
)
else
previous_diffs
=
fetch_latest_merge_request_diffs
previous_diffs
=
fetch_latest_merge_request_diffs
results
=
yield
results
=
yield
...
@@ -56,6 +61,7 @@ module EE
...
@@ -56,6 +61,7 @@ module EE
merge_request
.
sync_code_owners_with_approvers
merge_request
.
sync_code_owners_with_approvers
end
end
end
results
results
end
end
...
...
ee/app/services/ee/projects/update_service.rb
View file @
954f7c73
...
@@ -79,6 +79,7 @@ module EE
...
@@ -79,6 +79,7 @@ module EE
def
sync_approval_rules
def
sync_approval_rules
return
if
::
Feature
.
enabled?
(
:approval_rules
,
project
)
return
if
::
Feature
.
enabled?
(
:approval_rules
,
project
)
return
unless
project
.
previous_changes
.
include?
(
:approvals_before_merge
)
return
unless
project
.
previous_changes
.
include?
(
:approvals_before_merge
)
return
if
::
Feature
.
enabled?
(
:approval_rule
)
project
.
approval_rules
.
update_all
(
approvals_required:
project
.
approvals_before_merge
)
project
.
approval_rules
.
update_all
(
approvals_required:
project
.
approvals_before_merge
)
end
end
...
...
ee/lib/api/helpers/approval_helpers.rb
0 → 100644
View file @
954f7c73
# frozen_string_literal: true
module
API
module
Helpers
module
ApprovalHelpers
def
present_approval
(
merge_request
)
if
Feature
.
enabled?
(
:approval_rule
)
present
merge_request
.
approval_state
,
with:
::
EE
::
API
::
Entities
::
ApprovalState
,
current_user:
current_user
else
present
merge_request
.
present
(
current_user:
current_user
),
with:
::
EE
::
API
::
Entities
::
MergeRequestApprovals
,
current_user:
current_user
end
end
end
end
end
ee/lib/api/merge_request_approvals.rb
View file @
954f7c73
...
@@ -6,6 +6,7 @@ module API
...
@@ -6,6 +6,7 @@ module API
ARRAY_COERCION_LAMBDA
=
->
(
val
)
{
val
.
empty?
?
[]
:
Array
.
wrap
(
val
)
}
ARRAY_COERCION_LAMBDA
=
->
(
val
)
{
val
.
empty?
?
[]
:
Array
.
wrap
(
val
)
}
helpers
::
API
::
Helpers
::
ApprovalHelpers
helpers
do
helpers
do
def
handle_merge_request_errors!
(
errors
)
def
handle_merge_request_errors!
(
errors
)
if
errors
.
has_key?
:project_access
if
errors
.
has_key?
:project_access
...
@@ -38,18 +39,29 @@ module API
...
@@ -38,18 +39,29 @@ module API
# Examples:
# Examples:
# GET /projects/:id/merge_requests/:merge_request_iid/approvals
# GET /projects/:id/merge_requests/:merge_request_iid/approvals
#
#
# @deprecated
desc
'List approvals for merge request'
do
desc
'List approvals for merge request'
do
success
Entities
::
MergeRequestApprovals
success
::
EE
::
API
::
Entities
::
ApprovalState
end
end
get
'approvals'
do
get
'approvals'
do
merge_request
=
find_merge_request_with_access
(
params
[
:merge_request_iid
])
merge_request
=
find_merge_request_with_access
(
params
[
:merge_request_iid
])
present
merge_request
.
present
(
current_user:
current_user
),
with:
Entities
::
MergeRequestApprovals
,
current_user:
current_user
present_approval
(
merge_request
)
end
desc
'List approval rules for merge request'
,
{
success:
::
EE
::
API
::
Entities
::
MergeRequestApprovalRules
,
hidden:
true
}
get
'approval_rules'
do
merge_request
=
find_merge_request_with_access
(
params
[
:merge_request_iid
])
present
merge_request
.
approval_state
,
with:
::
EE
::
API
::
Entities
::
MergeRequestApprovalRules
,
current_user:
current_user
end
end
desc
'Change approval-related configuration'
do
desc
'Change approval-related configuration'
do
detail
'This feature was introduced in 10.6'
detail
'This feature was introduced in 10.6'
success
Entities
::
MergeRequestApprovals
success
::
EE
::
API
::
Entities
::
ApprovalState
end
end
params
do
params
do
requires
:approvals_required
,
type:
Integer
,
desc:
'The amount of approvals required. Must be higher than the project approvals'
requires
:approvals_required
,
type:
Integer
,
desc:
'The amount of approvals required. Must be higher than the project approvals'
...
@@ -62,7 +74,7 @@ module API
...
@@ -62,7 +74,7 @@ module API
merge_request
=
::
MergeRequests
::
UpdateService
.
new
(
user_project
,
current_user
,
approvals_before_merge:
params
[
:approvals_required
]).
execute
(
merge_request
)
merge_request
=
::
MergeRequests
::
UpdateService
.
new
(
user_project
,
current_user
,
approvals_before_merge:
params
[
:approvals_required
]).
execute
(
merge_request
)
if
merge_request
.
valid?
if
merge_request
.
valid?
present
merge_request
.
present
(
current_user:
current_user
),
with:
Entities
::
MergeRequestApprovals
,
current_user:
current_user
present
_approval
(
merge_request
)
else
else
handle_merge_request_errors!
merge_request
.
errors
handle_merge_request_errors!
merge_request
.
errors
end
end
...
@@ -70,7 +82,7 @@ module API
...
@@ -70,7 +82,7 @@ module API
desc
'Update approvers and approver groups'
do
desc
'Update approvers and approver groups'
do
detail
'This feature was introduced in 10.6'
detail
'This feature was introduced in 10.6'
success
Entities
::
MergeRequestApprovals
success
::
EE
::
API
::
Entities
::
ApprovalState
end
end
params
do
params
do
requires
:approver_ids
,
type:
Array
[
String
],
coerce_with:
ARRAY_COERCION_LAMBDA
,
desc:
'Array of User IDs to set as approvers.'
requires
:approver_ids
,
type:
Array
[
String
],
coerce_with:
ARRAY_COERCION_LAMBDA
,
desc:
'Array of User IDs to set as approvers.'
...
@@ -84,7 +96,7 @@ module API
...
@@ -84,7 +96,7 @@ module API
merge_request
=
::
MergeRequests
::
UpdateService
.
new
(
user_project
,
current_user
,
declared
(
params
,
include_parent_namespaces:
false
).
merge
(
remove_old_approvers:
true
)).
execute
(
merge_request
)
merge_request
=
::
MergeRequests
::
UpdateService
.
new
(
user_project
,
current_user
,
declared
(
params
,
include_parent_namespaces:
false
).
merge
(
remove_old_approvers:
true
)).
execute
(
merge_request
)
if
merge_request
.
valid?
if
merge_request
.
valid?
present
merge_request
.
present
(
current_user:
current_user
),
with:
Entities
::
MergeRequestApprovals
,
current_user:
current_user
present
_approval
(
merge_request
)
else
else
handle_merge_request_errors!
merge_request
.
errors
handle_merge_request_errors!
merge_request
.
errors
end
end
...
@@ -99,7 +111,7 @@ module API
...
@@ -99,7 +111,7 @@ module API
# POST /projects/:id/merge_requests/:merge_request_iid/approve
# POST /projects/:id/merge_requests/:merge_request_iid/approve
#
#
desc
'Approve a merge request'
do
desc
'Approve a merge request'
do
success
Entities
::
MergeRequestApprovals
success
::
EE
::
API
::
Entities
::
ApprovalState
end
end
params
do
params
do
optional
:sha
,
type:
String
,
desc:
'When present, must have the HEAD SHA of the source branch'
optional
:sha
,
type:
String
,
desc:
'When present, must have the HEAD SHA of the source branch'
...
@@ -115,11 +127,11 @@ module API
...
@@ -115,11 +127,11 @@ module API
.
new
(
user_project
,
current_user
)
.
new
(
user_project
,
current_user
)
.
execute
(
merge_request
)
.
execute
(
merge_request
)
present
merge_request
.
present
(
current_user:
current_user
),
with:
Entities
::
MergeRequestApprovals
,
current_user:
current_user
present
_approval
(
merge_request
)
end
end
desc
'Remove an approval from a merge request'
do
desc
'Remove an approval from a merge request'
do
success
Entities
::
MergeRequestApprovals
success
::
EE
::
API
::
Entities
::
ApprovalState
end
end
post
'unapprove'
do
post
'unapprove'
do
merge_request
=
find_project_merge_request
(
params
[
:merge_request_iid
])
merge_request
=
find_project_merge_request
(
params
[
:merge_request_iid
])
...
@@ -130,7 +142,7 @@ module API
...
@@ -130,7 +142,7 @@ module API
.
new
(
user_project
,
current_user
)
.
new
(
user_project
,
current_user
)
.
execute
(
merge_request
)
.
execute
(
merge_request
)
present
merge_request
.
present
(
current_user:
current_user
),
with:
Entities
::
MergeRequestApprovals
,
current_user:
current_user
present
_approval
(
merge_request
)
end
end
end
end
end
end
...
...
ee/lib/api/project_approval_rules.rb
0 → 100644
View file @
954f7c73
# frozen_string_literal: true
module
API
class
ProjectApprovalRules
<
::
Grape
::
API
before
{
authenticate!
}
ARRAY_COERCION_LAMBDA
=
->
(
val
)
{
val
.
empty?
?
[]
:
Array
.
wrap
(
val
)
}
params
do
requires
:id
,
type:
String
,
desc:
'The ID of a project'
end
resource
:projects
,
requirements:
::
API
::
API
::
NAMESPACE_OR_PROJECT_REQUIREMENTS
do
segment
':id/approval_rules'
do
desc
'Get all project approval rules'
do
detail
'This feature was introduced in 11.6'
success
EE
::
API
::
Entities
::
ProjectApprovalRules
end
get
do
authorize!
:create_merge_request_in
,
user_project
present
user_project
,
with:
EE
::
API
::
Entities
::
ProjectApprovalRules
,
current_user:
current_user
end
desc
'Create new approval rule'
do
detail
'This feature was introduced in 11.6'
success
EE
::
API
::
Entities
::
ApprovalRule
end
params
do
requires
:name
,
type:
String
,
desc:
'The name of the approval rule'
requires
:approvals_required
,
type:
Integer
,
desc:
'The number of required approvals for this rule'
optional
:users
,
as: :user_ids
,
type:
Array
,
coerce_with:
ARRAY_COERCION_LAMBDA
,
desc:
'The user ids for this rule'
optional
:groups
,
as: :group_ids
,
type:
Array
,
coerce_with:
ARRAY_COERCION_LAMBDA
,
desc:
'The group ids for this rule'
end
post
do
authorize!
:admin_project
,
user_project
result
=
::
ApprovalRules
::
CreateService
.
new
(
user_project
,
current_user
,
declared_params
(
include_missing:
false
)).
execute
if
result
[
:status
]
==
:success
present
result
[
:rule
],
with:
EE
::
API
::
Entities
::
ApprovalRule
,
current_user:
current_user
else
render_api_error!
(
result
[
:message
],
400
)
end
end
segment
':approval_rule_id'
do
desc
'Update approval rule'
do
detail
'This feature was introduced in 11.6'
success
EE
::
API
::
Entities
::
ApprovalRule
end
params
do
requires
:approval_rule_id
,
type:
Integer
,
desc:
'The ID of an approval_rule'
optional
:name
,
type:
String
,
desc:
'The name of the approval rule'
optional
:approvals_required
,
type:
Integer
,
desc:
'The number of required approvals for this rule'
optional
:users
,
as: :user_ids
,
type:
Array
,
coerce_with:
ARRAY_COERCION_LAMBDA
,
desc:
'The user ids for this rule'
optional
:groups
,
as: :group_ids
,
type:
Array
,
coerce_with:
ARRAY_COERCION_LAMBDA
,
desc:
'The group ids for this rule'
end
put
do
authorize!
:admin_project
,
user_project
params
=
declared_params
(
include_missing:
false
)
puts
params
.
inspect
approval_rule
=
user_project
.
approval_rules
.
find
(
params
.
delete
(
:approval_rule_id
))
result
=
::
ApprovalRules
::
UpdateService
.
new
(
approval_rule
,
current_user
,
params
).
execute
if
result
[
:status
]
==
:success
present
result
[
:rule
],
with:
EE
::
API
::
Entities
::
ApprovalRule
,
current_user:
current_user
else
render_api_error!
(
result
[
:message
],
400
)
end
end
desc
'Delete an approval rule'
do
detail
'This feature was introduced in 11.6'
end
params
do
requires
:approval_rule_id
,
type:
Integer
,
desc:
'The ID of an approval_rule'
end
delete
do
authorize!
:admin_project
,
user_project
approval_rule
=
user_project
.
approval_rules
.
find
(
params
[
:approval_rule_id
])
destroy_conditionally!
(
approval_rule
)
no_content!
end
end
end
end
end
end
ee/lib/api/project_approvals.rb
View file @
954f7c73
...
@@ -14,15 +14,15 @@ module API
...
@@ -14,15 +14,15 @@ module API
segment
':id/approvals'
do
segment
':id/approvals'
do
desc
'Get all project approvers and related configuration'
do
desc
'Get all project approvers and related configuration'
do
detail
'This feature was introduced in 10.6'
detail
'This feature was introduced in 10.6'
success
::
API
::
Entities
::
ApprovalSettings
success
EE
::
API
::
Entities
::
ApprovalSettings
end
end
get
'/'
do
get
'/'
do
present
user_project
.
present
(
current_user:
current_user
),
with:
::
API
::
Entities
::
ApprovalSettings
present
user_project
.
present
(
current_user:
current_user
),
with:
EE
::
API
::
Entities
::
ApprovalSettings
end
end
desc
'Change approval-related configuration'
do
desc
'Change approval-related configuration'
do
detail
'This feature was introduced in 10.6'
detail
'This feature was introduced in 10.6'
success
::
API
::
Entities
::
ApprovalSettings
success
EE
::
API
::
Entities
::
ApprovalSettings
end
end
params
do
params
do
optional
:approvals_before_merge
,
type:
Integer
,
desc:
'The amount of approvals required before an MR can be merged'
optional
:approvals_before_merge
,
type:
Integer
,
desc:
'The amount of approvals required before an MR can be merged'
...
@@ -36,7 +36,7 @@ module API
...
@@ -36,7 +36,7 @@ module API
result
=
::
Projects
::
UpdateService
.
new
(
user_project
,
current_user
,
project_params
).
execute
result
=
::
Projects
::
UpdateService
.
new
(
user_project
,
current_user
,
project_params
).
execute
if
result
[
:status
]
==
:success
if
result
[
:status
]
==
:success
present
user_project
.
present
(
current_user:
current_user
),
with:
::
API
::
Entities
::
ApprovalSettings
present
user_project
.
present
(
current_user:
current_user
),
with:
EE
::
API
::
Entities
::
ApprovalSettings
else
else
render_validation_error!
(
user_project
)
render_validation_error!
(
user_project
)
end
end
...
@@ -45,7 +45,7 @@ module API
...
@@ -45,7 +45,7 @@ module API
desc
'Update approvers and approver groups'
do
desc
'Update approvers and approver groups'
do
detail
'This feature was introduced in 10.6'
detail
'This feature was introduced in 10.6'
success
::
API
::
Entities
::
ApprovalSettings
success
EE
::
API
::
Entities
::
ApprovalSettings
end
end
params
do
params
do
requires
:approver_ids
,
type:
Array
[
String
],
coerce_with:
ARRAY_COERCION_LAMBDA
,
desc:
'Array of User IDs to set as approvers.'
requires
:approver_ids
,
type:
Array
[
String
],
coerce_with:
ARRAY_COERCION_LAMBDA
,
desc:
'Array of User IDs to set as approvers.'
...
@@ -55,7 +55,7 @@ module API
...
@@ -55,7 +55,7 @@ module API
result
=
::
Projects
::
UpdateService
.
new
(
user_project
,
current_user
,
declared
(
params
,
include_parent_namespaces:
false
).
merge
(
remove_old_approvers:
true
)).
execute
result
=
::
Projects
::
UpdateService
.
new
(
user_project
,
current_user
,
declared
(
params
,
include_parent_namespaces:
false
).
merge
(
remove_old_approvers:
true
)).
execute
if
result
[
:status
]
==
:success
if
result
[
:status
]
==
:success
present
user_project
.
present
(
current_user:
current_user
),
with:
::
API
::
Entities
::
ApprovalSettings
present
user_project
.
present
(
current_user:
current_user
),
with:
EE
::
API
::
Entities
::
ApprovalSettings
else
else
render_validation_error!
(
user_project
)
render_validation_error!
(
user_project
)
end
end
...
...
ee/lib/ee/api/endpoints.rb
View file @
954f7c73
...
@@ -9,6 +9,7 @@ module EE
...
@@ -9,6 +9,7 @@ module EE
mount
::
EE
::
API
::
Boards
mount
::
EE
::
API
::
Boards
mount
::
EE
::
API
::
GroupBoards
mount
::
EE
::
API
::
GroupBoards
mount
::
API
::
ProjectApprovalRules
mount
::
API
::
Unleash
mount
::
API
::
Unleash
mount
::
API
::
EpicIssues
mount
::
API
::
EpicIssues
mount
::
API
::
EpicLinks
mount
::
API
::
EpicLinks
...
...
ee/lib/ee/api/entities.rb
View file @
954f7c73
...
@@ -243,10 +243,57 @@ module EE
...
@@ -243,10 +243,57 @@ module EE
expose
:title
expose
:title
end
end
class
ApprovalRule
<
Grape
::
Entity
def
initialize
(
object
,
options
=
{})
presenter
=
::
ApprovalRulePresenter
.
new
(
object
,
current_user:
options
[
:current_user
])
super
(
presenter
,
options
)
end
expose
:id
,
:name
expose
:approvers
,
using:
::
API
::
Entities
::
UserBasic
expose
:approvals_required
expose
:users
,
using:
::
API
::
Entities
::
UserBasic
expose
:groups
,
using:
::
API
::
Entities
::
Group
end
class
MergeRequestApprovalRule
<
ApprovalRule
expose
:approved_approvers
,
as: :approved_by
,
using:
::
API
::
Entities
::
UserBasic
expose
:code_owner
end
# Decorates ApprovalState
class
MergeRequestApprovalRules
<
Grape
::
Entity
expose
:wrapped_approval_rules
,
as: :rules
,
using:
MergeRequestApprovalRule
end
# Decorates Project
class
ProjectApprovalRules
<
Grape
::
Entity
expose
:approval_rules
,
as: :rules
,
using:
ApprovalRule
end
# @deprecated
class
Approver
<
Grape
::
Entity
expose
:user
,
using:
::
API
::
Entities
::
UserBasic
end
# @deprecated
class
ApproverGroup
<
Grape
::
Entity
expose
:group
,
using:
::
API
::
Entities
::
Group
end
class
ApprovalSettings
<
Grape
::
Entity
expose
:approvers
,
using:
EE
::
API
::
Entities
::
Approver
expose
:approver_groups
,
using:
EE
::
API
::
Entities
::
ApproverGroup
expose
:approvals_before_merge
expose
:reset_approvals_on_push
expose
:disable_overriding_approvers_per_merge_request
end
class
Approvals
<
Grape
::
Entity
class
Approvals
<
Grape
::
Entity
expose
:user
,
using:
::
API
::
Entities
::
UserBasic
expose
:user
,
using:
::
API
::
Entities
::
UserBasic
end
end
# @deprecated, replaced with ApprovalState
class
MergeRequestApprovals
<
::
API
::
Entities
::
ProjectEntity
class
MergeRequestApprovals
<
::
API
::
Entities
::
ProjectEntity
def
initialize
(
merge_request
,
options
=
{})
def
initialize
(
merge_request
,
options
=
{})
presenter
=
merge_request
.
present
(
current_user:
options
[
:current_user
])
presenter
=
merge_request
.
present
(
current_user:
options
[
:current_user
])
...
@@ -259,6 +306,10 @@ module EE
...
@@ -259,6 +306,10 @@ module EE
expose
:approvals_left
expose
:approvals_left
expose
:approvals
,
as: :approved_by
,
using:
EE
::
API
::
Entities
::
Approvals
expose
:approvals
,
as: :approved_by
,
using:
EE
::
API
::
Entities
::
Approvals
expose
:approvers_left
,
as: :suggested_approvers
,
using:
::
API
::
Entities
::
UserBasic
expose
:approvers_left
,
as: :suggested_approvers
,
using:
::
API
::
Entities
::
UserBasic
# @deprecated
expose
:approvers
,
using:
EE
::
API
::
Entities
::
Approver
# @deprecated
expose
:approver_groups
,
using:
EE
::
API
::
Entities
::
ApproverGroup
expose
:user_has_approved
do
|
merge_request
,
options
|
expose
:user_has_approved
do
|
merge_request
,
options
|
merge_request
.
has_approved?
(
options
[
:current_user
])
merge_request
.
has_approved?
(
options
[
:current_user
])
...
@@ -269,6 +320,63 @@ module EE
...
@@ -269,6 +320,63 @@ module EE
end
end
end
end
class
ApprovalState
<
Grape
::
Entity
expose
:merge_request
,
merge:
true
,
using:
::
API
::
Entities
::
ProjectEntity
expose
(
:merge_status
)
{
|
approval_state
|
approval_state
.
merge_request
.
merge_status
}
expose
:approved?
,
as: :approved
# @deprecated, reads from first regular rule instead
expose
:approvals_required
do
|
approval_state
|
approval_state
.
first_regular_rule
&
.
approvals_required
end
expose
:approvals_left
expose
:approved_by
,
using:
EE
::
API
::
Entities
::
Approvals
do
|
approval_state
|
approval_state
.
merge_request
.
approvals
end
expose
:suggested_approvers
,
using:
::
API
::
Entities
::
UserBasic
do
|
approval_state
,
options
|
# TODO order by relevance
approval_state
.
unactioned_approvers
end
# @deprecated, reads from first regular rule instead
expose
:approvers
do
|
approval_state
|
if
rule
=
approval_state
.
first_regular_rule
rule
.
users
.
map
do
|
user
|
{
user:
::
API
::
Entities
::
UserBasic
.
represent
(
user
)
}
end
else
[]
end
end
# @deprecated, reads from first regular rule instead
expose
:approver_groups
do
|
approval_state
|
if
rule
=
approval_state
.
first_regular_rule
presenter
=
::
ApprovalRulePresenter
.
new
(
rule
,
current_user:
options
[
:current_user
])
presenter
.
groups
.
map
do
|
group
|
{
group:
::
API
::
Entities
::
Group
.
represent
(
group
)
}
end
else
[]
end
end
expose
:user_has_approved
do
|
approval_state
,
options
|
approval_state
.
has_approved?
(
options
[
:current_user
])
end
expose
:user_can_approve
do
|
approval_state
,
options
|
approval_state
.
can_approve?
(
options
[
:current_user
])
end
expose
:approval_rules_left
do
|
approval_state
,
options
|
approval_state
.
approval_rules_left
.
map
(
&
:name
)
end
end
class
LdapGroup
<
Grape
::
Entity
class
LdapGroup
<
Grape
::
Entity
expose
:cn
expose
:cn
end
end
...
...
ee/lib/ee/api/merge_requests.rb
View file @
954f7c73
...
@@ -9,6 +9,10 @@ module EE
...
@@ -9,6 +9,10 @@ module EE
helpers
do
helpers
do
params
:optional_params_ee
do
params
:optional_params_ee
do
optional
:approvals_before_merge
,
type:
Integer
,
desc:
'Number of approvals required before this can be merged'
optional
:approvals_before_merge
,
type:
Integer
,
desc:
'Number of approvals required before this can be merged'
optional
:approval_rules_attributes
,
type:
Array
,
documentation:
{
hidden:
true
}
do
optional
:id
,
type:
Integer
,
desc:
'The ID of a rule'
optional
:approvals_required
,
type:
Integer
,
desc:
'Total number of approvals required'
end
end
end
end
end
...
...
ee/spec/fixtures/api/schemas/public_api/v4/project_approval_rule.json
0 → 100644
View file @
954f7c73
{
"type"
:
"object"
,
"properties"
:
{
"id"
:
{
"type"
:
"integer"
},
"name"
:
{
"type"
:
"string"
},
"approvals_required"
:
{
"type"
:
"integer"
},
"approvers"
:
{
"type"
:
"array"
,
"items"
:
{
"type"
:
"object"
,
"properties"
:
{}
}
},
"groups"
:
{
"type"
:
"array"
,
"items"
:
{
"type"
:
"object"
,
"properties"
:
{}
}
},
"users"
:
{
"type"
:
"array"
,
"items"
:
{
"type"
:
"object"
,
"properties"
:
{}
}
}
},
"additionalProperties"
:
false
}
ee/spec/fixtures/api/schemas/public_api/v4/project_approval_rules.json
0 → 100644
View file @
954f7c73
{
"type"
:
"object"
,
"properties"
:
{
"rules"
:
{
"type"
:
"array"
,
"items"
:
{
"$ref"
:
"project_approval_rule.json"
}
}
},
"additionalProperties"
:
false
}
ee/spec/presenters/merge_request_presenter_spec.rb
View file @
954f7c73
...
@@ -42,6 +42,27 @@ describe MergeRequestPresenter do
...
@@ -42,6 +42,27 @@ describe MergeRequestPresenter do
end
end
end
end
describe
'#approvers_left with approval_rule enabled'
do
let!
(
:private_group
)
{
create
(
:group_with_members
,
:private
)
}
let!
(
:public_group
)
{
create
(
:group_with_members
)
}
let!
(
:public_approver_group
)
{
create
(
:approver_group
,
target:
resource
,
group:
public_group
)
}
let!
(
:private_approver_group
)
{
create
(
:approver_group
,
target:
resource
,
group:
private_group
)
}
let!
(
:approver
)
{
create
(
:approver
,
target:
resource
)
}
before
do
stub_feature_flags
approval_rule:
true
resource
.
approvals
.
create!
(
user:
approver
.
user
)
end
subject
{
described_class
.
new
(
resource
,
current_user:
user
).
approvers_left
}
it
'contains all approvers'
do
approvers
=
public_approver_group
.
users
+
private_approver_group
.
users
-
[
user
]
is_expected
.
to
match_array
(
approvers
)
end
end
describe
'#overall_approver_groups'
do
describe
'#overall_approver_groups'
do
let!
(
:private_group
)
{
create
(
:group_with_members
,
:private
)
}
let!
(
:private_group
)
{
create
(
:group_with_members
,
:private
)
}
let!
(
:public_group
)
{
create
(
:group_with_members
)
}
let!
(
:public_group
)
{
create
(
:group_with_members
)
}
...
@@ -88,4 +109,24 @@ describe MergeRequestPresenter do
...
@@ -88,4 +109,24 @@ describe MergeRequestPresenter do
end
end
end
end
end
end
describe
'#all_approvers_including_groups with approval_rule enabled'
do
let!
(
:private_group
)
{
create
(
:group_with_members
,
:private
)
}
let!
(
:public_group
)
{
create
(
:group_with_members
)
}
let!
(
:public_approver_group
)
{
create
(
:approver_group
,
target:
resource
,
group:
public_group
)
}
let!
(
:private_approver_group
)
{
create
(
:approver_group
,
target:
resource
,
group:
private_group
)
}
let!
(
:approver
)
{
create
(
:approver
,
target:
resource
)
}
before
do
stub_feature_flags
approval_rule:
true
end
subject
{
described_class
.
new
(
resource
,
current_user:
user
).
all_approvers_including_groups
}
it
do
approvers
=
[
public_approver_group
.
users
,
private_approver_group
.
users
,
approver
.
user
].
flatten
-
[
user
]
is_expected
.
to
match_array
(
approvers
)
end
end
end
end
ee/spec/requests/api/merge_request_approvals_spec.rb
View file @
954f7c73
...
@@ -388,3 +388,450 @@ describe API::MergeRequestApprovals do
...
@@ -388,3 +388,450 @@ describe API::MergeRequestApprovals do
end
end
end
end
end
end
describe
"API::MergeRequestApprovals with approval_rule enabled"
do
set
(
:user
)
{
create
(
:user
)
}
set
(
:user2
)
{
create
(
:user
)
}
set
(
:admin
)
{
create
(
:user
,
:admin
)
}
set
(
:project
)
{
create
(
:project
,
:public
,
:repository
,
creator:
user
,
namespace:
user
.
namespace
,
only_allow_merge_if_pipeline_succeeds:
false
)
}
set
(
:merge_request
)
{
create
(
:merge_request
,
:simple
,
author:
user
,
assignee:
user
,
source_project:
project
,
target_project:
project
,
title:
"Test"
,
created_at:
Time
.
now
)
}
set
(
:approver
)
{
create
:user
}
set
(
:group
)
{
create
:group
}
before
do
stub_feature_flags
(
approval_rule:
true
)
end
describe
'GET :id/merge_requests/:merge_request_iid/approvals'
do
let!
(
:rule
)
{
create
(
:approval_merge_request_rule
,
merge_request:
merge_request
,
approvals_required:
2
,
name:
'foo'
)
}
it
'retrieves the approval status'
do
project
.
add_developer
(
approver
)
project
.
add_developer
(
create
(
:user
))
merge_request
.
approvals
.
create
(
user:
approver
)
rule
.
users
<<
approver
rule
.
groups
<<
group
get
api
(
"/projects/
#{
project
.
id
}
/merge_requests/
#{
merge_request
.
iid
}
/approvals"
,
user
)
expect
(
response
).
to
have_gitlab_http_status
(
200
)
expect
(
json_response
[
'approvals_required'
]).
to
eq
2
expect
(
json_response
[
'approvals_left'
]).
to
eq
1
expect
(
json_response
[
'approval_rules_left'
]).
to
be_empty
expect
(
json_response
[
'approved_by'
][
0
][
'user'
][
'username'
]).
to
eq
(
approver
.
username
)
expect
(
json_response
[
'user_can_approve'
]).
to
be
false
expect
(
json_response
[
'user_has_approved'
]).
to
be
false
expect
(
json_response
[
'approvers'
][
0
][
'user'
][
'username'
]).
to
eq
(
approver
.
username
)
expect
(
json_response
[
'approver_groups'
][
0
][
'group'
][
'name'
]).
to
eq
(
group
.
name
)
expect
(
json_response
[
'approved'
]).
to
be
true
end
it
'lists unapproved rule names'
do
project
.
add_developer
(
approver
)
project
.
add_developer
(
create
(
:user
))
rule
.
users
<<
approver
rule
.
groups
<<
group
get
api
(
"/projects/
#{
project
.
id
}
/merge_requests/
#{
merge_request
.
iid
}
/approvals"
,
approver
)
expect
(
response
).
to
have_gitlab_http_status
(
200
)
expect
(
json_response
[
'approvals_required'
]).
to
eq
2
expect
(
json_response
[
'approvals_left'
]).
to
eq
2
expect
(
json_response
[
'approval_rules_left'
]).
to
eq
([
'foo'
])
expect
(
json_response
[
'approved_by'
]).
to
be_empty
expect
(
json_response
[
'user_can_approve'
]).
to
be
true
expect
(
json_response
[
'user_has_approved'
]).
to
be
false
expect
(
json_response
[
'approvers'
][
0
][
'user'
][
'username'
]).
to
eq
(
approver
.
username
)
expect
(
json_response
[
'approver_groups'
][
0
][
'group'
][
'name'
]).
to
eq
(
group
.
name
)
expect
(
json_response
[
'approved'
]).
to
be
false
end
context
'when private group approver'
do
before
do
private_group
=
create
(
:group
,
:private
)
merge_request
.
approver_groups
.
create
(
group:
private_group
)
end
it
'hides private group'
do
get
api
(
"/projects/
#{
project
.
id
}
/merge_requests/
#{
merge_request
.
iid
}
/approvals"
,
user
)
expect
(
response
).
to
have_gitlab_http_status
(
200
)
expect
(
json_response
[
'approver_groups'
].
size
).
to
eq
(
0
)
end
context
'when admin'
do
it
'shows all approver groups'
do
get
api
(
"/projects/
#{
project
.
id
}
/merge_requests/
#{
merge_request
.
iid
}
/approvals"
,
admin
)
expect
(
response
).
to
have_gitlab_http_status
(
200
)
expect
(
json_response
[
'approver_groups'
].
size
).
to
eq
(
1
)
end
end
end
context
'when approvers are set to zero'
do
before
do
create
(
:approval_project_rule
,
project:
project
,
approvals_required:
0
)
get
api
(
"/projects/
#{
project
.
id
}
/merge_requests/
#{
merge_request
.
iid
}
/approvals"
,
user
)
end
it
'returns a 200'
do
expect
(
response
).
to
have_gitlab_http_status
(
200
)
expect
(
json_response
[
'approved'
]).
to
be
true
expect
(
json_response
[
'message'
]).
to
eq
(
nil
)
end
end
end
describe
'GET :id/merge_requests/:merge_request_iid/approval_rules'
do
let!
(
:rule
)
{
create
(
:approval_merge_request_rule
,
merge_request:
merge_request
,
approvals_required:
2
,
name:
'foo'
)
}
it
'retrieves the approval rules details'
do
project
.
add_developer
(
approver
)
merge_request
.
approvals
.
create
(
user:
approver
)
rule
.
users
<<
approver
get
api
(
"/projects/
#{
project
.
id
}
/merge_requests/
#{
merge_request
.
iid
}
/approval_rules"
,
user
)
expect
(
response
).
to
have_gitlab_http_status
(
200
)
expect
(
json_response
[
'rules'
].
size
).
to
eq
(
1
)
rule_response
=
json_response
[
'rules'
].
first
expect
(
rule_response
[
'id'
]).
to
eq
(
rule
.
id
)
expect
(
rule_response
[
'name'
]).
to
eq
(
'foo'
)
expect
(
rule_response
[
'approvers'
][
0
][
'username'
]).
to
eq
(
approver
.
username
)
expect
(
rule_response
[
'approved_by'
][
0
][
'username'
]).
to
eq
(
approver
.
username
)
end
it
'excludes private groups'
do
private_group
=
create
:group
,
:private
rule
.
groups
<<
private_group
get
api
(
"/projects/
#{
project
.
id
}
/merge_requests/
#{
merge_request
.
iid
}
/approval_rules"
,
user
)
expect
(
response
).
to
have_gitlab_http_status
(
200
)
expect
(
json_response
[
'rules'
].
size
).
to
eq
(
1
)
rule_response
=
json_response
[
'rules'
].
first
expect
(
rule_response
[
'id'
]).
to
eq
(
rule
.
id
)
expect
(
rule_response
[
'groups'
].
size
).
to
eq
(
0
)
end
end
describe
'POST :id/merge_requests/:merge_request_iid/approvals'
do
shared_examples_for
'user allowed to override approvals required'
do
context
'when disable_overriding_approvers_per_merge_request is false on the project'
do
before
do
project
.
update
(
disable_overriding_approvers_per_merge_request:
false
)
create
(
:approval_merge_request_rule
,
merge_request:
merge_request
)
end
it
'allows you to override approvals required'
do
expect
do
post
api
(
"/projects/
#{
project
.
id
}
/merge_requests/
#{
merge_request
.
iid
}
/approvals"
,
current_user
),
approvals_required:
5
end
.
to
change
{
merge_request
.
reload
.
approvals_before_merge
}.
from
(
nil
).
to
(
5
)
expect
(
response
).
to
have_gitlab_http_status
(
201
)
expect
(
json_response
[
'approvals_required'
]).
to
eq
(
5
)
end
context
'when project approvals are zero'
do
before
do
project
.
update!
(
approvals_before_merge:
0
)
end
it
'does not include an error in the response'
do
expect
do
post
api
(
"/projects/
#{
project
.
id
}
/merge_requests/
#{
merge_request
.
iid
}
/approvals"
,
current_user
),
approvals_required:
0
end
.
to
change
{
merge_request
.
reload
.
approvals_before_merge
}.
from
(
nil
).
to
(
0
)
expect
(
json_response
[
'message'
]).
to
eq
(
nil
)
end
end
end
context
'when disable_overriding_approvers_per_merge_request is true on the project'
do
before
do
project
.
update
(
disable_overriding_approvers_per_merge_request:
true
)
end
it
'does not allow you to override approvals required'
do
expect
do
post
api
(
"/projects/
#{
project
.
id
}
/merge_requests/
#{
merge_request
.
iid
}
/approvals"
,
current_user
),
approvals_required:
5
end
.
not_to
change
{
merge_request
.
reload
.
approvals_before_merge
}
expect
(
response
).
to
have_gitlab_http_status
(
422
)
end
end
it
'only shows approver groups that are visible to current user'
do
private_group
=
create
(
:group
,
:private
)
merge_request
.
approver_groups
.
create
(
group:
private_group
)
post
api
(
"/projects/
#{
project
.
id
}
/merge_requests/
#{
merge_request
.
iid
}
/approvals"
,
current_user
),
approvals_required:
5
expect
(
response
).
to
have_gitlab_http_status
(
201
)
expect
(
json_response
[
'approver_groups'
].
size
).
to
eq
(
expected_approver_group_size
)
end
end
context
'as a project admin'
do
it_behaves_like
'user allowed to override approvals required'
do
let
(
:current_user
)
{
user
}
let
(
:expected_approver_group_size
)
{
0
}
end
end
context
'as a global admin'
do
it_behaves_like
'user allowed to override approvals required'
do
let
(
:current_user
)
{
admin
}
let
(
:expected_approver_group_size
)
{
1
}
end
end
context
'as a random user'
do
before
do
project
.
update
(
disable_overriding_approvers_per_merge_request:
false
)
end
it
'does not allow you to override approvals required'
do
expect
do
post
api
(
"/projects/
#{
project
.
id
}
/merge_requests/
#{
merge_request
.
iid
}
/approvals"
,
user2
),
approvals_required:
5
end
.
not_to
change
{
merge_request
.
reload
.
approvals_before_merge
}
expect
(
response
).
to
have_gitlab_http_status
(
403
)
end
end
end
describe
'PUT :id/merge_requests/:merge_request_iid/approvers'
do
RSpec
::
Matchers
.
define_negated_matcher
:not_change
,
:change
shared_examples_for
'user allowed to change approvers'
do
context
'when disable_overriding_approvers_per_merge_request is true on the project'
do
before
do
project
.
update
(
disable_overriding_approvers_per_merge_request:
true
)
end
it
'does not allow overriding approvers'
do
expect
do
put
api
(
"/projects/
#{
project
.
id
}
/merge_requests/
#{
merge_request
.
iid
}
/approvers"
,
current_user
),
approver_ids:
[
approver
.
id
],
approver_group_ids:
[
group
.
id
]
end
.
to
not_change
{
merge_request
.
approvers
.
count
}.
and
not_change
{
merge_request
.
approver_groups
.
count
}
end
end
context
'when disable_overriding_approvers_per_merge_request is false on the project'
do
before
do
project
.
update
(
disable_overriding_approvers_per_merge_request:
false
)
end
it
'allows overriding approvers'
do
expect
do
put
api
(
"/projects/
#{
project
.
id
}
/merge_requests/
#{
merge_request
.
iid
}
/approvers"
,
current_user
),
approver_ids:
[
approver
.
id
],
approver_group_ids:
[
group
.
id
]
end
.
to
change
{
merge_request
.
approvers
.
count
}.
from
(
0
).
to
(
1
)
.
and
change
{
merge_request
.
approver_groups
.
count
}.
from
(
0
).
to
(
1
)
expect
(
response
).
to
have_gitlab_http_status
(
200
)
expect
(
json_response
[
'approvers'
][
0
][
'user'
][
'username'
]).
to
eq
(
approver
.
username
)
expect
(
json_response
[
'approver_groups'
][
0
][
'group'
][
'name'
]).
to
eq
(
group
.
name
)
end
it
'removes approvers not in the payload'
do
merge_request
.
approvers
.
create
(
user:
approver
)
merge_request
.
approver_groups
.
create
(
group:
group
)
expect
do
put
api
(
"/projects/
#{
project
.
id
}
/merge_requests/
#{
merge_request
.
iid
}
/approvers"
,
current_user
),
{
approver_ids:
[],
approver_group_ids:
[]
}.
to_json
,
{
CONTENT_TYPE
:
'application/json'
}
end
.
to
change
{
merge_request
.
approvers
.
count
}.
from
(
1
).
to
(
0
)
.
and
change
{
merge_request
.
approver_groups
.
count
}.
from
(
1
).
to
(
0
)
expect
(
response
).
to
have_gitlab_http_status
(
200
)
expect
(
json_response
[
'approvers'
]).
to
eq
([])
end
context
'when sending form-encoded data'
do
it
'removes approvers not in the payload'
do
merge_request
.
approvers
.
create
(
user:
approver
)
merge_request
.
approver_groups
.
create
(
group:
group
)
expect
do
put
api
(
"/projects/
#{
project
.
id
}
/merge_requests/
#{
merge_request
.
iid
}
/approvers"
,
current_user
),
approver_ids:
''
,
approver_group_ids:
''
end
.
to
change
{
merge_request
.
approvers
.
count
}.
from
(
1
).
to
(
0
)
.
and
change
{
merge_request
.
approver_groups
.
count
}.
from
(
1
).
to
(
0
)
expect
(
response
).
to
have_gitlab_http_status
(
200
)
expect
(
json_response
[
'approvers'
]).
to
eq
([])
end
end
end
it
'only shows approver groups that are visible to current user'
do
private_group
=
create
(
:group
,
:private
)
merge_request
.
approver_groups
.
create
(
group:
private_group
)
put
api
(
"/projects/
#{
project
.
id
}
/merge_requests/
#{
merge_request
.
iid
}
/approvers"
,
current_user
),
approver_ids:
[
approver
.
id
],
approver_group_ids:
[
private_group
.
id
,
group
.
id
]
expect
(
response
).
to
have_gitlab_http_status
(
200
)
expect
(
json_response
[
'approver_groups'
].
size
).
to
eq
(
expected_group_size
)
end
end
context
'as a project admin'
do
it_behaves_like
'user allowed to change approvers'
do
let
(
:current_user
)
{
user
}
let
(
:expected_group_size
)
{
1
}
end
end
context
'as a global admin'
do
it_behaves_like
'user allowed to change approvers'
do
let
(
:current_user
)
{
admin
}
let
(
:expected_group_size
)
{
2
}
end
end
context
'as a random user'
do
before
do
project
.
update
(
disable_overriding_approvers_per_merge_request:
false
)
end
it
'does not allow overriding approvers'
do
expect
do
put
api
(
"/projects/
#{
project
.
id
}
/merge_requests/
#{
merge_request
.
iid
}
/approvers"
,
user2
),
approver_ids:
[
approver
.
id
],
approver_group_ids:
[
group
.
id
]
end
.
to
not_change
{
merge_request
.
approvers
.
count
}.
and
not_change
{
merge_request
.
approver_groups
.
count
}
expect
(
response
).
to
have_gitlab_http_status
(
403
)
end
end
end
describe
'POST :id/merge_requests/:merge_request_iid/approve'
do
let!
(
:rule
)
{
create
(
:approval_merge_request_rule
,
merge_request:
merge_request
,
approvals_required:
2
)
}
context
'as the author of the merge request'
do
before
do
post
api
(
"/projects/
#{
project
.
id
}
/merge_requests/
#{
merge_request
.
iid
}
/approve"
,
user
)
end
it
'returns a 401'
do
expect
(
response
).
to
have_gitlab_http_status
(
401
)
end
end
context
'as a valid approver'
do
set
(
:approver
)
{
create
(
:user
)
}
before
do
project
.
add_developer
(
approver
)
project
.
add_developer
(
create
(
:user
))
rule
.
users
<<
approver
end
def
approve
(
extra_params
=
{})
post
api
(
"/projects/
#{
project
.
id
}
/merge_requests/
#{
merge_request
.
iid
}
/approve"
,
approver
),
extra_params
end
context
'when the sha param is not set'
do
before
do
approve
end
it
'approves the merge request'
do
expect
(
response
).
to
have_gitlab_http_status
(
201
)
expect
(
json_response
[
'approvals_left'
]).
to
eq
(
1
)
expect
(
json_response
[
'approved_by'
][
0
][
'user'
][
'username'
]).
to
eq
(
approver
.
username
)
expect
(
json_response
[
'user_has_approved'
]).
to
be
true
expect
(
json_response
[
'approved'
]).
to
be
true
end
end
context
'when the sha param is correct'
do
before
do
approve
(
sha:
merge_request
.
diff_head_sha
)
end
it
'approves the merge request'
do
expect
(
response
).
to
have_gitlab_http_status
(
201
)
expect
(
json_response
[
'approvals_left'
]).
to
eq
(
1
)
expect
(
json_response
[
'approved_by'
][
0
][
'user'
][
'username'
]).
to
eq
(
approver
.
username
)
expect
(
json_response
[
'user_has_approved'
]).
to
be
true
expect
(
json_response
[
'approved'
]).
to
be
true
end
end
context
'when the sha param is incorrect'
do
before
do
approve
(
sha:
merge_request
.
diff_head_sha
.
reverse
)
end
it
'returns a 409'
do
expect
(
response
).
to
have_gitlab_http_status
(
409
)
end
it
'does not approve the merge request'
do
expect
(
merge_request
.
reload
.
approval_state
.
approvals_left
).
to
eq
(
2
)
end
end
it
'only shows group approvers visible to the user'
do
private_group
=
create
(
:group
,
:private
)
merge_request
.
approver_groups
.
create
(
group:
private_group
)
approve
expect
(
response
).
to
have_gitlab_http_status
(
201
)
expect
(
json_response
[
'approver_groups'
].
size
).
to
eq
(
0
)
end
end
end
describe
'POST :id/merge_requests/:merge_request_iid/unapprove'
do
let!
(
:rule
)
{
create
(
:approval_merge_request_rule
,
merge_request:
merge_request
,
approvals_required:
2
,
name:
'foo'
)
}
context
'as a user who has approved the merge request'
do
set
(
:approver
)
{
create
(
:user
)
}
set
(
:unapprover
)
{
create
(
:user
)
}
before
do
project
.
add_developer
(
approver
)
project
.
add_developer
(
unapprover
)
project
.
add_developer
(
create
(
:user
))
merge_request
.
approvals
.
create
(
user:
approver
)
merge_request
.
approvals
.
create
(
user:
unapprover
)
rule
.
users
=
[
approver
,
unapprover
]
end
it
'unapproves the merge request'
do
post
api
(
"/projects/
#{
project
.
id
}
/merge_requests/
#{
merge_request
.
iid
}
/unapprove"
,
unapprover
)
expect
(
response
).
to
have_gitlab_http_status
(
201
)
expect
(
json_response
[
'approvals_left'
]).
to
eq
(
1
)
usernames
=
json_response
[
'approved_by'
].
map
{
|
u
|
u
[
'user'
][
'username'
]
}
expect
(
usernames
).
not_to
include
(
unapprover
.
username
)
expect
(
usernames
.
size
).
to
be
1
expect
(
json_response
[
'user_has_approved'
]).
to
be
false
expect
(
json_response
[
'user_can_approve'
]).
to
be
true
expect
(
json_response
[
'user_can_approve'
]).
to
be
true
expect
(
json_response
[
'approved'
]).
to
be
false
end
it
'only shows group approvers visible to the user'
do
private_group
=
create
(
:group
,
:private
)
merge_request
.
approver_groups
.
create
(
group:
private_group
)
post
api
(
"/projects/
#{
project
.
id
}
/merge_requests/
#{
merge_request
.
iid
}
/unapprove"
,
unapprover
)
expect
(
response
).
to
have_gitlab_http_status
(
201
)
expect
(
json_response
[
'approver_groups'
].
size
).
to
eq
(
0
)
end
end
end
end
ee/spec/requests/api/merge_requests_spec.rb
View file @
954f7c73
...
@@ -18,6 +18,32 @@ describe API::MergeRequests do
...
@@ -18,6 +18,32 @@ describe API::MergeRequests do
project
.
add_reporter
(
user
)
project
.
add_reporter
(
user
)
end
end
describe
'PUT /projects/:id/merge_requests'
do
context
'when updating existing approval rules'
do
def
update_merge_request
(
params
)
put
api
(
"/projects/
#{
project
.
id
}
/merge_requests/
#{
merge_request
.
iid
}
"
,
user
),
params:
params
end
let!
(
:rule
)
{
create
(
:approval_merge_request_rule
,
merge_request:
merge_request
,
approvals_required:
1
)
}
it
'is successful'
do
update_merge_request
(
title:
"New title"
,
approval_rules_attributes:
[
{
id:
rule
.
id
,
approvals_required:
2
}
]
)
expect
(
response
).
to
have_gitlab_http_status
(
200
)
merge_request
.
reload
expect
(
merge_request
.
approval_rules
.
size
).
to
eq
(
1
)
expect
(
merge_request
.
approval_rules
.
first
.
approvals_required
).
to
eq
(
2
)
end
end
end
describe
"POST /projects/:id/merge_requests"
do
describe
"POST /projects/:id/merge_requests"
do
def
create_merge_request
(
args
)
def
create_merge_request
(
args
)
defaults
=
{
defaults
=
{
...
...
ee/spec/requests/api/project_approval_rules_spec.rb
0 → 100644
View file @
954f7c73
# frozen_string_literal: true
require
'spec_helper'
describe
API
::
ProjectApprovalRules
do
set
(
:group
)
{
create
(
:group_with_members
)
}
set
(
:user
)
{
create
(
:user
)
}
set
(
:user2
)
{
create
(
:user
)
}
set
(
:admin
)
{
create
(
:user
,
:admin
)
}
set
(
:project
)
{
create
(
:project
,
:public
,
:repository
,
creator:
user
,
namespace:
user
.
namespace
,
only_allow_merge_if_pipeline_succeeds:
false
)
}
set
(
:approver
)
{
create
(
:user
)
}
let
(
:url
)
{
"/projects/
#{
project
.
id
}
/approval_rules"
}
describe
'GET /projects/:id/approval_rules'
do
context
'when the request is correct'
do
let!
(
:rule
)
do
rule
=
create
(
:approval_project_rule
,
name:
'security'
,
project:
project
,
approvals_required:
7
)
rule
.
users
<<
approver
rule
end
let
(
:developer
)
do
user
=
create
(
:user
)
project
.
add_guest
(
user
)
user
end
it
'matches the response schema'
do
get
api
(
url
,
developer
)
expect
(
response
).
to
have_gitlab_http_status
(
200
)
expect
(
response
).
to
match_response_schema
(
'public_api/v4/project_approval_rules'
,
dir:
'ee'
)
json
=
json_response
expect
(
json
[
'rules'
].
size
).
to
eq
(
1
)
rule
=
json
[
'rules'
].
first
expect
(
rule
[
'approvals_required'
]).
to
eq
(
7
)
expect
(
rule
[
'name'
]).
to
eq
(
'security'
)
end
context
'private group filtering'
do
set
(
:private_group
)
{
create
:group
,
:private
}
before
do
rule
.
groups
<<
private_group
end
it
'excludes private groups if user has no access'
do
get
api
(
url
,
developer
)
json
=
json_response
rule
=
json
[
'rules'
].
first
expect
(
rule
[
'groups'
].
size
).
to
eq
(
0
)
end
it
'includes private groups if user has access'
do
private_group
.
add_owner
(
developer
)
get
api
(
url
,
developer
)
json
=
json_response
rule
=
json
[
'rules'
].
first
expect
(
rule
[
'groups'
].
size
).
to
eq
(
1
)
end
end
end
end
describe
'POST /projects/:id/approval_rules'
do
let
(
:current_user
)
{
user
}
let
(
:params
)
do
{
name:
'security'
,
approvals_required:
10
}
end
context
'when missing parameters'
do
it
'returns 400 status'
do
post
api
(
url
,
current_user
)
expect
(
response
).
to
have_gitlab_http_status
(
400
)
end
end
context
'when user is without access'
do
it
'returns 403'
do
post
api
(
url
,
user2
),
params
expect
(
response
).
to
have_gitlab_http_status
(
403
)
end
end
context
'when the request is correct'
do
it
'returns 201 status'
do
post
api
(
url
,
current_user
),
params
expect
(
response
).
to
have_gitlab_http_status
(
201
)
expect
(
response
).
to
match_response_schema
(
'public_api/v4/project_approval_rule'
,
dir:
'ee'
)
end
it
'changes settings properly'
do
create
(
:approval_project_rule
,
project:
project
,
approvals_required:
2
)
project
.
reset_approvals_on_push
=
false
project
.
disable_overriding_approvers_per_merge_request
=
true
project
.
save
post
api
(
url
,
current_user
),
params
expect
(
json_response
.
symbolize_keys
).
to
include
(
params
)
end
end
end
describe
'PUT /projects/:id/approval_rules/:approval_rule_id'
do
let!
(
:approval_rule
)
{
create
(
:approval_project_rule
,
project:
project
)
}
let
(
:url
)
{
"/projects/
#{
project
.
id
}
/approval_rules/
#{
approval_rule
.
id
}
"
}
shared_examples_for
'a user with access'
do
before
do
project
.
add_developer
(
approver
)
end
context
'when approver already exists'
do
before
do
approval_rule
.
users
<<
approver
approval_rule
.
groups
<<
group
end
context
'when sending json data'
do
it
'removes all approvers if empty params are given'
do
expect
do
put
api
(
url
,
current_user
),
params:
{
users:
[],
groups:
[]
}.
to_json
,
headers:
{
CONTENT_TYPE
:
'application/json'
}
end
.
to
change
{
approval_rule
.
users
.
count
+
approval_rule
.
groups
.
count
}.
from
(
2
).
to
(
0
)
expect
(
response
).
to
have_gitlab_http_status
(
200
)
end
end
end
it
'sets approvers'
do
expect
do
put
api
(
url
,
current_user
),
params:
{
users:
[
approver
.
id
]
}
end
.
to
change
{
approval_rule
.
users
.
count
}.
from
(
0
).
to
(
1
)
expect
(
approval_rule
.
users
).
to
contain_exactly
(
approver
)
expect
(
approval_rule
.
groups
).
to
be_empty
expect
(
response
).
to
have_gitlab_http_status
(
200
)
expect
(
json_response
[
'approvers'
][
0
][
'id'
]).
to
eq
(
approver
.
id
)
end
end
context
'as a project admin'
do
it_behaves_like
'a user with access'
do
let
(
:current_user
)
{
user
}
let
(
:visible_approver_groups_count
)
{
0
}
end
end
context
'as a global admin'
do
it_behaves_like
'a user with access'
do
let
(
:current_user
)
{
admin
}
let
(
:visible_approver_groups_count
)
{
1
}
end
end
context
'as a random user'
do
it
'returns 403'
do
project
.
approvers
.
create
(
user:
approver
)
expect
do
put
api
(
url
,
user2
),
{
users:
[],
groups:
[]
}.
to_json
,
{
CONTENT_TYPE
:
'application/json'
}
end
.
not_to
change
{
approval_rule
.
approvers
.
size
}
expect
(
response
).
to
have_gitlab_http_status
(
403
)
end
end
end
describe
'DELETE /projects/:id/approval_rules/:approval_rule_id'
do
let!
(
:approval_rule
)
{
create
(
:approval_project_rule
,
project:
project
)
}
let
(
:url
)
{
"/projects/
#{
project
.
id
}
/approval_rules/
#{
approval_rule
.
id
}
"
}
it
'destroys'
do
delete
api
(
url
,
user
)
expect
(
ApprovalProjectRule
.
exists?
(
id:
approval_rule
.
id
)).
to
eq
(
false
)
expect
(
response
).
to
have_gitlab_http_status
(
204
)
end
context
'when approval rule not found'
do
let!
(
:approval_rule_2
)
{
create
(
:approval_project_rule
)
}
let
(
:url
)
{
"/projects/
#{
project
.
id
}
/approval_rules/
#{
approval_rule_2
.
id
}
"
}
it
'returns not found'
do
delete
api
(
url
,
user
)
expect
(
response
).
to
have_gitlab_http_status
(
404
)
end
end
context
'when user is not eligible to delete'
do
it
'returns forbidden'
do
delete
api
(
url
,
user2
)
expect
(
response
).
to
have_gitlab_http_status
(
403
)
end
end
end
end
ee/spec/services/ee/merge_requests/refresh_service_spec.rb
View file @
954f7c73
...
@@ -137,6 +137,36 @@ describe MergeRequests::RefreshService do
...
@@ -137,6 +137,36 @@ describe MergeRequests::RefreshService do
end
end
end
end
end
end
context
'when code owners enabled, with approval_rule enabled'
do
let
(
:relevant_merge_requests
)
{
[
merge_request
,
another_merge_request
]
}
let
(
:new_owners
)
{
[
owner
]
}
before
do
stub_feature_flags
(
approval_rule:
true
)
relevant_merge_requests
.
each
do
|
merge_request
|
expect
(
::
Gitlab
::
CodeOwners
).
to
receive
(
:for_merge_request
).
with
(
merge_request
).
and_return
(
new_owners
)
end
[
forked_merge_request
].
each
do
|
merge_request
|
expect
(
::
Gitlab
::
CodeOwners
).
not_to
receive
(
:for_merge_request
)
end
end
it
'triggers syncing of code owners'
do
relevant_merge_requests
.
each
do
|
merge_request
|
expect
(
merge_request
.
approval_rules
.
code_owner
.
exists?
).
to
eq
(
false
)
end
subject
relevant_merge_requests
.
each
do
|
merge_request
|
code_owner_rule
=
merge_request
.
approval_rules
.
code_owner
.
first
expect
(
code_owner_rule
.
users
).
to
eq
(
new_owners
)
end
end
end
end
end
end
end
end
end
ee/spec/services/projects/update_service_spec.rb
View file @
954f7c73
...
@@ -251,6 +251,18 @@ describe Projects::UpdateService, '#execute' do
...
@@ -251,6 +251,18 @@ describe Projects::UpdateService, '#execute' do
expect
(
rule
.
reload
.
approvals_required
).
to
eq
(
0
)
expect
(
rule
.
reload
.
approvals_required
).
to
eq
(
0
)
end
end
end
end
context
'when approval_rule feature is enabled'
do
it
"does not update approval_rules' approvals_required"
do
stub_feature_flags
(
approval_rule:
true
)
rule
=
create
(
:approval_project_rule
,
project:
project
)
expect
do
update_project
(
project
,
user
,
approvals_before_merge:
42
)
end
.
not_to
change
{
rule
.
reload
.
approvals_required
}
end
end
end
end
def
update_project
(
project
,
user
,
opts
)
def
update_project
(
project
,
user
,
opts
)
...
...
lib/api/entities.rb
View file @
954f7c73
...
@@ -741,32 +741,6 @@ module API
...
@@ -741,32 +741,6 @@ module API
end
end
end
end
class
Approver
<
Grape
::
Entity
expose
:user
,
using:
Entities
::
UserBasic
end
class
ApproverGroup
<
Grape
::
Entity
expose
:group
,
using:
Entities
::
Group
end
class
MergeRequestApprovals
<
ProjectEntity
expose
:merge_status
expose
:approvals_required
expose
:approvals_left
expose
:approvals
,
as: :approved_by
,
using:
Entities
::
Approver
expose
:approvers_left
,
as: :suggested_approvers
,
using:
Entities
::
UserBasic
expose
:approvers
,
using:
Entities
::
Approver
expose
:approver_groups
,
using:
Entities
::
ApproverGroup
expose
:user_has_approved
do
|
merge_request
,
options
|
merge_request
.
has_approved?
(
options
[
:current_user
])
end
expose
:user_can_approve
do
|
merge_request
,
options
|
merge_request
.
can_approve?
(
options
[
:current_user
])
end
end
class
MergeRequestDiff
<
Grape
::
Entity
class
MergeRequestDiff
<
Grape
::
Entity
expose
:id
,
:head_commit_sha
,
:base_commit_sha
,
:start_commit_sha
,
expose
:id
,
:head_commit_sha
,
:base_commit_sha
,
:start_commit_sha
,
:created_at
,
:merge_request_id
,
:state
,
:real_size
:created_at
,
:merge_request_id
,
:state
,
:real_size
...
@@ -1560,14 +1534,6 @@ module API
...
@@ -1560,14 +1534,6 @@ module API
klass
.
prepend
(
with
)
klass
.
prepend
(
with
)
end
end
class
ApprovalSettings
<
Grape
::
Entity
expose
:approvers
,
using:
Entities
::
Approver
expose
:approver_groups
,
using:
Entities
::
ApproverGroup
expose
:approvals_before_merge
expose
:reset_approvals_on_push
expose
:disable_overriding_approvers_per_merge_request
end
class
ManagedLicense
<
Grape
::
Entity
class
ManagedLicense
<
Grape
::
Entity
expose
:id
,
:name
,
:approval_status
expose
:id
,
:name
,
:approval_status
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