Commit 5e470af3 authored by Douglas Barbosa Alexandre's avatar Douglas Barbosa Alexandre

Merge branch 'fj-add-snippet-input-file-action-to-create-mutation' into 'master'

Add FileInput argument to snippet create mutation

See merge request gitlab-org/gitlab!34449
parents e2801a62 50856523
...@@ -21,7 +21,7 @@ module Mutations ...@@ -21,7 +21,7 @@ module Mutations
description: 'File name of the snippet' description: 'File name of the snippet'
argument :content, GraphQL::STRING_TYPE, argument :content, GraphQL::STRING_TYPE,
required: true, required: false,
description: 'Content of the snippet' description: 'Content of the snippet'
argument :description, GraphQL::STRING_TYPE, argument :description, GraphQL::STRING_TYPE,
...@@ -40,6 +40,10 @@ module Mutations ...@@ -40,6 +40,10 @@ module Mutations
required: false, required: false,
description: 'The paths to files uploaded in the snippet description' description: 'The paths to files uploaded in the snippet description'
argument :files, [Types::Snippets::FileInputType],
description: "The snippet files to create",
required: false
def resolve(args) def resolve(args)
project_path = args.delete(:project_path) project_path = args.delete(:project_path)
...@@ -49,13 +53,9 @@ module Mutations ...@@ -49,13 +53,9 @@ module Mutations
raise_resource_not_available_error! raise_resource_not_available_error!
end end
# We need to rename `uploaded_files` into `files` because
# it's the expected key param
args[:files] = args.delete(:uploaded_files)
service_response = ::Snippets::CreateService.new(project, service_response = ::Snippets::CreateService.new(project,
context[:current_user], context[:current_user],
args).execute create_params(args)).execute
snippet = service_response.payload[:snippet] snippet = service_response.payload[:snippet]
...@@ -82,6 +82,18 @@ module Mutations ...@@ -82,6 +82,18 @@ module Mutations
def can_create_personal_snippet? def can_create_personal_snippet?
Ability.allowed?(context[:current_user], :create_snippet) Ability.allowed?(context[:current_user], :create_snippet)
end end
def create_params(args)
args.tap do |create_args|
# We need to rename `files` into `snippet_files` because
# it's the expected key param
create_args[:snippet_files] = create_args.delete(:files)&.map(&:to_h)
# We need to rename `uploaded_files` into `files` because
# it's the expected key param
create_args[:files] = create_args.delete(:uploaded_files)
end
end
end end
end end
end end
...@@ -12,7 +12,9 @@ module Snippets ...@@ -12,7 +12,9 @@ module Snippets
super super
@uploaded_assets = Array(@params.delete(:files).presence) @uploaded_assets = Array(@params.delete(:files).presence)
@snippet_files = SnippetInputActionCollection.new(Array(@params.delete(:snippet_files).presence))
input_actions = Array(@params.delete(:snippet_files).presence)
@snippet_files = SnippetInputActionCollection.new(input_actions, allowed_actions: restricted_files_actions)
filter_spam_check_params filter_spam_check_params
end end
...@@ -79,5 +81,9 @@ module Snippets ...@@ -79,5 +81,9 @@ module Snippets
def build_actions_from_params(snippet) def build_actions_from_params(snippet)
raise NotImplementedError raise NotImplementedError
end end
def restricted_files_actions
nil
end
end end
end end
...@@ -100,5 +100,9 @@ module Snippets ...@@ -100,5 +100,9 @@ module Snippets
def build_actions_from_params(_snippet) def build_actions_from_params(_snippet)
[{ file_path: params[:file_name], content: params[:content] }] [{ file_path: params[:file_name], content: params[:content] }]
end end
def restricted_files_actions
:create
end
end end
end end
---
title: Add files argument to snippet create mutation
merge_request: 34449
author:
type: changed
...@@ -1824,7 +1824,7 @@ input CreateSnippetInput { ...@@ -1824,7 +1824,7 @@ input CreateSnippetInput {
""" """
Content of the snippet Content of the snippet
""" """
content: String! content: String
""" """
Description of the snippet Description of the snippet
...@@ -1836,6 +1836,11 @@ input CreateSnippetInput { ...@@ -1836,6 +1836,11 @@ input CreateSnippetInput {
""" """
fileName: String fileName: String
"""
The snippet files to create
"""
files: [SnippetFileInputType!]
""" """
The project full path the snippet is associated with The project full path the snippet is associated with
""" """
......
...@@ -4825,13 +4825,9 @@ ...@@ -4825,13 +4825,9 @@
"name": "content", "name": "content",
"description": "Content of the snippet", "description": "Content of the snippet",
"type": { "type": {
"kind": "NON_NULL", "kind": "SCALAR",
"name": null, "name": "String",
"ofType": { "ofType": null
"kind": "SCALAR",
"name": "String",
"ofType": null
}
}, },
"defaultValue": null "defaultValue": null
}, },
...@@ -4887,6 +4883,24 @@ ...@@ -4887,6 +4883,24 @@
}, },
"defaultValue": null "defaultValue": null
}, },
{
"name": "files",
"description": "The snippet files to create",
"type": {
"kind": "LIST",
"name": null,
"ofType": {
"kind": "NON_NULL",
"name": null,
"ofType": {
"kind": "INPUT_OBJECT",
"name": "SnippetFileInputType",
"ofType": null
}
}
},
"defaultValue": null
},
{ {
"name": "clientMutationId", "name": "clientMutationId",
"description": "A unique identifier for the client performing the mutation.", "description": "A unique identifier for the client performing the mutation.",
...@@ -14,9 +14,8 @@ describe 'Creating a Snippet' do ...@@ -14,9 +14,8 @@ describe 'Creating a Snippet' do
let(:visibility_level) { 'public' } let(:visibility_level) { 'public' }
let(:project_path) { nil } let(:project_path) { nil }
let(:uploaded_files) { nil } let(:uploaded_files) { nil }
let(:mutation_vars) do
let(:mutation) do {
variables = {
content: content, content: content,
description: description, description: description,
visibility_level: visibility_level, visibility_level: visibility_level,
...@@ -25,8 +24,10 @@ describe 'Creating a Snippet' do ...@@ -25,8 +24,10 @@ describe 'Creating a Snippet' do
project_path: project_path, project_path: project_path,
uploaded_files: uploaded_files uploaded_files: uploaded_files
} }
end
graphql_mutation(:create_snippet, variables) let(:mutation) do
graphql_mutation(:create_snippet, mutation_vars)
end end
def mutation_response def mutation_response
...@@ -137,6 +138,47 @@ describe 'Creating a Snippet' do ...@@ -137,6 +138,47 @@ describe 'Creating a Snippet' do
end end
end end
context 'when snippet is created using the files param' do
let(:action) { :create }
let(:file_1) { { filePath: 'example_file1', content: 'This is the example file 1' }}
let(:file_2) { { filePath: 'example_file2', content: 'This is the example file 2' }}
let(:actions) { [{ action: action }.merge(file_1), { action: action }.merge(file_2)] }
let(:mutation_vars) do
{
description: description,
visibility_level: visibility_level,
project_path: project_path,
title: title,
files: actions
}
end
it 'creates the Snippet' do
expect do
subject
end.to change { Snippet.count }.by(1)
end
it 'returns the created Snippet' do
subject
expect(mutation_response['snippet']['title']).to eq(title)
expect(mutation_response['snippet']['description']).to eq(description)
expect(mutation_response['snippet']['visibilityLevel']).to eq(visibility_level)
expect(mutation_response['snippet']['blobs'][0]['plainData']).to match(file_1[:content])
expect(mutation_response['snippet']['blobs'][0]['fileName']).to match(file_1[:file_path])
expect(mutation_response['snippet']['blobs'][1]['plainData']).to match(file_2[:content])
expect(mutation_response['snippet']['blobs'][1]['fileName']).to match(file_2[:file_path])
end
context 'when action is invalid' do
let(:file_1) { { filePath: 'example_file1' }}
it_behaves_like 'a mutation that returns errors in the response', errors: ['Snippet files have invalid data']
it_behaves_like 'does not create snippet'
end
end
context 'when there are ActiveRecord validation errors' do context 'when there are ActiveRecord validation errors' do
let(:title) { '' } let(:title) { '' }
......
...@@ -283,6 +283,19 @@ describe Snippets::CreateService do ...@@ -283,6 +283,19 @@ describe Snippets::CreateService do
expect(snippet.repository.exists?).to be_falsey expect(snippet.repository.exists?).to be_falsey
end end
end end
context 'when snippet_files contain an action different from "create"' do
let(:snippet_files) { [{ action: 'delete', file_path: 'snippet_file_path.rb' }] }
it 'a validation error is raised' do
response = subject
snippet = response.payload[:snippet]
expect(response).to be_error
expect(snippet.errors.full_messages_for(:snippet_files)).to eq ['Snippet files have invalid data']
expect(snippet.repository.exists?).to be_falsey
end
end
end end
context 'when ProjectSnippet' do context 'when ProjectSnippet' 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