Currently, including a tree-sitter parser as a dependency in a zig
project and running `zig build test` on the project will fetch the
zig-tree-sitter dependency declared by the parser. This is a problem
because (a) consumers may not want this dependency for whatever reason
and (b) due to how often Zig breaks everything and how scarcely most
tree-sitter parsers are updated, the zig-tree-sitter version pinned
by the parser module will often be outdated and broken.
The workaround I used was taken from https://ziggit.dev/t/11234
(cherry picked from commit 107bd800b0)
93 lines
2.9 KiB
Zig
93 lines
2.9 KiB
Zig
const std = @import("std");
|
|
|
|
pub fn build(b: *std.Build) !void {
|
|
const target = b.standardTargetOptions(.{});
|
|
const optimize = b.standardOptimizeOption(.{});
|
|
|
|
const shared = b.option(bool, "build-shared", "Build a shared library") orelse true;
|
|
const reuse_alloc = b.option(bool, "reuse-allocator", "Reuse the library allocator") orelse false;
|
|
|
|
const library_name = "tree-sitter-PARSER_NAME";
|
|
|
|
const lib: *std.Build.Step.Compile = b.addLibrary(.{
|
|
.name = library_name,
|
|
.linkage = if (shared) .dynamic else .static,
|
|
.root_module = b.createModule(.{
|
|
.target = target,
|
|
.optimize = optimize,
|
|
.link_libc = true,
|
|
.pic = if (shared) true else null,
|
|
}),
|
|
});
|
|
|
|
lib.addCSourceFile(.{
|
|
.file = b.path("src/parser.c"),
|
|
.flags = &.{"-std=c11"},
|
|
});
|
|
if (fileExists(b, "src/scanner.c")) {
|
|
lib.addCSourceFile(.{
|
|
.file = b.path("src/scanner.c"),
|
|
.flags = &.{"-std=c11"},
|
|
});
|
|
}
|
|
|
|
if (reuse_alloc) {
|
|
lib.root_module.addCMacro("TREE_SITTER_REUSE_ALLOCATOR", "");
|
|
}
|
|
if (optimize == .Debug) {
|
|
lib.root_module.addCMacro("TREE_SITTER_DEBUG", "");
|
|
}
|
|
|
|
lib.addIncludePath(b.path("src"));
|
|
|
|
b.installArtifact(lib);
|
|
b.installFile("src/node-types.json", "node-types.json");
|
|
|
|
if (fileExists(b, "queries")) {
|
|
b.installDirectory(.{
|
|
.source_dir = b.path("queries"),
|
|
.install_dir = .prefix,
|
|
.install_subdir = "queries",
|
|
.include_extensions = &.{"scm"},
|
|
});
|
|
}
|
|
|
|
const module = b.addModule(library_name, .{
|
|
.root_source_file = b.path("bindings/zig/root.zig"),
|
|
.target = target,
|
|
.optimize = optimize,
|
|
});
|
|
module.linkLibrary(lib);
|
|
|
|
const tests = b.addTest(.{
|
|
.root_module = b.createModule(.{
|
|
.root_source_file = b.path("bindings/zig/test.zig"),
|
|
.target = target,
|
|
.optimize = optimize,
|
|
}),
|
|
});
|
|
tests.root_module.addImport(library_name, module);
|
|
|
|
// HACK: fetch tree-sitter dependency only when testing this module
|
|
if (b.pkg_hash.len == 0) {
|
|
var args = try std.process.argsWithAllocator(b.allocator);
|
|
defer args.deinit();
|
|
while (args.next()) |a| {
|
|
if (std.mem.eql(u8, a, "test")) {
|
|
const ts_dep = b.lazyDependency("tree_sitter", .{}) orelse continue;
|
|
tests.root_module.addImport("tree-sitter", ts_dep.module("tree-sitter"));
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
const run_tests = b.addRunArtifact(tests);
|
|
const test_step = b.step("test", "Run unit tests");
|
|
test_step.dependOn(&run_tests.step);
|
|
}
|
|
|
|
inline fn fileExists(b: *std.Build, filename: []const u8) bool {
|
|
const dir = b.build_root.handle;
|
|
dir.access(filename, .{}) catch return false;
|
|
return true;
|
|
}
|