mirror of
https://codeberg.org/ziglang/zig.git
synced 2025-12-06 13:54:21 +00:00
110 lines
4 KiB
Zig
110 lines
4 KiB
Zig
const std = @import("std");
|
|
const mem = std.mem;
|
|
const fs = std.fs;
|
|
const Step = std.Build.Step;
|
|
const InstallDir = std.Build.InstallDir;
|
|
const InstallDirStep = @This();
|
|
|
|
step: Step,
|
|
options: Options,
|
|
/// This is used by the build system when a file being installed comes from one
|
|
/// package but is being installed by another.
|
|
dest_builder: *std.Build,
|
|
|
|
pub const base_id = .install_dir;
|
|
|
|
pub const Options = struct {
|
|
source_dir: []const u8,
|
|
install_dir: InstallDir,
|
|
install_subdir: []const u8,
|
|
/// File paths which end in any of these suffixes will be excluded
|
|
/// from being installed.
|
|
exclude_extensions: []const []const u8 = &.{},
|
|
/// File paths which end in any of these suffixes will result in
|
|
/// empty files being installed. This is mainly intended for large
|
|
/// test.zig files in order to prevent needless installation bloat.
|
|
/// However if the files were not present at all, then
|
|
/// `@import("test.zig")` would be a compile error.
|
|
blank_extensions: []const []const u8 = &.{},
|
|
|
|
fn dupe(self: Options, b: *std.Build) Options {
|
|
return .{
|
|
.source_dir = b.dupe(self.source_dir),
|
|
.install_dir = self.install_dir.dupe(b),
|
|
.install_subdir = b.dupe(self.install_subdir),
|
|
.exclude_extensions = b.dupeStrings(self.exclude_extensions),
|
|
.blank_extensions = b.dupeStrings(self.blank_extensions),
|
|
};
|
|
}
|
|
};
|
|
|
|
pub fn init(owner: *std.Build, options: Options) InstallDirStep {
|
|
owner.pushInstalledFile(options.install_dir, options.install_subdir);
|
|
return .{
|
|
.step = Step.init(.{
|
|
.id = .install_dir,
|
|
.name = owner.fmt("install {s}/", .{options.source_dir}),
|
|
.owner = owner,
|
|
.makeFn = make,
|
|
}),
|
|
.options = options.dupe(owner),
|
|
.dest_builder = owner,
|
|
};
|
|
}
|
|
|
|
fn make(step: *Step, prog_node: *std.Progress.Node) !void {
|
|
_ = prog_node;
|
|
const self = @fieldParentPtr(InstallDirStep, "step", step);
|
|
const dest_builder = self.dest_builder;
|
|
const arena = dest_builder.allocator;
|
|
const dest_prefix = dest_builder.getInstallPath(self.options.install_dir, self.options.install_subdir);
|
|
const src_builder = self.step.owner;
|
|
var src_dir = src_builder.build_root.handle.openIterableDir(self.options.source_dir, .{}) catch |err| {
|
|
return step.fail("unable to open source directory '{}{s}': {s}", .{
|
|
src_builder.build_root, self.options.source_dir, @errorName(err),
|
|
});
|
|
};
|
|
defer src_dir.close();
|
|
var it = try src_dir.walk(arena);
|
|
var all_cached = true;
|
|
next_entry: while (try it.next()) |entry| {
|
|
for (self.options.exclude_extensions) |ext| {
|
|
if (mem.endsWith(u8, entry.path, ext)) {
|
|
continue :next_entry;
|
|
}
|
|
}
|
|
|
|
// relative to src build root
|
|
const src_sub_path = try fs.path.join(arena, &.{ self.options.source_dir, entry.path });
|
|
const dest_path = try fs.path.join(arena, &.{ dest_prefix, entry.path });
|
|
const cwd = fs.cwd();
|
|
|
|
switch (entry.kind) {
|
|
.Directory => try cwd.makePath(dest_path),
|
|
.File => {
|
|
for (self.options.blank_extensions) |ext| {
|
|
if (mem.endsWith(u8, entry.path, ext)) {
|
|
try dest_builder.truncateFile(dest_path);
|
|
continue :next_entry;
|
|
}
|
|
}
|
|
|
|
const prev_status = fs.Dir.updateFile(
|
|
src_builder.build_root.handle,
|
|
src_sub_path,
|
|
cwd,
|
|
dest_path,
|
|
.{},
|
|
) catch |err| {
|
|
return step.fail("unable to update file from '{}{s}' to '{s}': {s}", .{
|
|
src_builder.build_root, src_sub_path, dest_path, @errorName(err),
|
|
});
|
|
};
|
|
all_cached = all_cached and prev_status == .fresh;
|
|
},
|
|
else => continue,
|
|
}
|
|
}
|
|
|
|
step.result_cached = all_cached;
|
|
}
|