gitlab-org--gitlab-foss/lib/gitlab/workhorse.rb

299 lines
8.7 KiB
Ruby
Raw Normal View History

# frozen_string_literal: true
2016-02-01 05:33:22 -05:00
require 'base64'
require 'json'
require 'securerandom'
2017-03-24 13:22:42 -04:00
require 'uri'
2016-02-01 05:33:22 -05:00
module Gitlab
class Workhorse
2017-02-21 18:32:18 -05:00
SEND_DATA_HEADER = 'Gitlab-Workhorse-Send-Data'.freeze
VERSION_FILE = 'GITLAB_WORKHORSE_VERSION'.freeze
INTERNAL_API_CONTENT_TYPE = 'application/vnd.gitlab-workhorse+json'.freeze
INTERNAL_API_REQUEST_HEADER = 'Gitlab-Workhorse-Api-Request'.freeze
NOTIFICATION_CHANNEL = 'workhorse:notifications'.freeze
2018-03-07 10:22:37 -05:00
ALLOWED_GIT_HTTP_ACTIONS = %w[git_receive_pack git_upload_pack info_refs].freeze
DETECT_HEADER = 'Gitlab-Workhorse-Detect-Content-Type'.freeze
# Supposedly the effective key size for HMAC-SHA256 is 256 bits, i.e. 32
# bytes https://tools.ietf.org/html/rfc4868#section-2.6
SECRET_LENGTH = 32
2016-02-02 08:09:55 -05:00
2016-02-11 12:10:14 -05:00
class << self
def git_http_ok(repository, repo_type, user, action, show_all_refs: false)
2018-03-07 10:22:37 -05:00
raise "Unsupported action: #{action}" unless ALLOWED_GIT_HTTP_ACTIONS.include?(action.to_s)
2018-03-06 09:17:00 -05:00
project = repository.project
attrs = {
GL_ID: Gitlab::GlId.gl_id(user),
GL_REPOSITORY: repo_type.identifier_for_subject(project),
GL_USERNAME: user&.username,
ShowAllRefs: show_all_refs,
Repository: repository.gitaly_repository.to_h,
GitConfigOptions: [],
GitalyServer: {
address: Gitlab::GitalyClient.address(project.repository_storage),
token: Gitlab::GitalyClient.token(project.repository_storage)
}
}
# Custom option for git-receive-pack command
receive_max_input_size = Gitlab::CurrentSettings.receive_max_input_size.to_i
if receive_max_input_size > 0
attrs[:GitConfigOptions] << "receive.maxInputSize=#{receive_max_input_size.megabytes}"
end
attrs
2016-04-06 11:52:12 -04:00
end
2016-02-01 05:33:22 -05:00
def send_git_blob(repository, blob)
2018-07-09 06:02:02 -04:00
params = {
'GitalyServer' => gitaly_server_hash(repository),
'GetBlobRequest' => {
repository: repository.gitaly_repository.to_h,
oid: blob.id,
limit: -1
}
}
2016-02-01 05:33:22 -05:00
[
2016-02-02 08:09:55 -05:00
SEND_DATA_HEADER,
2016-06-08 08:30:15 -04:00
"git-blob:#{encode(params)}"
2016-02-01 05:33:22 -05:00
]
end
2016-02-02 08:09:55 -05:00
def send_git_archive(repository, ref:, format:, append_sha:, path: nil)
path_enabled = Feature.enabled?(:git_archive_path, default_enabled: true)
path = nil unless path_enabled
2016-02-02 08:09:55 -05:00
format ||= 'tar.gz'
format = format.downcase
metadata = repository.archive_metadata(
ref,
Gitlab.config.gitlab.repository_downloads_path,
format,
append_sha: append_sha,
path: path
)
2016-02-02 08:09:55 -05:00
raise "Repository or ref not found" if metadata.empty?
params =
if path_enabled
send_git_archive_params(repository, metadata, path, archive_format(format))
else
metadata
end
# If present, DisableCache must be a Boolean. Otherwise
# workhorse ignores it.
params['DisableCache'] = true if git_archive_cache_disabled?
params['GitalyServer'] = gitaly_server_hash(repository)
2016-02-02 08:09:55 -05:00
[
SEND_DATA_HEADER,
2016-06-08 08:30:15 -04:00
"git-archive:#{encode(params)}"
2016-02-02 08:09:55 -05:00
]
end
2016-05-12 14:50:49 -04:00
def send_git_snapshot(repository)
params = {
'GitalyServer' => gitaly_server_hash(repository),
'GetSnapshotRequest' => Gitaly::GetSnapshotRequest.new(
repository: repository.gitaly_repository
).to_json
}
[
SEND_DATA_HEADER,
"git-snapshot:#{encode(params)}"
]
end
2016-06-08 08:30:15 -04:00
def send_git_diff(repository, diff_refs)
2018-07-06 06:01:15 -04:00
params = {
'GitalyServer' => gitaly_server_hash(repository),
'RawDiffRequest' => Gitaly::RawDiffRequest.new(
gitaly_diff_or_patch_hash(repository, diff_refs)
).to_json
}
2016-05-12 14:50:49 -04:00
[
SEND_DATA_HEADER,
"git-diff:#{encode(params)}"
2016-02-02 08:09:55 -05:00
]
end
2016-07-03 17:01:13 -04:00
def send_git_patch(repository, diff_refs)
params = {
'GitalyServer' => gitaly_server_hash(repository),
'RawPatchRequest' => Gitaly::RawPatchRequest.new(
gitaly_diff_or_patch_hash(repository, diff_refs)
).to_json
}
[
2016-06-28 08:59:25 -04:00
SEND_DATA_HEADER,
"git-format-patch:#{encode(params)}"
]
end
def send_artifacts_entry(build, entry)
2018-01-29 12:57:34 -05:00
file = build.artifacts_file
archive = file.file_storage? ? file.path : file.url
params = {
2018-01-29 12:57:34 -05:00
'Archive' => archive,
'Entry' => Base64.encode64(entry.to_s)
}
[
SEND_DATA_HEADER,
"artifacts-entry:#{encode(params)}"
]
end
2018-02-06 09:31:39 -05:00
def send_url(url, allow_redirects: false)
params = {
'URL' => url,
'AllowRedirects' => allow_redirects
}
[
SEND_DATA_HEADER,
"send-url:#{encode(params)}"
]
end
def channel_websocket(channel)
details = {
'Channel' => {
'Subprotocols' => channel[:subprotocols],
'Url' => channel[:url],
'Header' => channel[:headers],
'MaxSessionTime' => channel[:max_session_time]
}
}
details['Channel']['CAPem'] = channel[:ca_pem] if channel.key?(:ca_pem)
details
end
def version
path = Rails.root.join(VERSION_FILE)
path.readable? ? path.read.chomp : 'unknown'
end
def secret
@secret ||= begin
bytes = Base64.strict_decode64(File.read(secret_path).chomp)
raise "#{secret_path} does not contain #{SECRET_LENGTH} bytes" if bytes.length != SECRET_LENGTH
bytes
end
end
def write_secret
bytes = SecureRandom.random_bytes(SECRET_LENGTH)
File.open(secret_path, 'w:BINARY', 0600) do |f|
2016-09-26 08:21:39 -04:00
f.chmod(0600) # If the file already existed, the '0600' passed to 'open' above was a no-op.
f.write(Base64.strict_encode64(bytes))
end
end
def verify_api_request!(request_headers)
2016-08-18 10:31:44 -04:00
decode_jwt(request_headers[INTERNAL_API_REQUEST_HEADER])
end
def decode_jwt(encoded_message)
JWT.decode(
2016-08-18 10:31:44 -04:00
encoded_message,
secret,
true,
{ iss: 'gitlab-workhorse', verify_iss: true, algorithm: 'HS256' }
)
end
def secret_path
Gitlab.config.workhorse.secret_file
end
2017-03-06 05:44:45 -05:00
def set_key_and_notify(key, value, expire: nil, overwrite: true)
Gitlab::Redis::SharedState.with do |redis|
result = redis.set(key, value, ex: expire, nx: !overwrite)
if result
2017-03-06 05:44:45 -05:00
redis.publish(NOTIFICATION_CHANNEL, "#{key}=#{value}")
value
else
redis.get(key)
end
end
end
2016-02-02 08:09:55 -05:00
protected
# This is the outermost encoding of a senddata: header. It is safe for
# inclusion in HTTP response headers
2016-02-02 08:09:55 -05:00
def encode(hash)
Base64.urlsafe_encode64(JSON.dump(hash))
end
2017-07-09 23:43:20 -04:00
# This is for encoding individual fields inside the senddata JSON that
# contain binary data. In workhorse, the corresponding struct field should
# be type []byte
def encode_binary(binary)
Base64.encode64(binary)
end
2017-07-09 23:43:20 -04:00
def gitaly_server_hash(repository)
{
address: Gitlab::GitalyClient.address(repository.project.repository_storage),
token: Gitlab::GitalyClient.token(repository.project.repository_storage)
}
end
def gitaly_diff_or_patch_hash(repository, diff_refs)
{
repository: repository.gitaly_repository,
left_commit_id: diff_refs.base_sha,
right_commit_id: diff_refs.head_sha
}
end
def git_archive_cache_disabled?
ENV['WORKHORSE_ARCHIVE_CACHE_DISABLED'].present? || Feature.enabled?(:workhorse_archive_cache_disabled)
end
def archive_format(format)
case format
when "tar.bz2", "tbz", "tbz2", "tb2", "bz2"
Gitaly::GetArchiveRequest::Format::TAR_BZ2
when "tar"
Gitaly::GetArchiveRequest::Format::TAR
when "zip"
Gitaly::GetArchiveRequest::Format::ZIP
else
Gitaly::GetArchiveRequest::Format::TAR_GZ
end
end
def send_git_archive_params(repository, metadata, path, format)
{
'ArchivePath' => metadata['ArchivePath'],
'GetArchiveRequest' => encode_binary(
Gitaly::GetArchiveRequest.new(
repository: repository.gitaly_repository,
commit_id: metadata['CommitId'],
prefix: metadata['ArchivePrefix'],
format: format,
path: path.presence || ""
).to_proto
)
}
end
2016-02-01 05:33:22 -05:00
end
end
2016-02-01 06:27:35 -05:00
end