Commit 0cc2caaf authored by Douwe Maan's avatar Douwe Maan

Merge branch '4266-board-with-config-api' into 'master'

Resolve "Board with config API"

Closes #4266

See merge request gitlab-org/gitlab-ee!5954
parents 873de89b a5521d78
...@@ -26,6 +26,8 @@ ee/db/**/* ...@@ -26,6 +26,8 @@ ee/db/**/*
ee/app/serializers/ee/merge_request_widget_entity.rb ee/app/serializers/ee/merge_request_widget_entity.rb
ee/lib/api/epics.rb ee/lib/api/epics.rb
ee/lib/api/geo_nodes.rb ee/lib/api/geo_nodes.rb
ee/lib/ee/api/group_boards.rb
ee/lib/ee/api/boards.rb
ee/lib/ee/gitlab/ldap/sync/admin_users.rb ee/lib/ee/gitlab/ldap/sync/admin_users.rb
ee/app/workers/geo/file_download_dispatch_worker/job_artifact_job_finder.rb ee/app/workers/geo/file_download_dispatch_worker/job_artifact_job_finder.rb
ee/app/workers/geo/file_download_dispatch_worker/lfs_object_job_finder.rb ee/app/workers/geo/file_download_dispatch_worker/lfs_object_job_finder.rb
......
...@@ -20,6 +20,7 @@ module Labels ...@@ -20,6 +20,7 @@ module Labels
@available_labels ||= LabelsFinder.new( @available_labels ||= LabelsFinder.new(
current_user, current_user,
"#{parent_type}_id".to_sym => parent.id, "#{parent_type}_id".to_sym => parent.id,
include_ancestor_groups: include_ancestor_groups?,
only_group_labels: parent_is_group? only_group_labels: parent_is_group?
).execute(skip_authorization: skip_authorization) ).execute(skip_authorization: skip_authorization)
end end
...@@ -30,7 +31,8 @@ module Labels ...@@ -30,7 +31,8 @@ module Labels
new_label = available_labels.find_by(title: title) new_label = available_labels.find_by(title: title)
if new_label.nil? && (skip_authorization || Ability.allowed?(current_user, :admin_label, parent)) if new_label.nil? && (skip_authorization || Ability.allowed?(current_user, :admin_label, parent))
new_label = Labels::CreateService.new(params).execute(parent_type.to_sym => parent) create_params = params.except(:include_ancestor_groups)
new_label = Labels::CreateService.new(create_params).execute(parent_type.to_sym => parent)
end end
new_label new_label
...@@ -47,5 +49,9 @@ module Labels ...@@ -47,5 +49,9 @@ module Labels
def parent_is_group? def parent_is_group?
parent_type == "group" parent_type == "group"
end end
def include_ancestor_groups?
params[:include_ancestor_groups] == true
end
end end
end end
...@@ -209,6 +209,86 @@ Example response: ...@@ -209,6 +209,86 @@ Example response:
} }
``` ```
## Update a board **[STARTER]**
Updates a board.
```
PUT /projects/:id/boards/:board_id
```
| Attribute | Type | Required | Description |
| ------------------- | -------------- | -------- | ----------- |
| `id` | integer/string | yes | The ID or [URL-encoded path of the project](README.md#namespaced-path-encoding) owned by the authenticated user |
| `board_id` | integer | yes | The ID of a board |
| `name` | string | no | The new name of the board |
| `assignee_id` | integer | no | The assignee the board should be scoped to |
| `milestone_id` | integer | no | The milestone the board should be scoped to |
| `labels` | string | no | Comma-separated list of label names which the board should be scoped to |
| `weight` | integer | no | The weight range from 0 to 9, to which the board should be scoped to |
```bash
curl --request PUT --header "PRIVATE-TOKEN: 9koXpg98eAheJpvBs5tK" https://gitlab.example.com/api/v4/projects/5/boards/1?name=new_name&milestone_id=43&assignee_id=1&labels=Doing&weight=4
```
Example response:
```json
{
"id": 1,
"project": {
"id": 5,
"name": "Diaspora Project Site",
"name_with_namespace": "Diaspora / Diaspora Project Site",
"path": "diaspora-project-site",
"path_with_namespace": "diaspora/diaspora-project-site",
"created_at": "2018-07-03T05:48:49.982Z",
"default_branch": null,
"tag_list": [],
"ssh_url_to_repo": "ssh://user@example.com/diaspora/diaspora-project-site.git",
"http_url_to_repo": "http://example.com/diaspora/diaspora-project-site.git",
"web_url": "http://example.com/diaspora/diaspora-project-site",
"readme_url": null,
"avatar_url": null,
"star_count": 0,
"forks_count": 0,
"last_activity_at": "2018-07-03T05:48:49.982Z"
},
"lists": [],
"name": "new_name",
"group": null,
"milestone": {
"id": 43,
"iid": 1,
"project_id": 15,
"title": "Milestone 1",
"description": "Milestone 1 desc",
"state": "active",
"created_at": "2018-07-03T06:36:42.618Z",
"updated_at": "2018-07-03T06:36:42.618Z",
"due_date": null,
"start_date": null,
"web_url": "http://example.com/root/board1/milestones/1"
},
"assignee": {
"id": 1,
"name": "Administrator",
"username": "root",
"state": "active",
"avatar_url": "https://www.gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon",
"web_url": "http://example.com/root"
},
"labels": [{
"id": 10,
"name": "Doing",
"color": "#5CB85C",
"description": null
}],
"weight": 4
}
```
## Delete a board **[STARTER]** ## Delete a board **[STARTER]**
Deletes a board. Deletes a board.
......
...@@ -28,7 +28,36 @@ Example response: ...@@ -28,7 +28,36 @@ Example response:
{ {
"id": 1, "id": 1,
"name:": "group issue board", "name:": "group issue board",
"group_id": 5, "group": {
"id": 5,
"name": "Documentcloud",
"path": "documentcloud",
"owner_id": null,
"created_at": "2018-05-07T06:52:45.788Z",
"updated_at": "2018-07-03T06:48:17.005Z",
"description": "Consequatur aut a aperiam ut.",
"avatar": {
"url": null
},
"membership_lock": false,
"share_with_group_lock": false,
"visibility_level": 20,
"request_access_enabled": false,
"ldap_sync_status": "ready",
"ldap_sync_error": null,
"ldap_sync_last_update_at": null,
"ldap_sync_last_successful_update_at": null,
"ldap_sync_last_sync_at": null,
"lfs_enabled": null,
"parent_id": null,
"shared_runners_minutes_limit": null,
"repository_size_limit": null,
"require_two_factor_authentication": false,
"two_factor_grace_period": 48,
"plan_id": null,
"project_creation_level": 2,
"runners_token": "rgeeL-nv4wa9YdRvuMid"
},
"milestone": { "milestone": {
"id": 12 "id": 12
"title": "10.0" "title": "10.0"
...@@ -89,7 +118,36 @@ Example response: ...@@ -89,7 +118,36 @@ Example response:
{ {
"id": 1, "id": 1,
"name:": "group issue board", "name:": "group issue board",
"group_id": 5, "group": {
"id": 5,
"name": "Documentcloud",
"path": "documentcloud",
"owner_id": null,
"created_at": "2018-05-07T06:52:45.788Z",
"updated_at": "2018-07-03T06:48:17.005Z",
"description": "Consequatur aut a aperiam ut.",
"avatar": {
"url": null
},
"membership_lock": false,
"share_with_group_lock": false,
"visibility_level": 20,
"request_access_enabled": false,
"ldap_sync_status": "ready",
"ldap_sync_error": null,
"ldap_sync_last_update_at": null,
"ldap_sync_last_successful_update_at": null,
"ldap_sync_last_sync_at": null,
"lfs_enabled": null,
"parent_id": null,
"shared_runners_minutes_limit": null,
"repository_size_limit": null,
"require_two_factor_authentication": false,
"two_factor_grace_period": 48,
"plan_id": null,
"project_creation_level": 2,
"runners_token": "rgeeL-nv4wa9YdRvuMid"
},
"milestone": { "milestone": {
"id": 12 "id": 12
"title": "10.0" "title": "10.0"
...@@ -149,7 +207,36 @@ Example response: ...@@ -149,7 +207,36 @@ Example response:
{ {
"id": 1, "id": 1,
"name": "newboard", "name": "newboard",
"group_id": 5, "group": {
"id": 5,
"name": "Documentcloud",
"path": "documentcloud",
"owner_id": null,
"created_at": "2018-05-07T06:52:45.788Z",
"updated_at": "2018-07-03T06:48:17.005Z",
"description": "Consequatur aut a aperiam ut.",
"avatar": {
"url": null
},
"membership_lock": false,
"share_with_group_lock": false,
"visibility_level": 20,
"request_access_enabled": false,
"ldap_sync_status": "ready",
"ldap_sync_error": null,
"ldap_sync_last_update_at": null,
"ldap_sync_last_successful_update_at": null,
"ldap_sync_last_sync_at": null,
"lfs_enabled": null,
"parent_id": null,
"shared_runners_minutes_limit": null,
"repository_size_limit": null,
"require_two_factor_authentication": false,
"two_factor_grace_period": 48,
"plan_id": null,
"project_creation_level": 2,
"runners_token": "rgeeL-nv4wa9YdRvuMid"
},
"milestone": { "milestone": {
"id": 12 "id": 12
"title": "10.0" "title": "10.0"
...@@ -186,6 +273,98 @@ Example response: ...@@ -186,6 +273,98 @@ Example response:
} }
``` ```
## Update a board
Updates a board.
```
PUT /groups/:id/boards/:board_id
```
| Attribute | Type | Required | Description |
| ------------------- | -------------- | -------- | ----------- |
| `id` | integer/string | yes | The ID or [URL-encoded path of the group](README.md#namespaced-path-encoding) owned by the authenticated user |
| `board_id` | integer | yes | The ID of a board |
| `name` | string | no | The new name of the board |
| `assignee_id` | integer | no | The assignee the board should be scoped to |
| `milestone_id` | integer | no | The milestone the board should be scoped to |
| `labels` | string | no | Comma-separated list of label names which the board should be scoped to |
| `weight` | integer | no | The weight range from 0 to 9, to which the board should be scoped to |
```bash
curl --request PUT --header "PRIVATE-TOKEN: 9koXpg98eAheJpvBs5tK" https://gitlab.example.com/api/v4/groups/5/boards/1?name=new_name&milestone_id=44&assignee_id=1&labels=GroupLabel&weight=4
```
Example response:
```json
{
"id": 1,
"project": null,
"lists": [],
"name": "new_name",
"group": {
"id": 5,
"name": "Documentcloud",
"path": "documentcloud",
"owner_id": null,
"created_at": "2018-05-07T06:52:45.788Z",
"updated_at": "2018-07-03T06:48:17.005Z",
"description": "Consequatur aut a aperiam ut.",
"avatar": {
"url": null
},
"membership_lock": false,
"share_with_group_lock": false,
"visibility_level": 20,
"request_access_enabled": false,
"ldap_sync_status": "ready",
"ldap_sync_error": null,
"ldap_sync_last_update_at": null,
"ldap_sync_last_successful_update_at": null,
"ldap_sync_last_sync_at": null,
"lfs_enabled": null,
"parent_id": null,
"shared_runners_minutes_limit": null,
"repository_size_limit": null,
"require_two_factor_authentication": false,
"two_factor_grace_period": 48,
"plan_id": null,
"project_creation_level": 2,
"runners_token": "rgeeL-nv3wa6YdRvuMid"
},
"milestone": {
"id": 44,
"iid": 1,
"group_id": 5,
"title": "Group Milestone",
"description": "Group Milestone Desc",
"state": "active",
"created_at": "2018-07-03T07:15:19.271Z",
"updated_at": "2018-07-03T07:15:19.271Z",
"due_date": null,
"start_date": null,
"web_url": "http://example.com/groups/documentcloud/-/milestones/1"
},
"assignee": {
"id": 1,
"name": "Administrator",
"username": "root",
"state": "active",
"avatar_url": "https://www.gravatar.com/avatar/e64c7d89f26bd1972efa854d13d7dd61?s=80&d=identicon",
"web_url": "http://example.com/root"
},
"labels": [{
"id": 11,
"name": "GroupLabel",
"color": "#428BCA",
"description": ""
}],
"weight": 4
}
```
## Delete a board ## Delete a board
Deletes a board. Deletes a board.
......
...@@ -10,6 +10,7 @@ module Boards ...@@ -10,6 +10,7 @@ module Boards
set_assignee set_assignee
set_milestone set_milestone
set_labels
board.update(params) board.update(params)
end end
......
...@@ -27,6 +27,23 @@ module EE ...@@ -27,6 +27,23 @@ module EE
params[:milestone_id] = milestone&.id params[:milestone_id] = milestone&.id
end end
def set_labels
labels = params.delete(:labels)
return unless labels
params[:label_ids] = labels.split(",").map do |label_name|
label = Labels::FindOrCreateService.new(
current_user,
parent,
title: label_name.strip,
include_ancestor_groups: true
).execute
label.try(:id)
end.compact
end
end end
end end
end end
---
title: Add API endpoint for viewing and editing board config
merge_request: 5954
author:
type: added
...@@ -18,29 +18,30 @@ module EE ...@@ -18,29 +18,30 @@ module EE
end end
resource :projects, requirements: ::API::API::PROJECT_ENDPOINT_REQUIREMENTS do resource :projects, requirements: ::API::API::PROJECT_ENDPOINT_REQUIREMENTS do
segment ':id/boards' do segment ':id/boards' do
desc 'Get all project boards' do desc 'Create a project board' do
detail 'This feature was introduced in 8.13' detail 'This feature was introduced in 10.4'
success ::API::Entities::Board success ::API::Entities::Board
end end
params do params do
use :pagination requires :name, type: String, desc: 'The board name'
end end
get '/' do post '/' do
authorize!(:read_board, user_project) authorize!(:admin_board, board_parent)
present paginate(board_parent.boards), with: ::API::Entities::Board
create_board
end end
desc 'Create a project board' do desc 'Update a project board' do
detail 'This feature was introduced in 10.4' detail 'This feature was introduced in 11.0'
success ::API::Entities::Board success ::API::Entities::Board
end end
params do params do
requires :name, type: String, desc: 'The board name' use :update_params
end end
post '/' do put '/:board_id' do
authorize!(:admin_board, board_parent) authorize!(:admin_board, board_parent)
create_board update_board
end end
desc 'Delete a project board' do desc 'Delete a project board' do
......
...@@ -14,6 +14,17 @@ module EE ...@@ -14,6 +14,17 @@ module EE
present board, with: ::API::Entities::Board present board, with: ::API::Entities::Board
end end
def update_board
service = ::Boards::UpdateService.new(board_parent, current_user, declared_params)
service.execute(board)
if board.valid?
present board, with: ::API::Entities::Board
else
bad_request!("Failed to save board #{board.errors.messages}")
end
end
def delete_board def delete_board
forbidden! unless board_parent.multiple_issue_boards_available? forbidden! unless board_parent.multiple_issue_boards_available?
...@@ -22,6 +33,14 @@ module EE ...@@ -22,6 +33,14 @@ module EE
service.execute(board) service.execute(board)
end end
end end
params :update_params do
optional :name, type: String, desc: 'The board name'
optional :assignee_id, type: Integer, desc: 'The ID of a user to associate with board'
optional :milestone_id, type: Integer, desc: 'The ID of a milestone to associate with board'
optional :labels, type: String, desc: 'Comma-separated list of label names'
optional :weight, type: Integer, desc: 'The weight of the board'
end
end end
end end
end end
......
...@@ -34,6 +34,19 @@ module EE ...@@ -34,6 +34,19 @@ module EE
create_board create_board
end end
desc 'Update a group board' do
detail 'This feature was introduced in 11.0'
success ::API::Entities::Board
end
params do
use :update_params
end
put '/:board_id' do
authorize!(:admin_board, board_parent)
update_board
end
desc 'Delete a group board' do desc 'Delete a group board' do
detail 'This feature was introduced in 10.4' detail 'This feature was introduced in 10.4'
success ::API::Entities::Board success ::API::Entities::Board
......
...@@ -14,7 +14,7 @@ ...@@ -14,7 +14,7 @@
"properties": { "properties": {
"group": { "type": ["object", null] }, "group": { "type": ["object", null] },
"name": { "type": "string" }, "name": { "type": "string" },
"weight": { "type": ["string", "null"] }, "weight": { "type": ["integer", "null"] },
"assignee": { "assignee": {
"type": ["object", "null"] "type": ["object", "null"]
}, },
......
...@@ -5,6 +5,18 @@ describe Board do ...@@ -5,6 +5,18 @@ describe Board do
it { is_expected.to include_module(EE::Board) } it { is_expected.to include_module(EE::Board) }
describe 'relationships' do
it { is_expected.to belong_to(:milestone) }
it { is_expected.to have_one(:board_assignee) }
it { is_expected.to have_one(:assignee).through(:board_assignee) }
it { is_expected.to have_many(:board_labels) }
it { is_expected.to have_many(:labels).through(:board_labels) }
end
describe 'validations' do
it { is_expected.to validate_presence_of(:name) }
end
context 'validations' do context 'validations' do
context 'when group is present' do context 'when group is present' do
subject { described_class.new(group: create(:group)) } subject { described_class.new(group: create(:group)) }
......
...@@ -79,8 +79,6 @@ describe Boards::UpdateService, services: true do ...@@ -79,8 +79,6 @@ describe Boards::UpdateService, services: true do
end end
context 'group board milestone' do context 'group board milestone' do
let(:group) { create(:group) }
let(:group_board) { create(:board, group: group, name: 'Backend Group') }
let!(:milestone) { create(:milestone) } let!(:milestone) { create(:milestone) }
before do before do
...@@ -90,18 +88,18 @@ describe Boards::UpdateService, services: true do ...@@ -90,18 +88,18 @@ describe Boards::UpdateService, services: true do
it 'is not updated if it is not within group milestones' do it 'is not updated if it is not within group milestones' do
service = described_class.new(group, double, milestone_id: milestone.id) service = described_class.new(group, double, milestone_id: milestone.id)
service.execute(group_board) service.execute(board)
expect(group_board.reload.milestone).to be_nil expect(board.reload.milestone).to be_nil
end end
it 'is updated if it is within group milestones' do it 'is updated if it is within group milestones' do
milestone.update!(project: nil, group: group) milestone.update!(project: nil, group: group)
service = described_class.new(group, double, milestone_id: milestone.id) service = described_class.new(group, double, milestone_id: milestone.id)
service.execute(group_board) service.execute(board)
expect(group_board.reload.milestone).to eq(milestone) expect(board.reload.milestone).to eq(milestone)
end end
end end
...@@ -141,5 +139,81 @@ describe Boards::UpdateService, services: true do ...@@ -141,5 +139,81 @@ describe Boards::UpdateService, services: true do
expect(board.reload.milestone).to eq(milestone) expect(board.reload.milestone).to eq(milestone)
end end
end end
context '#set_labels' do
def expect_label_assigned(user, board, input_labels, expected_labels)
service = described_class.new(board.parent, user, labels: input_labels.join(','))
service.execute(board)
expect(board.reload.labels.map(&:title)).to contain_exactly(*expected_labels)
end
let(:user) { create(:user) }
let(:role) { :guest }
context 'group board labels' do
let!(:board) { create(:board, group: group, name: 'Backend') }
let!(:group_label) { create(:group_label, title: 'group_label', group: group) }
before do
group.add_user(user, role)
stub_licensed_features(scoped_issue_board: true)
end
it 'updates using only existing label' do
expect_label_assigned(user, board, %w{group_label new_label}, %w{group_label})
end
context 'user with admin_label ability' do
let(:role) { :reporter }
it 'finds and creates labels' do
expect_label_assigned(user, board, %w{group_label new_label}, %w{group_label new_label})
end
end
context 'nested group' do
let!(:child_group) { create(:group, parent: group)}
let(:project) { create(:project, group: child_group) }
it "allows using ancestor group's label" do
expect_label_assigned(user, board, %w{group_label}, %w{group_label})
end
end
end
context 'project board labels' do
let(:project) { create(:project, group: group) }
let!(:board) { create(:board, project: project, name: 'Backend') }
let!(:group_label) { create(:group_label, title: 'group_label', group: group) }
let!(:label) { create(:label, title: 'project_label', project: project) }
before do
project.add_user(user, role)
stub_licensed_features(scoped_issue_board: true)
end
context 'user with admin_label ability' do
let(:role) { :reporter }
it 'finds and creates labels' do
expect_label_assigned(user, board, %w{group_label project_label new_label}, %w{group_label project_label new_label})
end
end
it 'updates using only existing label' do
expect_label_assigned(user, board, %w{group_label project_label new_label}, %w{group_label project_label})
end
context 'nested group' do
let!(:child_group) { create(:group, parent: group)}
let(:project) { create(:project, group: child_group) }
it "allows using ancestor group's label" do
expect_label_assigned(user, board, %w{group_label project_label new_label}, %w{group_label project_label})
end
end
end
end
end end
end end
...@@ -17,7 +17,25 @@ shared_examples_for 'multiple and scoped issue boards' do |route_definition| ...@@ -17,7 +17,25 @@ shared_examples_for 'multiple and scoped issue boards' do |route_definition|
end end
end end
describe "DELETE #{route_definition}" do describe "PUT #{route_definition}/:board_id" do
let(:url) { "#{root_url}/#{board.id}" }
it 'updates a board' do
put api(url, user), name: 'new name', weight: 4, labels: 'foo, bar'
expect(response).to have_gitlab_http_status(200)
expect(response).to match_response_schema('public_api/v4/board', dir: "ee")
board.reload
expect(board.name).to eq('new name')
expect(board.weight).to eq(4)
expect(board.labels.map(&:title)).to contain_exactly('foo', 'bar')
end
end
describe "DELETE #{route_definition}/:board_id" do
let(:url) { "#{root_url}/#{board.id}" } let(:url) { "#{root_url}/#{board.id}" }
it 'deletes a board' do it 'deletes a board' do
......
...@@ -33,6 +33,7 @@ module API ...@@ -33,6 +33,7 @@ module API
success Entities::Board success Entities::Board
end end
get '/:board_id' do get '/:board_id' do
authorize!(:read_board, user_project)
present board, with: Entities::Board present board, with: Entities::Board
end end
end end
......
...@@ -3,17 +3,10 @@ require 'rails_helper' ...@@ -3,17 +3,10 @@ require 'rails_helper'
describe Board do describe Board do
describe 'relationships' do describe 'relationships' do
it { is_expected.to belong_to(:project) } it { is_expected.to belong_to(:project) }
it { is_expected.to belong_to(:milestone) }
it { is_expected.to have_one(:board_assignee) }
it { is_expected.to have_one(:assignee).through(:board_assignee) }
it { is_expected.to have_many(:board_labels) }
it { is_expected.to have_many(:labels).through(:board_labels) }
it { is_expected.to have_many(:lists).order(list_type: :asc, position: :asc).dependent(:delete_all) } it { is_expected.to have_many(:lists).order(list_type: :asc, position: :asc).dependent(:delete_all) }
end end
describe 'validations' do describe 'validations' do
it { is_expected.to validate_presence_of(:name) }
it { is_expected.to validate_presence_of(:project) } it { is_expected.to validate_presence_of(:project) }
end end
end end
...@@ -44,6 +44,26 @@ describe Labels::FindOrCreateService do ...@@ -44,6 +44,26 @@ describe Labels::FindOrCreateService do
expect(service.execute).to eq project_label expect(service.execute).to eq project_label
end end
end end
context 'when include_ancestor_groups is true' do
let(:group) { create(:group, :nested) }
let(:params) do
{
title: 'Audit',
include_ancestor_groups: true
}
end
it 'returns the ancestor group labels' do
group_label = create(:group_label, group: group.parent, title: 'Audit')
expect(service.execute).to eq group_label
end
it 'creates new labels if labels are not found' do
expect { service.execute }.to change(project.labels, :count).by(1)
end
end
end end
context 'when finding labels on group level' do context 'when finding labels on group level' do
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment