159 lines
5.5 KiB
JavaScript
159 lines
5.5 KiB
JavaScript
class ArgParser {
|
|
/**
|
|
* Create a unix-like argument parser to extract flags from the argument list. Can also create help messages.
|
|
* @param opts - {examples: string[], arguments: {key: string, alias: string, type: string, optional: boolean, desc: string}[], desc: string}
|
|
*/
|
|
constructor(opts) {
|
|
this.examples = opts.examples ?? [];
|
|
this.arguments = opts.args ?? [];
|
|
this.description = opts.desc;
|
|
}
|
|
|
|
/**
|
|
* Parse the list for arguments & create a dictionary.
|
|
* @param args {any[]} - Array of arguments
|
|
* @returns Dictionary of matched flags + unmatched args under 'extra'
|
|
*/
|
|
parse(args) {
|
|
const req = this.arguments.filter(a => !a.optional && !a.skip);
|
|
const queue = [...args], parsed = {}, extra = [];
|
|
for(let i = 0; i < queue.length; i++) {
|
|
if(queue[i][0] != '-') {
|
|
extra.push(queue[i]);
|
|
continue;
|
|
}
|
|
let value = null, parse = queue[i].slice(queue[i][1] == '-' ? 2 : 1);
|
|
if(parse.indexOf('=')) {
|
|
const split = parse.split('=');
|
|
parse = split[0];
|
|
value = split[1];
|
|
}
|
|
let arg = this.arguments.find(a => a.key == parse) ?? this.arguments.find(a => a.alias == parse);
|
|
if(!arg) {
|
|
extra.push(queue[i]);
|
|
continue;
|
|
}
|
|
if(!value) {
|
|
value = arg.type == 'bool' ? true : queue[i + 1];
|
|
if(arg.type != 'bool') i++;
|
|
}
|
|
parsed[arg.key] = value;
|
|
}
|
|
req.forEach((a, i) => parsed[a.key] = extra[i]);
|
|
extra.splice(0, req.length);
|
|
return {...parsed, extra};
|
|
}
|
|
|
|
/**
|
|
* Create a help message of the expected paramters & usage.
|
|
* @param msg {String} - Optional message to display with help
|
|
*/
|
|
help(msg) {
|
|
let message = '\n\n';
|
|
message += msg ? msg : this.description;
|
|
if(this.examples.length) message += '\n\nUsage:\t' + this.examples.join('\n\t');
|
|
const required = this.arguments.filter(a => !a.optional);
|
|
if(required.length) message += '\n\n\t' + required.map(a => {
|
|
const padding = 3 - ~~(a.key.length / 8);
|
|
return `${a.key}${Array(padding).fill('\t').join('')} ${a.desc}`;
|
|
}).join('\n\t');
|
|
const optional = this.arguments.filter(a => a.optional);
|
|
if(optional.length) message += '\n\nOptions:\n\t' + optional.map(a => {
|
|
const flgs = `${a.alias ? `-${a.alias} ` : ''}--${a.key}${a.type && a.type != 'bool' ? `=${a.type}` : ''}`;
|
|
const padding = 3 - ~~(flgs.length / 8);
|
|
return `${flgs}${Array(padding).fill('\t').join('')} ${a.desc}`;
|
|
}).join('\n\t');
|
|
return `${message}\n\n`;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Pwn a target server will availible tools. Can also copy & execute a script after pwning.
|
|
*/
|
|
export async function main(ns) {
|
|
ns.disableLog('ALL');
|
|
|
|
/**
|
|
* Prints text and waits a random amount of time to emulate work being complete.
|
|
* @param text {String} - message to Display
|
|
* @param min {Number} - minimum amount of time to wait in seconds. Defaults to 1 second.
|
|
* @param max {Number} - maximum amount of time to wait in seconds. Defaults to 1 second.
|
|
*/
|
|
async function printWithDelay(text, min=1, max=1) {
|
|
ns.tprint(text);
|
|
await ns.sleep(~~(Math.random() * (max * 1000 - min * 1000)) + min * 1000);
|
|
}
|
|
|
|
// Initilize script arguments
|
|
const argParser = new ArgParser({
|
|
desc: 'Automatically gain root on a target machine. Optionaly after being rooted, a file can be coppied & executed.',
|
|
examples: [
|
|
'run auto-pwn.js [TARGET] [SCRIPT] [ARGS]...',
|
|
'run auto-pwn.js --help',
|
|
],
|
|
args: [
|
|
{key: 'TARGET', desc: 'Target machine to root. Defaults to localhost'},
|
|
{key: 'SCRIPT', desc: 'Script to copy & execute'},
|
|
{key: 'ARGS', skip: true, desc: 'Aditional arguments for SCRIPT. Forward the target with "{{TARGET}}"'},
|
|
{key: 'threads', alias: 't', type: 'num', optional: true, desc: 'Set number of threads for script'},
|
|
{key: 'help', alias: 'h', type: 'bool', optional: true, desc: 'Display help message'},
|
|
]
|
|
});
|
|
const args = argParser.parse(ns.args);
|
|
if(args['help']) return ns.tprint(argParser.help());
|
|
|
|
// Setup
|
|
const target = args['TARGET'] && args['TARGET'] != 'localhost' ? args['TARGET'] : ns.getHostname();
|
|
const threads = args['threads'] || !args['SCRIPT'] ? 1 : ~~(ns.getServerMaxRam(target) / ns.getScriptRam(args['SCRIPT'], 'home'));
|
|
|
|
// Banner
|
|
ns.tprint('===================================================');
|
|
ns.tprint(`🧑💻 Pwning: ${target}`);
|
|
await printWithDelay('===================================================');
|
|
|
|
try {
|
|
// Open as many ports as possible
|
|
ns.brutessh(target);
|
|
await printWithDelay(`Attacking (SSH) ⚔️ ${target}:22`, 3, 5);
|
|
ns.ftpcrack(target);
|
|
await printWithDelay(`Attacking (FTP) ⚔️ ${target}:24`, 3, 5);
|
|
} catch {
|
|
} finally {
|
|
// Attempt to root
|
|
try {
|
|
ns.nuke(target)
|
|
await printWithDelay(`💀 Root Granted 💀`);
|
|
} catch {
|
|
await printWithDelay(`⚠️ Failed to Root ⚠️`);
|
|
ns.exit();
|
|
}
|
|
}
|
|
|
|
if(args['SCRIPT']) {
|
|
// Copy scripts
|
|
ns.tprint('');
|
|
await printWithDelay('Copying script:');
|
|
const speed = ~~(Math.random() * 100) / 10
|
|
await ns.scp(args['SCRIPT'], target);
|
|
await printWithDelay(`${args['SCRIPT']} \t [==================>] 100% \t (${speed} MB/s)`);
|
|
|
|
// Run scripts
|
|
ns.tprint('');
|
|
ns.tprint('Executing:');
|
|
ns.scriptKill(args['SCRIPT'], target);
|
|
await printWithDelay(`ssh -c "run ${args['SCRIPT']} ${args['extra'].join(' ')} -t ${threads}" root@${target}`);
|
|
const pid = ns.exec(args['SCRIPT'], target, threads, ...args['extra'].map(a => a == '{{TARGET}}' ? target : a));
|
|
ns.tprint('');
|
|
ns.tprint(pid != null ? '✅ Done!' : '⚠️ Failed to start ⚠️');
|
|
ns.tprint('');
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Autocomplete script arguments
|
|
* @param data - provided by API
|
|
*/
|
|
export function autocomplete(data) {
|
|
return [...data.servers, ...data.scripts];
|
|
}
|