72 lines
1.5 KiB
Ruby
72 lines
1.5 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
require 'faye/websocket/client'
|
|
require 'json_schemer'
|
|
|
|
class RunnerConnection
|
|
EVENTS = %i[start output exit stdout stderr].freeze
|
|
BACKEND_OUTPUT_SCHEMA = JSONSchemer.schema(JSON.parse(File.read('lib/runner/backend-output.schema.json')))
|
|
|
|
def initialize(url)
|
|
@socket = Faye::WebSocket::Client.new(url, [], ping: 5)
|
|
|
|
%i[open message error close].each do |event_type|
|
|
@socket.on(event_type) { |event| __send__(:"on_#{event_type}", event) }
|
|
end
|
|
|
|
EVENTS.each { |event_type| instance_variable_set(:"@#{event_type}_callback", ->(e) {}) }
|
|
@start_callback = -> {}
|
|
@exit_code = 0
|
|
end
|
|
|
|
def on(event, &block)
|
|
return unless EVENTS.include? event
|
|
|
|
instance_variable_set(:"@#{event}_callback", block)
|
|
end
|
|
|
|
def send(data)
|
|
@socket.send(encode(data))
|
|
end
|
|
|
|
private
|
|
|
|
def decode(event)
|
|
JSON.parse(event).deep_symbolize_keys
|
|
end
|
|
|
|
def encode(data)
|
|
data
|
|
end
|
|
|
|
def on_message(event)
|
|
return unless BACKEND_OUTPUT_SCHEMA.valid?(JSON.parse(event.data))
|
|
|
|
event = decode(event.data)
|
|
|
|
# TODO: handle other events like timeout
|
|
case event[:type].to_sym
|
|
when :exit_code
|
|
@exit_code = event[:data]
|
|
when :stderr
|
|
@stderr_callback.call event[:data]
|
|
@output_callback.call event[:data]
|
|
when :stdout
|
|
@stdout_callback.call event[:data]
|
|
@output_callback.call event[:data]
|
|
else
|
|
:error
|
|
end
|
|
end
|
|
|
|
def on_open(_event)
|
|
@start_callback.call
|
|
end
|
|
|
|
def on_error(event); end
|
|
|
|
def on_close(_event)
|
|
@exit_callback.call @exit_code
|
|
end
|
|
end
|