gitlab-org--gitlab-foss/lib/api/jobs.rb

223 lines
6.8 KiB
Ruby
Raw Normal View History

# frozen_string_literal: true
module API
class Jobs < Grape::API
2016-12-04 17:11:19 +00:00
include PaginationParams
before { authenticate! }
2016-10-24 11:06:17 +00:00
params do
requires :id, type: String, desc: 'The ID of a project'
end
resource :projects, requirements: API::NAMESPACE_OR_PROJECT_REQUIREMENTS do
2016-10-24 11:06:17 +00:00
helpers do
params :optional_scope do
optional :scope, types: [String, Array[String]], desc: 'The scope of builds to show',
values: ::CommitStatus::AVAILABLE_STATUSES,
2016-10-24 11:06:17 +00:00
coerce_with: ->(scope) {
case scope
when String
2016-10-24 11:06:17 +00:00
[scope]
when ::Hash
2016-10-24 11:06:17 +00:00
scope.values
when ::Array
scope
2016-10-24 11:06:17 +00:00
else
['unknown']
end
}
end
end
desc 'Get a projects jobs' do
success Entities::Job
2016-10-24 11:06:17 +00:00
end
params do
use :optional_scope
2016-12-04 17:11:19 +00:00
use :pagination
2016-10-24 11:06:17 +00:00
end
# rubocop: disable CodeReuse/ActiveRecord
get ':id/jobs' do
authorize_read_builds!
builds = user_project.builds.order('id DESC')
builds = filter_builds(builds, params[:scope])
2016-01-08 21:57:42 +00:00
2018-07-18 21:46:56 +00:00
builds = builds.preload(:user, :job_artifacts_archive, :job_artifacts, :runner, pipeline: :project)
present paginate(builds), with: Entities::Job
end
# rubocop: enable CodeReuse/ActiveRecord
desc 'Get pipeline jobs' do
success Entities::Job
end
params do
requires :pipeline_id, type: Integer, desc: 'The pipeline ID'
use :optional_scope
use :pagination
end
# rubocop: disable CodeReuse/ActiveRecord
get ':id/pipelines/:pipeline_id/jobs' do
authorize!(:read_pipeline, user_project)
pipeline = user_project.ci_pipelines.find(params[:pipeline_id])
authorize!(:read_build, pipeline)
builds = pipeline.builds
builds = filter_builds(builds, params[:scope])
2018-07-18 21:46:56 +00:00
builds = builds.preload(:job_artifacts_archive, :job_artifacts, project: [:namespace])
present paginate(builds), with: Entities::Job
end
# rubocop: enable CodeReuse/ActiveRecord
desc 'Get pipeline bridge jobs' do
success Entities::Bridge
end
params do
requires :pipeline_id, type: Integer, desc: 'The pipeline ID'
use :optional_scope
use :pagination
end
# rubocop: disable CodeReuse/ActiveRecord
get ':id/pipelines/:pipeline_id/bridges' do
authorize!(:read_build, user_project)
pipeline = user_project.ci_pipelines.find(params[:pipeline_id])
authorize!(:read_pipeline, pipeline)
bridges = pipeline.bridges
bridges = filter_builds(bridges, params[:scope])
bridges = bridges.preload(
:metadata,
downstream_pipeline: [project: [:route, { namespace: :route }]],
project: [:namespace]
)
present paginate(bridges), with: Entities::Bridge
end
# rubocop: enable CodeReuse/ActiveRecord
desc 'Get a specific job of a project' do
success Entities::Job
2016-10-24 11:06:17 +00:00
end
params do
requires :job_id, type: Integer, desc: 'The ID of a job'
2016-10-24 11:06:17 +00:00
end
get ':id/jobs/:job_id' do
authorize_read_builds!
build = find_build!(params[:job_id])
present build, with: Entities::Job
end
# TODO: We should use `present_disk_file!` and leave this implementation for backward compatibility (when build trace
# is saved in the DB instead of file). But before that, we need to consider how to replace the value of
# `runners_token` with some mask (like `xxxxxx`) when sending trace file directly by workhorse.
desc 'Get a trace of a specific job of a project'
2016-10-24 11:06:17 +00:00
params do
requires :job_id, type: Integer, desc: 'The ID of a job'
2016-10-24 11:06:17 +00:00
end
get ':id/jobs/:job_id/trace' do
authorize_read_builds!
build = find_build!(params[:job_id])
2015-12-28 12:09:51 +00:00
header 'Content-Disposition', "infile; filename=\"#{build.id}.log\""
content_type 'text/plain'
env['api.format'] = :binary
trace = build.trace.raw
2015-12-28 12:09:51 +00:00
body trace
end
desc 'Cancel a specific job of a project' do
success Entities::Job
2016-10-24 11:06:17 +00:00
end
params do
requires :job_id, type: Integer, desc: 'The ID of a job'
2016-10-24 11:06:17 +00:00
end
post ':id/jobs/:job_id/cancel' do
authorize_update_builds!
build = find_build!(params[:job_id])
authorize!(:update_build, build)
build.cancel
present build, with: Entities::Job
end
2016-10-24 11:06:17 +00:00
desc 'Retry a specific build of a project' do
success Entities::Job
2016-10-24 11:06:17 +00:00
end
params do
requires :job_id, type: Integer, desc: 'The ID of a build'
2016-10-24 11:06:17 +00:00
end
post ':id/jobs/:job_id/retry' do
authorize_update_builds!
build = find_build!(params[:job_id])
authorize!(:update_build, build)
break forbidden!('Job is not retryable') unless build.retryable?
build = ::Ci::Build.retry(build, current_user)
present build, with: Entities::Job
end
desc 'Erase job (remove artifacts and the trace)' do
success Entities::Job
2016-10-24 11:06:17 +00:00
end
params do
requires :job_id, type: Integer, desc: 'The ID of a build'
2016-10-24 11:06:17 +00:00
end
post ':id/jobs/:job_id/erase' do
2016-02-10 13:09:11 +00:00
authorize_update_builds!
build = find_build!(params[:job_id])
2017-11-06 13:20:44 +00:00
authorize!(:erase_build, build)
break forbidden!('Job is not erasable!') unless build.erasable?
build.erase(erased_by: current_user)
present build, with: Entities::Job
end
2016-06-10 15:11:27 +00:00
desc 'Trigger a actionable job (manual, delayed, etc)' do
success Entities::Job
2016-08-15 13:58:22 +00:00
detail 'This feature was added in GitLab 8.11'
end
params do
requires :job_id, type: Integer, desc: 'The ID of a Job'
2016-08-15 13:58:22 +00:00
end
post ":id/jobs/:job_id/play" do
2016-08-15 13:58:22 +00:00
authorize_read_builds!
build = find_build!(params[:job_id])
2016-08-15 13:58:22 +00:00
authorize!(:update_build, build)
bad_request!("Unplayable Job") unless build.playable?
2016-08-18 09:42:37 +00:00
build.play(current_user)
status 200
present build, with: Entities::Job
2016-08-15 13:58:22 +00:00
end
end
helpers do
# rubocop: disable CodeReuse/ActiveRecord
def filter_builds(builds, scope)
2016-01-13 14:17:59 +00:00
return builds if scope.nil? || scope.empty?
available_statuses = ::CommitStatus::AVAILABLE_STATUSES
2016-01-13 14:17:59 +00:00
unknown = scope - available_statuses
render_api_error!('Scope contains invalid value(s)', 400) unless unknown.empty?
2016-01-13 14:17:59 +00:00
builds.where(status: available_statuses && scope)
end
# rubocop: enable CodeReuse/ActiveRecord
end
end
end