debian-mirror-gitlab/app/controllers/projects/jobs_controller.rb

271 lines
8.4 KiB
Ruby
Raw Normal View History

2018-12-05 23:21:45 +05:30
# frozen_string_literal: true
2017-09-10 17:25:29 +05:30
class Projects::JobsController < Projects::ApplicationController
2018-05-09 12:01:36 +05:30
include SendFileUpload
2018-12-13 13:39:08 +05:30
include ContinueParams
2017-09-10 17:25:29 +05:30
2022-01-26 12:08:38 +05:30
before_action :find_job_as_build, except: [:index, :play, :show]
before_action :find_job_as_processable, only: [:play, :show]
2021-02-22 17:27:13 +05:30
before_action :authorize_read_build_trace!, only: [:trace, :raw]
2018-11-08 19:23:39 +05:30
before_action :authorize_read_build!
2017-09-10 17:25:29 +05:30
before_action :authorize_update_build!,
2021-10-27 15:23:28 +05:30
except: [:index, :show, :status, :raw, :trace, :erase, :cancel, :unschedule]
2018-03-17 18:26:18 +05:30
before_action :authorize_erase_build!, only: [:erase]
2019-02-15 15:39:39 +05:30
before_action :authorize_use_build_terminal!, only: [:terminal, :terminal_websocket_authorize]
2018-11-08 19:23:39 +05:30
before_action :verify_api_request!, only: :terminal_websocket_authorize
2020-06-23 00:09:42 +05:30
before_action :authorize_create_proxy_build!, only: :proxy_websocket_authorize
before_action :verify_proxy_request!, only: :proxy_websocket_authorize
2021-04-29 21:17:54 +05:30
before_action :push_jobs_table_vue, only: [:index]
2017-09-10 17:25:29 +05:30
2021-09-30 23:02:18 +05:30
before_action do
push_frontend_feature_flag(:infinitely_collapsible_sections, @project, default_enabled: :yaml)
2022-03-02 08:16:31 +05:30
push_frontend_feature_flag(:trigger_job_retry_action, @project, default_enabled: :yaml)
2021-09-30 23:02:18 +05:30
end
2017-09-10 17:25:29 +05:30
layout 'project'
2021-01-03 14:25:43 +05:30
feature_category :continuous_integration
2017-09-10 17:25:29 +05:30
def index
2020-01-01 13:55:28 +05:30
# We need all builds for tabs counters
2020-04-08 14:13:33 +05:30
@all_builds = Ci::JobsFinder.new(current_user: current_user, project: @project).execute
2020-01-01 13:55:28 +05:30
2017-09-10 17:25:29 +05:30
@scope = params[:scope]
2020-04-08 14:13:33 +05:30
@builds = Ci::JobsFinder.new(current_user: current_user, project: @project, params: params).execute
2020-01-01 13:55:28 +05:30
@builds = @builds.eager_load_everything
2018-03-17 18:26:18 +05:30
@builds = @builds.page(params[:page]).per(30).without_count
2017-09-10 17:25:29 +05:30
end
2018-12-05 23:21:45 +05:30
# rubocop: disable CodeReuse/ActiveRecord
2017-09-10 17:25:29 +05:30
def show
respond_to do |format|
format.html
format.json do
Gitlab::PollingInterval.set_header(response, interval: 10_000)
2022-01-26 12:08:38 +05:30
render json: Ci::JobSerializer
2017-09-10 17:25:29 +05:30
.new(project: @project, current_user: @current_user)
2021-11-18 22:05:49 +05:30
.represent(@build.present(current_user: current_user), {}, BuildDetailsEntity)
2017-09-10 17:25:29 +05:30
end
end
end
2018-12-05 23:21:45 +05:30
# rubocop: enable CodeReuse/ActiveRecord
2017-09-10 17:25:29 +05:30
def trace
2021-02-22 17:27:13 +05:30
@build.trace.being_watched! if @build.running?
if @build.has_trace?
@build.trace.read do |stream|
respond_to do |format|
format.json do
build_trace = Ci::BuildTrace.new(
build: @build,
stream: stream,
state: params[:state])
render json: BuildTraceSerializer
.new(project: @project, current_user: @current_user)
.represent(build_trace)
end
2017-09-10 17:25:29 +05:30
end
end
2021-02-22 17:27:13 +05:30
else
head :no_content
2017-09-10 17:25:29 +05:30
end
end
def retry
return respond_422 unless @build.retryable?
build = Ci::Build.retry(@build, current_user)
redirect_to build_path(build)
end
def play
return respond_422 unless @build.playable?
2021-01-03 14:25:43 +05:30
job = @build.play(current_user, play_params[:job_variables_attributes])
if job.is_a?(Ci::Bridge)
redirect_to pipeline_path(job.pipeline)
else
redirect_to build_path(job)
end
2017-09-10 17:25:29 +05:30
end
def cancel
2021-10-27 15:23:28 +05:30
service_response = Ci::BuildCancelService.new(@build, current_user).execute
2017-09-10 17:25:29 +05:30
2021-10-27 15:23:28 +05:30
if service_response.success?
destination = continue_params[:to].presence || builds_project_pipeline_path(@project, @build.pipeline.id)
redirect_to destination
elsif service_response.http_status == :forbidden
access_denied!
2018-12-13 13:39:08 +05:30
else
2021-10-27 15:23:28 +05:30
head service_response.http_status
2018-12-13 13:39:08 +05:30
end
2017-09-10 17:25:29 +05:30
end
2018-12-05 23:21:45 +05:30
def unschedule
2021-10-27 15:23:28 +05:30
service_response = Ci::BuildUnscheduleService.new(@build, current_user).execute
2018-12-05 23:21:45 +05:30
2021-10-27 15:23:28 +05:30
if service_response.success?
redirect_to build_path(@build)
elsif service_response.http_status == :forbidden
access_denied!
else
head service_response.http_status
end
2018-12-05 23:21:45 +05:30
end
2017-09-10 17:25:29 +05:30
def status
2022-01-26 12:08:38 +05:30
render json: Ci::JobSerializer
2017-09-10 17:25:29 +05:30
.new(project: @project, current_user: @current_user)
2021-11-18 22:05:49 +05:30
.represent_status(@build.present(current_user: current_user))
2017-09-10 17:25:29 +05:30
end
def erase
if @build.erase(erased_by: current_user)
redirect_to project_job_path(project, @build),
2019-07-07 11:18:12 +05:30
notice: _("Job has been successfully erased!")
2017-09-10 17:25:29 +05:30
else
respond_422
end
end
def raw
2021-12-11 22:18:48 +05:30
if @build.trace.archived_trace_exist?
2019-02-15 15:39:39 +05:30
workhorse_set_content_type!
2021-12-11 22:18:48 +05:30
send_upload(@build.job_artifacts_trace.file,
2018-05-09 12:01:36 +05:30
send_params: raw_send_params,
redirect_params: raw_redirect_params)
else
2021-01-03 14:25:43 +05:30
@build.trace.read do |stream|
2018-05-09 12:01:36 +05:30
if stream.file?
2019-02-15 15:39:39 +05:30
workhorse_set_content_type!
2018-05-09 12:01:36 +05:30
send_file stream.path, type: 'text/plain; charset=utf-8', disposition: 'inline'
else
2019-02-15 15:39:39 +05:30
# 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-05-09 12:01:36 +05:30
end
2017-09-10 17:25:29 +05:30
end
end
end
2018-11-08 19:23:39 +05:30
def terminal
end
# GET .../terminal.ws : implemented in gitlab-workhorse
def terminal_websocket_authorize
set_workhorse_internal_api_content_type
2019-07-07 11:18:12 +05:30
render json: Gitlab::Workhorse.channel_websocket(@build.terminal_specification)
2018-11-08 19:23:39 +05:30
end
2020-06-23 00:09:42 +05:30
def proxy_websocket_authorize
render json: proxy_websocket_service(build_service_specification)
end
2017-09-10 17:25:29 +05:30
private
2021-02-22 17:27:13 +05:30
def authorize_read_build_trace!
return if can?(current_user, :read_build_trace, @build)
msg = _(
"You must have developer or higher permissions in the associated project to view job logs when debug trace is enabled. To disable debug trace, set the 'CI_DEBUG_TRACE' variable to 'false' in your pipeline configuration or CI/CD settings. " \
"If you need to view this job log, a project maintainer must add you to the project with developer permissions or higher."
)
return access_denied!(msg) if @build.debug_mode?
access_denied!(_('The current user is not authorized to access the job log.'))
end
2017-09-10 17:25:29 +05:30
def authorize_update_build!
2021-01-03 14:25:43 +05:30
return access_denied! unless can?(current_user, :update_build, @build)
2017-09-10 17:25:29 +05:30
end
2018-03-17 18:26:18 +05:30
def authorize_erase_build!
2021-01-03 14:25:43 +05:30
return access_denied! unless can?(current_user, :erase_build, @build)
2018-03-17 18:26:18 +05:30
end
2018-11-08 19:23:39 +05:30
def authorize_use_build_terminal!
2021-01-03 14:25:43 +05:30
return access_denied! unless can?(current_user, :create_build_terminal, @build)
2018-11-08 19:23:39 +05:30
end
2020-06-23 00:09:42 +05:30
def authorize_create_proxy_build!
2021-01-03 14:25:43 +05:30
return access_denied! unless can?(current_user, :create_build_service_proxy, @build)
2020-06-23 00:09:42 +05:30
end
2018-11-08 19:23:39 +05:30
def verify_api_request!
Gitlab::Workhorse.verify_api_request!(request.headers)
end
2020-06-23 00:09:42 +05:30
def verify_proxy_request!
verify_api_request!
set_workhorse_internal_api_content_type
end
2018-05-09 12:01:36 +05:30
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
2019-10-12 21:52:04 +05:30
def play_params
params.permit(job_variables_attributes: %i[key secret_value])
end
2021-01-03 14:25:43 +05:30
def find_job_as_build
@build = project.builds.find(params[:id])
2017-09-10 17:25:29 +05:30
end
2021-01-03 14:25:43 +05:30
def find_job_as_processable
2021-02-22 17:27:13 +05:30
@build = project.processables.find(params[:id])
2021-01-03 14:25:43 +05:30
end
2017-09-10 17:25:29 +05:30
def build_path(build)
project_job_path(build.project, build)
end
2019-02-15 15:39:39 +05:30
def raw_trace_content_disposition(raw_data)
2021-04-01 16:36:13 +05:30
mime_type = Gitlab::Utils::MimeType.from_string(raw_data)
2019-02-15 15:39:39 +05:30
# if mime_type is nil can also represent 'text/plain'
2021-04-01 16:36:13 +05:30
return 'inline' if mime_type.nil? || mime_type == 'text/plain'
2019-02-15 15:39:39 +05:30
'attachment'
end
2019-12-04 20:38:33 +05:30
2020-06-23 00:09:42 +05:30
def build_service_specification
2021-01-03 14:25:43 +05:30
@build.service_specification(service: params['service'],
port: params['port'],
path: params['path'],
subprotocols: proxy_subprotocol)
2020-06-23 00:09:42 +05:30
end
def proxy_subprotocol
# This will allow to reuse the same subprotocol set
# in the original websocket connection
request.headers['HTTP_SEC_WEBSOCKET_PROTOCOL'].presence || ::Ci::BuildRunnerSession::TERMINAL_SUBPROTOCOL
end
# This method provides the information to Workhorse
# about the service we want to proxy to.
# For security reasons, in case this operation is started by JS,
# it's important to use only sourced GitLab JS code
def proxy_websocket_service(service)
service[:url] = ::Gitlab::UrlHelpers.as_wss(service[:url])
::Gitlab::Workhorse.channel_websocket(service)
end
2021-04-29 21:17:54 +05:30
def push_jobs_table_vue
push_frontend_feature_flag(:jobs_table_vue, @project, default_enabled: :yaml)
end
2020-06-23 00:09:42 +05:30
end