Commit 7dee6324 authored by Igor Drozdov's avatar Igor Drozdov

Merge branch 'fj-214692-avoid-commit-when-content-file-name-not-updated' into 'master'

Avoid commit when snippet file_name and content are not present

Closes #214683 and #214692

See merge request gitlab-org/gitlab!29761
parents 5af54f98 81b275e7
...@@ -4,6 +4,8 @@ module Snippets ...@@ -4,6 +4,8 @@ module Snippets
class UpdateService < Snippets::BaseService class UpdateService < Snippets::BaseService
include SpamCheckMethods include SpamCheckMethods
COMMITTABLE_ATTRIBUTES = %w(file_name content).freeze
UpdateError = Class.new(StandardError) UpdateError = Class.new(StandardError)
CreateRepositoryError = Class.new(StandardError) CreateRepositoryError = Class.new(StandardError)
...@@ -37,6 +39,10 @@ module Snippets ...@@ -37,6 +39,10 @@ module Snippets
def save_and_commit(snippet) def save_and_commit(snippet)
return false unless snippet.save return false unless snippet.save
# If the updated attributes does not need to update
# the repository we can just return
return true unless committable_attributes?
# In order to avoid non migrated snippets scenarios, # In order to avoid non migrated snippets scenarios,
# if the snippet does not have a repository we created it # if the snippet does not have a repository we created it
# We don't need to check if the repository exists # We don't need to check if the repository exists
...@@ -104,5 +110,9 @@ module Snippets ...@@ -104,5 +110,9 @@ module Snippets
def repository_empty?(snippet) def repository_empty?(snippet)
snippet.repository._uncached_exists? && !snippet.repository._uncached_has_visible_content? snippet.repository._uncached_exists? && !snippet.repository._uncached_has_visible_content?
end end
def committable_attributes?
(params.stringify_keys.keys & COMMITTABLE_ATTRIBUTES).present?
end
end end
end end
---
title: Avoid commit when snippet file_name and content are not present
merge_request: 29761
author:
type: changed
...@@ -61,6 +61,7 @@ describe 'Projects > Snippets > User updates a snippet', :js do ...@@ -61,6 +61,7 @@ describe 'Projects > Snippets > User updates a snippet', :js do
end end
fill_in('project_snippet_title', with: 'Snippet new title') fill_in('project_snippet_title', with: 'Snippet new title')
fill_in('project_snippet_file_name', with: 'new_file_name')
click_button('Save') click_button('Save')
end end
......
...@@ -92,6 +92,7 @@ describe 'User edits snippet', :js do ...@@ -92,6 +92,7 @@ describe 'User edits snippet', :js do
end end
fill_in 'personal_snippet_title', with: 'New Snippet Title' fill_in 'personal_snippet_title', with: 'New Snippet Title'
fill_in 'personal_snippet_file_name', with: 'new_file_name'
click_button('Save changes') click_button('Save changes')
end end
......
...@@ -424,6 +424,32 @@ describe API::Snippets do ...@@ -424,6 +424,32 @@ describe API::Snippets do
end end
end end
context "when admin" do
let(:admin) { create(:admin) }
let(:token) { create(:personal_access_token, user: admin, scopes: [:sudo]) }
subject do
put api("/snippets/#{snippet.id}", admin, personal_access_token: token), params: { visibility: 'private', sudo: user.id }
end
context 'when sudo is defined' do
it 'returns 200 and updates snippet visibility' do
expect(snippet.visibility).not_to eq('private')
subject
expect(response).to have_gitlab_http_status(:success)
expect(json_response["visibility"]).to eq 'private'
end
it 'does not commit data' do
expect_any_instance_of(SnippetRepository).not_to receive(:multi_files_action)
subject
end
end
end
def update_snippet(snippet_id: snippet.id, params: {}, requester: user) def update_snippet(snippet_id: snippet.id, params: {}, requester: user)
put api("/snippets/#{snippet_id}", requester), params: params put api("/snippets/#{snippet_id}", requester), params: params
end end
......
...@@ -270,6 +270,35 @@ describe Snippets::UpdateService do ...@@ -270,6 +270,35 @@ describe Snippets::UpdateService do
end end
end end
shared_examples 'committable attributes' do
context 'when file_name is updated' do
let(:options) { { file_name: 'snippet.rb' } }
it 'commits to repository' do
expect(service).to receive(:create_commit)
expect(subject).to be_success
end
end
context 'when content is updated' do
let(:options) { { content: 'puts "hello world"' } }
it 'commits to repository' do
expect(service).to receive(:create_commit)
expect(subject).to be_success
end
end
context 'when content or file_name is not updated' do
let(:options) { { title: 'Test snippet' } }
it 'does not perform any commit' do
expect(service).not_to receive(:create_commit)
expect(subject).to be_success
end
end
end
context 'when Project Snippet' do context 'when Project Snippet' do
let_it_be(:project) { create(:project) } let_it_be(:project) { create(:project) }
let!(:snippet) { create(:project_snippet, :repository, author: user, project: project) } let!(:snippet) { create(:project_snippet, :repository, author: user, project: project) }
...@@ -283,6 +312,7 @@ describe Snippets::UpdateService do ...@@ -283,6 +312,7 @@ describe Snippets::UpdateService do
it_behaves_like 'snippet update data is tracked' it_behaves_like 'snippet update data is tracked'
it_behaves_like 'updates repository content' it_behaves_like 'updates repository content'
it_behaves_like 'commit operation fails' it_behaves_like 'commit operation fails'
it_behaves_like 'committable attributes'
context 'when snippet does not have a repository' do context 'when snippet does not have a repository' do
let!(:snippet) { create(:project_snippet, author: user, project: project) } let!(:snippet) { create(:project_snippet, author: user, project: project) }
...@@ -301,6 +331,7 @@ describe Snippets::UpdateService do ...@@ -301,6 +331,7 @@ describe Snippets::UpdateService do
it_behaves_like 'snippet update data is tracked' it_behaves_like 'snippet update data is tracked'
it_behaves_like 'updates repository content' it_behaves_like 'updates repository content'
it_behaves_like 'commit operation fails' it_behaves_like 'commit operation fails'
it_behaves_like 'committable attributes'
context 'when snippet does not have a repository' do context 'when snippet does not have a repository' do
let!(:snippet) { create(:personal_snippet, author: user, project: project) } let!(:snippet) { create(:personal_snippet, author: user, project: project) }
......
...@@ -23,9 +23,18 @@ RSpec.shared_examples 'update with repository actions' do ...@@ -23,9 +23,18 @@ RSpec.shared_examples 'update with repository actions' do
context 'when the repository does not exist' do context 'when the repository does not exist' do
let(:snippet) { snippet_without_repo } let(:snippet) { snippet_without_repo }
it 'creates the repository' do context 'when update attributes does not include file_name or content' do
it 'does not create the repository' do
update_snippet(snippet_id: snippet.id, params: { title: 'foo' }) update_snippet(snippet_id: snippet.id, params: { title: 'foo' })
expect(snippet.repository).not_to exist
end
end
context 'when update attributes include file_name or content' do
it 'creates the repository' do
update_snippet(snippet_id: snippet.id, params: { title: 'foo', file_name: 'foo' })
expect(snippet.repository).to exist expect(snippet.repository).to exist
end end
...@@ -40,6 +49,7 @@ RSpec.shared_examples 'update with repository actions' do ...@@ -40,6 +49,7 @@ RSpec.shared_examples 'update with repository actions' do
expect(blob.data).to eq content expect(blob.data).to eq content
end end
end end
end
end end
RSpec.shared_examples 'snippet response without repository URLs' do RSpec.shared_examples 'snippet response without repository URLs' 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