const std = @import("std"); const builtin = @import("builtin"); const PatchMap = std.StringHashMap([]const u8); const InitSystem = enum { systemd, openrc, runit, s6, dinit, }; const min_zig_string = "0.15.0"; const current_zig = builtin.zig_version; // Implementing zig version detection through compile time comptime { const min_zig = std.SemanticVersion.parse(min_zig_string) catch unreachable; if (current_zig.order(min_zig) == .lt) { @compileError(std.fmt.comptimePrint("Your Zig version v{} does not meet the minimum build requirement of v{}", .{ current_zig, min_zig })); } } const ly_version = std.SemanticVersion{ .major = 1, .minor = 2, .patch = 0 }; var dest_directory: []const u8 = undefined; var config_directory: []const u8 = undefined; var prefix_directory: []const u8 = undefined; var executable_name: []const u8 = undefined; var init_system: InitSystem = undefined; var default_tty_str: []const u8 = undefined; pub fn build(b: *std.Build) !void { dest_directory = b.option([]const u8, "dest_directory", "Specify a destination directory for installation") orelse ""; config_directory = b.option([]const u8, "config_directory", "Specify a default config directory (default is /etc). This path gets embedded into the binary") orelse "/etc"; prefix_directory = b.option([]const u8, "prefix_directory", "Specify a default prefix directory (default is /usr)") orelse "/usr"; executable_name = b.option([]const u8, "name", "Specify installed executable file name (default is ly)") orelse "ly"; init_system = b.option(InitSystem, "init_system", "Specify the target init system (default is systemd)") orelse .systemd; const build_options = b.addOptions(); const version_str = try getVersionStr(b, "ly", ly_version); const enable_x11_support = b.option(bool, "enable_x11_support", "Enable X11 support (default is on)") orelse true; const default_tty = b.option(u8, "default_tty", "Set the TTY (default is 2)") orelse 2; default_tty_str = try std.fmt.allocPrint(b.allocator, "{d}", .{default_tty}); build_options.addOption([]const u8, "config_directory", config_directory); build_options.addOption([]const u8, "prefix_directory", prefix_directory); build_options.addOption([]const u8, "version", version_str); build_options.addOption(u8, "tty", default_tty); build_options.addOption(bool, "enable_x11_support", enable_x11_support); const target = b.standardTargetOptions(.{}); const optimize = b.standardOptimizeOption(.{}); const exe = b.addExecutable(.{ .name = "ly", .root_module = b.createModule(.{ .root_source_file = b.path("src/main.zig"), .target = target, .optimize = optimize, }), }); const zigini = b.dependency("zigini", .{ .target = target, .optimize = optimize }); exe.root_module.addImport("zigini", zigini.module("zigini")); exe.root_module.addOptions("build_options", build_options); const clap = b.dependency("clap", .{ .target = target, .optimize = optimize }); exe.root_module.addImport("clap", clap.module("clap")); const termbox_dep = b.dependency("termbox2", .{ .target = target, .optimize = optimize, }); exe.linkSystemLibrary("pam"); if (enable_x11_support) exe.linkSystemLibrary("xcb"); exe.linkLibC(); const translate_c = b.addTranslateC(.{ .root_source_file = termbox_dep.path("termbox2.h"), .target = target, .optimize = optimize, }); translate_c.defineCMacroRaw("TB_IMPL"); translate_c.defineCMacro("TB_OPT_ATTR_W", "32"); // Enable 24-bit color support + styling (32-bit) const termbox2 = translate_c.addModule("termbox2"); exe.root_module.addImport("termbox2", termbox2); b.installArtifact(exe); const run_cmd = b.addRunArtifact(exe); run_cmd.step.dependOn(b.getInstallStep()); if (b.args) |args| run_cmd.addArgs(args); const run_step = b.step("run", "Run the app"); run_step.dependOn(&run_cmd.step); const installexe_step = b.step("installexe", "Install Ly and the selected init system service"); installexe_step.makeFn = Installer(true).make; installexe_step.dependOn(b.getInstallStep()); const installnoconf_step = b.step("installnoconf", "Install Ly and the selected init system service, but not the configuration file"); installnoconf_step.makeFn = Installer(false).make; installnoconf_step.dependOn(b.getInstallStep()); const uninstallexe_step = b.step("uninstallexe", "Uninstall Ly and remove the selected init system service"); uninstallexe_step.makeFn = Uninstaller(true).make; const uninstallnoconf_step = b.step("uninstallnoconf", "Uninstall Ly and remove the selected init system service, but keep the configuration directory"); uninstallnoconf_step.makeFn = Uninstaller(false).make; } pub fn Installer(install_config: bool) type { return struct { pub fn make(step: *std.Build.Step, _: std.Build.Step.MakeOptions) !void { const allocator = step.owner.allocator; var patch_map = PatchMap.init(allocator); defer patch_map.deinit(); try patch_map.put("$DEFAULT_TTY", default_tty_str); try patch_map.put("$CONFIG_DIRECTORY", config_directory); try patch_map.put("$PREFIX_DIRECTORY", prefix_directory); try patch_map.put("$EXECUTABLE_NAME", executable_name); try install_ly(allocator, patch_map, install_config); try install_service(allocator, patch_map); } }; } fn install_ly(allocator: std.mem.Allocator, patch_map: PatchMap, install_config: bool) !void { const ly_config_directory = try std.fs.path.join(allocator, &[_][]const u8{ dest_directory, config_directory, "/ly" }); std.fs.cwd().makePath(ly_config_directory) catch { std.debug.print("warn: {s} already exists as a directory.\n", .{ly_config_directory}); }; const ly_custom_sessions_directory = try std.fs.path.join(allocator, &[_][]const u8{ dest_directory, config_directory, "/ly/custom-sessions" }); std.fs.cwd().makePath(ly_custom_sessions_directory) catch { std.debug.print("warn: {s} already exists as a directory.\n", .{ly_custom_sessions_directory}); }; const ly_lang_path = try std.fs.path.join(allocator, &[_][]const u8{ dest_directory, config_directory, "/ly/lang" }); std.fs.cwd().makePath(ly_lang_path) catch { std.debug.print("warn: {s} already exists as a directory.\n", .{ly_lang_path}); }; { const exe_path = try std.fs.path.join(allocator, &[_][]const u8{ dest_directory, prefix_directory, "/bin" }); std.fs.cwd().makePath(exe_path) catch { if (!std.mem.eql(u8, dest_directory, "")) { std.debug.print("warn: {s} already exists as a directory.\n", .{exe_path}); } }; var executable_dir = std.fs.cwd().openDir(exe_path, .{}) catch unreachable; defer executable_dir.close(); try installFile("zig-out/bin/ly", executable_dir, exe_path, executable_name, .{}); } { var config_dir = std.fs.cwd().openDir(ly_config_directory, .{}) catch unreachable; defer config_dir.close(); if (install_config) { const patched_config = try patchFile(allocator, "res/config.ini", patch_map); try installText(patched_config, config_dir, ly_config_directory, "config.ini", .{}); } const patched_setup = try patchFile(allocator, "res/setup.sh", patch_map); try installText(patched_setup, config_dir, ly_config_directory, "setup.sh", .{ .mode = 0o755 }); } { var custom_sessions_dir = std.fs.cwd().openDir(ly_custom_sessions_directory, .{}) catch unreachable; defer custom_sessions_dir.close(); const patched_readme = try patchFile(allocator, "res/custom-sessions/README", patch_map); try installText(patched_readme, custom_sessions_dir, ly_custom_sessions_directory, "README", .{}); } { var lang_dir = std.fs.cwd().openDir(ly_lang_path, .{}) catch unreachable; defer lang_dir.close(); try installFile("res/lang/cat.ini", lang_dir, ly_lang_path, "cat.ini", .{}); try installFile("res/lang/cs.ini", lang_dir, ly_lang_path, "cs.ini", .{}); try installFile("res/lang/de.ini", lang_dir, ly_lang_path, "de.ini", .{}); try installFile("res/lang/en.ini", lang_dir, ly_lang_path, "en.ini", .{}); try installFile("res/lang/es.ini", lang_dir, ly_lang_path, "es.ini", .{}); try installFile("res/lang/fr.ini", lang_dir, ly_lang_path, "fr.ini", .{}); try installFile("res/lang/it.ini", lang_dir, ly_lang_path, "it.ini", .{}); try installFile("res/lang/pl.ini", lang_dir, ly_lang_path, "pl.ini", .{}); try installFile("res/lang/pt.ini", lang_dir, ly_lang_path, "pt.ini", .{}); try installFile("res/lang/pt_BR.ini", lang_dir, ly_lang_path, "pt_BR.ini", .{}); try installFile("res/lang/ro.ini", lang_dir, ly_lang_path, "ro.ini", .{}); try installFile("res/lang/ru.ini", lang_dir, ly_lang_path, "ru.ini", .{}); try installFile("res/lang/sr.ini", lang_dir, ly_lang_path, "sr.ini", .{}); try installFile("res/lang/sv.ini", lang_dir, ly_lang_path, "sv.ini", .{}); try installFile("res/lang/tr.ini", lang_dir, ly_lang_path, "tr.ini", .{}); try installFile("res/lang/uk.ini", lang_dir, ly_lang_path, "uk.ini", .{}); } { const pam_path = try std.fs.path.join(allocator, &[_][]const u8{ dest_directory, config_directory, "/pam.d" }); std.fs.cwd().makePath(pam_path) catch { if (!std.mem.eql(u8, dest_directory, "")) { std.debug.print("warn: {s} already exists as a directory.\n", .{pam_path}); } }; var pam_dir = std.fs.cwd().openDir(pam_path, .{}) catch unreachable; defer pam_dir.close(); try installFile("res/pam.d/ly", pam_dir, pam_path, "ly", .{ .override_mode = 0o644 }); } } fn install_service(allocator: std.mem.Allocator, patch_map: PatchMap) !void { switch (init_system) { .systemd => { const service_path = try std.fs.path.join(allocator, &[_][]const u8{ dest_directory, prefix_directory, "/lib/systemd/system" }); std.fs.cwd().makePath(service_path) catch {}; var service_dir = std.fs.cwd().openDir(service_path, .{}) catch unreachable; defer service_dir.close(); const patched_service = try patchFile(allocator, "res/ly.service", patch_map); try installText(patched_service, service_dir, service_path, "ly.service", .{ .mode = 0o644 }); }, .openrc => { const service_path = try std.fs.path.join(allocator, &[_][]const u8{ dest_directory, config_directory, "/init.d" }); std.fs.cwd().makePath(service_path) catch {}; var service_dir = std.fs.cwd().openDir(service_path, .{}) catch unreachable; defer service_dir.close(); const patched_service = try patchFile(allocator, "res/ly-openrc", patch_map); try installText(patched_service, service_dir, service_path, executable_name, .{ .mode = 0o755 }); }, .runit => { const service_path = try std.fs.path.join(allocator, &[_][]const u8{ dest_directory, config_directory, "/sv/ly" }); std.fs.cwd().makePath(service_path) catch {}; var service_dir = std.fs.cwd().openDir(service_path, .{}) catch unreachable; defer service_dir.close(); const supervise_path = try std.fs.path.join(allocator, &[_][]const u8{ service_path, "supervise" }); const patched_conf = try patchFile(allocator, "res/ly-runit-service/conf", patch_map); try installText(patched_conf, service_dir, service_path, "conf", .{}); try installFile("res/ly-runit-service/finish", service_dir, service_path, "finish", .{ .override_mode = 0o755 }); const patched_run = try patchFile(allocator, "res/ly-runit-service/run", patch_map); try installText(patched_run, service_dir, service_path, "run", .{ .mode = 0o755 }); try std.fs.cwd().symLink("/run/runit/supervise.ly", supervise_path, .{}); std.debug.print("info: installed symlink /run/runit/supervise.ly\n", .{}); }, .s6 => { const admin_service_path = try std.fs.path.join(allocator, &[_][]const u8{ dest_directory, config_directory, "/s6/adminsv/default/contents.d" }); std.fs.cwd().makePath(admin_service_path) catch {}; var admin_service_dir = std.fs.cwd().openDir(admin_service_path, .{}) catch unreachable; defer admin_service_dir.close(); const file = try admin_service_dir.createFile("ly-srv", .{}); file.close(); const service_path = try std.fs.path.join(allocator, &[_][]const u8{ dest_directory, config_directory, "/s6/sv/ly-srv" }); std.fs.cwd().makePath(service_path) catch {}; var service_dir = std.fs.cwd().openDir(service_path, .{}) catch unreachable; defer service_dir.close(); const patched_run = try patchFile(allocator, "res/ly-s6/run", patch_map); try installText(patched_run, service_dir, service_path, "run", .{ .mode = 0o755 }); try installFile("res/ly-s6/type", service_dir, service_path, "type", .{}); }, .dinit => { const service_path = try std.fs.path.join(allocator, &[_][]const u8{ dest_directory, config_directory, "/dinit.d" }); std.fs.cwd().makePath(service_path) catch {}; var service_dir = std.fs.cwd().openDir(service_path, .{}) catch unreachable; defer service_dir.close(); const patched_service = try patchFile(allocator, "res/ly-dinit", patch_map); try installText(patched_service, service_dir, service_path, "ly", .{}); }, } } pub fn Uninstaller(uninstall_config: bool) type { return struct { pub fn make(step: *std.Build.Step, _: std.Build.Step.MakeOptions) !void { const allocator = step.owner.allocator; if (uninstall_config) { try deleteTree(allocator, config_directory, "/ly", "ly config directory not found"); } const exe_path = try std.fs.path.join(allocator, &[_][]const u8{ prefix_directory, "/bin/", executable_name }); var success = true; std.fs.cwd().deleteFile(exe_path) catch { std.debug.print("warn: ly executable not found\n", .{}); success = false; }; if (success) std.debug.print("info: deleted {s}\n", .{exe_path}); try deleteFile(allocator, config_directory, "/pam.d/ly", "ly pam file not found"); switch (init_system) { .systemd => try deleteFile(allocator, prefix_directory, "/lib/systemd/system/ly.service", "systemd service not found"), .openrc => try deleteFile(allocator, config_directory, "/init.d/ly", "openrc service not found"), .runit => try deleteTree(allocator, config_directory, "/sv/ly", "runit service not found"), .s6 => { try deleteTree(allocator, config_directory, "/s6/sv/ly-srv", "s6 service not found"); try deleteFile(allocator, config_directory, "/s6/adminsv/default/contents.d/ly-srv", "s6 admin service not found"); }, .dinit => try deleteFile(allocator, config_directory, "/dinit.d/ly", "dinit service not found"), } } }; } fn getVersionStr(b: *std.Build, name: []const u8, version: std.SemanticVersion) ![]const u8 { const version_str = b.fmt("{d}.{d}.{d}", .{ version.major, version.minor, version.patch }); var status: u8 = undefined; const git_describe_raw = b.runAllowFail(&[_][]const u8{ "git", "-C", b.build_root.path orelse ".", "describe", "--match", "*.*.*", "--tags", }, &status, .Ignore) catch { return version_str; }; var git_describe = std.mem.trim(u8, git_describe_raw, " \n\r"); git_describe = std.mem.trimLeft(u8, git_describe, "v"); switch (std.mem.count(u8, git_describe, "-")) { 0 => { if (!std.mem.eql(u8, version_str, git_describe)) { std.debug.print("{s} version '{s}' does not match git tag: '{s}'\n", .{ name, version_str, git_describe }); std.process.exit(1); } return version_str; }, 2 => { // Untagged development build (e.g. 0.10.0-dev.2025+ecf0050a9). var it = std.mem.splitScalar(u8, git_describe, '-'); const tagged_ancestor = std.mem.trimLeft(u8, it.first(), "v"); const commit_height = it.next().?; const commit_id = it.next().?; const ancestor_ver = try std.SemanticVersion.parse(tagged_ancestor); if (version.order(ancestor_ver) != .gt) { std.debug.print("{s} version '{f}' must be greater than tagged ancestor '{f}'\n", .{ name, version, ancestor_ver }); std.process.exit(1); } // Check that the commit hash is prefixed with a 'g' (a Git convention). if (commit_id.len < 1 or commit_id[0] != 'g') { std.debug.print("Unexpected `git describe` output: {s}\n", .{git_describe}); return version_str; } // The version is reformatted in accordance with the https://semver.org specification. return b.fmt("{s}-dev.{s}+{s}", .{ version_str, commit_height, commit_id[1..] }); }, else => { std.debug.print("Unexpected `git describe` output: {s}\n", .{git_describe}); return version_str; }, } } fn installFile( source_file: []const u8, destination_directory: std.fs.Dir, destination_directory_path: []const u8, destination_file: []const u8, options: std.fs.Dir.CopyFileOptions, ) !void { try std.fs.cwd().copyFile(source_file, destination_directory, destination_file, options); std.debug.print("info: installed {s}/{s}\n", .{ destination_directory_path, destination_file }); } fn patchFile(allocator: std.mem.Allocator, source_file: []const u8, patch_map: PatchMap) ![]const u8 { var file = try std.fs.cwd().openFile(source_file, .{}); defer file.close(); const stat = try file.stat(); var buffer: [4096]u8 = undefined; var reader = file.reader(&buffer); var text = try reader.interface.readAlloc(allocator, stat.size); var iterator = patch_map.iterator(); while (iterator.next()) |kv| { const new_text = try std.mem.replaceOwned(u8, allocator, text, kv.key_ptr.*, kv.value_ptr.*); allocator.free(text); text = new_text; } return text; } fn installText( text: []const u8, destination_directory: std.fs.Dir, destination_directory_path: []const u8, destination_file: []const u8, options: std.fs.File.CreateFlags, ) !void { var file = try destination_directory.createFile(destination_file, options); defer file.close(); var buffer: [1024]u8 = undefined; var writer = file.writer(&buffer); try writer.interface.writeAll(text); try writer.interface.flush(); std.debug.print("info: installed {s}/{s}\n", .{ destination_directory_path, destination_file }); } fn deleteFile( allocator: std.mem.Allocator, prefix: []const u8, file: []const u8, warning: []const u8, ) !void { const path = try std.fs.path.join(allocator, &[_][]const u8{ dest_directory, prefix, file }); std.fs.cwd().deleteFile(path) catch |err| { if (err == error.FileNotFound) { std.debug.print("warn: {s}\n", .{warning}); return; } return err; }; std.debug.print("info: deleted {s}\n", .{path}); } fn deleteTree( allocator: std.mem.Allocator, prefix: []const u8, directory: []const u8, warning: []const u8, ) !void { const path = try std.fs.path.join(allocator, &[_][]const u8{ dest_directory, prefix, directory }); var dir = std.fs.cwd().openDir(path, .{}) catch |err| { if (err == error.FileNotFound) { std.debug.print("warn: {s}\n", .{warning}); return; } return err; }; dir.close(); try std.fs.cwd().deleteTree(path); std.debug.print("info: deleted {s}\n", .{path}); }