zig/lib/std / Build/Step/Fmt.zig

This step has two modes: * Modify mode: directly modify source files, formatting them in place. * Check mode: fail the step if a non-conforming file is found.

//! This step has two modes:
//! * Modify mode: directly modify source files, formatting them in place.
//! * Check mode: fail the step if a non-conforming file is found.
const std = @import("std");
const Step = std.Build.Step;
const Fmt = @This();

base_id

If true, fails the build step when any non-conforming files are encountered.


step: Step,
paths: []const []const u8,
exclude_paths: []const []const u8,
check: bool,

Options


pub const base_id = .fmt;

create()


pub const Options = struct {
    paths: []const []const u8 = &.{},
    exclude_paths: []const []const u8 = &.{},
    /// If true, fails the build step when any non-conforming files are encountered.
    check: bool = false,
};

pub fn create(owner: *std.Build, options: Options) *Fmt {
    const self = owner.allocator.create(Fmt) catch @panic("OOM");
    const name = if (options.check) "zig fmt --check" else "zig fmt";
    self.* = .{
        .step = Step.init(.{
            .id = base_id,
            .name = name,
            .owner = owner,
            .makeFn = make,
        }),
        .paths = options.paths,
        .exclude_paths = options.exclude_paths,
        .check = options.check,
    };
    return self;
}

fn make(step: *Step, prog_node: *std.Progress.Node) !void {
    // zig fmt is fast enough that no progress is needed.
    _ = prog_node;

    // TODO: if check=false, this means we are modifying source files in place, which
    // is an operation that could race against other operations also modifying source files
    // in place. In this case, this step should obtain a write lock while making those
    // modifications.

    const b = step.owner;
    const arena = b.allocator;
    const self = @fieldParentPtr(Fmt, "step", step);

    var argv: std.ArrayListUnmanaged([]const u8) = .{};
    try argv.ensureUnusedCapacity(arena, 2 + 1 + self.paths.len + 2 * self.exclude_paths.len);

    argv.appendAssumeCapacity(b.zig_exe);
    argv.appendAssumeCapacity("fmt");

    if (self.check) {
        argv.appendAssumeCapacity("--check");
    }

    for (self.paths) |p| {
        argv.appendAssumeCapacity(b.pathFromRoot(p));
    }

    for (self.exclude_paths) |p| {
        argv.appendAssumeCapacity("--exclude");
        argv.appendAssumeCapacity(b.pathFromRoot(p));
    }

    return step.evalChildProcess(argv.items);
}