gitlab-org--gitlab-foss/app/controllers/projects/jobs_controller.rb

206 lines
5.8 KiB
Ruby
Raw Normal View History

# frozen_string_literal: true
class Projects::JobsController < Projects::ApplicationController
2018-03-02 16:19:17 -05:00
include SendFileUpload
include ContinueParams
2018-03-02 16:19:17 -05:00
before_action :build, except: [:index]
2018-07-05 09:55:10 -04:00
before_action :authorize_read_build!
before_action :authorize_update_build!,
except: [:index, :show, :status, :raw, :trace, :erase]
2017-11-06 08:20:44 -05:00
before_action :authorize_erase_build!, only: [:erase]
2018-12-17 13:22:03 -05:00
before_action :authorize_use_build_terminal!, only: [:terminal, :terminal_websocket_authorize]
2018-07-05 09:55:10 -04:00
before_action :verify_api_request!, only: :terminal_websocket_authorize
before_action only: [:show] do
push_frontend_feature_flag(:job_log_json, project, default_enabled: true)
end
layout 'project'
2015-10-14 06:15:03 -04:00
def index
# We need all builds for tabs counters
@all_builds = JobsFinder.new(current_user: current_user, project: @project).execute
2015-10-14 06:15:03 -04:00
@scope = params[:scope]
@builds = JobsFinder.new(current_user: current_user, project: @project, params: params).execute
@builds = @builds.eager_load_everything
@builds = @builds.page(params[:page]).per(30).without_count
2015-10-14 06:15:03 -04:00
end
# rubocop: disable CodeReuse/ActiveRecord
def show
2018-06-29 11:45:32 -04:00
@pipeline = @build.pipeline
@builds = @pipeline.builds
.order('id DESC')
.present(current_user: current_user)
respond_to do |format|
format.html
format.json do
Gitlab::PollingInterval.set_header(response, interval: 10_000)
render json: BuildSerializer
.new(project: @project, current_user: @current_user)
2017-05-31 16:10:00 -04:00
.represent(@build, {}, BuildDetailsEntity)
end
end
end
# rubocop: enable CodeReuse/ActiveRecord
def trace
build.trace.read do |stream|
respond_to do |format|
format.json do
# TODO: when the feature flag is removed we should not pass
# content_format to serialize method.
content_format = Feature.enabled?(:job_log_json, @project, default_enabled: true) ? :json : :html
build_trace = Ci::BuildTrace.new(
build: @build,
stream: stream,
state: params[:state],
content_format: content_format)
render json: BuildTraceSerializer
.new(project: @project, current_user: @current_user)
.represent(build_trace)
end
end
end
end
def retry
return respond_422 unless @build.retryable?
2016-06-10 17:36:54 -04:00
build = Ci::Build.retry(@build, current_user)
redirect_to build_path(build)
end
2016-07-16 12:39:58 -04:00
def play
return respond_422 unless @build.playable?
2016-07-16 12:39:58 -04:00
build = @build.play(current_user, play_params[:job_variables_attributes])
2016-07-16 12:39:58 -04:00
redirect_to build_path(build)
end
def cancel
return respond_422 unless @build.cancelable?
@build.cancel
if continue_params[:to]
redirect_to continue_params[:to]
else
redirect_to builds_project_pipeline_path(@project, @build.pipeline.id)
end
end
def unschedule
return respond_422 unless @build.scheduled?
2018-09-20 22:17:37 -04:00
@build.unschedule!
redirect_to build_path(@build)
end
def status
render json: BuildSerializer
2017-05-09 00:15:34 -04:00
.new(project: @project, current_user: @current_user)
2017-03-10 12:44:41 -05:00
.represent_status(@build)
end
def erase
if @build.erase(erased_by: current_user)
redirect_to project_job_path(project, @build),
notice: _("Job has been successfully erased!")
else
respond_422
end
end
def raw
2018-04-03 08:30:14 -04:00
if trace_artifact_file
workhorse_set_content_type!
2018-04-03 08:30:14 -04:00
send_upload(trace_artifact_file,
send_params: raw_send_params,
redirect_params: raw_redirect_params)
else
build.trace.read do |stream|
if stream.file?
workhorse_set_content_type!
2018-04-03 08:30:14 -04:00
send_file stream.path, type: 'text/plain; charset=utf-8', disposition: 'inline'
else
# In this case we can't use workhorse_set_content_type! and let
# Workhorse handle the response because the data is streamed directly
# to the user but, because we have the trace content, we can calculate
# the proper content type and disposition here.
raw_data = stream.raw
send_data raw_data, type: 'text/plain; charset=utf-8', disposition: raw_trace_content_disposition(raw_data), filename: 'job.log'
2018-04-03 08:30:14 -04:00
end
end
end
end
2018-07-05 09:55:10 -04:00
def terminal
end
# GET .../terminal.ws : implemented in gitlab-workhorse
def terminal_websocket_authorize
set_workhorse_internal_api_content_type
render json: Gitlab::Workhorse.channel_websocket(@build.terminal_specification)
2018-07-05 09:55:10 -04:00
end
private
def authorize_update_build!
return access_denied! unless can?(current_user, :update_build, build)
end
2017-11-06 08:20:44 -05:00
def authorize_erase_build!
return access_denied! unless can?(current_user, :erase_build, build)
end
2018-07-05 09:55:10 -04:00
def authorize_use_build_terminal!
return access_denied! unless can?(current_user, :create_build_terminal, build)
end
def verify_api_request!
Gitlab::Workhorse.verify_api_request!(request.headers)
end
2018-04-03 08:30:14 -04:00
def raw_send_params
{ type: 'text/plain; charset=utf-8', disposition: 'inline' }
end
def raw_redirect_params
{ query: { 'response-content-type' => 'text/plain; charset=utf-8', 'response-content-disposition' => 'inline' } }
end
def play_params
params.permit(job_variables_attributes: %i[key secret_value])
end
2018-03-01 16:30:31 -05:00
def trace_artifact_file
@trace_artifact_file ||= build.job_artifacts_trace&.file
end
def build
@build ||= project.builds.find(params[:id])
2018-06-25 13:33:12 -04:00
.present(current_user: current_user)
end
def build_path(build)
project_job_path(build.project, build)
end
def raw_trace_content_disposition(raw_data)
mime_type = MimeMagic.by_magic(raw_data)
# if mime_type is nil can also represent 'text/plain'
return 'inline' if mime_type.nil? || mime_type.type == 'text/plain'
'attachment'
end
end
Projects::JobsController.prepend_if_ee('EE::Projects::JobsController')