Files
bun.sh/src/tmp.zig
2023-12-02 06:54:33 -08:00

84 lines
2.9 KiB
Zig

const bun = @import("root").bun;
const std = @import("std");
const Environment = bun.Environment;
const O = std.os.O;
// O_TMPFILE doesn't seem to work very well.
const allow_tmpfile = false;
// To be used with files
// not folders!
pub const Tmpfile = struct {
destination_dir: bun.FileDescriptor = bun.invalid_fd,
tmpfilename: [:0]const u8 = "",
fd: bun.FileDescriptor = bun.invalid_fd,
using_tmpfile: bool = allow_tmpfile,
pub fn create(
destination_dir: bun.FileDescriptor,
tmpfilename: [:0]const u8,
) bun.JSC.Maybe(Tmpfile) {
const perm = 0o644;
var tmpfile = Tmpfile{
.destination_dir = destination_dir,
.tmpfilename = tmpfilename,
};
open: while (true) {
if (comptime allow_tmpfile) {
switch (bun.sys.openat(destination_dir, ".", O.WRONLY | O.TMPFILE | O.CLOEXEC, perm)) {
.result => |fd| {
tmpfile.fd = fd;
break :open;
},
.err => |err| {
switch (err.getErrno()) {
.INVAL, .OPNOTSUPP, .NOSYS => {
tmpfile.using_tmpfile = false;
},
else => return .{ .err = err },
}
},
}
}
tmpfile.fd = switch (bun.sys.openat(destination_dir, tmpfilename, O.CREAT | O.CLOEXEC | O.WRONLY, perm)) {
.result => |fd| fd,
.err => |err| return .{ .err = err },
};
break :open;
}
return .{ .result = tmpfile };
}
pub fn finish(this: *Tmpfile, destname: [:0]const u8) !void {
if (comptime allow_tmpfile) {
if (this.using_tmpfile) {
var retry = true;
const basename: [:0]const u8 = @ptrCast(std.fs.path.basename(destname));
while (retry) {
const ret = bun.sys.linkatTmpfile(this.fd, this.destination_dir, basename);
switch (ret) {
.result => {
return;
},
.err => |err| {
if (err.getErrno() == .EXIST and retry) {
_ = bun.sys.unlinkat(this.destination_dir, basename);
retry = false;
continue;
} else {
try ret.unwrap();
return;
}
},
}
}
}
}
try bun.C.moveFileZWithHandle(bun.fdcast(this.fd), this.destination_dir, this.tmpfilename, bun.fdcast(this.destination_dir), destname);
}
};