mirror of
https://github.com/jetzig-framework/jetzig.git
synced 2025-05-14 14:06:08 +00:00

Use in-memory KV store (JetKV) for simple job queue. Build script generates an array of Zig modules in `src/app/jobs/` and stores their name + run function (`run(allocator, params, logger)`). View functions schedule jobs with arbitrary params. Thread pool spawns a (configurable) number of workers and pops jobs from the queue, then invokes the appropriate run function.
203 lines
7.6 KiB
Zig
203 lines
7.6 KiB
Zig
const std = @import("std");
|
|
|
|
pub const Routes = @import("src/Routes.zig");
|
|
pub const GenerateMimeTypes = @import("src/GenerateMimeTypes.zig");
|
|
pub const TemplateFn = @import("src/jetzig.zig").TemplateFn;
|
|
pub const StaticRequest = @import("src/jetzig.zig").StaticRequest;
|
|
pub const http = @import("src/jetzig/http.zig");
|
|
pub const data = @import("src/jetzig/data.zig");
|
|
pub const views = @import("src/jetzig/views.zig");
|
|
pub const Route = views.Route;
|
|
pub const Job = @import("src/jetzig.zig").Job;
|
|
|
|
const zmpl_build = @import("zmpl");
|
|
|
|
pub fn build(b: *std.Build) !void {
|
|
const target = b.standardTargetOptions(.{});
|
|
const optimize = b.standardOptimizeOption(.{});
|
|
const template_path_option = b.option([]const u8, "zmpl_templates_path", "Path to templates") orelse
|
|
"src/app/views/";
|
|
const template_path: []const u8 = if (std.fs.path.isAbsolute(template_path_option))
|
|
try b.allocator.dupe(u8, template_path_option)
|
|
else
|
|
std.fs.cwd().realpathAlloc(b.allocator, template_path_option) catch |err| switch (err) {
|
|
error.FileNotFound => "",
|
|
else => return err,
|
|
};
|
|
|
|
const lib = b.addStaticLibrary(.{
|
|
.name = "jetzig",
|
|
.root_source_file = .{ .path = "src/jetzig.zig" },
|
|
.target = target,
|
|
.optimize = optimize,
|
|
});
|
|
|
|
const mime_module = try GenerateMimeTypes.generateMimeModule(b);
|
|
|
|
const zig_args_dep = b.dependency("args", .{ .target = target, .optimize = optimize });
|
|
|
|
const jetzig_module = b.addModule("jetzig", .{ .root_source_file = .{ .path = "src/jetzig.zig" } });
|
|
jetzig_module.addImport("mime_types", mime_module);
|
|
lib.root_module.addImport("jetzig", jetzig_module);
|
|
|
|
const zmpl_version = b.option(
|
|
enum { v1, v2 },
|
|
"zmpl_version",
|
|
"Zmpl syntax version (default: v1)",
|
|
) orelse .v2;
|
|
|
|
if (zmpl_version == .v1) {
|
|
std.debug.print(
|
|
\\[WARN] Zmpl v1 is deprecated and will soon be removed.
|
|
\\ Update to v2 by modifying `jetzigInit` in your `build.zig`:
|
|
\\
|
|
\\ try jetzig.jetzigInit(b, exe, .{{ .zmpl_version = .v2 }});
|
|
\\
|
|
\\ See https://jetzig.dev/documentation.html for information on migrating to Zmpl v2.
|
|
\\
|
|
, .{});
|
|
}
|
|
|
|
const zmpl_dep = b.dependency(
|
|
"zmpl",
|
|
.{
|
|
.target = target,
|
|
.optimize = optimize,
|
|
.zmpl_templates_path = template_path,
|
|
.zmpl_auto_build = false,
|
|
.zmpl_version = zmpl_version,
|
|
.zmpl_constants = try zmpl_build.addTemplateConstants(b, struct {
|
|
jetzig_view: []const u8,
|
|
jetzig_action: []const u8,
|
|
}),
|
|
},
|
|
);
|
|
|
|
const zmpl_module = zmpl_dep.module("zmpl");
|
|
|
|
const jetkv_dep = b.dependency("jetkv", .{ .target = target, .optimize = optimize });
|
|
|
|
// This is the way to make it look nice in the zig build script
|
|
// If we would do it the other way around, we would have to do
|
|
// b.dependency("jetzig",.{}).builder.dependency("zmpl",.{}).module("zmpl");
|
|
b.modules.put("zmpl", zmpl_dep.module("zmpl")) catch @panic("Out of memory");
|
|
|
|
const zmd_dep = b.dependency("zmd", .{ .target = target, .optimize = optimize });
|
|
|
|
lib.root_module.addImport("zmpl", zmpl_module);
|
|
jetzig_module.addImport("zmpl", zmpl_module);
|
|
jetzig_module.addImport("args", zig_args_dep.module("args"));
|
|
jetzig_module.addImport("zmd", zmd_dep.module("zmd"));
|
|
jetzig_module.addImport("jetkv", jetkv_dep.module("jetkv"));
|
|
|
|
const main_tests = b.addTest(.{
|
|
.root_source_file = .{ .path = "src/tests.zig" },
|
|
.target = target,
|
|
.optimize = optimize,
|
|
});
|
|
|
|
const docs_step = b.step("docs", "Generate documentation");
|
|
const docs_install = b.addInstallDirectory(.{
|
|
.source_dir = lib.getEmittedDocs(),
|
|
.install_dir = .prefix,
|
|
.install_subdir = "docs",
|
|
});
|
|
|
|
docs_step.dependOn(&docs_install.step);
|
|
|
|
main_tests.root_module.addImport("zmpl", zmpl_dep.module("zmpl"));
|
|
main_tests.root_module.addImport("jetkv", jetkv_dep.module("jetkv"));
|
|
const run_main_tests = b.addRunArtifact(main_tests);
|
|
|
|
const test_step = b.step("test", "Run library tests");
|
|
test_step.dependOn(&run_main_tests.step);
|
|
}
|
|
|
|
/// Build-time options for Jetzig.
|
|
pub const JetzigInitOptions = struct {
|
|
zmpl_version: enum { v1, v2 } = .v1,
|
|
};
|
|
|
|
pub fn jetzigInit(b: *std.Build, exe: *std.Build.Step.Compile, options: JetzigInitOptions) !void {
|
|
const target = b.host;
|
|
const optimize = exe.root_module.optimize orelse .Debug;
|
|
const jetzig_dep = b.dependency(
|
|
"jetzig",
|
|
.{ .optimize = optimize, .target = target, .zmpl_version = options.zmpl_version },
|
|
);
|
|
const jetzig_module = jetzig_dep.module("jetzig");
|
|
const zmpl_module = jetzig_dep.module("zmpl");
|
|
|
|
exe.root_module.addImport("jetzig", jetzig_module);
|
|
exe.root_module.addImport("zmpl", zmpl_module);
|
|
|
|
if (b.option(bool, "jetzig_runner", "Used internally by `jetzig server` command.")) |jetzig_runner| {
|
|
if (jetzig_runner) {
|
|
const file = try std.fs.cwd().createFile(".jetzig", .{ .truncate = true });
|
|
defer file.close();
|
|
try file.writeAll(exe.name);
|
|
}
|
|
}
|
|
|
|
const root_path = b.build_root.path orelse try std.fs.cwd().realpathAlloc(b.allocator, ".");
|
|
const templates_path = try std.fs.path.join(
|
|
b.allocator,
|
|
&[_][]const u8{ root_path, "src", "app", "views" },
|
|
);
|
|
|
|
const jobs_path = try std.fs.path.join(
|
|
b.allocator,
|
|
&[_][]const u8{ root_path, "src", "app", "jobs" },
|
|
);
|
|
|
|
var generate_routes = try Routes.init(b.allocator, root_path, templates_path, jobs_path);
|
|
try generate_routes.generateRoutes();
|
|
const write_files = b.addWriteFiles();
|
|
const routes_file = write_files.add("routes.zig", generate_routes.buffer.items);
|
|
const routes_module = b.createModule(.{ .root_source_file = routes_file });
|
|
|
|
var src_dir = try std.fs.openDirAbsolute(b.pathFromRoot("src"), .{ .iterate = true });
|
|
defer src_dir.close();
|
|
var walker = try src_dir.walk(b.allocator);
|
|
defer walker.deinit();
|
|
|
|
while (try walker.next()) |entry| {
|
|
if (entry.kind == .file) {
|
|
if (!std.mem.eql(u8, ".zig", std.fs.path.extension(entry.path))) continue;
|
|
|
|
const stat = try src_dir.statFile(entry.path);
|
|
const src_data = try src_dir.readFileAlloc(b.allocator, entry.path, stat.size);
|
|
defer b.allocator.free(src_data);
|
|
|
|
const relpath = try std.fs.path.join(b.allocator, &[_][]const u8{ "src", entry.path });
|
|
defer b.allocator.free(relpath);
|
|
|
|
_ = write_files.add(relpath, src_data);
|
|
}
|
|
}
|
|
|
|
const exe_static_routes = b.addExecutable(.{
|
|
.name = "static",
|
|
.root_source_file = jetzig_dep.path("src/compile_static_routes.zig"),
|
|
.target = target,
|
|
.optimize = optimize,
|
|
});
|
|
|
|
exe.root_module.addImport("routes", routes_module);
|
|
|
|
var it = exe.root_module.import_table.iterator();
|
|
while (it.next()) |import| {
|
|
routes_module.addImport(import.key_ptr.*, import.value_ptr.*);
|
|
exe_static_routes.root_module.addImport(import.key_ptr.*, import.value_ptr.*);
|
|
}
|
|
|
|
exe_static_routes.root_module.addImport("routes", routes_module);
|
|
exe_static_routes.root_module.addImport("jetzig", jetzig_module);
|
|
exe_static_routes.root_module.addImport("zmpl", zmpl_module);
|
|
exe_static_routes.root_module.addImport("jetzig_app", &exe.root_module);
|
|
|
|
const run_static_routes_cmd = b.addRunArtifact(exe_static_routes);
|
|
run_static_routes_cmd.expectExitCode(0);
|
|
exe.step.dependOn(&run_static_routes_cmd.step);
|
|
}
|