mirror of
https://git.robbyzambito.me/zaprus
synced 2026-02-04 16:34:51 +00:00
Compare commits
6 Commits
19d4e88c33
...
09152377ed
| Author | SHA1 | Date | |
|---|---|---|---|
| 09152377ed | |||
| c3b17f8267 | |||
| cf365673b5 | |||
| 16fd65e281 | |||
| 8965a4d5d4 | |||
| ba8a84c478 |
17
build.zig
17
build.zig
@@ -41,6 +41,23 @@ pub fn build(b: *std.Build) void {
|
|||||||
.target = target,
|
.target = target,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Create static library
|
||||||
|
const lib = b.addLibrary(.{
|
||||||
|
.name = "zaprus",
|
||||||
|
.root_module = b.createModule(.{
|
||||||
|
.root_source_file = b.path("src/c_api.zig"),
|
||||||
|
.target = target,
|
||||||
|
.optimize = optimize,
|
||||||
|
.link_libc = true,
|
||||||
|
.imports = &.{
|
||||||
|
.{ .name = "zaprus", .module = mod },
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
|
||||||
|
b.installArtifact(lib);
|
||||||
|
lib.installHeader(b.path("include/zaprus.h"), "zaprus.h");
|
||||||
|
|
||||||
// Here we define an executable. An executable needs to have a root module
|
// Here we define an executable. An executable needs to have a root module
|
||||||
// which needs to expose a `main` function. While we could add a main function
|
// which needs to expose a `main` function. While we could add a main function
|
||||||
// to the module defined above, it's sometimes preferable to split business
|
// to the module defined above, it's sometimes preferable to split business
|
||||||
|
|||||||
33
include/zaprus.h
Normal file
33
include/zaprus.h
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
#ifndef ZAPRUS_H
|
||||||
|
#define ZAPRUS_H
|
||||||
|
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
typedef void* zaprus_client;
|
||||||
|
typedef void* zaprus_connection;
|
||||||
|
|
||||||
|
// Returns NULL if there was an error.
|
||||||
|
zaprus_client zaprus_init_client(void);
|
||||||
|
|
||||||
|
void zaprus_deinit_client(zaprus_client client);
|
||||||
|
|
||||||
|
// Returns 0 on success, else returns 1.
|
||||||
|
int zaprus_client_send_relay(zaprus_client client, const char* payload, size_t payload_len, const char dest[4]);
|
||||||
|
|
||||||
|
// Returns NULL if there was an error.
|
||||||
|
// Caller should call zaprus_deinit_connection when done with the connection.
|
||||||
|
zaprus_connection zaprus_connect(zaprus_client client, const char* payload, size_t payload_len);
|
||||||
|
|
||||||
|
void zaprus_deinit_connection(zaprus_connection connection);
|
||||||
|
|
||||||
|
// Capacity is the maximum length of the output buffer.
|
||||||
|
// out_len is modified to specify how much of the capacity is used by the response.
|
||||||
|
// Blocks until the next message is available, or returns 1 if the underlying socket times out.
|
||||||
|
// Returns 0 on success, else returns 1.
|
||||||
|
int zaprus_connection_next(zaprus_connection connection, char *out, size_t capacity, size_t *out_len);
|
||||||
|
|
||||||
|
// Returns 0 on success, else returns 1.
|
||||||
|
int zaprus_connection_send(zaprus_connection connection, const char *payload, size_t payload_len);
|
||||||
|
|
||||||
|
#endif // ZAPRUS_H
|
||||||
@@ -22,10 +22,8 @@ pub fn deinit(self: *Client) void {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn sendRelay(self: *Client, io: Io, payload: []const u8, dest: [4]u8) !void {
|
pub fn sendRelay(self: *Client, io: Io, payload: []const u8, dest: [4]u8) !void {
|
||||||
const rand = blk: {
|
|
||||||
const io_source: std.Random.IoSource = .{ .io = io };
|
const io_source: std.Random.IoSource = .{ .io = io };
|
||||||
break :blk io_source.interface();
|
const rand = io_source.interface();
|
||||||
};
|
|
||||||
|
|
||||||
var headers: EthIpUdp = .{
|
var headers: EthIpUdp = .{
|
||||||
.src_mac = self.socket.mac,
|
.src_mac = self.socket.mac,
|
||||||
@@ -53,22 +51,18 @@ pub fn sendRelay(self: *Client, io: Io, payload: []const u8, dest: [4]u8) !void
|
|||||||
const relay_bytes = relay.toBytes(&relay_buf);
|
const relay_bytes = relay.toBytes(&relay_buf);
|
||||||
headers.setPayloadLen(relay_bytes.len);
|
headers.setPayloadLen(relay_bytes.len);
|
||||||
|
|
||||||
const full_msg = blk: {
|
|
||||||
var msg_buf: [max_message_size]u8 = undefined;
|
var msg_buf: [max_message_size]u8 = undefined;
|
||||||
var msg_w: Writer = .fixed(&msg_buf);
|
var msg_w: Writer = .fixed(&msg_buf);
|
||||||
msg_w.writeAll(&headers.toBytes()) catch unreachable;
|
msg_w.writeAll(&headers.toBytes()) catch unreachable;
|
||||||
msg_w.writeAll(relay_bytes) catch unreachable;
|
msg_w.writeAll(relay_bytes) catch unreachable;
|
||||||
break :blk msg_w.buffered();
|
const full_msg = msg_w.buffered();
|
||||||
};
|
|
||||||
|
|
||||||
try self.socket.send(full_msg);
|
try self.socket.send(full_msg);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn connect(self: Client, io: Io, payload: []const u8) !SaprusConnection {
|
pub fn connect(self: Client, io: Io, payload: []const u8) !SaprusConnection {
|
||||||
const rand = blk: {
|
|
||||||
const io_source: std.Random.IoSource = .{ .io = io };
|
const io_source: std.Random.IoSource = .{ .io = io };
|
||||||
break :blk io_source.interface();
|
const rand = io_source.interface();
|
||||||
};
|
|
||||||
|
|
||||||
var headers: EthIpUdp = .{
|
var headers: EthIpUdp = .{
|
||||||
.src_mac = self.socket.mac,
|
.src_mac = self.socket.mac,
|
||||||
@@ -98,7 +92,10 @@ pub fn connect(self: Client, io: Io, payload: []const u8) !SaprusConnection {
|
|||||||
};
|
};
|
||||||
|
|
||||||
log.debug("Setting bpf filter to port {}", .{connection.connection.src});
|
log.debug("Setting bpf filter to port {}", .{connection.connection.src});
|
||||||
try self.socket.attachSaprusPortFilter(connection.connection.src);
|
self.socket.attachSaprusPortFilter(connection.connection.src) catch |err| {
|
||||||
|
log.err("Failed to set port filter: {t}", .{err});
|
||||||
|
return err;
|
||||||
|
};
|
||||||
log.debug("bpf set", .{});
|
log.debug("bpf set", .{});
|
||||||
|
|
||||||
var connection_buf: [2048]u8 = undefined;
|
var connection_buf: [2048]u8 = undefined;
|
||||||
@@ -106,13 +103,11 @@ pub fn connect(self: Client, io: Io, payload: []const u8) !SaprusConnection {
|
|||||||
headers.setPayloadLen(connection_bytes.len);
|
headers.setPayloadLen(connection_bytes.len);
|
||||||
|
|
||||||
log.debug("Building full message", .{});
|
log.debug("Building full message", .{});
|
||||||
var full_msg = blk: {
|
|
||||||
var msg_buf: [2048]u8 = undefined;
|
var msg_buf: [2048]u8 = undefined;
|
||||||
var msg_w: Writer = .fixed(&msg_buf);
|
var msg_w: Writer = .fixed(&msg_buf);
|
||||||
msg_w.writeAll(&headers.toBytes()) catch unreachable;
|
msg_w.writeAll(&headers.toBytes()) catch unreachable;
|
||||||
msg_w.writeAll(connection_bytes) catch unreachable;
|
msg_w.writeAll(connection_bytes) catch unreachable;
|
||||||
break :blk msg_w.buffered();
|
var full_msg = msg_w.buffered();
|
||||||
};
|
|
||||||
log.debug("Built full message. Sending message", .{});
|
log.debug("Built full message. Sending message", .{});
|
||||||
|
|
||||||
try self.socket.send(full_msg);
|
try self.socket.send(full_msg);
|
||||||
@@ -128,13 +123,13 @@ pub fn connect(self: Client, io: Io, payload: []const u8) !SaprusConnection {
|
|||||||
headers.setPayloadLen(connection_bytes.len);
|
headers.setPayloadLen(connection_bytes.len);
|
||||||
|
|
||||||
log.debug("Building final handshake message", .{});
|
log.debug("Building final handshake message", .{});
|
||||||
full_msg = blk: {
|
|
||||||
var msg_buf: [2048]u8 = undefined;
|
msg_w.end = 0;
|
||||||
var msg_w: Writer = .fixed(&msg_buf);
|
|
||||||
msg_w.writeAll(&headers.toBytes()) catch unreachable;
|
msg_w.writeAll(&headers.toBytes()) catch unreachable;
|
||||||
msg_w.writeAll(connection_bytes) catch unreachable;
|
msg_w.writeAll(connection_bytes) catch unreachable;
|
||||||
break :blk msg_w.buffered();
|
full_msg = msg_w.buffered();
|
||||||
};
|
|
||||||
try self.socket.send(full_msg);
|
try self.socket.send(full_msg);
|
||||||
|
|
||||||
return .init(self.socket, headers, connection);
|
return .init(self.socket, headers, connection);
|
||||||
|
|||||||
@@ -17,10 +17,8 @@ pub fn next(self: Connection, io: Io, buf: []u8) ![]const u8 {
|
|||||||
log.debug("Awaiting connection message", .{});
|
log.debug("Awaiting connection message", .{});
|
||||||
const res = try self.socket.receive(buf);
|
const res = try self.socket.receive(buf);
|
||||||
log.debug("Received {} byte connection message", .{res.len});
|
log.debug("Received {} byte connection message", .{res.len});
|
||||||
const connection_res = blk: {
|
|
||||||
const msg: SaprusMessage = try .parse(res[42..]);
|
const msg: SaprusMessage = try .parse(res[42..]);
|
||||||
break :blk msg.connection;
|
const connection_res = msg.connection;
|
||||||
};
|
|
||||||
|
|
||||||
log.debug("Payload was {s}", .{connection_res.payload});
|
log.debug("Payload was {s}", .{connection_res.payload});
|
||||||
|
|
||||||
@@ -28,29 +26,23 @@ pub fn next(self: Connection, io: Io, buf: []u8) ![]const u8 {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn send(self: *Connection, io: Io, buf: []const u8) !void {
|
pub fn send(self: *Connection, io: Io, buf: []const u8) !void {
|
||||||
const rand = blk: {
|
|
||||||
const io_source: std.Random.IoSource = .{ .io = io };
|
const io_source: std.Random.IoSource = .{ .io = io };
|
||||||
break :blk io_source.interface();
|
const rand = io_source.interface();
|
||||||
};
|
|
||||||
|
|
||||||
log.debug("Sending connection message", .{});
|
log.debug("Sending connection message", .{});
|
||||||
|
|
||||||
self.connection.connection.payload = buf;
|
self.connection.connection.payload = buf;
|
||||||
const connection_bytes = blk: {
|
var connection_bytes_buf: [2048]u8 = undefined;
|
||||||
var connection_bytes: [2048]u8 = undefined;
|
const connection_bytes = self.connection.toBytes(&connection_bytes_buf);
|
||||||
break :blk self.connection.toBytes(&connection_bytes);
|
|
||||||
};
|
|
||||||
|
|
||||||
self.headers.ip.id = rand.int(u16);
|
self.headers.ip.id = rand.int(u16);
|
||||||
self.headers.setPayloadLen(connection_bytes.len);
|
self.headers.setPayloadLen(connection_bytes.len);
|
||||||
|
|
||||||
const full_msg = blk: {
|
|
||||||
var msg_buf: [2048]u8 = undefined;
|
var msg_buf: [2048]u8 = undefined;
|
||||||
var msg_w: Writer = .fixed(&msg_buf);
|
var msg_w: Writer = .fixed(&msg_buf);
|
||||||
try msg_w.writeAll(&self.headers.toBytes());
|
try msg_w.writeAll(&self.headers.toBytes());
|
||||||
try msg_w.writeAll(connection_bytes);
|
try msg_w.writeAll(connection_bytes);
|
||||||
break :blk msg_w.buffered();
|
const full_msg = msg_w.buffered();
|
||||||
};
|
|
||||||
|
|
||||||
try self.socket.send(full_msg);
|
try self.socket.send(full_msg);
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
const RawSocket = @This();
|
const RawSocket = @This();
|
||||||
|
|
||||||
|
const is_debug = builtin.mode == .Debug;
|
||||||
|
|
||||||
fd: i32,
|
fd: i32,
|
||||||
sockaddr_ll: std.posix.sockaddr.ll,
|
sockaddr_ll: std.posix.sockaddr.ll,
|
||||||
mac: [6]u8,
|
mac: [6]u8,
|
||||||
@@ -15,7 +17,7 @@ const Ifconf = extern struct {
|
|||||||
};
|
};
|
||||||
|
|
||||||
pub fn init() !RawSocket {
|
pub fn init() !RawSocket {
|
||||||
const socket: i32 = @intCast(std.os.linux.socket(std.os.linux.AF.PACKET, std.os.linux.SOCK.RAW, 0));
|
const socket: i32 = std.math.cast(i32, std.os.linux.socket(std.os.linux.AF.PACKET, std.os.linux.SOCK.RAW, 0)) orelse return error.SocketError;
|
||||||
if (socket < 0) return error.SocketError;
|
if (socket < 0) return error.SocketError;
|
||||||
|
|
||||||
var ifreq_storage: [16]std.os.linux.ifreq = undefined;
|
var ifreq_storage: [16]std.os.linux.ifreq = undefined;
|
||||||
@@ -70,7 +72,7 @@ pub fn init() !RawSocket {
|
|||||||
const bind_ret = std.os.linux.bind(socket, @ptrCast(&sockaddr_ll), @sizeOf(@TypeOf(sockaddr_ll)));
|
const bind_ret = std.os.linux.bind(socket, @ptrCast(&sockaddr_ll), @sizeOf(@TypeOf(sockaddr_ll)));
|
||||||
if (bind_ret != 0) return error.BindError;
|
if (bind_ret != 0) return error.BindError;
|
||||||
|
|
||||||
const timeout: std.os.linux.timeval = .{ .sec = 600, .usec = 0 };
|
const timeout: std.os.linux.timeval = .{ .sec = 60 * if (is_debug) 1 else 10, .usec = 0 };
|
||||||
const timeout_ret = std.os.linux.setsockopt(socket, std.os.linux.SOL.SOCKET, std.os.linux.SO.RCVTIMEO, @ptrCast(&timeout), @sizeOf(@TypeOf(timeout)));
|
const timeout_ret = std.os.linux.setsockopt(socket, std.os.linux.SOL.SOCKET, std.os.linux.SO.RCVTIMEO, @ptrCast(&timeout), @sizeOf(@TypeOf(timeout)));
|
||||||
if (timeout_ret != 0) return error.SetTimeoutError;
|
if (timeout_ret != 0) return error.SetTimeoutError;
|
||||||
|
|
||||||
|
|||||||
88
src/c_api.zig
Normal file
88
src/c_api.zig
Normal file
@@ -0,0 +1,88 @@
|
|||||||
|
const std = @import("std");
|
||||||
|
const zaprus = @import("zaprus");
|
||||||
|
|
||||||
|
// Opaque types for C API
|
||||||
|
const ZaprusClient = opaque {};
|
||||||
|
const ZaprusConnection = opaque {};
|
||||||
|
|
||||||
|
const alloc = std.heap.c_allocator;
|
||||||
|
const io = std.Io.Threaded.global_single_threaded.io();
|
||||||
|
|
||||||
|
export fn zaprus_init_client() ?*ZaprusClient {
|
||||||
|
const client = alloc.create(zaprus.Client) catch return null;
|
||||||
|
client.* = zaprus.Client.init() catch {
|
||||||
|
alloc.destroy(client);
|
||||||
|
return null;
|
||||||
|
};
|
||||||
|
return @ptrCast(client);
|
||||||
|
}
|
||||||
|
|
||||||
|
export fn zaprus_deinit_client(client: ?*ZaprusClient) void {
|
||||||
|
const c: ?*zaprus.Client = @ptrCast(@alignCast(client));
|
||||||
|
if (c) |zc| {
|
||||||
|
zc.deinit();
|
||||||
|
alloc.destroy(zc);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export fn zaprus_client_send_relay(
|
||||||
|
client: ?*ZaprusClient,
|
||||||
|
payload: [*c]const u8,
|
||||||
|
payload_len: usize,
|
||||||
|
dest: [*c]const u8,
|
||||||
|
) c_int {
|
||||||
|
const c: ?*zaprus.Client = @ptrCast(@alignCast(client));
|
||||||
|
const zc = c orelse return 1;
|
||||||
|
|
||||||
|
zc.sendRelay(io, payload[0..payload_len], dest[0..4].*) catch return 1;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
export fn zaprus_connect(
|
||||||
|
client: ?*ZaprusClient,
|
||||||
|
payload: [*c]const u8,
|
||||||
|
payload_len: usize,
|
||||||
|
) ?*ZaprusConnection {
|
||||||
|
const c: ?*zaprus.Client = @ptrCast(@alignCast(client));
|
||||||
|
const zc = c orelse return null;
|
||||||
|
|
||||||
|
const connection = alloc.create(zaprus.Connection) catch return null;
|
||||||
|
connection.* = zc.connect(io, payload[0..payload_len]) catch {
|
||||||
|
alloc.destroy(connection);
|
||||||
|
return null;
|
||||||
|
};
|
||||||
|
return @ptrCast(connection);
|
||||||
|
}
|
||||||
|
|
||||||
|
export fn zaprus_deinit_connection(connection: ?*ZaprusConnection) void {
|
||||||
|
const c: ?*zaprus.Connection = @ptrCast(@alignCast(connection));
|
||||||
|
if (c) |zc| {
|
||||||
|
alloc.destroy(zc);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export fn zaprus_connection_next(
|
||||||
|
connection: ?*ZaprusConnection,
|
||||||
|
out: [*c]u8,
|
||||||
|
capacity: usize,
|
||||||
|
out_len: *usize,
|
||||||
|
) c_int {
|
||||||
|
const c: ?*zaprus.Connection = @ptrCast(@alignCast(connection));
|
||||||
|
const zc = c orelse return 1;
|
||||||
|
|
||||||
|
const result = zc.next(io, out[0..capacity]) catch return 1;
|
||||||
|
out_len.* = result.len;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
export fn zaprus_connection_send(
|
||||||
|
connection: ?*ZaprusConnection,
|
||||||
|
payload: [*c]const u8,
|
||||||
|
payload_len: usize,
|
||||||
|
) c_int {
|
||||||
|
const c: ?*zaprus.Connection = @ptrCast(@alignCast(connection));
|
||||||
|
const zc = c orelse return 1;
|
||||||
|
|
||||||
|
zc.send(io, payload[0..payload_len]) catch return 1;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
70
src/main.zig
70
src/main.zig
@@ -43,7 +43,7 @@ pub fn main(init: std.process.Init) !void {
|
|||||||
if (to_option.get(args[i])) |opt| {
|
if (to_option.get(args[i])) |opt| {
|
||||||
switch (opt) {
|
switch (opt) {
|
||||||
.help => {
|
.help => {
|
||||||
std.debug.print("{s}\n", .{help});
|
std.debug.print("{s}", .{help});
|
||||||
return;
|
return;
|
||||||
},
|
},
|
||||||
.relay => {
|
.relay => {
|
||||||
@@ -66,12 +66,7 @@ pub fn main(init: std.process.Init) !void {
|
|||||||
.connect => {
|
.connect => {
|
||||||
i += 1;
|
i += 1;
|
||||||
if (i < args.len) {
|
if (i < args.len) {
|
||||||
var w: Writer = blk: {
|
flags.connect = args[i];
|
||||||
var buf: [2048]u8 = undefined;
|
|
||||||
break :blk .fixed(&buf);
|
|
||||||
};
|
|
||||||
try w.printBase64(args[i]);
|
|
||||||
flags.connect = w.buffered();
|
|
||||||
} else {
|
} else {
|
||||||
flags.connect = "";
|
flags.connect = "";
|
||||||
}
|
}
|
||||||
@@ -89,10 +84,11 @@ pub fn main(init: std.process.Init) !void {
|
|||||||
return error.InvalidArguments;
|
return error.InvalidArguments;
|
||||||
}
|
}
|
||||||
|
|
||||||
var client: SaprusClient = try .init();
|
var client: SaprusClient = undefined;
|
||||||
defer client.deinit();
|
|
||||||
|
|
||||||
if (flags.relay != null) {
|
if (flags.relay != null) {
|
||||||
|
client = try .init();
|
||||||
|
defer client.deinit();
|
||||||
var chunk_writer_buf: [2048]u8 = undefined;
|
var chunk_writer_buf: [2048]u8 = undefined;
|
||||||
var chunk_writer: Writer = .fixed(&chunk_writer_buf);
|
var chunk_writer: Writer = .fixed(&chunk_writer_buf);
|
||||||
if (flags.relay.?.len > 0) {
|
if (flags.relay.?.len > 0) {
|
||||||
@@ -101,7 +97,7 @@ pub fn main(init: std.process.Init) !void {
|
|||||||
chunk_writer.end = 0;
|
chunk_writer.end = 0;
|
||||||
try chunk_writer.print("{b64}", .{chunk});
|
try chunk_writer.print("{b64}", .{chunk});
|
||||||
try client.sendRelay(init.io, chunk_writer.buffered(), parseDest(flags.dest));
|
try client.sendRelay(init.io, chunk_writer.buffered(), parseDest(flags.dest));
|
||||||
try init.io.sleep(.fromMilliseconds(40), .real);
|
try init.io.sleep(.fromMilliseconds(40), .boot);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
var stdin_file: std.Io.File = .stdin();
|
var stdin_file: std.Io.File = .stdin();
|
||||||
@@ -112,10 +108,14 @@ pub fn main(init: std.process.Init) !void {
|
|||||||
var stdin = &limited.interface;
|
var stdin = &limited.interface;
|
||||||
|
|
||||||
while (stdin.fillMore()) {
|
while (stdin.fillMore()) {
|
||||||
|
// Sometimes fillMore will return 0 bytes.
|
||||||
|
// Skip these
|
||||||
|
if (stdin.seek == stdin.end) continue;
|
||||||
|
|
||||||
chunk_writer.end = 0;
|
chunk_writer.end = 0;
|
||||||
try chunk_writer.print("{b64}", .{stdin.buffered()});
|
try chunk_writer.print("{b64}", .{stdin.buffered()});
|
||||||
try client.sendRelay(init.io, chunk_writer.buffered(), parseDest(flags.dest));
|
try client.sendRelay(init.io, chunk_writer.buffered(), parseDest(flags.dest));
|
||||||
try init.io.sleep(.fromMilliseconds(40), .real);
|
try init.io.sleep(.fromMilliseconds(40), .boot);
|
||||||
try stdin.discardAll(stdin.end);
|
try stdin.discardAll(stdin.end);
|
||||||
} else |err| switch (err) {
|
} else |err| switch (err) {
|
||||||
error.EndOfStream => {
|
error.EndOfStream => {
|
||||||
@@ -127,21 +127,39 @@ pub fn main(init: std.process.Init) !void {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var retry_seconds: u16 = 12 * if (is_debug) 1 else 10;
|
||||||
|
|
||||||
|
var init_con_buf: [SaprusClient.max_payload_len]u8 = undefined;
|
||||||
|
var w: Writer = .fixed(&init_con_buf);
|
||||||
|
try w.print("{b64}", .{flags.connect.?});
|
||||||
|
|
||||||
if (flags.connect != null) {
|
if (flags.connect != null) {
|
||||||
reconnect: while (true) {
|
reconnect: while (true) {
|
||||||
|
client = try .init();
|
||||||
|
defer client.deinit();
|
||||||
log.debug("Starting connection", .{});
|
log.debug("Starting connection", .{});
|
||||||
var connection = try client.connect(init.io, flags.connect.?);
|
|
||||||
|
var connection = client.connect(init.io, w.buffered()) catch {
|
||||||
|
try init.io.sleep(.fromSeconds(retry_seconds), .boot);
|
||||||
|
continue;
|
||||||
|
};
|
||||||
|
|
||||||
|
retry_seconds = 60 * if (is_debug) 1 else 10;
|
||||||
|
|
||||||
log.debug("Connection started", .{});
|
log.debug("Connection started", .{});
|
||||||
|
|
||||||
while (true) {
|
next_message: while (true) {
|
||||||
var res_buf: [2048]u8 = undefined;
|
var res_buf: [2048]u8 = undefined;
|
||||||
const next = connection.next(init.io, &res_buf) catch continue :reconnect;
|
const next = connection.next(init.io, &res_buf) catch {
|
||||||
|
try init.io.sleep(.fromSeconds(retry_seconds), .boot);
|
||||||
|
continue :reconnect;
|
||||||
|
};
|
||||||
|
|
||||||
const b64d = std.base64.standard.Decoder;
|
const b64d = std.base64.standard.Decoder;
|
||||||
var connection_payload_buf: [2048]u8 = undefined;
|
var connection_payload_buf: [2048]u8 = undefined;
|
||||||
const connection_payload = connection_payload_buf[0..try b64d.calcSizeForSlice(next)];
|
const connection_payload = connection_payload_buf[0..try b64d.calcSizeForSlice(next)];
|
||||||
b64d.decode(connection_payload, next) catch {
|
b64d.decode(connection_payload, next) catch {
|
||||||
// TODO: debug log
|
log.debug("Failed to decode message, skipping: '{s}'", .{connection_payload});
|
||||||
continue;
|
continue;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -156,20 +174,24 @@ pub fn main(init: std.process.Init) !void {
|
|||||||
var child_stderr: std.ArrayList(u8) = .empty;
|
var child_stderr: std.ArrayList(u8) = .empty;
|
||||||
defer child_stderr.deinit(init.gpa);
|
defer child_stderr.deinit(init.gpa);
|
||||||
|
|
||||||
try child.collectOutput(init.gpa, &child_stdout, &child_stderr, std.math.maxInt(usize));
|
child.collectOutput(init.gpa, &child_stdout, &child_stderr, std.math.maxInt(usize)) catch |err| {
|
||||||
|
log.debug("Failed to collect output: {t}", .{err});
|
||||||
// const b64e = std.base64.standard.Encoder;
|
continue;
|
||||||
var cmd_output: Writer = blk: {
|
|
||||||
var cmd_output_buf: [2048]u8 = undefined;
|
|
||||||
break :blk .fixed(&cmd_output_buf);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
var cmd_output_buf: [SaprusClient.max_payload_len * 2]u8 = undefined;
|
||||||
|
var cmd_output: Writer = .fixed(&cmd_output_buf);
|
||||||
|
|
||||||
var cmd_output_window_iter = std.mem.window(u8, child_stdout.items, SaprusClient.max_payload_len, SaprusClient.max_payload_len);
|
var cmd_output_window_iter = std.mem.window(u8, child_stdout.items, SaprusClient.max_payload_len, SaprusClient.max_payload_len);
|
||||||
while (cmd_output_window_iter.next()) |chunk| {
|
while (cmd_output_window_iter.next()) |chunk| {
|
||||||
cmd_output.end = 0;
|
cmd_output.end = 0;
|
||||||
try cmd_output.print("{b64}", .{chunk});
|
// Unreachable because the cmd_output_buf is twice the size of the chunk.
|
||||||
try connection.send(init.io, cmd_output.buffered());
|
cmd_output.print("{b64}", .{chunk}) catch unreachable;
|
||||||
try init.io.sleep(.fromMilliseconds(40), .real);
|
connection.send(init.io, cmd_output.buffered()) catch |err| {
|
||||||
|
log.debug("Failed to send connection chunk: {t}", .{err});
|
||||||
|
continue :next_message;
|
||||||
|
};
|
||||||
|
try init.io.sleep(.fromMilliseconds(40), .boot);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user