BUGFIX: Fix issues and edge-cases with rm (#1404)

This commit is contained in:
David Walker 2024-06-16 18:27:46 -07:00 committed by GitHub
parent 4382f860db
commit 99b22a221c
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194

@ -1,18 +1,19 @@
import { Terminal } from "../../Terminal";
import { BaseServer } from "../../Server/BaseServer";
import { PromptEvent } from "../../ui/React/PromptManager";
import { hasScriptExtension } from "../../Paths/ScriptFilePath";
import { hasTextExtension } from "../../Paths/TextFilePath";
import type { Directory } from "../../Paths/Directory";
import { getAllDirectories, type Directory } from "../../Paths/Directory";
import type { ProgramFilePath } from "../../Paths/ProgramFilePath";
import type { IReturnStatus } from "../../types";
import type { FilePath } from "../../Paths/FilePath";
export function rm(args: (string | number | boolean)[], server: BaseServer): void {
const errors = {
arg: (reason: string) => `Incorrect usage of rm command. ${reason}. Usage: rm [OPTION]... [FILE]...`,
dirsProvided: () => "Incorrect usage of rm command. To delete directories, use the -r flag",
invalidDir: (name: string) => `Invalid directory: ${name}`,
invalidFile: (name: string) => `Invalid file: ${name}`,
dirsProvided: (name: string) =>
`Incorrect usage of rm command. To delete directories, use the -r flag. Failing directory: ${name}`,
invalidFile: (name: string) => `Invalid filename: ${name}`,
noSuchFile: (name: string) => `File does not exist: ${name}`,
noSuchDir: (name: string) => `Directory does not exist: ${name}`,
deleteFailed: (name: string, reason?: string) => `Failed to delete "${name}". ${reason ?? "Uncaught error"}`,
rootDeletion: () =>
"You are trying to delete all files within the root directory. If this is intentional, use the --no-preserve-root flag",
@ -37,25 +38,74 @@ export function rm(args: (string | number | boolean)[], server: BaseServer): voi
const directories: Directory[] = [];
const files: FilePath[] = [];
const allDirs: Set<Directory> = getAllDirectories(server);
const allFiles: Set<FilePath> = new Set([
...server.scripts.keys(),
...server.textFiles.keys(),
...(server.programs as ProgramFilePath[]),
]);
for (const file of server.contracts) {
allFiles.add(file.fn);
}
for (const file of server.messages) {
if (file.endsWith(".lit")) {
allFiles.add(file as FilePath);
}
}
for (const target of targets) {
if (!hasTextExtension(target) && !hasScriptExtension(target)) {
const dirPath = Terminal.getDirectory(target);
if (dirPath === null) return Terminal.error(errors["invalidDir"](target));
if (!recursive) return Terminal.error(errors["dirsProvided"]());
directories.push(dirPath);
// Directories can be specified with or without a trailing slash. However,
// trying to remove a file with a trailing slash is an error.
const fileDir = Terminal.getDirectory(target + (target[target.length - 1] === "/" ? "" : "/"));
const file = Terminal.getFilepath(target);
const fileExists = file !== null && allFiles.has(file);
if (fileDir === null) return Terminal.error(errors.invalidFile(target));
const dirExists = allDirs.has(fileDir);
if (file === null || dirExists) {
// If file === null, it means we specified a trailing-slash directory/,
// or something that does not have an extension or otherwise isn't file-like.
if (fileExists) {
// We have this early case here specifically to handle situations where
// a file and a directory with the same name exist. That's right, you
// can have *both* /foo.txt *and* /foo.txt/bar.txt.
//
// In this case, we need to treat filenames preferrentially as files first.
// If we have -r, we will *also* delete the directory.
files.push(file);
}
if (!recursive) {
if (fileExists) {
// This is valid, but we shouldn't touch the directory.
continue;
} else {
// Only exists as a directory (maybe).
return Terminal.error(errors.dirsProvided(target));
}
}
if (!dirExists && !force) {
return Terminal.error(errors.noSuchDir(target));
}
// If we pass -f and pass a non-existing directory, we will add it
// here and then it will match no files, producing no errors. This
// aligns with Unix rm.
directories.push(fileDir);
continue;
}
const file = Terminal.getFilepath(target);
if (file === null) return Terminal.error(errors["invalidFile"](target));
if (!force && !allFiles.has(file)) {
// With -f, we ignore file-not-found and try to delete everything at the end.
return Terminal.error(errors.noSuchFile(target));
}
files.push(file);
}
for (const dir of directories) {
for (const file of server.scripts.keys()) {
if (file.startsWith(dir)) files.push(file);
for (const file of allFiles) {
for (const dir of directories) {
if (file.startsWith(dir)) {
files.push(file);
}
}
}
@ -72,16 +122,23 @@ export function rm(args: (string | number | boolean)[], server: BaseServer): voi
if (report.result.res) {
Terminal.success(`Deleted: ${report.target}`);
} else {
Terminal.error(errors["deleteFailed"](report.target, report.result.msg));
Terminal.error(errors.deleteFailed(report.target, report.result.msg));
}
}
};
if (force || files.length === 1) {
if (
force ||
(files.length === 1 && !files[0].endsWith(".exe") && !files[0].endsWith(".lit") && !files[0].endsWith(".cct"))
) {
deleteSelectedTargets();
} else {
const promptText = `Are you sure you want to delete ${
files.length === 1 ? files[0] : "these files"
}? This is irreversible.${files.length > 1 ? "\n\nDeleting:\n" + targetList : ""}`;
PromptEvent.emit({
txt: "Are you sure you want to delete these files? This is irreversible.\n\nDeleting:\n" + targetList,
txt: promptText,
resolve: (value: string | boolean) => {
if (typeof value === "string") throw new Error("PromptEvent got a string, expected boolean");
if (value) deleteSelectedTargets();