Commit 4a5f4c65 authored by Rémy Coutable's avatar Rémy Coutable

Merge branch 'lock-unlock-quick-actions' into 'master'

Add /lock and /unlock quick actions

Closes #39173

See merge request gitlab-org/gitlab-ce!15197
parents d3ff9e9c e39962b0
...@@ -489,6 +489,30 @@ module QuickActions ...@@ -489,6 +489,30 @@ module QuickActions
"#{comment} #{TABLEFLIP}" "#{comment} #{TABLEFLIP}"
end end
desc "Lock the discussion"
explanation "Locks the discussion"
condition do
issuable.is_a?(Issuable) &&
issuable.persisted? &&
!issuable.discussion_locked? &&
current_user.can?(:"admin_#{issuable.to_ability_name}", issuable)
end
command :lock do
@updates[:discussion_locked] = true
end
desc "Unlock the discussion"
explanation "Unlocks the discussion"
condition do
issuable.is_a?(Issuable) &&
issuable.persisted? &&
issuable.discussion_locked? &&
current_user.can?(:"admin_#{issuable.to_ability_name}", issuable)
end
command :unlock do
@updates[:discussion_locked] = false
end
# This is a dummy command, so that it appears in the autocomplete commands # This is a dummy command, so that it appears in the autocomplete commands
desc 'CC' desc 'CC'
params '@user' params '@user'
......
---
title: Add /lock and /unlock quick actions
merge_request: 15197
author: Mehdi Lahmam (@mehlah)
type: added
...@@ -44,3 +44,5 @@ do. ...@@ -44,3 +44,5 @@ do.
| `/shrug` | Append the comment with `¯\_(ツ)_/¯` | | `/shrug` | Append the comment with `¯\_(ツ)_/¯` |
| <code>/copy_metadata #issue &#124; !merge_request</code> | Copy labels and milestone from other issue or merge request | | <code>/copy_metadata #issue &#124; !merge_request</code> | Copy labels and milestone from other issue or merge request |
| `/confidential` | Makes the issue confidential | | `/confidential` | Makes the issue confidential |
| `/lock` | Lock the discussion |
| `/unlock` | Unlock the discussion |
...@@ -272,6 +272,28 @@ describe QuickActions::InterpretService do ...@@ -272,6 +272,28 @@ describe QuickActions::InterpretService do
end end
end end
shared_examples 'lock command' do
let(:issue) { create(:issue, project: project, discussion_locked: false) }
let(:merge_request) { create(:merge_request, source_project: project, discussion_locked: false) }
it 'returns discussion_locked: true if content contains /lock' do
_, updates = service.execute(content, issuable)
expect(updates).to eq(discussion_locked: true)
end
end
shared_examples 'unlock command' do
let(:issue) { create(:issue, project: project, discussion_locked: true) }
let(:merge_request) { create(:merge_request, source_project: project, discussion_locked: true) }
it 'returns discussion_locked: true if content contains /unlock' do
_, updates = service.execute(content, issuable)
expect(updates).to eq(discussion_locked: false)
end
end
shared_examples 'empty command' do shared_examples 'empty command' do
it 'populates {} if content contains an unsupported command' do it 'populates {} if content contains an unsupported command' do
_, updates = service.execute(content, issuable) _, updates = service.execute(content, issuable)
...@@ -786,6 +808,26 @@ describe QuickActions::InterpretService do ...@@ -786,6 +808,26 @@ describe QuickActions::InterpretService do
let(:issuable) { issue } let(:issuable) { issue }
end end
it_behaves_like 'lock command' do
let(:content) { '/lock' }
let(:issuable) { issue }
end
it_behaves_like 'lock command' do
let(:content) { '/lock' }
let(:issuable) { merge_request }
end
it_behaves_like 'unlock command' do
let(:content) { '/unlock' }
let(:issuable) { issue }
end
it_behaves_like 'unlock command' do
let(:content) { '/unlock' }
let(:issuable) { merge_request }
end
context '/todo' do context '/todo' do
let(:content) { '/todo' } let(:content) { '/todo' }
...@@ -961,6 +1003,16 @@ describe QuickActions::InterpretService do ...@@ -961,6 +1003,16 @@ describe QuickActions::InterpretService do
let(:content) { '/duplicate #{issue.to_reference}' } let(:content) { '/duplicate #{issue.to_reference}' }
let(:issuable) { issue } let(:issuable) { issue }
end end
it_behaves_like 'empty command' do
let(:content) { '/lock' }
let(:issuable) { issue }
end
it_behaves_like 'empty command' do
let(:content) { '/unlock' }
let(:issuable) { issue }
end
end end
context '/award command' do context '/award command' do
......
...@@ -55,7 +55,7 @@ shared_examples 'issuable record that supports quick actions in its description ...@@ -55,7 +55,7 @@ shared_examples 'issuable record that supports quick actions in its description
describe "note on #{issuable_type}", :js do describe "note on #{issuable_type}", :js do
before do before do
visit public_send("namespace_project_#{issuable_type}_path", project.namespace, project, issuable) visit public_send("project_#{issuable_type}_path", project, issuable)
end end
context 'with a note containing commands' do context 'with a note containing commands' do
...@@ -121,7 +121,7 @@ shared_examples 'issuable record that supports quick actions in its description ...@@ -121,7 +121,7 @@ shared_examples 'issuable record that supports quick actions in its description
gitlab_sign_out gitlab_sign_out
gitlab_sign_in(guest) gitlab_sign_in(guest)
visit public_send("namespace_project_#{issuable_type}_path", project.namespace, project, issuable) visit public_send("project_#{issuable_type}_path", project, issuable)
end end
it "does not close the #{issuable_type}" do it "does not close the #{issuable_type}" do
...@@ -158,7 +158,7 @@ shared_examples 'issuable record that supports quick actions in its description ...@@ -158,7 +158,7 @@ shared_examples 'issuable record that supports quick actions in its description
gitlab_sign_out gitlab_sign_out
gitlab_sign_in(guest) gitlab_sign_in(guest)
visit public_send("namespace_project_#{issuable_type}_path", project.namespace, project, issuable) visit public_send("project_#{issuable_type}_path", project, issuable)
end end
it "does not reopen the #{issuable_type}" do it "does not reopen the #{issuable_type}" do
...@@ -190,7 +190,7 @@ shared_examples 'issuable record that supports quick actions in its description ...@@ -190,7 +190,7 @@ shared_examples 'issuable record that supports quick actions in its description
gitlab_sign_out gitlab_sign_out
gitlab_sign_in(guest) gitlab_sign_in(guest)
visit public_send("namespace_project_#{issuable_type}_path", project.namespace, project, issuable) visit public_send("project_#{issuable_type}_path", project, issuable)
end end
it "does not change the #{issuable_type} title" do it "does not change the #{issuable_type} title" do
...@@ -285,13 +285,87 @@ shared_examples 'issuable record that supports quick actions in its description ...@@ -285,13 +285,87 @@ shared_examples 'issuable record that supports quick actions in its description
expect(issuable.reload.assignees).to eq [maintainer] expect(issuable.reload.assignees).to eq [maintainer]
end end
end end
context "with a note locking the #{issuable_type} discussion" do
before do
issuable.update(discussion_locked: false)
expect(issuable).not_to be_discussion_locked
end
context "when current user can lock #{issuable_type} discussion" do
it "locks the #{issuable_type} discussion" do
add_note("/lock")
expect(page).not_to have_content '/lock'
expect(page).to have_content 'Commands applied'
expect(issuable.reload).to be_discussion_locked
end
end
context "when current user cannot lock #{issuable_type}" do
before do
guest = create(:user)
project.add_guest(guest)
gitlab_sign_out
sign_in(guest)
visit public_send("project_#{issuable_type}_path", project, issuable)
end
it "does not lock the #{issuable_type} discussion" do
add_note("/lock")
expect(page).not_to have_content 'Commands applied'
expect(issuable).not_to be_discussion_locked
end
end
end
context "with a note unlocking the #{issuable_type} discussion" do
before do
issuable.update(discussion_locked: true)
expect(issuable).to be_discussion_locked
end
context "when current user can unlock #{issuable_type} discussion" do
it "unlocks the #{issuable_type} discussion" do
add_note("/unlock")
expect(page).not_to have_content '/unlock'
expect(page).to have_content 'Commands applied'
expect(issuable.reload).not_to be_discussion_locked
end
end
context "when current user cannot unlock #{issuable_type}" do
before do
guest = create(:user)
project.add_guest(guest)
gitlab_sign_out
sign_in(guest)
visit public_send("project_#{issuable_type}_path", project, issuable)
end
it "does not unlock the #{issuable_type} discussion" do
add_note("/unlock")
expect(page).not_to have_content 'Commands applied'
expect(issuable).to be_discussion_locked
end
end
end
end end
describe "preview of note on #{issuable_type}", :js do describe "preview of note on #{issuable_type}", :js do
it 'removes quick actions from note and explains them' do it 'removes quick actions from note and explains them' do
create(:user, username: 'bob') create(:user, username: 'bob')
visit public_send("namespace_project_#{issuable_type}_path", project.namespace, project, issuable) visit public_send("project_#{issuable_type}_path", project, issuable)
page.within('.js-main-target-form') do page.within('.js-main-target-form') do
fill_in 'note[note]', with: "Awesome!\n/assign @bob " fill_in 'note[note]', with: "Awesome!\n/assign @bob "
......
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