mirror of
https://github.com/oven-sh/bun
synced 2026-02-09 10:28:47 +00:00
Compare commits
6 Commits
claude/tes
...
ali/test-f
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
5395767685 | ||
|
|
352cecde32 | ||
|
|
2cca4efbe7 | ||
|
|
eb5cb58e56 | ||
|
|
6c6d46333f | ||
|
|
0c40eaf8ab |
@@ -554,6 +554,7 @@ JSC_DEFINE_HOST_FUNCTION(jsFunction_validateEncoding, (JSC::JSGlobalObject * glo
|
||||
auto encoding = callFrame->argument(1);
|
||||
|
||||
auto normalized = WebCore::parseEnumeration<BufferEncodingType>(*globalObject, encoding);
|
||||
|
||||
if (normalized == BufferEncodingType::hex) {
|
||||
auto data = callFrame->argument(0);
|
||||
|
||||
@@ -572,6 +573,7 @@ JSC_DEFINE_HOST_FUNCTION(jsFunction_validateEncoding, (JSC::JSGlobalObject * glo
|
||||
length = lengthValue.toLength(globalObject);
|
||||
RETURN_IF_EXCEPTION(scope, {});
|
||||
}
|
||||
|
||||
if (length % 2 != 0) {
|
||||
return Bun::ERR::INVALID_ARG_VALUE(scope, globalObject, "encoding"_s, encoding, makeString("is invalid for data of length "_s, length));
|
||||
}
|
||||
|
||||
@@ -2519,6 +2519,32 @@ pub const Arguments = struct {
|
||||
}
|
||||
}
|
||||
|
||||
if (buffer_value.?.isString() and args.encoding == .hex) {
|
||||
var str = try bun.String.fromJS(buffer_value.?, ctx);
|
||||
defer str.deref();
|
||||
var utf8_bytes = str.toUTF8(bun.default_allocator);
|
||||
defer utf8_bytes.deinit();
|
||||
|
||||
const input = utf8_bytes.slice();
|
||||
const len = input.len;
|
||||
if (len % 2 != 0) {
|
||||
return ctx.ERR(.INVALID_ARG_VALUE, "'encoding' is invalid for data of length {d}", .{len}).throw();
|
||||
}
|
||||
|
||||
const dest_len = len / 2;
|
||||
const buf = try bun.default_allocator.alloc(u8, dest_len);
|
||||
|
||||
const written = strings.decodeHexToBytes(buf, u8, input) catch
|
||||
return ctx.ERR(.INVALID_ARG_VALUE, "'encoding' is invalid for data of length {d}", .{len}).throw();
|
||||
bun.assert(written == dest_len);
|
||||
|
||||
const slice = JSC.ZigString.Slice.init(bun.default_allocator, buf);
|
||||
defer ctx.vm().reportExtraMemory(slice.len);
|
||||
|
||||
args.buffer.deinit();
|
||||
args.buffer = .{ .encoded_slice = slice };
|
||||
}
|
||||
|
||||
return args;
|
||||
}
|
||||
};
|
||||
|
||||
@@ -4,6 +4,7 @@ const EventEmitter = require("node:events");
|
||||
const fs = $zig("node_fs_binding.zig", "createBinding") as $ZigGeneratedClasses.NodeJSFS;
|
||||
const { glob } = require("internal/fs/glob");
|
||||
const constants = $processBindingConstants.fs;
|
||||
const { validateInteger, validateEncoding } = require("internal/validators");
|
||||
|
||||
var PromisePrototypeFinally = Promise.prototype.finally; //TODO
|
||||
var SymbolAsyncDispose = Symbol.asyncDispose;
|
||||
@@ -22,8 +23,6 @@ const kDeserialize = Symbol("kDeserialize");
|
||||
const kEmptyObject = ObjectFreeze({ __proto__: null });
|
||||
const kFlag = Symbol("kFlag");
|
||||
|
||||
const { validateInteger } = require("internal/validators");
|
||||
|
||||
function watch(
|
||||
filename: string | Buffer | URL,
|
||||
options: { encoding?: BufferEncoding; persistent?: boolean; recursive?: boolean; signal?: AbortSignal } = {},
|
||||
@@ -418,6 +417,13 @@ function asyncWrap(fn: any, name: string) {
|
||||
}
|
||||
|
||||
async stat(options) {
|
||||
if (!(this instanceof FileHandle)) {
|
||||
const err = new TypeError("handle must be an instance of FileHandle");
|
||||
err.code = "ERR_INTERNAL_ASSERTION";
|
||||
err.name = "AssertionError";
|
||||
throw err;
|
||||
}
|
||||
|
||||
const fd = this[kFd];
|
||||
throwEBADFIfNecessary("fstat", fd);
|
||||
|
||||
@@ -468,9 +474,13 @@ function asyncWrap(fn: any, name: string) {
|
||||
if (offset == null) {
|
||||
offset = 0;
|
||||
}
|
||||
|
||||
if (typeof length !== "number") length = buffer.byteLength - offset;
|
||||
if (typeof position !== "number") position = null;
|
||||
} else {
|
||||
validateEncoding(buffer, length); // length is the encoding in this overload
|
||||
}
|
||||
|
||||
try {
|
||||
this[kRef]();
|
||||
return { buffer, bytesWritten: await write(fd, buffer, offset, length, position) };
|
||||
|
||||
512
test/js/node/test/parallel/test-fs-promises.js
Normal file
512
test/js/node/test/parallel/test-fs-promises.js
Normal file
@@ -0,0 +1,512 @@
|
||||
'use strict';
|
||||
|
||||
const common = require('../common');
|
||||
const assert = require('assert');
|
||||
const tmpdir = require('../common/tmpdir');
|
||||
const fixtures = require('../common/fixtures');
|
||||
const path = require('path');
|
||||
const fs = require('fs');
|
||||
const fsPromises = fs.promises;
|
||||
const {
|
||||
access,
|
||||
chmod,
|
||||
chown,
|
||||
copyFile,
|
||||
lchown,
|
||||
link,
|
||||
lchmod,
|
||||
lstat,
|
||||
lutimes,
|
||||
mkdir,
|
||||
mkdtemp,
|
||||
open,
|
||||
readFile,
|
||||
readdir,
|
||||
readlink,
|
||||
realpath,
|
||||
rename,
|
||||
rmdir,
|
||||
stat,
|
||||
statfs,
|
||||
symlink,
|
||||
truncate,
|
||||
unlink,
|
||||
utimes,
|
||||
writeFile
|
||||
} = fsPromises;
|
||||
|
||||
const tmpDir = tmpdir.path;
|
||||
|
||||
let dirc = 0;
|
||||
function nextdir() {
|
||||
return `test${++dirc}`;
|
||||
}
|
||||
|
||||
// fs.promises should be enumerable.
|
||||
assert.strictEqual(
|
||||
Object.prototype.propertyIsEnumerable.call(fs, 'promises'),
|
||||
true
|
||||
);
|
||||
|
||||
{
|
||||
access(__filename, 0)
|
||||
.then(common.mustCall());
|
||||
|
||||
assert.rejects(
|
||||
access('this file does not exist', 0),
|
||||
{
|
||||
code: 'ENOENT',
|
||||
name: 'Error',
|
||||
message: /^ENOENT: no such file or directory, access/,
|
||||
// stack: /at async ok\.rejects/ // TODO: Bun async stack trace
|
||||
}
|
||||
).then(common.mustCall());
|
||||
|
||||
assert.rejects(
|
||||
access(__filename, 8),
|
||||
{
|
||||
code: 'ERR_OUT_OF_RANGE',
|
||||
}
|
||||
).then(common.mustCall());
|
||||
|
||||
assert.rejects(
|
||||
access(__filename, { [Symbol.toPrimitive]() { return 5; } }),
|
||||
{
|
||||
code: 'ERR_INVALID_ARG_TYPE',
|
||||
}
|
||||
).then(common.mustCall());
|
||||
}
|
||||
|
||||
function verifyStatObject(stat) {
|
||||
assert.strictEqual(typeof stat, 'object');
|
||||
assert.strictEqual(typeof stat.dev, 'number');
|
||||
assert.strictEqual(typeof stat.mode, 'number');
|
||||
}
|
||||
|
||||
function verifyStatFsObject(stat, isBigint = false) {
|
||||
const valueType = isBigint ? 'bigint' : 'number';
|
||||
|
||||
assert.strictEqual(typeof stat, 'object');
|
||||
assert.strictEqual(typeof stat.type, valueType);
|
||||
assert.strictEqual(typeof stat.bsize, valueType);
|
||||
assert.strictEqual(typeof stat.blocks, valueType);
|
||||
assert.strictEqual(typeof stat.bfree, valueType);
|
||||
assert.strictEqual(typeof stat.bavail, valueType);
|
||||
assert.strictEqual(typeof stat.files, valueType);
|
||||
assert.strictEqual(typeof stat.ffree, valueType);
|
||||
}
|
||||
|
||||
async function getHandle(dest) {
|
||||
await copyFile(fixtures.path('baz.js'), dest);
|
||||
await access(dest);
|
||||
|
||||
return open(dest, 'r+');
|
||||
}
|
||||
|
||||
async function executeOnHandle(dest, func) {
|
||||
let handle;
|
||||
try {
|
||||
handle = await getHandle(dest);
|
||||
await func(handle);
|
||||
} finally {
|
||||
if (handle) {
|
||||
await handle.close();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
{
|
||||
async function doTest() {
|
||||
tmpdir.refresh();
|
||||
|
||||
const dest = path.resolve(tmpDir, 'baz.js');
|
||||
|
||||
// handle is object
|
||||
{
|
||||
await executeOnHandle(dest, async (handle) => {
|
||||
assert.strictEqual(typeof handle, 'object');
|
||||
});
|
||||
}
|
||||
|
||||
// file stats
|
||||
{
|
||||
await executeOnHandle(dest, async (handle) => {
|
||||
let stats = await handle.stat();
|
||||
verifyStatObject(stats);
|
||||
assert.strictEqual(stats.size, 35);
|
||||
|
||||
await handle.truncate(1);
|
||||
|
||||
stats = await handle.stat();
|
||||
verifyStatObject(stats);
|
||||
assert.strictEqual(stats.size, 1);
|
||||
|
||||
stats = await stat(dest);
|
||||
verifyStatObject(stats);
|
||||
|
||||
stats = await handle.stat();
|
||||
verifyStatObject(stats);
|
||||
|
||||
await handle.datasync();
|
||||
await handle.sync();
|
||||
});
|
||||
}
|
||||
|
||||
// File system stats
|
||||
{
|
||||
const statFs = await statfs(dest);
|
||||
verifyStatFsObject(statFs);
|
||||
}
|
||||
|
||||
// File system stats bigint
|
||||
{
|
||||
const statFs = await statfs(dest, { bigint: true });
|
||||
verifyStatFsObject(statFs, true);
|
||||
}
|
||||
|
||||
// Test fs.read promises when length to read is zero bytes
|
||||
{
|
||||
const dest = path.resolve(tmpDir, 'test1.js');
|
||||
await executeOnHandle(dest, async (handle) => {
|
||||
const buf = Buffer.from('DAWGS WIN');
|
||||
const bufLen = buf.length;
|
||||
await handle.write(buf);
|
||||
const ret = await handle.read(Buffer.alloc(bufLen), 0, 0, 0);
|
||||
assert.strictEqual(ret.bytesRead, 0);
|
||||
|
||||
await unlink(dest);
|
||||
});
|
||||
}
|
||||
|
||||
// Use fallback buffer allocation when first argument is null
|
||||
{
|
||||
await executeOnHandle(dest, async (handle) => {
|
||||
const ret = await handle.read(null, 0, 0, 0);
|
||||
assert.strictEqual(ret.buffer.length, 16384);
|
||||
});
|
||||
}
|
||||
|
||||
// TypeError if buffer is not ArrayBufferView or nullable object
|
||||
{
|
||||
await executeOnHandle(dest, async (handle) => {
|
||||
await assert.rejects(
|
||||
async () => handle.read(0, 0, 0, 0),
|
||||
{ code: 'ERR_INVALID_ARG_TYPE' }
|
||||
);
|
||||
});
|
||||
}
|
||||
|
||||
// Bytes written to file match buffer
|
||||
{
|
||||
await executeOnHandle(dest, async (handle) => {
|
||||
const buf = Buffer.from('hello fsPromises');
|
||||
const bufLen = buf.length;
|
||||
await handle.write(buf);
|
||||
const ret = await handle.read(Buffer.alloc(bufLen), 0, bufLen, 0);
|
||||
assert.strictEqual(ret.bytesRead, bufLen);
|
||||
assert.deepStrictEqual(ret.buffer, buf);
|
||||
});
|
||||
}
|
||||
|
||||
// Truncate file to specified length
|
||||
{
|
||||
await executeOnHandle(dest, async (handle) => {
|
||||
const buf = Buffer.from('hello FileHandle');
|
||||
const bufLen = buf.length;
|
||||
await handle.write(buf, 0, bufLen, 0);
|
||||
const ret = await handle.read(Buffer.alloc(bufLen), 0, bufLen, 0);
|
||||
assert.strictEqual(ret.bytesRead, bufLen);
|
||||
assert.deepStrictEqual(ret.buffer, buf);
|
||||
await truncate(dest, 5);
|
||||
assert.strictEqual((await readFile(dest)).toString(), 'hello');
|
||||
});
|
||||
}
|
||||
|
||||
// Invalid change of ownership
|
||||
{
|
||||
await executeOnHandle(dest, async (handle) => {
|
||||
await chmod(dest, 0o666);
|
||||
await handle.chmod(0o666);
|
||||
|
||||
await chmod(dest, (0o10777));
|
||||
await handle.chmod(0o10777);
|
||||
|
||||
if (!common.isWindows) {
|
||||
await chown(dest, process.getuid(), process.getgid());
|
||||
await handle.chown(process.getuid(), process.getgid());
|
||||
}
|
||||
|
||||
await assert.rejects(
|
||||
async () => {
|
||||
await chown(dest, 1, -2);
|
||||
},
|
||||
{
|
||||
code: 'ERR_OUT_OF_RANGE',
|
||||
name: 'RangeError',
|
||||
message: 'The value of "gid" is out of range. ' +
|
||||
'It must be >= -1 and <= 4294967295. Received -2'
|
||||
});
|
||||
|
||||
await assert.rejects(
|
||||
async () => {
|
||||
await handle.chown(1, -2);
|
||||
},
|
||||
{
|
||||
code: 'ERR_OUT_OF_RANGE',
|
||||
name: 'RangeError',
|
||||
message: 'The value of "gid" is out of range. ' +
|
||||
'It must be >= -1 and <= 4294967295. Received -2'
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
// Set modification times
|
||||
{
|
||||
await executeOnHandle(dest, async (handle) => {
|
||||
|
||||
await utimes(dest, new Date(), new Date());
|
||||
|
||||
try {
|
||||
await handle.utimes(new Date(), new Date());
|
||||
} catch (err) {
|
||||
// Some systems do not have futimes. If there is an error,
|
||||
// expect it to be ENOSYS
|
||||
common.expectsError({
|
||||
code: 'ENOSYS',
|
||||
name: 'Error'
|
||||
})(err);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
// Set modification times with lutimes
|
||||
{
|
||||
const a_time = new Date();
|
||||
a_time.setMinutes(a_time.getMinutes() - 1);
|
||||
const m_time = new Date();
|
||||
m_time.setHours(m_time.getHours() - 1);
|
||||
await lutimes(dest, a_time, m_time);
|
||||
const stats = await stat(dest);
|
||||
|
||||
assert.strictEqual(a_time.toString(), stats.atime.toString());
|
||||
assert.strictEqual(m_time.toString(), stats.mtime.toString());
|
||||
}
|
||||
|
||||
// create symlink
|
||||
{
|
||||
const newPath = path.resolve(tmpDir, 'baz2.js');
|
||||
await rename(dest, newPath);
|
||||
let stats = await stat(newPath);
|
||||
verifyStatObject(stats);
|
||||
|
||||
if (common.canCreateSymLink()) {
|
||||
const newLink = path.resolve(tmpDir, 'baz3.js');
|
||||
await symlink(newPath, newLink);
|
||||
if (!common.isWindows) {
|
||||
await lchown(newLink, process.getuid(), process.getgid());
|
||||
}
|
||||
stats = await lstat(newLink);
|
||||
verifyStatObject(stats);
|
||||
|
||||
assert.strictEqual(newPath.toLowerCase(),
|
||||
(await realpath(newLink)).toLowerCase());
|
||||
assert.strictEqual(newPath.toLowerCase(),
|
||||
(await readlink(newLink)).toLowerCase());
|
||||
|
||||
const newMode = 0o666;
|
||||
if (common.isMacOS) {
|
||||
// `lchmod` is only available on macOS.
|
||||
await lchmod(newLink, newMode);
|
||||
stats = await lstat(newLink);
|
||||
assert.strictEqual(stats.mode & 0o777, newMode);
|
||||
} else {
|
||||
await Promise.all([
|
||||
assert.rejects(
|
||||
lchmod(newLink, newMode),
|
||||
common.expectsError({
|
||||
code: 'ERR_METHOD_NOT_IMPLEMENTED',
|
||||
name: 'Error',
|
||||
message: 'The lchmod() method is not implemented'
|
||||
})
|
||||
),
|
||||
]);
|
||||
}
|
||||
|
||||
await unlink(newLink);
|
||||
}
|
||||
}
|
||||
|
||||
// specify symlink type
|
||||
{
|
||||
const dir = path.join(tmpDir, nextdir());
|
||||
await symlink(tmpDir, dir, 'dir');
|
||||
const stats = await lstat(dir);
|
||||
assert.strictEqual(stats.isSymbolicLink(), true);
|
||||
await unlink(dir);
|
||||
}
|
||||
|
||||
// create hard link
|
||||
{
|
||||
const newPath = path.resolve(tmpDir, 'baz2.js');
|
||||
const newLink = path.resolve(tmpDir, 'baz4.js');
|
||||
await link(newPath, newLink);
|
||||
|
||||
await unlink(newLink);
|
||||
}
|
||||
|
||||
// Testing readdir lists both files and directories
|
||||
{
|
||||
const newDir = path.resolve(tmpDir, 'dir');
|
||||
const newFile = path.resolve(tmpDir, 'foo.js');
|
||||
|
||||
await mkdir(newDir);
|
||||
await writeFile(newFile, 'DAWGS WIN!', 'utf8');
|
||||
|
||||
const stats = await stat(newDir);
|
||||
assert(stats.isDirectory());
|
||||
const list = await readdir(tmpDir);
|
||||
assert.notStrictEqual(list.indexOf('dir'), -1);
|
||||
assert.notStrictEqual(list.indexOf('foo.js'), -1);
|
||||
await rmdir(newDir);
|
||||
await unlink(newFile);
|
||||
}
|
||||
|
||||
// Use fallback encoding when input is null
|
||||
{
|
||||
const newFile = path.resolve(tmpDir, 'dogs_running.js');
|
||||
await writeFile(newFile, 'dogs running', { encoding: null });
|
||||
const fileExists = fs.existsSync(newFile);
|
||||
assert.strictEqual(fileExists, true);
|
||||
}
|
||||
|
||||
// `mkdir` when options is number.
|
||||
{
|
||||
const dir = path.join(tmpDir, nextdir());
|
||||
await mkdir(dir, 777);
|
||||
const stats = await stat(dir);
|
||||
assert(stats.isDirectory());
|
||||
}
|
||||
|
||||
// `mkdir` when options is string.
|
||||
{
|
||||
const dir = path.join(tmpDir, nextdir());
|
||||
await mkdir(dir, '777');
|
||||
const stats = await stat(dir);
|
||||
assert(stats.isDirectory());
|
||||
}
|
||||
|
||||
// `mkdirp` when folder does not yet exist.
|
||||
{
|
||||
const dir = path.join(tmpDir, nextdir(), nextdir());
|
||||
await mkdir(dir, { recursive: true });
|
||||
const stats = await stat(dir);
|
||||
assert(stats.isDirectory());
|
||||
}
|
||||
|
||||
// `mkdirp` when path is a file.
|
||||
{
|
||||
const dir = path.join(tmpDir, nextdir(), nextdir());
|
||||
await mkdir(path.dirname(dir));
|
||||
await writeFile(dir, '');
|
||||
await assert.rejects(
|
||||
mkdir(dir, { recursive: true }),
|
||||
{
|
||||
code: 'EEXIST',
|
||||
message: /EEXIST: .*mkdir/,
|
||||
name: 'Error',
|
||||
syscall: 'mkdir',
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
// `mkdirp` when part of the path is a file.
|
||||
{
|
||||
const file = path.join(tmpDir, nextdir(), nextdir());
|
||||
const dir = path.join(file, nextdir(), nextdir());
|
||||
await mkdir(path.dirname(file));
|
||||
await writeFile(file, '');
|
||||
await assert.rejects(
|
||||
mkdir(dir, { recursive: true }),
|
||||
{
|
||||
code: 'ENOTDIR',
|
||||
message: /ENOTDIR: .*mkdir/,
|
||||
name: 'Error',
|
||||
syscall: 'mkdir',
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
// mkdirp ./
|
||||
{
|
||||
const dir = path.resolve(tmpDir, `${nextdir()}/./${nextdir()}`);
|
||||
await mkdir(dir, { recursive: true });
|
||||
const stats = await stat(dir);
|
||||
assert(stats.isDirectory());
|
||||
}
|
||||
|
||||
// mkdirp ../
|
||||
{
|
||||
const dir = path.resolve(tmpDir, `${nextdir()}/../${nextdir()}`);
|
||||
await mkdir(dir, { recursive: true });
|
||||
const stats = await stat(dir);
|
||||
assert(stats.isDirectory());
|
||||
}
|
||||
|
||||
// fs.mkdirp requires the recursive option to be of type boolean.
|
||||
// Everything else generates an error.
|
||||
{
|
||||
const dir = path.join(tmpDir, nextdir(), nextdir());
|
||||
['', 1, {}, [], null, Symbol('test'), () => {}].forEach((recursive) => {
|
||||
assert.rejects(
|
||||
// mkdir() expects to get a boolean value for options.recursive.
|
||||
async () => mkdir(dir, { recursive }),
|
||||
{
|
||||
code: 'ERR_INVALID_ARG_TYPE',
|
||||
name: 'TypeError'
|
||||
}
|
||||
).then(common.mustCall());
|
||||
});
|
||||
}
|
||||
|
||||
// `mkdtemp` with invalid numeric prefix
|
||||
{
|
||||
await mkdtemp(path.resolve(tmpDir, 'FOO'));
|
||||
await assert.rejects(
|
||||
// mkdtemp() expects to get a string prefix.
|
||||
async () => mkdtemp(1),
|
||||
{
|
||||
code: 'ERR_INVALID_ARG_TYPE',
|
||||
name: 'TypeError'
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
// Regression test for https://github.com/nodejs/node/issues/38168
|
||||
{
|
||||
await executeOnHandle(dest, async (handle) => {
|
||||
await assert.rejects(
|
||||
async () => handle.write('abc', 0, 'hex'),
|
||||
{
|
||||
code: 'ERR_INVALID_ARG_VALUE',
|
||||
message: /'encoding' is invalid for data of length 3/
|
||||
}
|
||||
);
|
||||
|
||||
const ret = await handle.write('abcd', 0, 'hex');
|
||||
assert.strictEqual(ret.bytesWritten, 2);
|
||||
});
|
||||
}
|
||||
|
||||
// Test prototype methods calling with contexts other than FileHandle
|
||||
{
|
||||
await executeOnHandle(dest, async (handle) => {
|
||||
await assert.rejects(() => handle.stat.call({}), {
|
||||
code: 'ERR_INTERNAL_ASSERTION',
|
||||
message: /handle must be an instance of FileHandle/
|
||||
});
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
doTest().then(common.mustCall());
|
||||
}
|
||||
Reference in New Issue
Block a user