
In order to provide an alternative to Poseidon, a strategy for the DockerContainerPool is added that is used by the runner model. Co-authored-by: Sebastian Serth <Sebastian.Serth@hpi.de>
145 lines
4.5 KiB
Ruby
145 lines
4.5 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
class Runner::Strategy::Poseidon < Runner::Strategy
|
|
HEADERS = {'Content-Type' => 'application/json'}.freeze
|
|
ERRORS = %w[NOMAD_UNREACHABLE NOMAD_OVERLOAD NOMAD_INTERNAL_SERVER_ERROR UNKNOWN].freeze
|
|
|
|
ERRORS.each do |error|
|
|
define_singleton_method :"error_#{error.downcase}" do
|
|
error
|
|
end
|
|
end
|
|
|
|
def self.sync_environment(environment)
|
|
environment.copy_to_poseidon
|
|
end
|
|
|
|
def self.request_from_management(environment)
|
|
url = "#{Runner::BASE_URL}/runners"
|
|
body = {executionEnvironmentId: environment.id, inactivityTimeout: Runner::UNUSED_EXPIRATION_TIME}
|
|
response = Faraday.post(url, body.to_json, HEADERS)
|
|
|
|
case response.status
|
|
when 200
|
|
response_body = parse response
|
|
runner_id = response_body[:runnerId]
|
|
runner_id.presence || raise(Runner::Error::Unknown.new('Poseidon did not send a runner id'))
|
|
when 404
|
|
raise Runner::Error::EnvironmentNotFound.new
|
|
else
|
|
handle_error response
|
|
end
|
|
rescue Faraday::Error => e
|
|
raise Runner::Error::Unknown.new("Faraday request to runner management failed: #{e.inspect}")
|
|
end
|
|
|
|
def self.handle_error(response)
|
|
case response.status
|
|
when 400
|
|
response_body = parse response
|
|
raise Runner::Error::BadRequest.new(response_body[:message])
|
|
when 401
|
|
raise Runner::Error::Unauthorized.new('Authentication with Poseidon failed')
|
|
when 404
|
|
raise Runner::Error::RunnerNotFound.new
|
|
when 500
|
|
response_body = parse response
|
|
error_code = response_body[:errorCode]
|
|
if error_code == error_nomad_overload
|
|
raise Runner::Error::NotAvailable.new("Poseidon has no runner available (#{error_code}): #{response_body[:message]}")
|
|
else
|
|
raise Runner::Error::InternalServerError.new("Poseidon sent #{response_body[:errorCode]}: #{response_body[:message]}")
|
|
end
|
|
else
|
|
raise Runner::Error::Unknown.new("Poseidon sent unexpected response status code #{response.status}")
|
|
end
|
|
end
|
|
|
|
def self.parse(response)
|
|
JSON.parse(response.body).deep_symbolize_keys
|
|
rescue JSON::ParserError => e
|
|
# Poseidon should not send invalid json
|
|
raise Runner::Error::Unknown.new("Error parsing response from Poseidon: #{e.message}")
|
|
end
|
|
|
|
def initialize(runner_id, _environment)
|
|
super
|
|
@allocation_id = runner_id
|
|
end
|
|
|
|
def copy_files(files)
|
|
copy = files.map do |file|
|
|
{
|
|
path: file.filepath,
|
|
content: Base64.strict_encode64(file.content),
|
|
}
|
|
end
|
|
url = "#{runner_url}/files"
|
|
body = {copy: copy}
|
|
response = Faraday.patch(url, body.to_json, HEADERS)
|
|
return if response.status == 204
|
|
|
|
Runner.destroy(@allocation_id) if response.status == 400
|
|
self.class.handle_error response
|
|
rescue Faraday::Error => e
|
|
raise Runner::Error::Unknown.new("Faraday request to runner management failed: #{e.inspect}")
|
|
end
|
|
|
|
def attach_to_execution(command)
|
|
starting_time = Time.zone.now
|
|
websocket_url = execute_command(command)
|
|
EventMachine.run do
|
|
socket = Connection.new(websocket_url, self)
|
|
yield(socket)
|
|
end
|
|
Time.zone.now - starting_time # execution duration
|
|
end
|
|
|
|
def destroy_at_management
|
|
response = Faraday.delete runner_url
|
|
self.class.handle_error response unless response.status == 204
|
|
rescue Faraday::Error => e
|
|
raise Runner::Error::Unknown.new("Faraday request to runner management failed: #{e.inspect}")
|
|
end
|
|
|
|
private
|
|
|
|
def execute_command(command)
|
|
url = "#{runner_url}/execute"
|
|
body = {command: command, timeLimit: @execution_environment.permitted_execution_time}
|
|
response = Faraday.post(url, body.to_json, HEADERS)
|
|
case response.status
|
|
when 200
|
|
response_body = self.class.parse response
|
|
websocket_url = response_body[:websocketUrl]
|
|
if websocket_url.present?
|
|
return websocket_url
|
|
else
|
|
raise Runner::Error::Unknown.new('Poseidon did not send websocket url')
|
|
end
|
|
when 400
|
|
Runner.destroy(@allocation_id)
|
|
end
|
|
|
|
self.class.handle_error response
|
|
rescue Faraday::Error => e
|
|
raise Runner::Error::Unknown.new("Faraday request to runner management failed: #{e.inspect}")
|
|
end
|
|
|
|
def runner_url
|
|
"#{Runner::BASE_URL}/runners/#{@allocation_id}"
|
|
end
|
|
|
|
class Connection < Runner::Connection
|
|
def decode(raw_event)
|
|
JSON.parse(raw_event.data)
|
|
rescue JSON::ParserError => e
|
|
raise Runner::Error::Unknown.new("The websocket message from Poseidon could not be decoded to JSON: #{e.inspect}")
|
|
end
|
|
|
|
def encode(data)
|
|
data
|
|
end
|
|
end
|
|
end
|