mirror of
https://github.com/oven-sh/bun
synced 2026-02-13 12:29:07 +00:00
326 lines
13 KiB
Zig
326 lines
13 KiB
Zig
//! Process information and control APIs (`globalThis.process` / `node:process`)
|
|
comptime {
|
|
@export(&getTitle, .{ .name = "Bun__Process__getTitle" });
|
|
@export(&setTitle, .{ .name = "Bun__Process__setTitle" });
|
|
@export(&getArgv, .{ .name = "Bun__Process__getArgv" });
|
|
@export(&getCwd, .{ .name = "Bun__Process__getCwd" });
|
|
@export(&setCwd, .{ .name = "Bun__Process__setCwd" });
|
|
@export(&exit, .{ .name = "Bun__Process__exit" });
|
|
@export(&getArgv0, .{ .name = "Bun__Process__getArgv0" });
|
|
@export(&getExecPath, .{ .name = "Bun__Process__getExecPath" });
|
|
@export(&getExecArgv, .{ .name = "Bun__Process__getExecArgv" });
|
|
}
|
|
|
|
var title_mutex = bun.Mutex{};
|
|
|
|
pub fn getTitle(_: *JSGlobalObject, title: *ZigString) callconv(.C) void {
|
|
title_mutex.lock();
|
|
defer title_mutex.unlock();
|
|
const str = bun.CLI.Bun__Node__ProcessTitle;
|
|
title.* = ZigString.init(str orelse "bun");
|
|
}
|
|
|
|
// TODO: https://github.com/nodejs/node/blob/master/deps/uv/src/unix/darwin-proctitle.c
|
|
pub fn setTitle(globalObject: *JSGlobalObject, newvalue: *ZigString) callconv(.C) JSValue {
|
|
title_mutex.lock();
|
|
defer title_mutex.unlock();
|
|
if (bun.CLI.Bun__Node__ProcessTitle) |_| bun.default_allocator.free(bun.CLI.Bun__Node__ProcessTitle.?);
|
|
bun.CLI.Bun__Node__ProcessTitle = newvalue.dupe(bun.default_allocator) catch bun.outOfMemory();
|
|
return newvalue.toJS(globalObject);
|
|
}
|
|
|
|
pub fn getArgv0(globalObject: *JSC.JSGlobalObject) callconv(.C) JSC.JSValue {
|
|
return JSC.ZigString.fromUTF8(bun.argv[0]).toJS(globalObject);
|
|
}
|
|
|
|
pub fn getExecPath(globalObject: *JSC.JSGlobalObject) callconv(.C) JSC.JSValue {
|
|
const out = bun.selfExePath() catch {
|
|
// if for any reason we are unable to get the executable path, we just return argv[0]
|
|
return getArgv0(globalObject);
|
|
};
|
|
|
|
return JSC.ZigString.fromUTF8(out).toJS(globalObject);
|
|
}
|
|
|
|
pub fn getExecArgv(globalObject: *JSC.JSGlobalObject) callconv(.C) JSC.JSValue {
|
|
var sfb = std.heap.stackFallback(4096, globalObject.allocator());
|
|
const temp_alloc = sfb.get();
|
|
const vm = globalObject.bunVM();
|
|
|
|
if (vm.worker) |worker| {
|
|
// was explicitly overridden for the worker?
|
|
if (worker.execArgv) |execArgv| {
|
|
const array = JSC.JSValue.createEmptyArray(globalObject, execArgv.len);
|
|
for (0..execArgv.len) |i| {
|
|
array.putIndex(globalObject, @intCast(i), bun.String.init(execArgv[i]).toJS(globalObject));
|
|
}
|
|
return array;
|
|
}
|
|
}
|
|
|
|
var args = std.ArrayList(bun.String).initCapacity(temp_alloc, bun.argv.len - 1) catch bun.outOfMemory();
|
|
defer args.deinit();
|
|
defer for (args.items) |*arg| arg.deref();
|
|
|
|
var seen_run = false;
|
|
var prev: ?[]const u8 = null;
|
|
|
|
// we re-parse the process argv to extract execArgv, since this is a very uncommon operation
|
|
// it isn't worth doing this as a part of the CLI
|
|
for (bun.argv[@min(1, bun.argv.len)..]) |arg| {
|
|
defer prev = arg;
|
|
|
|
if (arg.len >= 1 and arg[0] == '-') {
|
|
args.append(bun.String.createUTF8(arg)) catch bun.outOfMemory();
|
|
continue;
|
|
}
|
|
|
|
if (!seen_run and bun.strings.eqlComptime(arg, "run")) {
|
|
seen_run = true;
|
|
continue;
|
|
}
|
|
|
|
// A set of execArgv args consume an extra argument, so we do not want to
|
|
// confuse these with script names.
|
|
const map = bun.ComptimeStringMap(void, comptime brk: {
|
|
const auto_params = bun.CLI.Arguments.auto_params;
|
|
const KV = struct { []const u8, void };
|
|
var entries: [auto_params.len]KV = undefined;
|
|
var i = 0;
|
|
for (auto_params) |param| {
|
|
if (param.takes_value != .none) {
|
|
if (param.names.long) |name| {
|
|
entries[i] = .{ "--" ++ name, {} };
|
|
i += 1;
|
|
}
|
|
if (param.names.short) |name| {
|
|
entries[i] = .{ &[_]u8{ '-', name }, {} };
|
|
i += 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
var result: [i]KV = undefined;
|
|
@memcpy(&result, entries[0..i]);
|
|
break :brk result;
|
|
});
|
|
|
|
if (prev) |p| if (map.has(p)) {
|
|
args.append(bun.String.createUTF8(arg)) catch @panic("OOM");
|
|
continue;
|
|
};
|
|
|
|
// we hit the script name
|
|
break;
|
|
}
|
|
|
|
return bun.String.toJSArray(globalObject, args.items);
|
|
}
|
|
|
|
pub fn getArgv(globalObject: *JSC.JSGlobalObject) callconv(.C) JSC.JSValue {
|
|
const vm = globalObject.bunVM();
|
|
|
|
// Allocate up to 32 strings in stack
|
|
var stack_fallback_allocator = std.heap.stackFallback(
|
|
32 * @sizeOf(JSC.ZigString) + (bun.MAX_PATH_BYTES + 1) + 32,
|
|
bun.default_allocator,
|
|
);
|
|
const allocator = stack_fallback_allocator.get();
|
|
|
|
var args_count: usize = vm.argv.len;
|
|
if (vm.worker) |worker| {
|
|
args_count = worker.argv.len;
|
|
}
|
|
|
|
const args = allocator.alloc(
|
|
bun.String,
|
|
// argv omits "bun" because it could be "bun run" or "bun" and it's kind of ambiguous
|
|
// argv also omits the script name
|
|
args_count + 2,
|
|
) catch bun.outOfMemory();
|
|
var args_list: std.ArrayListUnmanaged(bun.String) = .initBuffer(args);
|
|
|
|
if (vm.standalone_module_graph != null) {
|
|
// Don't break user's code because they did process.argv.slice(2)
|
|
// Even if they didn't type "bun", we still want to add it as argv[0]
|
|
args_list.appendAssumeCapacity(
|
|
bun.String.static("bun"),
|
|
);
|
|
} else {
|
|
const exe_path = bun.selfExePath() catch null;
|
|
args_list.appendAssumeCapacity(
|
|
if (exe_path) |str| bun.String.fromUTF8(str) else bun.String.static("bun"),
|
|
);
|
|
}
|
|
|
|
if (vm.main.len > 0 and
|
|
!strings.endsWithComptime(vm.main, bun.pathLiteral("/[eval]")) and
|
|
!strings.endsWithComptime(vm.main, bun.pathLiteral("/[stdin]")))
|
|
{
|
|
if (vm.worker != null and vm.worker.?.eval_mode) {
|
|
args_list.appendAssumeCapacity(bun.String.static("[worker eval]"));
|
|
} else {
|
|
args_list.appendAssumeCapacity(bun.String.fromUTF8(vm.main));
|
|
}
|
|
}
|
|
|
|
defer allocator.free(args);
|
|
|
|
if (vm.worker) |worker| {
|
|
for (worker.argv) |arg| {
|
|
args_list.appendAssumeCapacity(bun.String.init(arg));
|
|
}
|
|
} else {
|
|
for (vm.argv) |arg| {
|
|
const str = bun.String.fromUTF8(arg);
|
|
// https://github.com/yargs/yargs/blob/adb0d11e02c613af3d9427b3028cc192703a3869/lib/utils/process-argv.ts#L1
|
|
args_list.appendAssumeCapacity(str);
|
|
}
|
|
}
|
|
|
|
return bun.String.toJSArray(globalObject, args_list.items);
|
|
}
|
|
|
|
pub fn getCwd(globalObject: *JSC.JSGlobalObject) callconv(.C) JSC.JSValue {
|
|
return JSC.toJSHostValue(globalObject, getCwd_(globalObject));
|
|
}
|
|
fn getCwd_(globalObject: *JSC.JSGlobalObject) bun.JSError!JSC.JSValue {
|
|
var buf: bun.PathBuffer = undefined;
|
|
switch (bun.api.node.path.getCwd(&buf)) {
|
|
.result => |r| return JSC.ZigString.init(r).withEncoding().toJS(globalObject),
|
|
.err => |e| {
|
|
return globalObject.throwValue(e.toJSC(globalObject));
|
|
},
|
|
}
|
|
}
|
|
|
|
pub fn setCwd(globalObject: *JSC.JSGlobalObject, to: *JSC.ZigString) callconv(.C) JSC.JSValue {
|
|
return JSC.toJSHostValue(globalObject, setCwd_(globalObject, to));
|
|
}
|
|
fn setCwd_(globalObject: *JSC.JSGlobalObject, to: *JSC.ZigString) bun.JSError!JSC.JSValue {
|
|
if (to.len == 0) {
|
|
return globalObject.throwInvalidArguments("Expected path to be a non-empty string", .{});
|
|
}
|
|
const vm = globalObject.bunVM();
|
|
const fs = vm.transpiler.fs;
|
|
|
|
var buf: bun.PathBuffer = undefined;
|
|
const slice = to.sliceZBuf(&buf) catch return globalObject.throw("Invalid path", .{});
|
|
|
|
switch (Syscall.chdir(fs.top_level_dir, slice)) {
|
|
.result => {
|
|
// When we update the cwd from JS, we have to update the bundler's version as well
|
|
// However, this might be called many times in a row, so we use a pre-allocated buffer
|
|
// that way we don't have to worry about garbage collector
|
|
const into_cwd_buf = switch (bun.sys.getcwd(&buf)) {
|
|
.result => |r| r,
|
|
.err => |err| {
|
|
_ = Syscall.chdir(fs.top_level_dir, fs.top_level_dir);
|
|
return globalObject.throwValue(err.toJSC(globalObject));
|
|
},
|
|
};
|
|
@memcpy(fs.top_level_dir_buf[0..into_cwd_buf.len], into_cwd_buf);
|
|
fs.top_level_dir_buf[into_cwd_buf.len] = 0;
|
|
fs.top_level_dir = fs.top_level_dir_buf[0..into_cwd_buf.len :0];
|
|
|
|
const len = fs.top_level_dir.len;
|
|
// Ensure the path ends with a slash
|
|
if (fs.top_level_dir_buf[len - 1] != std.fs.path.sep) {
|
|
fs.top_level_dir_buf[len] = std.fs.path.sep;
|
|
fs.top_level_dir_buf[len + 1] = 0;
|
|
fs.top_level_dir = fs.top_level_dir_buf[0 .. len + 1 :0];
|
|
}
|
|
const withoutTrailingSlash = if (Environment.isWindows) strings.withoutTrailingSlashWindowsPath else strings.withoutTrailingSlash;
|
|
var str = bun.String.createUTF8(withoutTrailingSlash(fs.top_level_dir));
|
|
return str.transferToJS(globalObject);
|
|
},
|
|
.err => |e| {
|
|
return globalObject.throwValue(e.toJSC(globalObject));
|
|
},
|
|
}
|
|
}
|
|
|
|
// TODO(@190n) this may need to be noreturn
|
|
pub fn exit(globalObject: *JSC.JSGlobalObject, code: u8) callconv(.c) void {
|
|
var vm = globalObject.bunVM();
|
|
if (vm.worker) |worker| {
|
|
vm.exit_handler.exit_code = code;
|
|
worker.requestTerminate();
|
|
return;
|
|
}
|
|
|
|
vm.exit_handler.exit_code = code;
|
|
vm.onExit();
|
|
vm.globalExit();
|
|
}
|
|
|
|
// TODO: switch this to using *bun.wtf.String when it is added
|
|
pub fn Bun__Process__editWindowsEnvVar(k: bun.String, v: bun.String) callconv(.C) void {
|
|
comptime bun.assert(bun.Environment.isWindows);
|
|
if (k.tag == .Empty) return;
|
|
const wtf1 = k.value.WTFStringImpl;
|
|
var fixed_stack_allocator = std.heap.stackFallback(1025, bun.default_allocator);
|
|
const allocator = fixed_stack_allocator.get();
|
|
var buf1 = allocator.alloc(u16, k.utf16ByteLength() + 1) catch bun.outOfMemory();
|
|
defer allocator.free(buf1);
|
|
var buf2 = allocator.alloc(u16, v.utf16ByteLength() + 1) catch bun.outOfMemory();
|
|
defer allocator.free(buf2);
|
|
const len1: usize = switch (wtf1.is8Bit()) {
|
|
true => bun.strings.copyLatin1IntoUTF16([]u16, buf1, []const u8, wtf1.latin1Slice()).written,
|
|
false => b: {
|
|
@memcpy(buf1[0..wtf1.length()], wtf1.utf16Slice());
|
|
break :b wtf1.length();
|
|
},
|
|
};
|
|
buf1[len1] = 0;
|
|
const str2: ?[*:0]const u16 = if (v.tag != .Dead) str: {
|
|
if (v.tag == .Empty) break :str (&[_]u16{0})[0..0 :0];
|
|
const wtf2 = v.value.WTFStringImpl;
|
|
const len2: usize = switch (wtf2.is8Bit()) {
|
|
true => bun.strings.copyLatin1IntoUTF16([]u16, buf2, []const u8, wtf2.latin1Slice()).written,
|
|
false => b: {
|
|
@memcpy(buf2[0..wtf2.length()], wtf2.utf16Slice());
|
|
break :b wtf2.length();
|
|
},
|
|
};
|
|
buf2[len2] = 0;
|
|
break :str buf2[0..len2 :0].ptr;
|
|
} else null;
|
|
_ = bun.c.SetEnvironmentVariableW(buf1[0..len1 :0].ptr, str2);
|
|
}
|
|
|
|
comptime {
|
|
if (Environment.export_cpp_apis and Environment.isWindows) {
|
|
@export(&Bun__Process__editWindowsEnvVar, .{ .name = "Bun__Process__editWindowsEnvVar" });
|
|
}
|
|
}
|
|
|
|
pub export const Bun__version: [*:0]const u8 = "v" ++ bun.Global.package_json_version;
|
|
pub export const Bun__version_with_sha: [*:0]const u8 = "v" ++ bun.Global.package_json_version_with_sha;
|
|
pub export const Bun__versions_boringssl: [*:0]const u8 = bun.Global.versions.boringssl;
|
|
pub export const Bun__versions_libarchive: [*:0]const u8 = bun.Global.versions.libarchive;
|
|
pub export const Bun__versions_mimalloc: [*:0]const u8 = bun.Global.versions.mimalloc;
|
|
pub export const Bun__versions_picohttpparser: [*:0]const u8 = bun.Global.versions.picohttpparser;
|
|
pub export const Bun__versions_uws: [*:0]const u8 = bun.Environment.git_sha;
|
|
pub export const Bun__versions_webkit: [*:0]const u8 = bun.Global.versions.webkit;
|
|
pub export const Bun__versions_zig: [*:0]const u8 = bun.Global.versions.zig;
|
|
pub export const Bun__versions_zlib: [*:0]const u8 = bun.Global.versions.zlib;
|
|
pub export const Bun__versions_tinycc: [*:0]const u8 = bun.Global.versions.tinycc;
|
|
pub export const Bun__versions_lolhtml: [*:0]const u8 = bun.Global.versions.lolhtml;
|
|
pub export const Bun__versions_c_ares: [*:0]const u8 = bun.Global.versions.c_ares;
|
|
pub export const Bun__versions_libdeflate: [*:0]const u8 = bun.Global.versions.libdeflate;
|
|
pub export const Bun__versions_usockets: [*:0]const u8 = bun.Environment.git_sha;
|
|
pub export const Bun__version_sha: [*:0]const u8 = bun.Environment.git_sha;
|
|
pub export const Bun__versions_lshpack: [*:0]const u8 = bun.Global.versions.lshpack;
|
|
pub export const Bun__versions_zstd: [*:0]const u8 = bun.Global.versions.zstd;
|
|
|
|
const std = @import("std");
|
|
const Environment = bun.Environment;
|
|
const bun = @import("bun");
|
|
const JSC = bun.JSC;
|
|
const JSGlobalObject = JSC.JSGlobalObject;
|
|
const JSValue = JSC.JSValue;
|
|
const ZigString = JSC.ZigString;
|
|
const Syscall = bun.sys;
|
|
const strings = bun.strings;
|