2018-09-29 18:34:47 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2016-11-26 10:37:26 -05:00
|
|
|
module API
|
|
|
|
# Snippets API
|
|
|
|
class Snippets < Grape::API
|
|
|
|
include PaginationParams
|
|
|
|
|
|
|
|
before { authenticate! }
|
|
|
|
|
|
|
|
resource :snippets do
|
|
|
|
helpers do
|
|
|
|
def snippets_for_current_user
|
2017-04-28 18:06:27 -04:00
|
|
|
SnippetsFinder.new(current_user, author: current_user).execute
|
2016-11-26 10:37:26 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def public_snippets
|
Rewrite SnippetsFinder to improve performance
This completely rewrites the SnippetsFinder class from the ground up in
order to improve its performance. The old code was beyond salvaging. It
was complex, included various Rails 5 workarounds, comments that
shouldn't be necessary, and most important of all: it produced a really
poorly performing database query.
As a result, I opted for rewriting the finder from scratch, instead of
trying to patch the existing code. Instead of trying to reuse as many
existing methods as possible, I opted for defining new methods
specifically meant for the SnippetsFinder. This requires some extra code
here and there, but allows us to have much more control over the
resulting SQL queries. It is these changes that then allow us to produce
a _much_ more efficient query.
To illustrate how bad the old query was, we will use my own snippets as
an example. Currently I have 52 snippets, most of which are global ones.
To retrieve these, you would run the following Ruby code:
user = User.find_by(username: 'yorickpeterse')
SnippetsFinder.new(user, author: user).execute
On GitLab.com the resulting query will take between 10 and 15 seconds to
run, producing the query plan found at
https://explain.depesz.com/s/Y5IX. Apart from the long execution time,
the total number of buffers (the sum of all shared hits) is around 185
GB, though the real number is probably (hopefully) much lower as I doubt
simply summing these numbers produces the true total number of buffers
used.
The new query's plan can be found at https://explain.depesz.com/s/wHdN,
and this query takes between 10 and 100-ish milliseconds to run. The
total number of buffers used is only about 30 MB.
Fixes https://gitlab.com/gitlab-org/gitlab-ce/issues/52639
2018-10-25 11:35:31 -04:00
|
|
|
SnippetsFinder.new(current_user, scope: :are_public).execute
|
2016-11-26 10:37:26 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
desc 'Get a snippets list for authenticated user' do
|
|
|
|
detail 'This feature was introduced in GitLab 8.15.'
|
|
|
|
success Entities::PersonalSnippet
|
|
|
|
end
|
|
|
|
params do
|
|
|
|
use :pagination
|
|
|
|
end
|
|
|
|
get do
|
|
|
|
present paginate(snippets_for_current_user), with: Entities::PersonalSnippet
|
|
|
|
end
|
|
|
|
|
|
|
|
desc 'List all public snippets current_user has access to' do
|
|
|
|
detail 'This feature was introduced in GitLab 8.15.'
|
|
|
|
success Entities::PersonalSnippet
|
|
|
|
end
|
|
|
|
params do
|
|
|
|
use :pagination
|
|
|
|
end
|
|
|
|
get 'public' do
|
|
|
|
present paginate(public_snippets), with: Entities::PersonalSnippet
|
|
|
|
end
|
|
|
|
|
|
|
|
desc 'Get a single snippet' do
|
|
|
|
detail 'This feature was introduced in GitLab 8.15.'
|
|
|
|
success Entities::PersonalSnippet
|
|
|
|
end
|
|
|
|
params do
|
|
|
|
requires :id, type: Integer, desc: 'The ID of a snippet'
|
|
|
|
end
|
|
|
|
get ':id' do
|
|
|
|
snippet = snippets_for_current_user.find(params[:id])
|
|
|
|
present snippet, with: Entities::PersonalSnippet
|
|
|
|
end
|
|
|
|
|
|
|
|
desc 'Create new snippet' do
|
|
|
|
detail 'This feature was introduced in GitLab 8.15.'
|
|
|
|
success Entities::PersonalSnippet
|
|
|
|
end
|
|
|
|
params do
|
|
|
|
requires :title, type: String, desc: 'The title of a snippet'
|
|
|
|
requires :file_name, type: String, desc: 'The name of a snippet file'
|
|
|
|
requires :content, type: String, desc: 'The content of a snippet'
|
2017-05-03 11:26:49 -04:00
|
|
|
optional :description, type: String, desc: 'The description of a snippet'
|
2017-02-16 10:42:17 -05:00
|
|
|
optional :visibility, type: String,
|
|
|
|
values: Gitlab::VisibilityLevel.string_values,
|
|
|
|
default: 'internal',
|
|
|
|
desc: 'The visibility of the snippet'
|
2016-11-26 10:37:26 -05:00
|
|
|
end
|
|
|
|
post do
|
2017-03-01 15:23:00 -05:00
|
|
|
attrs = declared_params(include_missing: false).merge(request: request, api: true)
|
2016-11-26 10:37:26 -05:00
|
|
|
snippet = CreateSnippetService.new(nil, current_user, attrs).execute
|
|
|
|
|
2017-02-14 14:07:11 -05:00
|
|
|
render_spam_error! if snippet.spam?
|
|
|
|
|
2016-11-26 10:37:26 -05:00
|
|
|
if snippet.persisted?
|
|
|
|
present snippet, with: Entities::PersonalSnippet
|
|
|
|
else
|
|
|
|
render_validation_error!(snippet)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
desc 'Update an existing snippet' do
|
|
|
|
detail 'This feature was introduced in GitLab 8.15.'
|
|
|
|
success Entities::PersonalSnippet
|
|
|
|
end
|
|
|
|
params do
|
|
|
|
requires :id, type: Integer, desc: 'The ID of a snippet'
|
|
|
|
optional :title, type: String, desc: 'The title of a snippet'
|
|
|
|
optional :file_name, type: String, desc: 'The name of a snippet file'
|
|
|
|
optional :content, type: String, desc: 'The content of a snippet'
|
2017-05-03 11:26:49 -04:00
|
|
|
optional :description, type: String, desc: 'The description of a snippet'
|
2017-02-16 10:42:17 -05:00
|
|
|
optional :visibility, type: String,
|
|
|
|
values: Gitlab::VisibilityLevel.string_values,
|
|
|
|
desc: 'The visibility of the snippet'
|
|
|
|
at_least_one_of :title, :file_name, :content, :visibility
|
2016-11-26 10:37:26 -05:00
|
|
|
end
|
2018-08-27 11:31:01 -04:00
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
2016-11-26 10:37:26 -05:00
|
|
|
put ':id' do
|
|
|
|
snippet = snippets_for_current_user.find_by(id: params.delete(:id))
|
2018-04-18 05:19:40 -04:00
|
|
|
break not_found!('Snippet') unless snippet
|
2017-11-14 04:02:39 -05:00
|
|
|
|
2016-11-26 10:37:26 -05:00
|
|
|
authorize! :update_personal_snippet, snippet
|
|
|
|
|
2017-03-01 15:23:00 -05:00
|
|
|
attrs = declared_params(include_missing: false).merge(request: request, api: true)
|
2016-11-26 10:37:26 -05:00
|
|
|
|
|
|
|
UpdateSnippetService.new(nil, current_user, snippet, attrs).execute
|
2017-02-14 14:07:11 -05:00
|
|
|
|
|
|
|
render_spam_error! if snippet.spam?
|
|
|
|
|
2016-11-26 10:37:26 -05:00
|
|
|
if snippet.persisted?
|
|
|
|
present snippet, with: Entities::PersonalSnippet
|
|
|
|
else
|
|
|
|
render_validation_error!(snippet)
|
|
|
|
end
|
|
|
|
end
|
2018-08-27 11:31:01 -04:00
|
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
2016-11-26 10:37:26 -05:00
|
|
|
|
|
|
|
desc 'Remove snippet' do
|
|
|
|
detail 'This feature was introduced in GitLab 8.15.'
|
|
|
|
success Entities::PersonalSnippet
|
|
|
|
end
|
|
|
|
params do
|
|
|
|
requires :id, type: Integer, desc: 'The ID of a snippet'
|
|
|
|
end
|
2018-08-27 11:31:01 -04:00
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
2016-11-26 10:37:26 -05:00
|
|
|
delete ':id' do
|
|
|
|
snippet = snippets_for_current_user.find_by(id: params.delete(:id))
|
2018-04-18 05:19:40 -04:00
|
|
|
break not_found!('Snippet') unless snippet
|
2017-02-20 13:18:12 -05:00
|
|
|
|
2016-11-26 10:37:26 -05:00
|
|
|
authorize! :destroy_personal_snippet, snippet
|
2017-02-20 13:18:12 -05:00
|
|
|
|
2017-03-02 07:14:13 -05:00
|
|
|
destroy_conditionally!(snippet)
|
2016-11-26 10:37:26 -05:00
|
|
|
end
|
2018-08-27 11:31:01 -04:00
|
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
2016-11-26 10:37:26 -05:00
|
|
|
|
|
|
|
desc 'Get a raw snippet' do
|
|
|
|
detail 'This feature was introduced in GitLab 8.15.'
|
|
|
|
end
|
|
|
|
params do
|
|
|
|
requires :id, type: Integer, desc: 'The ID of a snippet'
|
|
|
|
end
|
2018-08-27 11:31:01 -04:00
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
2016-11-26 10:37:26 -05:00
|
|
|
get ":id/raw" do
|
|
|
|
snippet = snippets_for_current_user.find_by(id: params.delete(:id))
|
2018-04-18 05:19:40 -04:00
|
|
|
break not_found!('Snippet') unless snippet
|
2016-11-26 10:37:26 -05:00
|
|
|
|
|
|
|
env['api.format'] = :txt
|
|
|
|
content_type 'text/plain'
|
2018-11-23 11:44:09 -05:00
|
|
|
header['Content-Disposition'] = 'attachment'
|
2016-11-26 10:37:26 -05:00
|
|
|
present snippet.content
|
|
|
|
end
|
2018-08-27 11:31:01 -04:00
|
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
2017-07-05 10:27:32 -04:00
|
|
|
|
|
|
|
desc 'Get the user agent details for a snippet' do
|
|
|
|
success Entities::UserAgentDetail
|
|
|
|
end
|
|
|
|
params do
|
|
|
|
requires :id, type: Integer, desc: 'The ID of a snippet'
|
|
|
|
end
|
2018-08-27 11:31:01 -04:00
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
2017-07-05 10:27:32 -04:00
|
|
|
get ":id/user_agent_detail" do
|
|
|
|
authenticated_as_admin!
|
|
|
|
|
2017-07-06 09:19:14 -04:00
|
|
|
snippet = Snippet.find_by!(id: params[:id])
|
2017-07-05 10:27:32 -04:00
|
|
|
|
2018-04-18 05:19:40 -04:00
|
|
|
break not_found!('UserAgentDetail') unless snippet.user_agent_detail
|
2017-07-05 10:27:32 -04:00
|
|
|
|
|
|
|
present snippet.user_agent_detail, with: Entities::UserAgentDetail
|
|
|
|
end
|
2018-08-27 11:31:01 -04:00
|
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
2016-11-26 10:37:26 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|