2023-12-31 00:14:30 +08:00
|
|
|
const std = @import("std");
|
|
|
|
const builtin = @import("builtin");
|
|
|
|
|
|
|
|
const Build = std.Build;
|
|
|
|
const OptimizeMode = std.builtin.OptimizeMode;
|
|
|
|
const Compile = Build.Step.Compile;
|
|
|
|
const Module = Build.Module;
|
|
|
|
|
2024-05-27 04:31:42 +02:00
|
|
|
const lib_name = "webui";
|
2024-05-29 23:27:29 +02:00
|
|
|
const zig_ver = builtin.zig_version.minor;
|
2024-10-12 13:27:50 +09:00
|
|
|
var global_log_level: std.log.Level = .warn;
|
2023-12-31 00:14:30 +08:00
|
|
|
|
2024-05-29 23:27:29 +02:00
|
|
|
pub fn build(b: *Build) !void {
|
2024-05-03 21:15:13 +08:00
|
|
|
const target = b.standardTargetOptions(.{});
|
|
|
|
const optimize = b.standardOptimizeOption(.{});
|
2024-05-29 23:27:29 +02:00
|
|
|
|
2024-05-27 04:31:42 +02:00
|
|
|
const is_dynamic = b.option(bool, "dynamic", "build the dynamic library") orelse false;
|
|
|
|
const enable_tls = b.option(bool, "enable-tls", "enable TLS support") orelse false;
|
2024-10-12 13:27:50 +09:00
|
|
|
const verbose = b.option(std.log.Level, "verbose", "set verbose output") orelse .warn;
|
2024-05-03 21:15:13 +08:00
|
|
|
|
2024-10-12 13:27:50 +09:00
|
|
|
global_log_level = verbose;
|
|
|
|
|
|
|
|
switch (comptime zig_ver) {
|
2024-05-29 23:27:29 +02:00
|
|
|
11 => {
|
|
|
|
if (enable_tls and !target.isNative()) {
|
2024-10-12 13:27:50 +09:00
|
|
|
log(.err, .WebUI, "cross compilation is not supported with TLS enabled", .{});
|
|
|
|
return error.InvalidBuildConfiguration;
|
2024-05-29 23:27:29 +02:00
|
|
|
}
|
|
|
|
},
|
2024-06-09 11:35:29 +08:00
|
|
|
12, 13, 14 => {
|
2024-05-29 23:27:29 +02:00
|
|
|
if (enable_tls and !target.query.isNative()) {
|
2024-10-12 13:27:50 +09:00
|
|
|
log(.err, .WebUI, "cross compilation is not supported with TLS enabled", .{});
|
|
|
|
return error.InvalidBuildConfiguration;
|
2024-05-29 23:27:29 +02:00
|
|
|
}
|
|
|
|
},
|
2024-10-12 13:27:50 +09:00
|
|
|
else => return error.UnsupportedZigVersion,
|
2024-05-27 04:31:42 +02:00
|
|
|
}
|
2024-05-03 21:15:13 +08:00
|
|
|
|
2024-10-12 13:27:50 +09:00
|
|
|
log(.info, .WebUI, "Building {s} WebUI library{s}...", .{
|
|
|
|
if (is_dynamic) "dynamic" else "static",
|
|
|
|
if (enable_tls) " with TLS support" else "",
|
|
|
|
});
|
|
|
|
defer {
|
|
|
|
log(.info, .WebUI, "Done.", .{});
|
2024-05-29 18:10:29 +02:00
|
|
|
}
|
2024-05-27 05:18:21 +02:00
|
|
|
|
2024-05-27 04:31:42 +02:00
|
|
|
const webui = if (is_dynamic) b.addSharedLibrary(.{
|
|
|
|
.name = lib_name,
|
|
|
|
.target = target,
|
|
|
|
.optimize = optimize,
|
|
|
|
}) else b.addStaticLibrary(.{
|
|
|
|
.name = lib_name,
|
|
|
|
.target = target,
|
|
|
|
.optimize = optimize,
|
|
|
|
});
|
2024-06-01 05:38:28 +02:00
|
|
|
try addLinkerFlags(b, webui, enable_tls);
|
2024-05-29 23:27:29 +02:00
|
|
|
|
|
|
|
b.installArtifact(webui);
|
|
|
|
|
2024-10-12 13:27:50 +09:00
|
|
|
try build_examples(b, webui);
|
2024-05-29 23:27:29 +02:00
|
|
|
}
|
|
|
|
|
2024-06-01 05:38:28 +02:00
|
|
|
fn addLinkerFlags(b: *Build, webui: *Compile, enable_tls: bool) !void {
|
2024-05-29 23:27:29 +02:00
|
|
|
const webui_target = if (zig_ver < 12) webui.target else webui.rootModuleTarget();
|
|
|
|
const is_windows = if (zig_ver < 12) webui_target.isWindows() else webui_target.os.tag == .windows;
|
|
|
|
|
|
|
|
// Prepare compiler flags.
|
2024-10-12 13:27:50 +09:00
|
|
|
const no_tls_flags: []const []const u8 = &.{"-DNO_SSL"};
|
|
|
|
const tls_flags: []const []const u8 = &.{ "-DWEBUI_TLS", "-DNO_SSL_DL", "-DOPENSSL_API_1_1" };
|
|
|
|
const civetweb_flags: []const []const u8 = &.{
|
|
|
|
"-DNDEBUG",
|
|
|
|
"-DNO_CACHING",
|
|
|
|
"-DNO_CGI",
|
|
|
|
"-DUSE_WEBSOCKET",
|
|
|
|
"-Wno-error=date-time",
|
|
|
|
};
|
2024-05-03 21:15:13 +08:00
|
|
|
|
|
|
|
webui.addCSourceFile(.{
|
2024-06-01 05:38:28 +02:00
|
|
|
.file = if (zig_ver < 12) .{ .path = "src/webui.c" } else b.path("src/webui.c"),
|
2024-10-12 13:27:50 +09:00
|
|
|
.flags = if (enable_tls) tls_flags else no_tls_flags,
|
2024-05-03 21:15:13 +08:00
|
|
|
});
|
|
|
|
webui.addCSourceFile(.{
|
2024-06-01 05:38:28 +02:00
|
|
|
.file = if (zig_ver < 12) .{ .path = "src/civetweb/civetweb.c" } else b.path("src/civetweb/civetweb.c"),
|
2024-10-12 13:27:50 +09:00
|
|
|
.flags = if (enable_tls) civetweb_flags ++ tls_flags else civetweb_flags ++ .{"-DUSE_WEBSOCKET"} ++ no_tls_flags,
|
2024-05-03 21:15:13 +08:00
|
|
|
});
|
|
|
|
webui.linkLibC();
|
2024-06-01 05:38:28 +02:00
|
|
|
webui.addIncludePath(if (zig_ver < 12) .{ .path = "include" } else b.path("include"));
|
2024-05-29 23:27:29 +02:00
|
|
|
if (zig_ver < 12) {
|
|
|
|
webui.installHeader("include/webui.h", "webui.h");
|
|
|
|
} else {
|
2024-06-01 05:38:28 +02:00
|
|
|
webui.installHeader(b.path("include/webui.h"), "webui.h");
|
2024-05-29 23:27:29 +02:00
|
|
|
}
|
|
|
|
if (webui_target.isDarwin()) {
|
2024-05-28 03:39:33 +02:00
|
|
|
webui.addCSourceFile(.{
|
2024-06-01 05:38:28 +02:00
|
|
|
.file = if (zig_ver < 12) .{ .path = "src/webview/wkwebview.m" } else b.path("src/webview/wkwebview.m"),
|
2024-05-28 03:39:33 +02:00
|
|
|
.flags = &.{},
|
|
|
|
});
|
|
|
|
webui.linkFramework("Cocoa");
|
|
|
|
webui.linkFramework("WebKit");
|
2024-05-29 23:27:29 +02:00
|
|
|
} else if (is_windows) {
|
2024-05-03 21:15:13 +08:00
|
|
|
webui.linkSystemLibrary("ws2_32");
|
2024-07-08 08:57:19 +08:00
|
|
|
webui.linkSystemLibrary("ole32");
|
2024-05-29 23:27:29 +02:00
|
|
|
if (webui_target.abi == .msvc) {
|
2024-05-27 04:31:42 +02:00
|
|
|
webui.linkSystemLibrary("Advapi32");
|
2024-05-28 03:39:33 +02:00
|
|
|
webui.linkSystemLibrary("Shell32");
|
2024-05-27 04:31:42 +02:00
|
|
|
webui.linkSystemLibrary("user32");
|
|
|
|
}
|
2024-05-03 21:15:13 +08:00
|
|
|
if (enable_tls) {
|
|
|
|
webui.linkSystemLibrary("bcrypt");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (enable_tls) {
|
|
|
|
webui.linkSystemLibrary("ssl");
|
|
|
|
webui.linkSystemLibrary("crypto");
|
|
|
|
}
|
2024-10-12 13:27:50 +09:00
|
|
|
|
|
|
|
for (if (zig_ver < 12) webui.link_objects.items else webui.root_module.link_objects.items) |lo| {
|
|
|
|
switch (lo) {
|
|
|
|
.c_source_file => |csf| {
|
|
|
|
log(.debug, .WebUI, "{s} linker flags: {s}", .{
|
|
|
|
if (zig_ver < 12) csf.file.path else csf.file.src_path.sub_path,
|
|
|
|
csf.flags,
|
|
|
|
});
|
|
|
|
},
|
|
|
|
else => {},
|
|
|
|
}
|
|
|
|
}
|
2024-05-03 21:15:13 +08:00
|
|
|
}
|
|
|
|
|
2024-05-29 23:27:29 +02:00
|
|
|
fn build_examples(b: *Build, webui: *Compile) !void {
|
|
|
|
const build_examples_step = b.step("examples", "builds the library and its examples");
|
|
|
|
const target = if (zig_ver < 12) webui.target else webui.root_module.resolved_target.?;
|
|
|
|
const optimize = if (zig_ver < 12) webui.optimize else webui.root_module.optimize.?;
|
2024-05-03 21:15:13 +08:00
|
|
|
|
2024-06-01 05:38:28 +02:00
|
|
|
const examples_path = (if (zig_ver < 12) (Build.LazyPath{ .path = "examples/C" }) else b.path("examples/C")).getPath(b);
|
2024-05-29 23:27:29 +02:00
|
|
|
var examples_dir = if (zig_ver < 12)
|
2024-10-12 14:07:28 +09:00
|
|
|
std.fs.cwd().openIterableDir(examples_path, .{}) catch |e| switch (e) {
|
|
|
|
// Do not attempt building examples if directory does not exist.
|
|
|
|
error.FileNotFound => return,
|
|
|
|
else => return e,
|
|
|
|
}
|
2024-05-29 23:27:29 +02:00
|
|
|
else
|
2024-10-12 14:07:28 +09:00
|
|
|
std.fs.cwd().openDir(examples_path, .{ .iterate = true }) catch |e| switch (e) {
|
|
|
|
// Do not attempt building examples if directory does not exist.
|
|
|
|
error.FileNotFound => return,
|
|
|
|
else => return e,
|
|
|
|
};
|
2024-05-27 04:31:42 +02:00
|
|
|
defer examples_dir.close();
|
2024-05-03 21:15:13 +08:00
|
|
|
|
2024-05-27 04:31:42 +02:00
|
|
|
var paths = examples_dir.iterate();
|
|
|
|
while (try paths.next()) |val| {
|
|
|
|
if (val.kind != .directory) {
|
2024-05-26 05:00:14 +02:00
|
|
|
continue;
|
2024-05-03 21:15:13 +08:00
|
|
|
}
|
2024-05-27 04:31:42 +02:00
|
|
|
const example_name = val.name;
|
2024-10-12 13:27:50 +09:00
|
|
|
|
2024-05-27 04:31:42 +02:00
|
|
|
const exe = b.addExecutable(.{ .name = example_name, .target = target, .optimize = optimize });
|
|
|
|
const path = try std.fmt.allocPrint(b.allocator, "examples/C/{s}/main.c", .{example_name});
|
2024-10-12 13:27:50 +09:00
|
|
|
defer b.allocator.free(path);
|
2024-05-27 04:31:42 +02:00
|
|
|
|
2024-06-01 05:38:28 +02:00
|
|
|
exe.addCSourceFile(.{ .file = if (zig_ver < 12) .{ .path = path } else b.path(path), .flags = &.{} });
|
2024-05-29 23:27:29 +02:00
|
|
|
exe.linkLibrary(webui);
|
2024-05-26 05:00:14 +02:00
|
|
|
|
|
|
|
const exe_install = b.addInstallArtifact(exe, .{});
|
|
|
|
const exe_run = b.addRunArtifact(exe);
|
2024-05-29 18:10:29 +02:00
|
|
|
const step_name = try std.fmt.allocPrint(b.allocator, "run_{s}", .{example_name});
|
2024-10-12 13:27:50 +09:00
|
|
|
defer b.allocator.free(step_name);
|
2024-05-29 18:10:29 +02:00
|
|
|
const step_desc = try std.fmt.allocPrint(b.allocator, "run example {s}", .{example_name});
|
2024-10-12 13:27:50 +09:00
|
|
|
defer b.allocator.free(step_desc);
|
2024-05-27 04:31:42 +02:00
|
|
|
|
2024-06-01 05:38:28 +02:00
|
|
|
const cwd = try std.fmt.allocPrint(b.allocator, "src/examples/{s}", .{example_name});
|
2024-10-12 13:27:50 +09:00
|
|
|
defer b.allocator.free(cwd);
|
2024-06-01 05:38:28 +02:00
|
|
|
if (zig_ver < 12) exe_run.cwd = cwd else exe_run.setCwd(b.path(cwd));
|
2024-05-26 05:00:14 +02:00
|
|
|
|
2024-05-29 18:10:29 +02:00
|
|
|
exe_run.step.dependOn(&exe_install.step);
|
2024-05-29 23:27:29 +02:00
|
|
|
build_examples_step.dependOn(&exe_install.step);
|
2024-05-27 04:31:42 +02:00
|
|
|
b.step(step_name, step_desc).dependOn(&exe_run.step);
|
2023-12-31 00:14:30 +08:00
|
|
|
}
|
|
|
|
}
|
2024-10-12 13:27:50 +09:00
|
|
|
|
|
|
|
/// Function to runtime-scope log levels based on build flag, for all scopes.
|
|
|
|
fn log(
|
|
|
|
comptime level: std.log.Level,
|
|
|
|
comptime scope: @TypeOf(.EnumLiteral),
|
|
|
|
comptime format: []const u8,
|
|
|
|
args: anytype,
|
|
|
|
) void {
|
|
|
|
const should_print: bool = @intFromEnum(global_log_level) >= @intFromEnum(level);
|
|
|
|
if (should_print) {
|
|
|
|
switch (comptime level) {
|
|
|
|
.err => std.log.scoped(scope).err(format, args),
|
|
|
|
.warn => std.log.scoped(scope).warn(format, args),
|
|
|
|
.info => std.log.scoped(scope).info(format, args),
|
|
|
|
.debug => std.log.scoped(scope).debug(format, args),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|