mirror of
https://codeberg.org/ziglang/zig.git
synced 2025-12-06 13:54:21 +00:00
442 lines
16 KiB
Zig
442 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.ArrayList([]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 => {
|
|
var r: std.Io.Reader = .fixed(body);
|
|
const emit_digest = r.takeStruct(std.zig.Server.Message.EmitDigest, .little) catch unreachable;
|
|
if (!emit_digest.flags.cache_hit) {
|
|
std.log.info("source changes detected; rebuilt wasm component", .{});
|
|
}
|
|
const digest = r.takeArray(Cache.bin_digest_len) catch unreachable;
|
|
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,
|
|
};
|
|
var w = file.writer(&.{});
|
|
w.interface.writeStruct(header, .little) catch |err| switch (err) {
|
|
error.WriteFailed => return w.err.?,
|
|
};
|
|
}
|
|
|
|
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();
|
|
}
|