2024-01-01 14:58:21 -05:00
|
|
|
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
2023-03-15 23:16:03 -04:00
|
|
|
|
2024-10-17 15:51:15 -04:00
|
|
|
/// <reference lib="deno.ns" />
|
|
|
|
import { assert, assertEquals, assertRejects, assertThrows } from "@std/assert";
|
2023-03-15 23:16:03 -04:00
|
|
|
import { join } from "node:path";
|
2024-11-27 02:35:39 -05:00
|
|
|
import { fileURLToPath } from "node:url";
|
2023-03-15 23:16:03 -04:00
|
|
|
import { tmpdir } from "node:os";
|
2024-01-21 15:48:48 -05:00
|
|
|
import {
|
2024-10-08 06:41:32 -04:00
|
|
|
closeSync,
|
2024-01-21 15:48:48 -05:00
|
|
|
constants,
|
2024-10-17 15:51:15 -04:00
|
|
|
copyFileSync,
|
2024-04-07 23:47:34 -04:00
|
|
|
createWriteStream,
|
2024-01-21 15:48:48 -05:00
|
|
|
existsSync,
|
2024-05-28 06:24:54 -04:00
|
|
|
lstatSync,
|
2024-01-21 15:48:48 -05:00
|
|
|
mkdtempSync,
|
2024-10-08 06:41:32 -04:00
|
|
|
openSync,
|
2024-01-21 15:48:48 -05:00
|
|
|
promises,
|
|
|
|
readFileSync,
|
2024-10-08 06:41:32 -04:00
|
|
|
readSync,
|
2024-03-06 07:59:10 -05:00
|
|
|
Stats,
|
|
|
|
statSync,
|
2024-01-21 15:48:48 -05:00
|
|
|
writeFileSync,
|
|
|
|
} from "node:fs";
|
2024-05-28 08:30:31 -04:00
|
|
|
import {
|
|
|
|
constants as fsPromiseConstants,
|
2024-10-17 15:51:15 -04:00
|
|
|
copyFile,
|
2024-05-28 08:30:31 -04:00
|
|
|
cp,
|
|
|
|
FileHandle,
|
|
|
|
open,
|
2024-10-26 23:04:35 -04:00
|
|
|
stat,
|
2024-05-28 08:30:31 -04:00
|
|
|
writeFile,
|
|
|
|
} from "node:fs/promises";
|
2024-04-07 23:47:34 -04:00
|
|
|
import process from "node:process";
|
2023-12-04 16:05:40 -05:00
|
|
|
import { pathToAbsoluteFileUrl } from "../unit/test_util.ts";
|
2023-03-15 23:16:03 -04:00
|
|
|
|
|
|
|
Deno.test(
|
|
|
|
"[node/fs writeFileSync] write file without option",
|
|
|
|
() => {
|
|
|
|
const data = "Hello";
|
|
|
|
const filename = mkdtempSync(join(tmpdir(), "foo-")) + "/test.txt";
|
|
|
|
|
|
|
|
writeFileSync(filename, data);
|
|
|
|
const dataRead = readFileSync(filename, "utf8");
|
|
|
|
|
|
|
|
assert(dataRead === "Hello");
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
Deno.test(
|
|
|
|
"[node/fs writeFileSync] write file with option ASCII",
|
|
|
|
() => {
|
|
|
|
const data = "Hello";
|
|
|
|
const filename = mkdtempSync(join(tmpdir(), "foo-")) + "/test.txt";
|
|
|
|
|
|
|
|
writeFileSync(filename, data, { encoding: "ascii" });
|
|
|
|
const dataRead = readFileSync(filename, "utf8");
|
|
|
|
|
|
|
|
assert(dataRead === "Hello");
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
Deno.test(
|
|
|
|
"[node/fs writeFileSync] write file throws error when encoding is not implemented",
|
|
|
|
() => {
|
|
|
|
const data = "Hello";
|
|
|
|
const filename = mkdtempSync(join(tmpdir(), "foo-")) + "/test.txt";
|
|
|
|
|
|
|
|
assertThrows(
|
|
|
|
() => writeFileSync(filename, data, { encoding: "utf16le" }),
|
|
|
|
'The value "utf16le" is invalid for option "encoding"',
|
|
|
|
);
|
|
|
|
},
|
|
|
|
);
|
2023-12-04 16:05:40 -05:00
|
|
|
|
|
|
|
Deno.test(
|
|
|
|
"[node/fs existsSync] path",
|
|
|
|
{ permissions: { read: true } },
|
|
|
|
() => {
|
chore: move cli/tests/ -> tests/ (#22369)
This looks like a massive PR, but it's only a move from cli/tests ->
tests, and updates of relative paths for files.
This is the first step towards aggregate all of the integration test
files under tests/, which will lead to a set of integration tests that
can run without the CLI binary being built.
While we could leave these tests under `cli`, it would require us to
keep a more complex directory structure for the various test runners. In
addition, we have a lot of complexity to ignore various test files in
the `cli` project itself (cargo publish exclusion rules, autotests =
false, etc).
And finally, the `tests/` folder will eventually house the `test_ffi`,
`test_napi` and other testing code, reducing the size of the root repo
directory.
For easier review, the extremely large and noisy "move" is in the first
commit (with no changes -- just a move), while the remainder of the
changes to actual files is in the second commit.
2024-02-10 15:22:13 -05:00
|
|
|
assert(existsSync("tests/testdata/assets/fixture.json"));
|
2023-12-04 16:05:40 -05:00
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
Deno.test(
|
|
|
|
"[node/fs existsSync] url",
|
|
|
|
{ permissions: { read: true } },
|
|
|
|
() => {
|
|
|
|
assert(existsSync(
|
chore: move cli/tests/ -> tests/ (#22369)
This looks like a massive PR, but it's only a move from cli/tests ->
tests, and updates of relative paths for files.
This is the first step towards aggregate all of the integration test
files under tests/, which will lead to a set of integration tests that
can run without the CLI binary being built.
While we could leave these tests under `cli`, it would require us to
keep a more complex directory structure for the various test runners. In
addition, we have a lot of complexity to ignore various test files in
the `cli` project itself (cargo publish exclusion rules, autotests =
false, etc).
And finally, the `tests/` folder will eventually house the `test_ffi`,
`test_napi` and other testing code, reducing the size of the root repo
directory.
For easier review, the extremely large and noisy "move" is in the first
commit (with no changes -- just a move), while the remainder of the
changes to actual files is in the second commit.
2024-02-10 15:22:13 -05:00
|
|
|
pathToAbsoluteFileUrl("tests/testdata/assets/fixture.json"),
|
2023-12-04 16:05:40 -05:00
|
|
|
));
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
Deno.test(
|
|
|
|
"[node/fs existsSync] no permission",
|
|
|
|
{ permissions: { read: false } },
|
|
|
|
() => {
|
|
|
|
assertThrows(() => {
|
chore: move cli/tests/ -> tests/ (#22369)
This looks like a massive PR, but it's only a move from cli/tests ->
tests, and updates of relative paths for files.
This is the first step towards aggregate all of the integration test
files under tests/, which will lead to a set of integration tests that
can run without the CLI binary being built.
While we could leave these tests under `cli`, it would require us to
keep a more complex directory structure for the various test runners. In
addition, we have a lot of complexity to ignore various test files in
the `cli` project itself (cargo publish exclusion rules, autotests =
false, etc).
And finally, the `tests/` folder will eventually house the `test_ffi`,
`test_napi` and other testing code, reducing the size of the root repo
directory.
For easier review, the extremely large and noisy "move" is in the first
commit (with no changes -- just a move), while the remainder of the
changes to actual files is in the second commit.
2024-02-10 15:22:13 -05:00
|
|
|
existsSync("tests/testdata/assets/fixture.json");
|
2024-09-10 14:12:24 -04:00
|
|
|
}, Deno.errors.NotCapable);
|
2023-12-04 16:05:40 -05:00
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
Deno.test(
|
|
|
|
"[node/fs existsSync] not exists",
|
|
|
|
{ permissions: { read: true } },
|
|
|
|
() => {
|
|
|
|
assert(!existsSync("bad_filename"));
|
|
|
|
},
|
|
|
|
);
|
2024-01-21 15:48:48 -05:00
|
|
|
|
|
|
|
Deno.test(
|
|
|
|
"[node/fs/promises constants] is the same as from node:fs",
|
|
|
|
() => {
|
|
|
|
assertEquals(constants, fsPromiseConstants);
|
|
|
|
assertEquals(constants, promises.constants);
|
|
|
|
},
|
|
|
|
);
|
2024-02-06 02:02:58 -05:00
|
|
|
|
2024-03-06 07:59:10 -05:00
|
|
|
Deno.test(
|
|
|
|
"[node/fs statSync] instanceof fs.Stats",
|
|
|
|
() => {
|
|
|
|
const stat = statSync("tests/testdata/assets/fixture.json");
|
|
|
|
assert(stat);
|
|
|
|
assert(stat instanceof Stats);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
2024-10-26 23:04:35 -04:00
|
|
|
Deno.test(
|
|
|
|
"[node/fs statSync] throw error with path information",
|
|
|
|
() => {
|
|
|
|
const file = "non-exist-file";
|
|
|
|
const fileUrl = new URL(file, import.meta.url);
|
|
|
|
|
|
|
|
assertThrows(() => {
|
|
|
|
statSync(file);
|
|
|
|
}, "Error: ENOENT: no such file or directory, stat 'non-exist-file'");
|
|
|
|
|
|
|
|
assertThrows(() => {
|
|
|
|
statSync(fileUrl);
|
|
|
|
}, `Error: ENOENT: no such file or directory, stat '${fileUrl.pathname}'`);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
Deno.test(
|
|
|
|
"[node/fs/promises stat] throw error with path information",
|
|
|
|
async () => {
|
|
|
|
const file = "non-exist-file";
|
|
|
|
const fileUrl = new URL(file, import.meta.url);
|
|
|
|
|
|
|
|
try {
|
|
|
|
await stat(file);
|
|
|
|
} catch (error: unknown) {
|
|
|
|
assertEquals(
|
|
|
|
`${error}`,
|
|
|
|
"Error: ENOENT: no such file or directory, stat 'non-exist-file'",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
await stat(fileUrl);
|
|
|
|
} catch (error: unknown) {
|
|
|
|
assertEquals(
|
|
|
|
`${error}`,
|
2024-11-27 02:35:39 -05:00
|
|
|
`Error: ENOENT: no such file or directory, stat '${
|
|
|
|
fileURLToPath(fileUrl)
|
|
|
|
}'`,
|
2024-10-26 23:04:35 -04:00
|
|
|
);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
2024-02-06 02:02:58 -05:00
|
|
|
Deno.test(
|
|
|
|
"[node/fs/promises cp] copy file",
|
|
|
|
async () => {
|
|
|
|
const src = mkdtempSync(join(tmpdir(), "foo-")) + "/test.txt";
|
|
|
|
const dest = mkdtempSync(join(tmpdir(), "foo-")) + "/test.txt";
|
|
|
|
writeFileSync(src, "Hello");
|
|
|
|
|
|
|
|
await cp(src, dest);
|
|
|
|
|
|
|
|
const dataRead = readFileSync(dest, "utf8");
|
|
|
|
assert(dataRead === "Hello");
|
|
|
|
},
|
|
|
|
);
|
2024-04-07 23:47:34 -04:00
|
|
|
|
|
|
|
// TODO(kt3k): Delete this test case, and instead enable the compat case
|
|
|
|
// `test/parallel/test-fs-writestream-open-write.js`, when we update
|
|
|
|
// `tests/node_compat/runner/suite`.
|
|
|
|
Deno.test("[node/fs createWriteStream", async () => {
|
|
|
|
const { promise, resolve, reject } = Promise.withResolvers<void>();
|
|
|
|
const tempDir = await Deno.makeTempDir();
|
|
|
|
const file = join(tempDir, "file.txt");
|
|
|
|
try {
|
|
|
|
const w = createWriteStream(file);
|
|
|
|
|
|
|
|
w.on("open", () => {
|
|
|
|
w.write("hello, ");
|
|
|
|
|
|
|
|
process.nextTick(() => {
|
|
|
|
w.write("world");
|
|
|
|
w.end();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
w.on("close", async () => {
|
|
|
|
try {
|
|
|
|
assertEquals(await Deno.readTextFile(file), "hello, world");
|
|
|
|
resolve();
|
|
|
|
} catch (e) {
|
|
|
|
reject(e);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
await promise;
|
|
|
|
} finally {
|
|
|
|
await Deno.remove(tempDir, { recursive: true });
|
|
|
|
}
|
|
|
|
});
|
2024-05-28 06:24:54 -04:00
|
|
|
|
|
|
|
Deno.test(
|
|
|
|
"[node/fs lstatSync] supports throwIfNoEntry option",
|
|
|
|
() => {
|
|
|
|
const result = lstatSync("non-existing-path", { throwIfNoEntry: false });
|
|
|
|
assertEquals(result, undefined);
|
|
|
|
},
|
|
|
|
);
|
2024-05-28 08:30:31 -04:00
|
|
|
|
|
|
|
// Test for https://github.com/denoland/deno/issues/23707
|
|
|
|
Deno.test(
|
|
|
|
"[node/fs/promises read] respect position argument",
|
|
|
|
async () => {
|
|
|
|
const file = mkdtempSync(join(tmpdir(), "foo-")) + "/test.bin";
|
|
|
|
await writeFile(file, "");
|
|
|
|
|
|
|
|
const res: number[] = [];
|
|
|
|
let fd: FileHandle | undefined;
|
|
|
|
try {
|
|
|
|
fd = await open(file, "r+");
|
|
|
|
|
|
|
|
for (let i = 0; i <= 5; i++) {
|
|
|
|
const buffer = new Uint8Array([i]);
|
|
|
|
await fd.write(buffer, 0, 1, i + 10);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (let i = 10; i <= 15; i++) {
|
|
|
|
const buffer = new Uint8Array(1);
|
|
|
|
await fd.read(buffer, 0, 1, i);
|
|
|
|
res.push(Number(buffer.toString()));
|
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
await fd?.close();
|
|
|
|
}
|
|
|
|
|
|
|
|
assertEquals(res, [0, 1, 2, 3, 4, 5]);
|
|
|
|
},
|
|
|
|
);
|
2024-10-08 06:41:32 -04:00
|
|
|
|
|
|
|
Deno.test("[node/fs] readSync works", () => {
|
|
|
|
const fd = openSync("tests/testdata/assets/hello.txt", "r");
|
|
|
|
const buf = new Uint8Array(256);
|
|
|
|
const bytesRead = readSync(fd!, buf);
|
|
|
|
assertEquals(bytesRead, 12);
|
|
|
|
closeSync(fd!);
|
|
|
|
});
|
2024-10-17 15:51:15 -04:00
|
|
|
|
|
|
|
Deno.test("[node/fs] copyFile COPYFILE_EXCL works", async () => {
|
|
|
|
const dir = mkdtempSync(join(tmpdir(), "foo-"));
|
|
|
|
const src = join(dir, "src.txt");
|
|
|
|
const dest = join(dir, "dest.txt");
|
|
|
|
await writeFile(src, "");
|
|
|
|
await copyFile(src, dest, fsPromiseConstants.COPYFILE_EXCL);
|
|
|
|
assert(existsSync(dest));
|
2024-10-17 18:19:37 -04:00
|
|
|
await assertRejects(() =>
|
|
|
|
copyFile(src, dest, fsPromiseConstants.COPYFILE_EXCL)
|
|
|
|
);
|
2024-10-17 15:51:15 -04:00
|
|
|
const dest2 = join(dir, "dest2.txt");
|
|
|
|
copyFileSync(src, dest2, fsPromiseConstants.COPYFILE_EXCL);
|
|
|
|
assert(existsSync(dest2));
|
|
|
|
assertThrows(() =>
|
|
|
|
copyFileSync(src, dest2, fsPromiseConstants.COPYFILE_EXCL)
|
|
|
|
);
|
|
|
|
});
|
2024-10-22 22:28:04 -04:00
|
|
|
|
|
|
|
Deno.test("[node/fs] statSync throws ENOENT for invalid path containing colon in it", () => {
|
|
|
|
// deno-lint-ignore no-explicit-any
|
|
|
|
const err: any = assertThrows(() => {
|
|
|
|
// Note: Deno.stat throws ERROR_INVALID_NAME (os error 123) instead of
|
|
|
|
// ERROR_FILE_NOT_FOUND (os error 2) on windows. This case checks that
|
|
|
|
// ERROR_INVALID_NAME is mapped to ENOENT correctly on node compat layer.
|
|
|
|
statSync("jsr:@std/assert");
|
|
|
|
});
|
|
|
|
assertEquals(err.code, "ENOENT");
|
|
|
|
});
|