mirror of
https://github.com/denoland/deno.git
synced 2024-10-31 09:14:20 -04:00
38213f1142
When listening on a UNIX socket path, Deno currently tries to unlink this path prior to actually listening. The implementation of this behaviour is VERY racy, involves 2 additional syscalls, and does not match the behaviour of any other runtime (Node.js, Go, Rust, etc). This commit removes this behaviour. If a user wants to listen on an existing socket, they must now unlink the file themselves prior to listening. This change in behaviour only impacts --unstable APIs, so it is not a breaking change.
291 lines
8.9 KiB
TypeScript
291 lines
8.9 KiB
TypeScript
// Copyright 2018-2022 the Deno authors. All rights reserved. MIT license.
|
|
import { assert, assertRejects, assertThrows } from "./test_util.ts";
|
|
|
|
const REMOVE_METHODS = ["remove", "removeSync"] as const;
|
|
|
|
Deno.test(
|
|
{ permissions: { write: true, read: true } },
|
|
async function removeDirSuccess() {
|
|
for (const method of REMOVE_METHODS) {
|
|
// REMOVE EMPTY DIRECTORY
|
|
const path = Deno.makeTempDirSync() + "/subdir";
|
|
Deno.mkdirSync(path);
|
|
const pathInfo = Deno.statSync(path);
|
|
assert(pathInfo.isDirectory); // check exist first
|
|
await Deno[method](path); // remove
|
|
// We then check again after remove
|
|
assertThrows(() => {
|
|
Deno.statSync(path);
|
|
}, Deno.errors.NotFound);
|
|
}
|
|
},
|
|
);
|
|
|
|
Deno.test(
|
|
{ permissions: { write: true, read: true } },
|
|
async function removeFileSuccess() {
|
|
for (const method of REMOVE_METHODS) {
|
|
// REMOVE FILE
|
|
const enc = new TextEncoder();
|
|
const data = enc.encode("Hello");
|
|
const filename = Deno.makeTempDirSync() + "/test.txt";
|
|
Deno.writeFileSync(filename, data, { mode: 0o666 });
|
|
const fileInfo = Deno.statSync(filename);
|
|
assert(fileInfo.isFile); // check exist first
|
|
await Deno[method](filename); // remove
|
|
// We then check again after remove
|
|
assertThrows(() => {
|
|
Deno.statSync(filename);
|
|
}, Deno.errors.NotFound);
|
|
}
|
|
},
|
|
);
|
|
|
|
Deno.test(
|
|
{ permissions: { write: true, read: true } },
|
|
async function removeFileByUrl() {
|
|
for (const method of REMOVE_METHODS) {
|
|
// REMOVE FILE
|
|
const enc = new TextEncoder();
|
|
const data = enc.encode("Hello");
|
|
|
|
const tempDir = Deno.makeTempDirSync();
|
|
const fileUrl = new URL(
|
|
`file://${Deno.build.os === "windows" ? "/" : ""}${tempDir}/test.txt`,
|
|
);
|
|
|
|
Deno.writeFileSync(fileUrl, data, { mode: 0o666 });
|
|
const fileInfo = Deno.statSync(fileUrl);
|
|
assert(fileInfo.isFile); // check exist first
|
|
await Deno[method](fileUrl); // remove
|
|
// We then check again after remove
|
|
assertThrows(() => {
|
|
Deno.statSync(fileUrl);
|
|
}, Deno.errors.NotFound);
|
|
}
|
|
},
|
|
);
|
|
|
|
Deno.test(
|
|
{ permissions: { write: true, read: true } },
|
|
async function removeFail() {
|
|
for (const method of REMOVE_METHODS) {
|
|
// NON-EMPTY DIRECTORY
|
|
const path = Deno.makeTempDirSync() + "/dir/subdir";
|
|
const subPath = path + "/subsubdir";
|
|
Deno.mkdirSync(path, { recursive: true });
|
|
Deno.mkdirSync(subPath);
|
|
const pathInfo = Deno.statSync(path);
|
|
assert(pathInfo.isDirectory); // check exist first
|
|
const subPathInfo = Deno.statSync(subPath);
|
|
assert(subPathInfo.isDirectory); // check exist first
|
|
|
|
await assertRejects(
|
|
async () => {
|
|
await Deno[method](path);
|
|
},
|
|
Error,
|
|
`remove '${path}'`,
|
|
);
|
|
// TODO(ry) Is Other really the error we should get here? What would Go do?
|
|
|
|
// NON-EXISTENT DIRECTORY/FILE
|
|
await assertRejects(
|
|
async () => {
|
|
await Deno[method]("/baddir");
|
|
},
|
|
Deno.errors.NotFound,
|
|
`remove '/baddir'`,
|
|
);
|
|
}
|
|
},
|
|
);
|
|
|
|
Deno.test(
|
|
{ permissions: { write: true, read: true } },
|
|
async function removeDanglingSymlinkSuccess() {
|
|
for (const method of REMOVE_METHODS) {
|
|
const danglingSymlinkPath = Deno.makeTempDirSync() + "/dangling_symlink";
|
|
if (Deno.build.os === "windows") {
|
|
Deno.symlinkSync("unexistent_file", danglingSymlinkPath, {
|
|
type: "file",
|
|
});
|
|
} else {
|
|
Deno.symlinkSync("unexistent_file", danglingSymlinkPath);
|
|
}
|
|
const pathInfo = Deno.lstatSync(danglingSymlinkPath);
|
|
assert(pathInfo.isSymlink);
|
|
await Deno[method](danglingSymlinkPath);
|
|
assertThrows(() => {
|
|
Deno.lstatSync(danglingSymlinkPath);
|
|
}, Deno.errors.NotFound);
|
|
}
|
|
},
|
|
);
|
|
|
|
Deno.test(
|
|
{ permissions: { write: true, read: true } },
|
|
async function removeValidSymlinkSuccess() {
|
|
for (const method of REMOVE_METHODS) {
|
|
const encoder = new TextEncoder();
|
|
const data = encoder.encode("Test");
|
|
const tempDir = Deno.makeTempDirSync();
|
|
const filePath = tempDir + "/test.txt";
|
|
const validSymlinkPath = tempDir + "/valid_symlink";
|
|
Deno.writeFileSync(filePath, data, { mode: 0o666 });
|
|
if (Deno.build.os === "windows") {
|
|
Deno.symlinkSync(filePath, validSymlinkPath, { type: "file" });
|
|
} else {
|
|
Deno.symlinkSync(filePath, validSymlinkPath);
|
|
}
|
|
const symlinkPathInfo = Deno.statSync(validSymlinkPath);
|
|
assert(symlinkPathInfo.isFile);
|
|
await Deno[method](validSymlinkPath);
|
|
assertThrows(() => {
|
|
Deno.statSync(validSymlinkPath);
|
|
}, Deno.errors.NotFound);
|
|
await Deno[method](filePath);
|
|
}
|
|
},
|
|
);
|
|
|
|
Deno.test({ permissions: { write: false } }, async function removePerm() {
|
|
for (const method of REMOVE_METHODS) {
|
|
await assertRejects(async () => {
|
|
await Deno[method]("/baddir");
|
|
}, Deno.errors.PermissionDenied);
|
|
}
|
|
});
|
|
|
|
Deno.test(
|
|
{ permissions: { write: true, read: true } },
|
|
async function removeAllDirSuccess() {
|
|
for (const method of REMOVE_METHODS) {
|
|
// REMOVE EMPTY DIRECTORY
|
|
let path = Deno.makeTempDirSync() + "/dir/subdir";
|
|
Deno.mkdirSync(path, { recursive: true });
|
|
let pathInfo = Deno.statSync(path);
|
|
assert(pathInfo.isDirectory); // check exist first
|
|
await Deno[method](path, { recursive: true }); // remove
|
|
// We then check again after remove
|
|
assertThrows(
|
|
() => {
|
|
Deno.statSync(path);
|
|
}, // Directory is gone
|
|
Deno.errors.NotFound,
|
|
);
|
|
|
|
// REMOVE NON-EMPTY DIRECTORY
|
|
path = Deno.makeTempDirSync() + "/dir/subdir";
|
|
const subPath = path + "/subsubdir";
|
|
Deno.mkdirSync(path, { recursive: true });
|
|
Deno.mkdirSync(subPath);
|
|
pathInfo = Deno.statSync(path);
|
|
assert(pathInfo.isDirectory); // check exist first
|
|
const subPathInfo = Deno.statSync(subPath);
|
|
assert(subPathInfo.isDirectory); // check exist first
|
|
await Deno[method](path, { recursive: true }); // remove
|
|
// We then check parent directory again after remove
|
|
assertThrows(() => {
|
|
Deno.statSync(path);
|
|
}, Deno.errors.NotFound);
|
|
// Directory is gone
|
|
}
|
|
},
|
|
);
|
|
|
|
Deno.test(
|
|
{ permissions: { write: true, read: true } },
|
|
async function removeAllFileSuccess() {
|
|
for (const method of REMOVE_METHODS) {
|
|
// REMOVE FILE
|
|
const enc = new TextEncoder();
|
|
const data = enc.encode("Hello");
|
|
const filename = Deno.makeTempDirSync() + "/test.txt";
|
|
Deno.writeFileSync(filename, data, { mode: 0o666 });
|
|
const fileInfo = Deno.statSync(filename);
|
|
assert(fileInfo.isFile); // check exist first
|
|
await Deno[method](filename, { recursive: true }); // remove
|
|
// We then check again after remove
|
|
assertThrows(() => {
|
|
Deno.statSync(filename);
|
|
}, Deno.errors.NotFound);
|
|
// File is gone
|
|
}
|
|
},
|
|
);
|
|
|
|
Deno.test({ permissions: { write: true } }, async function removeAllFail() {
|
|
for (const method of REMOVE_METHODS) {
|
|
// NON-EXISTENT DIRECTORY/FILE
|
|
await assertRejects(
|
|
async () => {
|
|
// Non-existent
|
|
await Deno[method]("/baddir", { recursive: true });
|
|
},
|
|
Deno.errors.NotFound,
|
|
`remove '/baddir'`,
|
|
);
|
|
}
|
|
});
|
|
|
|
Deno.test({ permissions: { write: false } }, async function removeAllPerm() {
|
|
for (const method of REMOVE_METHODS) {
|
|
await assertRejects(async () => {
|
|
await Deno[method]("/baddir", { recursive: true });
|
|
}, Deno.errors.PermissionDenied);
|
|
}
|
|
});
|
|
|
|
Deno.test(
|
|
{
|
|
ignore: Deno.build.os === "windows",
|
|
permissions: { write: true, read: true },
|
|
},
|
|
async function removeUnixSocketSuccess() {
|
|
for (const method of REMOVE_METHODS) {
|
|
// MAKE TEMPORARY UNIX SOCKET
|
|
const path = Deno.makeTempDirSync() + "/test.sock";
|
|
const listener = Deno.listen({ transport: "unix", path });
|
|
listener.close();
|
|
Deno.statSync(path); // check if unix socket exists
|
|
|
|
await Deno[method](path);
|
|
assertThrows(() => Deno.statSync(path), Deno.errors.NotFound);
|
|
}
|
|
},
|
|
);
|
|
|
|
if (Deno.build.os === "windows") {
|
|
Deno.test(
|
|
{ permissions: { run: true, write: true, read: true } },
|
|
async function removeFileSymlink() {
|
|
const { success } = await Deno.spawn("cmd", {
|
|
args: ["/c", "mklink", "file_link", "bar"],
|
|
stdout: "null",
|
|
});
|
|
|
|
assert(success);
|
|
await Deno.remove("file_link");
|
|
await assertRejects(async () => {
|
|
await Deno.lstat("file_link");
|
|
}, Deno.errors.NotFound);
|
|
},
|
|
);
|
|
|
|
Deno.test(
|
|
{ permissions: { run: true, write: true, read: true } },
|
|
async function removeDirSymlink() {
|
|
const { success } = await Deno.spawn("cmd", {
|
|
args: ["/c", "mklink", "/d", "dir_link", "bar"],
|
|
stdout: "null",
|
|
});
|
|
|
|
assert(success);
|
|
await Deno.remove("dir_link");
|
|
await assertRejects(async () => {
|
|
await Deno.lstat("dir_link");
|
|
}, Deno.errors.NotFound);
|
|
},
|
|
);
|
|
}
|