mirror of
https://codeberg.org/ziglang/zig.git
synced 2025-12-06 13:54:21 +00:00
`std.Io.tty.Config.detect` may be an expensive check (e.g. involving syscalls), and doing it every time we need to print isn't really necessary; under normal usage, we can compute the value once and cache it for the whole program's execution. Since anyone outputting to stderr may reasonably want this information (in fact they are very likely to), it makes sense to cache it and return it from `lockStderrWriter`. Call sites who do not need it will experience no significant overhead, and can just ignore the TTY config with a `const w, _` destructure.
439 lines
16 KiB
Zig
439 lines
16 KiB
Zig
const builtin = @import("builtin");
|
|
const std = @import("std");
|
|
const mem = std.mem;
|
|
const Allocator = std.mem.Allocator;
|
|
const assert = std.debug.assert;
|
|
const Cache = std.Build.Cache;
|
|
|
|
fn usage() noreturn {
|
|
std.fs.File.stdout().writeAll(
|
|
\\Usage: zig std [options]
|
|
\\
|
|
\\Options:
|
|
\\ -h, --help Print this help and exit
|
|
\\ -p [port], --port [port] Port to listen on. Default is 0, meaning an ephemeral port chosen by the system.
|
|
\\ --[no-]open-browser Force enabling or disabling opening a browser tab to the served website.
|
|
\\ By default, enabled unless a port is specified.
|
|
\\
|
|
) catch {};
|
|
std.process.exit(1);
|
|
}
|
|
|
|
pub fn main() !void {
|
|
var arena_instance = std.heap.ArenaAllocator.init(std.heap.page_allocator);
|
|
defer arena_instance.deinit();
|
|
const arena = arena_instance.allocator();
|
|
|
|
var general_purpose_allocator: std.heap.GeneralPurposeAllocator(.{}) = .init;
|
|
const gpa = general_purpose_allocator.allocator();
|
|
|
|
var argv = try std.process.argsWithAllocator(arena);
|
|
defer argv.deinit();
|
|
assert(argv.skip());
|
|
const zig_lib_directory = argv.next().?;
|
|
const zig_exe_path = argv.next().?;
|
|
const global_cache_path = argv.next().?;
|
|
|
|
var lib_dir = try std.fs.cwd().openDir(zig_lib_directory, .{});
|
|
defer lib_dir.close();
|
|
|
|
var listen_port: u16 = 0;
|
|
var force_open_browser: ?bool = null;
|
|
while (argv.next()) |arg| {
|
|
if (mem.eql(u8, arg, "-h") or mem.eql(u8, arg, "--help")) {
|
|
usage();
|
|
} else if (mem.eql(u8, arg, "-p") or mem.eql(u8, arg, "--port")) {
|
|
listen_port = std.fmt.parseInt(u16, argv.next() orelse usage(), 10) catch |err| {
|
|
std.log.err("expected port number: {}", .{err});
|
|
usage();
|
|
};
|
|
} else if (mem.eql(u8, arg, "--open-browser")) {
|
|
force_open_browser = true;
|
|
} else if (mem.eql(u8, arg, "--no-open-browser")) {
|
|
force_open_browser = false;
|
|
} else {
|
|
std.log.err("unrecognized argument: {s}", .{arg});
|
|
usage();
|
|
}
|
|
}
|
|
const should_open_browser = force_open_browser orelse (listen_port == 0);
|
|
|
|
const address = std.net.Address.parseIp("127.0.0.1", listen_port) catch unreachable;
|
|
var http_server = try address.listen(.{
|
|
.reuse_address = true,
|
|
});
|
|
const port = http_server.listen_address.in.getPort();
|
|
const url_with_newline = try std.fmt.allocPrint(arena, "http://127.0.0.1:{d}/\n", .{port});
|
|
std.fs.File.stdout().writeAll(url_with_newline) catch {};
|
|
if (should_open_browser) {
|
|
openBrowserTab(gpa, url_with_newline[0 .. url_with_newline.len - 1 :'\n']) catch |err| {
|
|
std.log.err("unable to open browser: {s}", .{@errorName(err)});
|
|
};
|
|
}
|
|
|
|
var context: Context = .{
|
|
.gpa = gpa,
|
|
.zig_exe_path = zig_exe_path,
|
|
.global_cache_path = global_cache_path,
|
|
.lib_dir = lib_dir,
|
|
.zig_lib_directory = zig_lib_directory,
|
|
};
|
|
|
|
while (true) {
|
|
const connection = try http_server.accept();
|
|
_ = std.Thread.spawn(.{}, accept, .{ &context, connection }) catch |err| {
|
|
std.log.err("unable to accept connection: {s}", .{@errorName(err)});
|
|
connection.stream.close();
|
|
continue;
|
|
};
|
|
}
|
|
}
|
|
|
|
fn accept(context: *Context, connection: std.net.Server.Connection) void {
|
|
defer connection.stream.close();
|
|
|
|
var recv_buffer: [4000]u8 = undefined;
|
|
var send_buffer: [4000]u8 = undefined;
|
|
var conn_reader = connection.stream.reader(&recv_buffer);
|
|
var conn_writer = connection.stream.writer(&send_buffer);
|
|
var server = std.http.Server.init(conn_reader.interface(), &conn_writer.interface);
|
|
while (server.reader.state == .ready) {
|
|
var request = server.receiveHead() catch |err| switch (err) {
|
|
error.HttpConnectionClosing => return,
|
|
else => {
|
|
std.log.err("closing http connection: {s}", .{@errorName(err)});
|
|
return;
|
|
},
|
|
};
|
|
serveRequest(&request, context) catch |err| switch (err) {
|
|
error.WriteFailed => {
|
|
if (conn_writer.err) |e| {
|
|
std.log.err("unable to serve {s}: {s}", .{ request.head.target, @errorName(e) });
|
|
} else {
|
|
std.log.err("unable to serve {s}: {s}", .{ request.head.target, @errorName(err) });
|
|
}
|
|
return;
|
|
},
|
|
else => {
|
|
std.log.err("unable to serve {s}: {s}", .{ request.head.target, @errorName(err) });
|
|
return;
|
|
},
|
|
};
|
|
}
|
|
}
|
|
|
|
const Context = struct {
|
|
gpa: Allocator,
|
|
lib_dir: std.fs.Dir,
|
|
zig_lib_directory: []const u8,
|
|
zig_exe_path: []const u8,
|
|
global_cache_path: []const u8,
|
|
};
|
|
|
|
fn serveRequest(request: *std.http.Server.Request, context: *Context) !void {
|
|
if (std.mem.eql(u8, request.head.target, "/") or
|
|
std.mem.eql(u8, request.head.target, "/debug") or
|
|
std.mem.eql(u8, request.head.target, "/debug/"))
|
|
{
|
|
try serveDocsFile(request, context, "docs/index.html", "text/html");
|
|
} else if (std.mem.eql(u8, request.head.target, "/main.js") or
|
|
std.mem.eql(u8, request.head.target, "/debug/main.js"))
|
|
{
|
|
try serveDocsFile(request, context, "docs/main.js", "application/javascript");
|
|
} else if (std.mem.eql(u8, request.head.target, "/main.wasm")) {
|
|
try serveWasm(request, context, .ReleaseFast);
|
|
} else if (std.mem.eql(u8, request.head.target, "/debug/main.wasm")) {
|
|
try serveWasm(request, context, .Debug);
|
|
} else if (std.mem.eql(u8, request.head.target, "/sources.tar") or
|
|
std.mem.eql(u8, request.head.target, "/debug/sources.tar"))
|
|
{
|
|
try serveSourcesTar(request, context);
|
|
} else {
|
|
try request.respond("not found", .{
|
|
.status = .not_found,
|
|
.extra_headers = &.{
|
|
.{ .name = "content-type", .value = "text/plain" },
|
|
},
|
|
});
|
|
}
|
|
}
|
|
|
|
const cache_control_header: std.http.Header = .{
|
|
.name = "cache-control",
|
|
.value = "max-age=0, must-revalidate",
|
|
};
|
|
|
|
fn serveDocsFile(
|
|
request: *std.http.Server.Request,
|
|
context: *Context,
|
|
name: []const u8,
|
|
content_type: []const u8,
|
|
) !void {
|
|
const gpa = context.gpa;
|
|
// The desired API is actually sendfile, which will require enhancing std.http.Server.
|
|
// We load the file with every request so that the user can make changes to the file
|
|
// and refresh the HTML page without restarting this server.
|
|
const file_contents = try context.lib_dir.readFileAlloc(name, gpa, .limited(10 * 1024 * 1024));
|
|
defer gpa.free(file_contents);
|
|
try request.respond(file_contents, .{
|
|
.extra_headers = &.{
|
|
.{ .name = "content-type", .value = content_type },
|
|
cache_control_header,
|
|
},
|
|
});
|
|
}
|
|
|
|
fn serveSourcesTar(request: *std.http.Server.Request, context: *Context) !void {
|
|
const gpa = context.gpa;
|
|
|
|
var send_buffer: [0x4000]u8 = undefined;
|
|
var response = try request.respondStreaming(&send_buffer, .{
|
|
.respond_options = .{
|
|
.extra_headers = &.{
|
|
.{ .name = "content-type", .value = "application/x-tar" },
|
|
cache_control_header,
|
|
},
|
|
},
|
|
});
|
|
|
|
var std_dir = try context.lib_dir.openDir("std", .{ .iterate = true });
|
|
defer std_dir.close();
|
|
|
|
var walker = try std_dir.walk(gpa);
|
|
defer walker.deinit();
|
|
|
|
var archiver: std.tar.Writer = .{ .underlying_writer = &response.writer };
|
|
archiver.prefix = "std";
|
|
|
|
while (try walker.next()) |entry| {
|
|
switch (entry.kind) {
|
|
.file => {
|
|
if (!std.mem.endsWith(u8, entry.basename, ".zig"))
|
|
continue;
|
|
if (std.mem.endsWith(u8, entry.basename, "test.zig"))
|
|
continue;
|
|
},
|
|
else => continue,
|
|
}
|
|
var file = try entry.dir.openFile(entry.basename, .{});
|
|
defer file.close();
|
|
const stat = try file.stat();
|
|
var file_reader: std.fs.File.Reader = .{
|
|
.file = file,
|
|
.interface = std.fs.File.Reader.initInterface(&.{}),
|
|
.size = stat.size,
|
|
};
|
|
try archiver.writeFile(entry.path, &file_reader, stat.mtime);
|
|
}
|
|
|
|
{
|
|
// Since this command is JIT compiled, the builtin module available in
|
|
// this source file corresponds to the user's host system.
|
|
const builtin_zig = @embedFile("builtin");
|
|
archiver.prefix = "builtin";
|
|
try archiver.writeFileBytes("builtin.zig", builtin_zig, .{});
|
|
}
|
|
|
|
// intentionally omitting the pointless trailer
|
|
//try archiver.finish();
|
|
try response.end();
|
|
}
|
|
|
|
fn serveWasm(
|
|
request: *std.http.Server.Request,
|
|
context: *Context,
|
|
optimize_mode: std.builtin.OptimizeMode,
|
|
) !void {
|
|
const gpa = context.gpa;
|
|
|
|
var arena_instance = std.heap.ArenaAllocator.init(gpa);
|
|
defer arena_instance.deinit();
|
|
const arena = arena_instance.allocator();
|
|
|
|
// Do the compilation every request, so that the user can edit the files
|
|
// and see the changes without restarting the server.
|
|
const wasm_base_path = try buildWasmBinary(arena, context, optimize_mode);
|
|
const bin_name = try std.zig.binNameAlloc(arena, .{
|
|
.root_name = autodoc_root_name,
|
|
.target = &(std.zig.system.resolveTargetQuery(std.Build.parseTargetQuery(.{
|
|
.arch_os_abi = autodoc_arch_os_abi,
|
|
.cpu_features = autodoc_cpu_features,
|
|
}) catch unreachable) catch unreachable),
|
|
.output_mode = .Exe,
|
|
});
|
|
// std.http.Server does not have a sendfile API yet.
|
|
const bin_path = try wasm_base_path.join(arena, bin_name);
|
|
const file_contents = try bin_path.root_dir.handle.readFileAlloc(bin_path.sub_path, gpa, .limited(10 * 1024 * 1024));
|
|
defer gpa.free(file_contents);
|
|
try request.respond(file_contents, .{
|
|
.extra_headers = &.{
|
|
.{ .name = "content-type", .value = "application/wasm" },
|
|
cache_control_header,
|
|
},
|
|
});
|
|
}
|
|
|
|
const autodoc_root_name = "autodoc";
|
|
const autodoc_arch_os_abi = "wasm32-freestanding";
|
|
const autodoc_cpu_features = "baseline+atomics+bulk_memory+multivalue+mutable_globals+nontrapping_fptoint+reference_types+sign_ext";
|
|
|
|
fn buildWasmBinary(
|
|
arena: Allocator,
|
|
context: *Context,
|
|
optimize_mode: std.builtin.OptimizeMode,
|
|
) !Cache.Path {
|
|
const gpa = context.gpa;
|
|
|
|
var argv: std.ArrayListUnmanaged([]const u8) = .empty;
|
|
|
|
try argv.appendSlice(arena, &.{
|
|
context.zig_exe_path, //
|
|
"build-exe", //
|
|
"-fno-entry", //
|
|
"-O", @tagName(optimize_mode), //
|
|
"-target", autodoc_arch_os_abi, //
|
|
"-mcpu", autodoc_cpu_features, //
|
|
"--cache-dir", context.global_cache_path, //
|
|
"--global-cache-dir", context.global_cache_path, //
|
|
"--name", autodoc_root_name, //
|
|
"-rdynamic", //
|
|
"--dep", "Walk", //
|
|
try std.fmt.allocPrint(
|
|
arena,
|
|
"-Mroot={s}/docs/wasm/main.zig",
|
|
.{context.zig_lib_directory},
|
|
),
|
|
try std.fmt.allocPrint(
|
|
arena,
|
|
"-MWalk={s}/docs/wasm/Walk.zig",
|
|
.{context.zig_lib_directory},
|
|
),
|
|
"--listen=-", //
|
|
});
|
|
|
|
var child = std.process.Child.init(argv.items, gpa);
|
|
child.stdin_behavior = .Pipe;
|
|
child.stdout_behavior = .Pipe;
|
|
child.stderr_behavior = .Pipe;
|
|
try child.spawn();
|
|
|
|
var poller = std.Io.poll(gpa, enum { stdout, stderr }, .{
|
|
.stdout = child.stdout.?,
|
|
.stderr = child.stderr.?,
|
|
});
|
|
defer poller.deinit();
|
|
|
|
try sendMessage(child.stdin.?, .update);
|
|
try sendMessage(child.stdin.?, .exit);
|
|
|
|
var result: ?Cache.Path = null;
|
|
var result_error_bundle = std.zig.ErrorBundle.empty;
|
|
|
|
const stdout = poller.reader(.stdout);
|
|
|
|
poll: while (true) {
|
|
const Header = std.zig.Server.Message.Header;
|
|
while (stdout.buffered().len < @sizeOf(Header)) if (!try poller.poll()) break :poll;
|
|
const header = stdout.takeStruct(Header, .little) catch unreachable;
|
|
while (stdout.buffered().len < header.bytes_len) if (!try poller.poll()) break :poll;
|
|
const body = stdout.take(header.bytes_len) catch unreachable;
|
|
|
|
switch (header.tag) {
|
|
.zig_version => {
|
|
if (!std.mem.eql(u8, builtin.zig_version_string, body)) {
|
|
return error.ZigProtocolVersionMismatch;
|
|
}
|
|
},
|
|
.error_bundle => {
|
|
result_error_bundle = try std.zig.Server.allocErrorBundle(arena, body);
|
|
},
|
|
.emit_digest => {
|
|
const EmitDigest = std.zig.Server.Message.EmitDigest;
|
|
const emit_digest = @as(*align(1) const EmitDigest, @ptrCast(body));
|
|
if (!emit_digest.flags.cache_hit) {
|
|
std.log.info("source changes detected; rebuilt wasm component", .{});
|
|
}
|
|
const digest = body[@sizeOf(EmitDigest)..][0..Cache.bin_digest_len];
|
|
result = .{
|
|
.root_dir = Cache.Directory.cwd(),
|
|
.sub_path = try std.fs.path.join(arena, &.{
|
|
context.global_cache_path, "o" ++ std.fs.path.sep_str ++ Cache.binToHex(digest.*),
|
|
}),
|
|
};
|
|
},
|
|
else => {}, // ignore other messages
|
|
}
|
|
}
|
|
|
|
const stderr = poller.reader(.stderr);
|
|
if (stderr.bufferedLen() > 0) {
|
|
std.debug.print("{s}", .{stderr.buffered()});
|
|
}
|
|
|
|
// Send EOF to stdin.
|
|
child.stdin.?.close();
|
|
child.stdin = null;
|
|
|
|
switch (try child.wait()) {
|
|
.Exited => |code| {
|
|
if (code != 0) {
|
|
std.log.err(
|
|
"the following command exited with error code {d}:\n{s}",
|
|
.{ code, try std.Build.Step.allocPrintCmd(arena, null, argv.items) },
|
|
);
|
|
return error.WasmCompilationFailed;
|
|
}
|
|
},
|
|
.Signal, .Stopped, .Unknown => {
|
|
std.log.err(
|
|
"the following command terminated unexpectedly:\n{s}",
|
|
.{try std.Build.Step.allocPrintCmd(arena, null, argv.items)},
|
|
);
|
|
return error.WasmCompilationFailed;
|
|
},
|
|
}
|
|
|
|
if (result_error_bundle.errorMessageCount() > 0) {
|
|
result_error_bundle.renderToStdErr(.{}, true);
|
|
std.log.err("the following command failed with {d} compilation errors:\n{s}", .{
|
|
result_error_bundle.errorMessageCount(),
|
|
try std.Build.Step.allocPrintCmd(arena, null, argv.items),
|
|
});
|
|
return error.WasmCompilationFailed;
|
|
}
|
|
|
|
return result orelse {
|
|
std.log.err("child process failed to report result\n{s}", .{
|
|
try std.Build.Step.allocPrintCmd(arena, null, argv.items),
|
|
});
|
|
return error.WasmCompilationFailed;
|
|
};
|
|
}
|
|
|
|
fn sendMessage(file: std.fs.File, tag: std.zig.Client.Message.Tag) !void {
|
|
const header: std.zig.Client.Message.Header = .{
|
|
.tag = tag,
|
|
.bytes_len = 0,
|
|
};
|
|
try file.writeAll(std.mem.asBytes(&header));
|
|
}
|
|
|
|
fn openBrowserTab(gpa: Allocator, url: []const u8) !void {
|
|
// Until https://github.com/ziglang/zig/issues/19205 is implemented, we
|
|
// spawn a thread for this child process.
|
|
_ = try std.Thread.spawn(.{}, openBrowserTabThread, .{ gpa, url });
|
|
}
|
|
|
|
fn openBrowserTabThread(gpa: Allocator, url: []const u8) !void {
|
|
const main_exe = switch (builtin.os.tag) {
|
|
.windows => "explorer",
|
|
.macos => "open",
|
|
else => "xdg-open",
|
|
};
|
|
var child = std.process.Child.init(&.{ main_exe, url }, gpa);
|
|
child.stdin_behavior = .Ignore;
|
|
child.stdout_behavior = .Ignore;
|
|
child.stderr_behavior = .Ignore;
|
|
try child.spawn();
|
|
_ = try child.wait();
|
|
}
|