2017-04-06 16:20:27 +00:00
|
|
|
module Gitlab
|
|
|
|
module Ci
|
|
|
|
class Trace
|
|
|
|
attr_reader :job
|
|
|
|
|
|
|
|
delegate :old_trace, to: :job
|
|
|
|
|
|
|
|
def initialize(job)
|
|
|
|
@job = job
|
|
|
|
end
|
|
|
|
|
|
|
|
def html(last_lines: nil)
|
|
|
|
read do |stream|
|
|
|
|
stream.html(last_lines: last_lines)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def raw(last_lines: nil)
|
|
|
|
read do |stream|
|
|
|
|
stream.raw(last_lines: last_lines)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def extract_coverage(regex)
|
|
|
|
read do |stream|
|
|
|
|
stream.extract_coverage(regex)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-09-25 16:54:08 +00:00
|
|
|
def extract_sections
|
|
|
|
read do |stream|
|
|
|
|
stream.extract_sections
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-04-06 16:20:27 +00:00
|
|
|
def set(data)
|
|
|
|
write do |stream|
|
|
|
|
data = job.hide_secrets(data)
|
|
|
|
stream.set(data)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def append(data, offset)
|
|
|
|
write do |stream|
|
|
|
|
current_length = stream.size
|
|
|
|
return -current_length unless current_length == offset
|
|
|
|
|
|
|
|
data = job.hide_secrets(data)
|
|
|
|
stream.append(data, offset)
|
|
|
|
stream.size
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def exist?
|
2018-01-25 09:50:56 +00:00
|
|
|
trace_artifact&.exists? || current_path.present? || old_trace.present?
|
2017-04-06 16:20:27 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
def read
|
|
|
|
stream = Gitlab::Ci::Trace::Stream.new do
|
2018-01-26 17:38:54 +00:00
|
|
|
if trace_artifact
|
2018-01-25 09:50:56 +00:00
|
|
|
trace_artifact.open
|
|
|
|
elsif current_path
|
2017-04-06 16:20:27 +00:00
|
|
|
File.open(current_path, "rb")
|
|
|
|
elsif old_trace
|
|
|
|
StringIO.new(old_trace)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
yield stream
|
|
|
|
ensure
|
|
|
|
stream&.close
|
|
|
|
end
|
|
|
|
|
|
|
|
def write
|
|
|
|
stream = Gitlab::Ci::Trace::Stream.new do
|
|
|
|
File.open(ensure_path, "a+b")
|
|
|
|
end
|
|
|
|
|
|
|
|
yield(stream).tap do
|
|
|
|
job.touch if job.needs_touch?
|
|
|
|
end
|
|
|
|
ensure
|
|
|
|
stream&.close
|
|
|
|
end
|
|
|
|
|
|
|
|
def erase!
|
2018-01-26 17:38:54 +00:00
|
|
|
trace_artifact&.destroy
|
2018-01-25 13:46:38 +00:00
|
|
|
|
2017-04-06 16:20:27 +00:00
|
|
|
paths.each do |trace_path|
|
|
|
|
FileUtils.rm(trace_path, force: true)
|
|
|
|
end
|
|
|
|
|
|
|
|
job.erase_old_trace!
|
|
|
|
end
|
|
|
|
|
2018-02-23 12:08:38 +00:00
|
|
|
def archive!
|
|
|
|
return if trace_artifact
|
|
|
|
|
|
|
|
if current_path
|
|
|
|
File.open(current_path) do |f|
|
|
|
|
archive_stream!(f)
|
|
|
|
f.unlink
|
|
|
|
end
|
|
|
|
elsif old_trace
|
|
|
|
StringIO(old_trace).tap do |stream|
|
|
|
|
archive_stream!(stream)
|
|
|
|
job.erase_old_trace!
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-04-06 16:20:27 +00:00
|
|
|
private
|
|
|
|
|
2018-02-23 12:08:38 +00:00
|
|
|
def archive_stream!(stream)
|
|
|
|
file = Tempfile.new('trace.log')
|
|
|
|
size = IO.copy_stream(file, stream)
|
|
|
|
raise 'Not all saved' unless size == stream.size
|
|
|
|
file.close
|
|
|
|
|
|
|
|
job.create_job_artifacts_trace!(
|
|
|
|
project: job.project,
|
|
|
|
file_type: :trace,
|
|
|
|
file: file)
|
|
|
|
ensure
|
|
|
|
file&.close
|
|
|
|
file&.unlink
|
|
|
|
end
|
|
|
|
|
2017-04-06 16:20:27 +00:00
|
|
|
def ensure_path
|
|
|
|
return current_path if current_path
|
|
|
|
|
|
|
|
ensure_directory
|
2018-01-26 17:38:54 +00:00
|
|
|
default_path
|
2017-04-06 16:20:27 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
def ensure_directory
|
2018-01-26 17:38:54 +00:00
|
|
|
unless Dir.exist?(default_directory)
|
|
|
|
FileUtils.mkdir_p(default_directory)
|
2017-04-06 16:20:27 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-01-26 17:00:29 +00:00
|
|
|
def current_path
|
|
|
|
@current_path ||= paths.find do |trace_path|
|
|
|
|
File.exist?(trace_path)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-04-06 16:20:27 +00:00
|
|
|
def paths
|
|
|
|
[
|
2018-01-26 17:38:54 +00:00
|
|
|
default_path,
|
2017-04-06 16:20:27 +00:00
|
|
|
deprecated_path
|
|
|
|
].compact
|
|
|
|
end
|
|
|
|
|
2018-01-26 17:38:54 +00:00
|
|
|
def default_directory
|
2017-04-06 16:20:27 +00:00
|
|
|
File.join(
|
|
|
|
Settings.gitlab_ci.builds_path,
|
2018-01-26 17:38:54 +00:00
|
|
|
job.created_at.utc.strftime("%Y_%m"),
|
|
|
|
job.project_id.to_s
|
2017-04-06 16:20:27 +00:00
|
|
|
)
|
|
|
|
end
|
|
|
|
|
2018-01-26 17:38:54 +00:00
|
|
|
def default_path
|
|
|
|
File.join(default_directory, "#{job.id}.log")
|
2017-04-06 16:20:27 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
def deprecated_path
|
|
|
|
File.join(
|
|
|
|
Settings.gitlab_ci.builds_path,
|
|
|
|
job.created_at.utc.strftime("%Y_%m"),
|
|
|
|
job.project.ci_id.to_s,
|
|
|
|
"#{job.id}.log"
|
|
|
|
) if job.project&.ci_id
|
|
|
|
end
|
2018-01-25 09:50:56 +00:00
|
|
|
|
|
|
|
def trace_artifact
|
|
|
|
job.job_artifacts_trace
|
|
|
|
end
|
2017-04-06 16:20:27 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|