import { evaluateDirectoryPath, getAllParentDirectories } from "./DirectoryHelpers"; import { getSubdirectories } from "./DirectoryServerHelpers"; import { Aliases, GlobalAliases, substituteAliases } from "../Alias"; import { DarkWebItems } from "../DarkWeb/DarkWebItems"; import { IPlayer } from "../PersonObjects/IPlayer"; import { GetServer, GetAllServers } from "../Server/AllServers"; import { ParseCommand, ParseCommands } from "./Parser"; import { isScriptFilename } from "../Script/isScriptFilename"; import { compile } from "../NetscriptJSEvaluator"; import { Flags } from "../NetscriptFunctions/Flags"; import * as libarg from "arg"; // An array of all Terminal commands const commands = [ "alias", "analyze", "backdoor", "cat", "cd", "check", "clear", "cls", "connect", "cp", "download", "expr", "free", "grow", "hack", "help", "home", "hostname", "ifconfig", "kill", "killall", "ls", "lscpu", "mem", "mv", "nano", "ps", "rm", "run", "scan-analyze", "scan", "scp", "sudov", "tail", "theme", "top", "weaken", ]; export async function determineAllPossibilitiesForTabCompletion( p: IPlayer, input: string, index: number, currPath = "", ): Promise { input = substituteAliases(input); let allPos: string[] = []; allPos = allPos.concat(Object.keys(GlobalAliases)); const currServ = p.getCurrentServer(); const homeComputer = p.getHomeComputer(); let parentDirPath = ""; let evaledParentDirPath: string | null = null; // Helper functions function addAllCodingContracts(): void { for (const cct of currServ.contracts) { allPos.push(cct.fn); } } function addAllLitFiles(): void { for (const file of currServ.messages) { if (!file.endsWith(".msg")) { allPos.push(file); } } } function addAllMessages(): void { for (const file of currServ.messages) { if (file.endsWith(".msg")) { allPos.push(file); } } } function addAllPrograms(): void { for (const program of homeComputer.programs) { allPos.push(program); } } function addAllScripts(): void { for (const script of currServ.scripts) { const res = processFilepath(script.filename); if (res) { allPos.push(res); } } } function addAllTextFiles(): void { for (const txt of currServ.textFiles) { const res = processFilepath(txt.fn); if (res) { allPos.push(res); } } } function addAllDirectories(): void { // Directories are based on the currently evaluated path const subdirs = getSubdirectories(currServ, evaledParentDirPath == null ? "/" : evaledParentDirPath); for (let i = 0; i < subdirs.length; ++i) { const assembledDirPath = evaledParentDirPath == null ? subdirs[i] : evaledParentDirPath + subdirs[i]; const res = processFilepath(assembledDirPath); if (res != null) { subdirs[i] = res; } } allPos = allPos.concat(subdirs); } // Convert from the real absolute path back to the original path used in the input function convertParentPath(filepath: string): string { if (parentDirPath == null || evaledParentDirPath == null) { console.warn(`convertParentPath() called when paths are null`); return filepath; } if (!filepath.startsWith(evaledParentDirPath)) { console.warn( `convertParentPath() called for invalid path. (filepath=${filepath}) (evaledParentDirPath=${evaledParentDirPath})`, ); return filepath; } return parentDirPath + filepath.slice(evaledParentDirPath.length); } // Given an a full, absolute filepath, converts it to the proper value // for autocompletion purposes function processFilepath(filepath: string): string | null { if (evaledParentDirPath) { if (filepath.startsWith(evaledParentDirPath)) { return convertParentPath(filepath); } } else if (parentDirPath !== "") { // If the parent directory is the root directory, but we're not searching // it from the root directory, we have to add the original path let t_parentDirPath = parentDirPath; if (!t_parentDirPath.endsWith("/")) { t_parentDirPath += "/"; } return parentDirPath + filepath; } else { return filepath; } return null; } function isCommand(cmd: string): boolean { let t_cmd = cmd; if (!t_cmd.endsWith(" ")) { t_cmd += " "; } return input.startsWith(t_cmd); } /** * If the command starts with './' and the index == -1, then the user * has input ./partialexecutablename so autocomplete the script or program. * Put './' in front of each script/executable */ if (isCommand("./") && index == -1) { //All programs and scripts for (let i = 0; i < currServ.scripts.length; ++i) { allPos.push("./" + currServ.scripts[i].filename); } //Programs are on home computer for (let i = 0; i < homeComputer.programs.length; ++i) { allPos.push("./" + homeComputer.programs[i]); } return allPos; } // Autocomplete the command if (index === -1) { return commands.concat(Object.keys(Aliases)).concat(Object.keys(GlobalAliases)); } // Since we're autocompleting an argument and not a command, the argument might // be a file/directory path. We have to account for that when autocompleting const commandArray = input.split(" "); if (commandArray.length === 0) { console.warn(`Tab autocompletion logic reached invalid branch`); return allPos; } const arg = commandArray[commandArray.length - 1]; parentDirPath = getAllParentDirectories(arg); evaledParentDirPath = evaluateDirectoryPath(parentDirPath, currPath); if (evaledParentDirPath === "/") { evaledParentDirPath = null; } else if (evaledParentDirPath == null) { return allPos; // Invalid path } else { evaledParentDirPath += "/"; } if (isCommand("buy")) { const options = []; for (const i in DarkWebItems) { const item = DarkWebItems[i]; options.push(item.program); } return options.concat(Object.keys(GlobalAliases)); } if (isCommand("scp") && index === 1) { for (const server of GetAllServers()) { allPos.push(server.hostname); } return allPos; } if (isCommand("scp") && index === 0) { addAllScripts(); addAllLitFiles(); addAllTextFiles(); addAllDirectories(); return allPos; } if (isCommand("cp") && index === 0) { addAllScripts(); addAllTextFiles(); addAllDirectories(); return allPos; } if (isCommand("connect")) { // All network connections for (let i = 0; i < currServ.serversOnNetwork.length; ++i) { const serv = GetServer(currServ.serversOnNetwork[i]); if (serv == null) { continue; } allPos.push(serv.hostname); } return allPos; } if (isCommand("kill") || isCommand("tail") || isCommand("mem") || isCommand("check")) { addAllScripts(); addAllDirectories(); return allPos; } if (isCommand("nano")) { addAllScripts(); addAllTextFiles(); allPos.push(".fconf"); addAllDirectories(); return allPos; } if (isCommand("rm")) { addAllScripts(); addAllPrograms(); addAllLitFiles(); addAllTextFiles(); addAllCodingContracts(); addAllDirectories(); return allPos; } async function scriptAutocomplete(): Promise { if (!isCommand("run")) return; const commands = ParseCommands(input); if (commands.length === 0) return; const command = ParseCommand(commands[commands.length - 1]); const filename = command[1] + ""; if (!isScriptFilename(filename)) return; // Not a script. if (filename.endsWith(".script")) return; // Doesn't work with ns1. const script = currServ.scripts.find((script) => script.filename === filename); if (!script) return; // Doesn't exist. if (!script.module) { await compile(script, currServ.scripts); } const loadedModule = await script.module; if (!loadedModule.autocomplete) return; // Doesn't have an autocomplete function. const runArgs = { "--tail": Boolean, "-t": Number }; const flags = libarg(runArgs, { permissive: true, argv: command.slice(2), }); const flagFunc = Flags(flags._); let pos: string[] = []; let pos2: string[] = []; pos = pos.concat( loadedModule.autocomplete( { servers: GetAllServers().map((server) => server.hostname), scripts: currServ.scripts.map((script) => script.filename), txts: currServ.textFiles.map((txt) => txt.fn), flags: (schema: any) => { pos2 = schema.map((f: any) => { if (f[0].length === 1) return "-" + f[0]; return "--" + f[0]; }); try { return flagFunc(schema); } catch (err) { return undefined; } }, }, flags._, ), ); return pos.concat(pos2); } const pos = await scriptAutocomplete(); if (pos) return pos; if (isCommand("run")) { addAllScripts(); addAllPrograms(); addAllCodingContracts(); addAllDirectories(); } if (isCommand("cat")) { addAllMessages(); addAllLitFiles(); addAllTextFiles(); addAllDirectories(); return allPos; } if (isCommand("download") || isCommand("mv")) { addAllScripts(); addAllTextFiles(); addAllDirectories(); return allPos; } if (isCommand("cd")) { addAllDirectories(); return allPos; } if (isCommand("ls") && index === 0) { addAllDirectories(); } return allPos; }