mirror of
https://github.com/lightpanda-io/browser.git
synced 2026-03-22 04:34:44 +00:00
Used ring buffer for telemetry events buffer
This commit is contained in:
@@ -67,7 +67,7 @@ pub fn init(allocator: Allocator, config: *const Config) !*App {
|
|||||||
app.app_dir_path = getAndMakeAppDir(allocator);
|
app.app_dir_path = getAndMakeAppDir(allocator);
|
||||||
|
|
||||||
app.telemetry = try Telemetry.init(app, config.mode);
|
app.telemetry = try Telemetry.init(app, config.mode);
|
||||||
errdefer app.telemetry.deinit();
|
errdefer app.telemetry.deinit(allocator);
|
||||||
|
|
||||||
app.arena_pool = ArenaPool.init(allocator, 512, 1024 * 16);
|
app.arena_pool = ArenaPool.init(allocator, 512, 1024 * 16);
|
||||||
errdefer app.arena_pool.deinit();
|
errdefer app.arena_pool.deinit();
|
||||||
@@ -85,7 +85,7 @@ pub fn deinit(self: *App) void {
|
|||||||
allocator.free(app_dir_path);
|
allocator.free(app_dir_path);
|
||||||
self.app_dir_path = null;
|
self.app_dir_path = null;
|
||||||
}
|
}
|
||||||
self.telemetry.deinit();
|
self.telemetry.deinit(allocator);
|
||||||
self.network.deinit();
|
self.network.deinit();
|
||||||
self.snapshot.deinit();
|
self.snapshot.deinit();
|
||||||
self.platform.deinit();
|
self.platform.deinit();
|
||||||
|
|||||||
@@ -41,6 +41,8 @@ const Listener = struct {
|
|||||||
// Number of fixed pollfds entries (wakeup pipe + listener).
|
// Number of fixed pollfds entries (wakeup pipe + listener).
|
||||||
const PSEUDO_POLLFDS = 2;
|
const PSEUDO_POLLFDS = 2;
|
||||||
|
|
||||||
|
const MAX_TICK_CALLBACKS = 16;
|
||||||
|
|
||||||
allocator: Allocator,
|
allocator: Allocator,
|
||||||
|
|
||||||
config: *const Config,
|
config: *const Config,
|
||||||
@@ -67,6 +69,15 @@ multi: ?*libcurl.CurlM = null,
|
|||||||
submission_mutex: std.Thread.Mutex = .{},
|
submission_mutex: std.Thread.Mutex = .{},
|
||||||
submission_queue: std.DoublyLinkedList = .{},
|
submission_queue: std.DoublyLinkedList = .{},
|
||||||
|
|
||||||
|
callbacks: [MAX_TICK_CALLBACKS]TickCallback = undefined,
|
||||||
|
callbacks_len: usize = 0,
|
||||||
|
callbacks_mutex: std.Thread.Mutex = .{},
|
||||||
|
|
||||||
|
const TickCallback = struct {
|
||||||
|
ctx: *anyopaque,
|
||||||
|
fun: *const fn (*anyopaque) void,
|
||||||
|
};
|
||||||
|
|
||||||
const ZigToCurlAllocator = struct {
|
const ZigToCurlAllocator = struct {
|
||||||
// C11 requires malloc to return memory aligned to max_align_t (16 bytes on x86_64).
|
// C11 requires malloc to return memory aligned to max_align_t (16 bytes on x86_64).
|
||||||
// We match this guarantee since libcurl expects malloc-compatible alignment.
|
// We match this guarantee since libcurl expects malloc-compatible alignment.
|
||||||
@@ -302,6 +313,30 @@ pub fn bind(
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn onTick(self: *Runtime, ctx: *anyopaque, callback: *const fn (*anyopaque) void) void {
|
||||||
|
self.callbacks_mutex.lock();
|
||||||
|
defer self.callbacks_mutex.unlock();
|
||||||
|
|
||||||
|
lp.assert(self.callbacks_len < MAX_TICK_CALLBACKS, "too many ticks", .{});
|
||||||
|
|
||||||
|
self.callbacks[self.callbacks_len] = .{
|
||||||
|
.ctx = ctx,
|
||||||
|
.fun = callback,
|
||||||
|
};
|
||||||
|
self.callbacks_len += 1;
|
||||||
|
|
||||||
|
self.wakeupPoll();
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn fireTicks(self: *Runtime) void {
|
||||||
|
self.callbacks_mutex.lock();
|
||||||
|
defer self.callbacks_mutex.unlock();
|
||||||
|
|
||||||
|
for (self.callbacks[0..self.callbacks_len]) |*callback| {
|
||||||
|
callback.fun(callback.ctx);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub fn run(self: *Runtime) void {
|
pub fn run(self: *Runtime) void {
|
||||||
var drain_buf: [64]u8 = undefined;
|
var drain_buf: [64]u8 = undefined;
|
||||||
var running_handles: c_int = 0;
|
var running_handles: c_int = 0;
|
||||||
@@ -326,7 +361,20 @@ pub fn run(self: *Runtime) void {
|
|||||||
self.preparePollFds(multi);
|
self.preparePollFds(multi);
|
||||||
}
|
}
|
||||||
|
|
||||||
const timeout = if (self.multi != null) self.getCurlTimeout() else @as(i32, -1);
|
// for ontick to work, you need to wake up periodically
|
||||||
|
const timeout = blk: {
|
||||||
|
const min_timeout = 250; // 250ms
|
||||||
|
if (self.multi == null) {
|
||||||
|
break :blk min_timeout;
|
||||||
|
}
|
||||||
|
|
||||||
|
const curl_timeout = self.getCurlTimeout();
|
||||||
|
if (curl_timeout == 0) {
|
||||||
|
break :blk min_timeout;
|
||||||
|
}
|
||||||
|
|
||||||
|
break :blk @min(min_timeout, curl_timeout);
|
||||||
|
};
|
||||||
|
|
||||||
_ = posix.poll(self.pollfds, timeout) catch |err| {
|
_ = posix.poll(self.pollfds, timeout) catch |err| {
|
||||||
lp.log.err(.app, "poll", .{ .err = err });
|
lp.log.err(.app, "poll", .{ .err = err });
|
||||||
@@ -354,8 +402,16 @@ pub fn run(self: *Runtime) void {
|
|||||||
self.processCompletions(multi);
|
self.processCompletions(multi);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (self.shutdown.load(.acquire) and running_handles == 0)
|
self.fireTicks();
|
||||||
break;
|
|
||||||
|
if (self.shutdown.load(.acquire) and running_handles == 0) {
|
||||||
|
// Check if fireTicks submitted new requests (e.g. telemetry flush).
|
||||||
|
// If so, continue the loop to drain and send them before exiting.
|
||||||
|
self.submission_mutex.lock();
|
||||||
|
const has_pending = self.submission_queue.first != null;
|
||||||
|
self.submission_mutex.unlock();
|
||||||
|
if (!has_pending) break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (self.listener) |listener| {
|
if (self.listener) |listener| {
|
||||||
|
|||||||
@@ -11,84 +11,97 @@ const telemetry = @import("telemetry.zig");
|
|||||||
const Runtime = @import("../network/Runtime.zig");
|
const Runtime = @import("../network/Runtime.zig");
|
||||||
const Connection = @import("../network/http.zig").Connection;
|
const Connection = @import("../network/http.zig").Connection;
|
||||||
|
|
||||||
const URL = "https://telemetry.lightpanda.io";
|
// const URL = "https://telemetry.lightpanda.io";
|
||||||
const BATCH_SIZE = 20;
|
const URL = "http://localhost:9876";
|
||||||
const BUFFER_SIZE = BATCH_SIZE * 2;
|
const BUFFER_SIZE = 1024;
|
||||||
|
|
||||||
const LightPanda = @This();
|
const LightPanda = @This();
|
||||||
|
|
||||||
allocator: Allocator,
|
allocator: Allocator,
|
||||||
runtime: *Runtime,
|
runtime: *Runtime,
|
||||||
|
|
||||||
|
/// Protects concurrent producers in send().
|
||||||
mutex: std.Thread.Mutex = .{},
|
mutex: std.Thread.Mutex = .{},
|
||||||
|
|
||||||
pcount: usize = 0,
|
iid: ?[36]u8 = null,
|
||||||
pending: [BUFFER_SIZE]LightPandaEvent = undefined,
|
run_mode: Config.RunMode = .serve,
|
||||||
|
|
||||||
pub fn init(app: *App) !LightPanda {
|
head: std.atomic.Value(usize) = .init(0),
|
||||||
return .{
|
tail: std.atomic.Value(usize) = .init(0),
|
||||||
|
dropped: std.atomic.Value(usize) = .init(0),
|
||||||
|
buffer: [BUFFER_SIZE]telemetry.Event = undefined,
|
||||||
|
|
||||||
|
pub fn init(self: *LightPanda, app: *App, iid: ?[36]u8, run_mode: Config.RunMode) !void {
|
||||||
|
self.* = .{
|
||||||
.allocator = app.allocator,
|
.allocator = app.allocator,
|
||||||
.runtime = &app.network,
|
.runtime = &app.network,
|
||||||
|
.iid = iid,
|
||||||
|
.run_mode = run_mode,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
self.runtime.onTick(@ptrCast(self), flushCallback);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn deinit(self: *LightPanda) void {
|
pub fn deinit(_: *LightPanda) void {}
|
||||||
self.flush();
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn send(self: *LightPanda, iid: ?[]const u8, run_mode: Config.RunMode, raw_event: telemetry.Event) !void {
|
pub fn send(self: *LightPanda, raw_event: telemetry.Event) !void {
|
||||||
const pending_count = blk: {
|
|
||||||
self.mutex.lock();
|
self.mutex.lock();
|
||||||
defer self.mutex.unlock();
|
defer self.mutex.unlock();
|
||||||
|
|
||||||
if (self.pcount == BUFFER_SIZE) {
|
const t = self.tail.load(.monotonic);
|
||||||
log.err(.telemetry, "telemetry buffer exhausted", .{});
|
const h = self.head.load(.acquire);
|
||||||
|
if (t - h >= BUFFER_SIZE) {
|
||||||
|
_ = self.dropped.fetchAdd(1, .monotonic);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
self.pending[self.pcount] = .{
|
self.buffer[t % BUFFER_SIZE] = raw_event;
|
||||||
.iid = iid,
|
self.tail.store(t + 1, .release);
|
||||||
.mode = run_mode,
|
|
||||||
.event = raw_event,
|
|
||||||
};
|
|
||||||
self.pcount += 1;
|
|
||||||
|
|
||||||
break :blk self.pcount;
|
|
||||||
};
|
|
||||||
|
|
||||||
if (pending_count >= BATCH_SIZE) {
|
|
||||||
self.flush();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn flush(self: *LightPanda) void {
|
fn flushCallback(ctx: *anyopaque) void {
|
||||||
|
const self: *LightPanda = @ptrCast(@alignCast(ctx));
|
||||||
self.postEvent() catch |err| {
|
self.postEvent() catch |err| {
|
||||||
log.warn(.telemetry, "flush error", .{ .err = err });
|
log.warn(.telemetry, "flush error", .{ .err = err });
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
fn postEvent(self: *LightPanda) !void {
|
fn postEvent(self: *LightPanda) !void {
|
||||||
|
const h = self.head.load(.monotonic);
|
||||||
|
const t = self.tail.load(.acquire);
|
||||||
|
const dropped = self.dropped.swap(0, .monotonic);
|
||||||
|
|
||||||
|
if (h == t and dropped == 0) return;
|
||||||
|
errdefer _ = self.dropped.fetchAdd(dropped, .monotonic);
|
||||||
|
|
||||||
var writer = std.Io.Writer.Allocating.init(self.allocator);
|
var writer = std.Io.Writer.Allocating.init(self.allocator);
|
||||||
defer writer.deinit();
|
defer writer.deinit();
|
||||||
|
|
||||||
self.mutex.lock();
|
const iid: ?[]const u8 = if (self.iid) |*id| id else null;
|
||||||
defer self.mutex.unlock();
|
|
||||||
|
|
||||||
const events = self.pending[0..self.pcount];
|
for (h..t) |i| {
|
||||||
if (events.len == 0) return;
|
const wrapped = LightPandaEvent{ .iid = iid, .mode = self.run_mode, .event = self.buffer[i % BUFFER_SIZE] };
|
||||||
|
try std.json.Stringify.value(&wrapped, .{ .emit_null_optional_fields = false }, &writer.writer);
|
||||||
for (events) |*event| {
|
|
||||||
try std.json.Stringify.value(event, .{ .emit_null_optional_fields = false }, &writer.writer);
|
|
||||||
try writer.writer.writeByte('\n');
|
try writer.writer.writeByte('\n');
|
||||||
}
|
}
|
||||||
|
|
||||||
const conn = self.runtime.getConnection() orelse return;
|
if (dropped > 0) {
|
||||||
|
const wrapped = LightPandaEvent{ .iid = iid, .mode = self.run_mode, .event = .{ .buffer_overflow = .{ .dropped = dropped } } };
|
||||||
|
try std.json.Stringify.value(&wrapped, .{ .emit_null_optional_fields = false }, &writer.writer);
|
||||||
|
try writer.writer.writeByte('\n');
|
||||||
|
}
|
||||||
|
|
||||||
|
const conn = self.runtime.getConnection() orelse {
|
||||||
|
_ = self.dropped.fetchAdd(dropped, .monotonic);
|
||||||
|
return;
|
||||||
|
};
|
||||||
errdefer self.runtime.releaseConnection(conn);
|
errdefer self.runtime.releaseConnection(conn);
|
||||||
|
|
||||||
try conn.setURL(URL);
|
try conn.setURL(URL);
|
||||||
try conn.setMethod(.POST);
|
try conn.setMethod(.POST);
|
||||||
try conn.setBody(writer.written());
|
try conn.setBody(writer.written());
|
||||||
|
|
||||||
self.pcount = 0;
|
self.head.store(t, .release);
|
||||||
self.runtime.submitRequest(conn);
|
self.runtime.submitRequest(conn);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -11,26 +11,21 @@ const uuidv4 = @import("../id.zig").uuidv4;
|
|||||||
const IID_FILE = "iid";
|
const IID_FILE = "iid";
|
||||||
|
|
||||||
pub fn isDisabled() bool {
|
pub fn isDisabled() bool {
|
||||||
|
if (builtin.mode == .Debug or builtin.is_test) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
return std.process.hasEnvVarConstant("LIGHTPANDA_DISABLE_TELEMETRY");
|
return std.process.hasEnvVarConstant("LIGHTPANDA_DISABLE_TELEMETRY");
|
||||||
}
|
}
|
||||||
|
|
||||||
pub const Telemetry = TelemetryT(blk: {
|
pub const Telemetry = TelemetryT(@import("lightpanda.zig"));
|
||||||
if (builtin.mode == .Debug or builtin.is_test) break :blk NoopProvider;
|
|
||||||
break :blk @import("lightpanda.zig");
|
|
||||||
});
|
|
||||||
|
|
||||||
fn TelemetryT(comptime P: type) type {
|
fn TelemetryT(comptime P: type) type {
|
||||||
return struct {
|
return struct {
|
||||||
// an "install" id that we [try to] persist and re-use between runs
|
provider: *P,
|
||||||
// null on IO error
|
|
||||||
iid: ?[36]u8,
|
|
||||||
|
|
||||||
provider: P,
|
|
||||||
|
|
||||||
disabled: bool,
|
disabled: bool,
|
||||||
|
|
||||||
run_mode: Config.RunMode,
|
|
||||||
|
|
||||||
const Self = @This();
|
const Self = @This();
|
||||||
|
|
||||||
pub fn init(app: *App, run_mode: Config.RunMode) !Self {
|
pub fn init(app: *App, run_mode: Config.RunMode) !Self {
|
||||||
@@ -39,31 +34,29 @@ fn TelemetryT(comptime P: type) type {
|
|||||||
log.info(.telemetry, "telemetry status", .{ .disabled = disabled });
|
log.info(.telemetry, "telemetry status", .{ .disabled = disabled });
|
||||||
}
|
}
|
||||||
|
|
||||||
const provider = try P.init(app);
|
const iid: ?[36]u8 = if (disabled) null else getOrCreateId(app.app_dir_path);
|
||||||
errdefer provider.deinit();
|
|
||||||
|
const provider = try app.allocator.create(P);
|
||||||
|
errdefer app.allocator.destroy(provider);
|
||||||
|
|
||||||
|
try P.init(provider, app, iid, run_mode);
|
||||||
|
|
||||||
return .{
|
return .{
|
||||||
.disabled = disabled,
|
.disabled = disabled,
|
||||||
.run_mode = run_mode,
|
|
||||||
.provider = provider,
|
.provider = provider,
|
||||||
.iid = if (disabled) null else getOrCreateId(app.app_dir_path),
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn flush(self: *Self) void {
|
pub fn deinit(self: *Self, allocator: Allocator) void {
|
||||||
self.provider.flush();
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn deinit(self: *Self) void {
|
|
||||||
self.provider.deinit();
|
self.provider.deinit();
|
||||||
|
allocator.destroy(self.provider);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn record(self: *Self, event: Event) void {
|
pub fn record(self: *Self, event: Event) void {
|
||||||
if (self.disabled) {
|
if (self.disabled) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const iid: ?[]const u8 = if (self.iid) |*iid| iid else null;
|
self.provider.send(event) catch |err| {
|
||||||
self.provider.send(iid, self.run_mode, event) catch |err| {
|
|
||||||
log.warn(.telemetry, "record error", .{ .err = err, .type = @tagName(std.meta.activeTag(event)) });
|
log.warn(.telemetry, "record error", .{ .err = err, .type = @tagName(std.meta.activeTag(event)) });
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
@@ -109,6 +102,7 @@ fn getOrCreateId(app_dir_path_: ?[]const u8) ?[36]u8 {
|
|||||||
pub const Event = union(enum) {
|
pub const Event = union(enum) {
|
||||||
run: void,
|
run: void,
|
||||||
navigate: Navigate,
|
navigate: Navigate,
|
||||||
|
buffer_overflow: BufferOverflow,
|
||||||
flag: []const u8, // used for testing
|
flag: []const u8, // used for testing
|
||||||
|
|
||||||
const Navigate = struct {
|
const Navigate = struct {
|
||||||
@@ -116,37 +110,35 @@ pub const Event = union(enum) {
|
|||||||
proxy: bool,
|
proxy: bool,
|
||||||
driver: []const u8 = "cdp",
|
driver: []const u8 = "cdp",
|
||||||
};
|
};
|
||||||
};
|
|
||||||
|
|
||||||
const NoopProvider = struct {
|
const BufferOverflow = struct {
|
||||||
fn init(_: *App) !NoopProvider {
|
dropped: usize,
|
||||||
return .{};
|
};
|
||||||
}
|
|
||||||
fn flush(_: NoopProvider) void {}
|
|
||||||
fn deinit(_: NoopProvider) void {}
|
|
||||||
pub fn send(_: NoopProvider, _: ?[]const u8, _: Config.RunMode, _: Event) !void {}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
extern fn setenv(name: [*:0]u8, value: [*:0]u8, override: c_int) c_int;
|
extern fn setenv(name: [*:0]u8, value: [*:0]u8, override: c_int) c_int;
|
||||||
extern fn unsetenv(name: [*:0]u8) c_int;
|
extern fn unsetenv(name: [*:0]u8) c_int;
|
||||||
|
|
||||||
const testing = @import("../testing.zig");
|
const testing = @import("../testing.zig");
|
||||||
test "telemetry: disabled by environment" {
|
test "telemetry: always disabled in debug builds" {
|
||||||
|
// Must be disabled regardless of environment variable.
|
||||||
|
_ = unsetenv(@constCast("LIGHTPANDA_DISABLE_TELEMETRY"));
|
||||||
|
try testing.expectEqual(true, isDisabled());
|
||||||
|
|
||||||
_ = setenv(@constCast("LIGHTPANDA_DISABLE_TELEMETRY"), @constCast(""), 0);
|
_ = setenv(@constCast("LIGHTPANDA_DISABLE_TELEMETRY"), @constCast(""), 0);
|
||||||
defer _ = unsetenv(@constCast("LIGHTPANDA_DISABLE_TELEMETRY"));
|
defer _ = unsetenv(@constCast("LIGHTPANDA_DISABLE_TELEMETRY"));
|
||||||
|
try testing.expectEqual(true, isDisabled());
|
||||||
|
|
||||||
const FailingProvider = struct {
|
const FailingProvider = struct {
|
||||||
fn init(_: *App) !@This() {
|
fn init(_: *@This(), _: *App, _: ?[36]u8, _: Config.RunMode) !void {}
|
||||||
return .{};
|
fn deinit(_: *@This()) void {}
|
||||||
}
|
pub fn send(_: *@This(), _: Event) !void {
|
||||||
fn deinit(_: @This()) void {}
|
|
||||||
pub fn send(_: @This(), _: ?[]const u8, _: Config.RunMode, _: Event) !void {
|
|
||||||
unreachable;
|
unreachable;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
var telemetry = try TelemetryT(FailingProvider).init(undefined, .serve);
|
var telemetry = try TelemetryT(FailingProvider).init(testing.test_app, .serve);
|
||||||
defer telemetry.deinit();
|
defer telemetry.deinit(testing.test_app.allocator);
|
||||||
telemetry.record(.{ .run = {} });
|
telemetry.record(.{ .run = {} });
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -170,8 +162,9 @@ test "telemetry: getOrCreateId" {
|
|||||||
|
|
||||||
test "telemetry: sends event to provider" {
|
test "telemetry: sends event to provider" {
|
||||||
var telemetry = try TelemetryT(MockProvider).init(testing.test_app, .serve);
|
var telemetry = try TelemetryT(MockProvider).init(testing.test_app, .serve);
|
||||||
defer telemetry.deinit();
|
defer telemetry.deinit(testing.test_app.allocator);
|
||||||
const mock = &telemetry.provider;
|
telemetry.disabled = false;
|
||||||
|
const mock = telemetry.provider;
|
||||||
|
|
||||||
telemetry.record(.{ .flag = "1" });
|
telemetry.record(.{ .flag = "1" });
|
||||||
telemetry.record(.{ .flag = "2" });
|
telemetry.record(.{ .flag = "2" });
|
||||||
@@ -184,32 +177,19 @@ test "telemetry: sends event to provider" {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const MockProvider = struct {
|
const MockProvider = struct {
|
||||||
iid: ?[]const u8,
|
|
||||||
run_mode: ?Config.RunMode,
|
|
||||||
allocator: Allocator,
|
allocator: Allocator,
|
||||||
events: std.ArrayList(Event),
|
events: std.ArrayList(Event),
|
||||||
|
|
||||||
fn init(app: *App) !@This() {
|
fn init(self: *MockProvider, app: *App, _: ?[36]u8, _: Config.RunMode) !void {
|
||||||
return .{
|
self.* = .{
|
||||||
.iid = null,
|
|
||||||
.run_mode = null,
|
|
||||||
.events = .{},
|
.events = .{},
|
||||||
.allocator = app.allocator,
|
.allocator = app.allocator,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
fn flush(_: *MockProvider) void {}
|
|
||||||
fn deinit(self: *MockProvider) void {
|
fn deinit(self: *MockProvider) void {
|
||||||
self.events.deinit(self.allocator);
|
self.events.deinit(self.allocator);
|
||||||
}
|
}
|
||||||
pub fn send(self: *MockProvider, iid: ?[]const u8, run_mode: Config.RunMode, events: Event) !void {
|
pub fn send(self: *MockProvider, event: Event) !void {
|
||||||
if (self.iid == null) {
|
try self.events.append(self.allocator, event);
|
||||||
try testing.expectEqual(null, self.run_mode);
|
|
||||||
self.iid = iid.?;
|
|
||||||
self.run_mode = run_mode;
|
|
||||||
} else {
|
|
||||||
try testing.expectEqual(self.iid.?, iid.?);
|
|
||||||
try testing.expectEqual(self.run_mode.?, run_mode);
|
|
||||||
}
|
|
||||||
try self.events.append(self.allocator, events);
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|||||||
Reference in New Issue
Block a user