(originally posted here: https://zig.news/liyu1981/how-to-use-your-fav-pkg-in-buildzig-3ni8)
This post is summary of my learning from this excellent comment: https://ziggit.dev/t/can-i-use-packages-inside-build-zig/2892/6 of @kristoff. Thanks!
(If you just want to know the solution, go directly to TLDR;)
The problem
As I asked in my post, the problem is actually simple and very practical:
how to use a pkg in our build.zig
?
The more detail version of this problem is:
I have previously written a few functions in various build.zig
. I want to reuse them instead of copying them around. So I make those functions a pkg called zcmd.zig
. Then I want to use it in a new project like below
const std = @import("std");
const zcmd = @import("zcmd");
pub fn build(b: *std.Build) !void {
// ...omit normal build setups...
const exe = b.addExecutable(.{
.name = "build_use_package",
.root_source_file = .{ .path = "src/main.zig" },
.target = target,
.optimize = optimize,
});
// do something with zcmd like
const result = try zcmd.run(.{
.allocator = std.heap.page_allocator,
.commands = &[_][]const []const u8{
&.{ "uname", "-a" },
&.{ "grep", "-Eo", "Version\s[\d\.]+" },
},
});
defer result.deinit();
// next do something with result.stdout like extract part of the OS's info
// ...omit rest of build setups...
}
zcmd
is a small pkg contains my function to run a bash-like pipeline, which here I use to extract my OS’s version.
and I have placed zcmd
in my build.zig.zon
as follows
// ...omit not relevant lines...
.dependencies = .{
.zcmd = .{
.url = "https://github.com/liyu1981/zcmd.zig/archive/refs/tags/v0.2.0.tar.gz",
.hash = "1220bb5963c28e563ed010e5d622611ec0cb711ba8c6644ab44a22955957b1b8fe1a",
},
},
// ...omit not relevant lines...
and when I zig build
, the result is expected: zig will complain to me that zcmd
is not found in module root.
The first question: is this usage possible?
That’s immediately what I was thinking. Follow excellent WTF is Build.Zig.Zon,
- in order to make my exe know about
zcmd
, I will need add it (throughaddModule
). But seems my exe does not rely onzcmd
, but the exe ofbuild.zig
relies on it. - so in fact exe of
build.zig
need to notified aboutzcmd
, but we do not have abuild.zig
for ourbuild.zig
, where to add?
Seems to be a dead end, then I posted for help. And luckily few of you knows about it, like this comment by @ktz alias, showed to me that with a local pkg (pkg added with file:///
protocol in build.zig.zon
), it can work.
That’s good news. So I tried his approach. By adding my zcmd
as a gitsubmodule, seems I did can make it work. But I still need to @import("<path_to_gitsubmodule>/src/zcmd.zig")
, which is good but not good enough.
The second question: @kristoff told me it will work, then how it is working?
Then I see the comment from @kristoff, it is working, and his repo is using the url
+ hash
way of declaring the dependency. So, this should work, the rest is just how I will replicate it, and also establish my concept modal on why this works.
After some try, I make my copy work, and also managed to streamline the concepts behind it. I feel I should write here, as it may be also useful to others.
TLDR; How to provide a pkg for build.zig
and how to use it in build.zig
Below points are for the busy people, and I will explain them after that with my example.
-
Provide a pkg for
build.zig
. A pkg has to expose itself in its ownbuild.zig
for using later in otherbuild.zig
, usually by declaringpub const pkg = @import("<path-to>/pkg_main.zig");
inbuild.zig
. -
Use a pkg later in other
build.zig
. After the normalzig fetch --save <url>
, inbuild.zig
, then can useconst pkg = @import("pkg_name").pkg;
to get a reference and use it. Whether there is a nested struct, depends on previous step how it is exposed.
My example step by step
Now I can come back to my example. My task is to use zcmd
in my kcov
‘s build.zig
.
First I go back to my zcmd
pkg to make sure that I have provided it for build.zig
In my build.zig
of zcmd
(check the source here)
const std = @import("std");
pub const zcmd = @import("src/zcmd.zig"); // mark 1
pub fn build(b: *std.Build) !void {
_ = b.addModule("zcmd", .{ // mark 2
.source_file = .{ .path = "src/zcmd.zig" },
});
}
mark 1
is the line I expose myzcmd
for using inbuild.zig
.mark 2
is the line I exposezcmd
as a normal zig pkg throughaddModule
.
Note: the reason behind that I should expose zcmd
in a special way is that when zig runs another build.zig
, it will check corresponding build.zig.zon
to find the dependencies, but it will only load dep’s build.zig
(in our case is zcmd
‘s build.zig
) because it is build time. (again thanks
@kristoff’s explaining)
Second make kcov
‘s build.zig.zon
knows about this new version
Publish and get a new zig fetch line like zig fetch --save https://github.com/liyu1981/zcmd.zig/archive/refs/tags/v0.2.1.tar.gz
, do it inside kcov
‘s folder. It will save something like below in kcov
‘s build.zig.zon
.
// ...omit not relevant lines...
.dependencies = .{
.zcmd = .{
.url = "https://github.com/liyu1981/zcmd.zig/archive/refs/tags/v0.2.1.tar.gz",
.hash = "12205e6bd4374c56bcea698e36309d141cfe9fc760ec79d715a0d54f632b999f39dc",
},
},
// ...omit not relevant lines...
not much changed right? Yes, usually it is just the hash
changed, and I will see
warning: overwriting existing dependency named 'zcmd'
after zig fetch
Third, use zcmd
in kcov
‘s build.zig
in my kcov
‘s build.zig
I added this line in the header
const zcmd = @import("zcmd").zcmd;
pay attention that for build usage, I need to get zcmd
again from the nested exposure of zcmd
because I expose in that way. (while normally const zcmd = @import("zcmd");
is enough.)
Then in rest of part of build.zig
for kcov
, I can do follows
const result = try zcmd.run(.{
.allocator = allocator,
.commands = &[_][]const []const u8{
&.{ "codesign", "-s", "-", "--entitlements", "osx-entitlements.xml", "-f", "zig-out/bin/kcov" },
},
});
result.assertSucceededPanic(.{ .check_stdout_not_empty = false, .check_stderr_empty = false });
above code will call codesign
util of macos after kcov
is built from source (so it can be used).
Note: why in this way, zcmd
works? because zig will load zcmd
‘s build.zig
when try to compile kcov
‘s build.zig
. zcmd
‘s buidl.zig
exposed zcmd
by loading the source, so the whole thing just go through: just like doing a zig run build.zig
, no building of kcov
involved, and zcmd
is already injected when build build.zig
.
Hope this can help! and thanks Zig community 🙂