Hacknet Node Automation 1.1
user_3576964
javascript
a year ago
6.1 kB
157
Indexable
Never
/** @param {NS} ns */ export async function main(ns) { ns.disableLog('sleep'); if (!ns.args.length) { return ns.tprint('Supply a target Cluster size'); } const target = ns.args[0]; const cluster = getMaxedNodes(ns); if (cluster >= target) { ns.tprint('Cluster target is less than current cluster size'); ns.tprint(`Target: ${target} | Cluster: ${cluster}`); return; } const killMode = ns.args[1] === 'no-kill' ? false : true; if (!killMode) { ns.tprint('Starting in no-kill mode'); ns.tprint('Running until manually terminated or target reached'); } ns.print('Starting Hacknet node automation'); let maxedCluster = await checkClusterStatus(ns, target); while (!maxedCluster) { await ns.sleep(500); await automateNodes(ns, target, killMode); maxedCluster = await checkClusterStatus(ns, target); } if (maxedCluster) { ns.print(`Cluster target of ${target} reached, exiting`); } } /** @param {NS} ns @param {number} target @param {boolean} killMode @returns {Promise<void>} */ const automateNodes = async (ns, target, killMode) => { return new Promise((resolve) => { const { hacknet: hn } = ns; const { ownedNodes, maxedNodes, nextNodeCost, money } = getStats(ns); let currentNode = maxedNodes; if (maxedNodes === ownedNodes && ownedNodes < target) { if (money < nextNodeCost) { if (!killMode) { (async () => { await handleNoKill(ns, resolve, money, 'New Node', nextNodeCost); })() return; } else { return terminate(ns); } } else { ns.print(`Upgrades finished for Node #${currentNode}`); currentNode = hn.purchaseNode(); ns.print(`Purchased New Node: #${currentNode}`); } } const { level, ram, cores } = hn.getNodeStats(currentNode); const canUpgrade = getCanUpgrade(ns, currentNode, money); if (canUpgrade.all) { upgradeNode(ns, currentNode); handleSuccess(ns, currentNode, level, ram, cores); return resolve(); } for (const component in canUpgrade) { if (component !== 'all') { if (!canUpgrade[component].isAffordable) { if (!killMode) { (async () => { await handleNoKill(ns, resolve, money, 'New Node', nextNodeCost); })() return resolve(); } else { terminate(ns); return resolve(); } } } } return resolve(); }); }; /** @param {NS} ns @param {number} target @returns {Promise<boolean>} */ const checkClusterStatus = async (ns, target) => { return new Promise((resolve) => { const { hacknet: hn } = ns; const ownedNodes = hn.numNodes(); if (ownedNodes === target) { const maxedNodes = getMaxedNodes(ns); if (maxedNodes === target) { return resolve(true); } else { ns.print('Maxed Nodes: ', maxedNodes); } } return resolve(false); }); }; /** @param {NS} ns */ const getMaxedNodes = (ns) => { const { hacknet: hn } = ns; const ownedNodes = hn.numNodes(); let maxedNodes = 0; for (let i = 0; i < ownedNodes; i++) { const { level, cores, ram } = hn.getNodeStats(i); if (level === 200 && cores === 16 && ram === 64) { maxedNodes++; } } return maxedNodes; }; /** @param {NS} ns @param {number} currentNode */ const getUpgradeCosts = (ns, currentNode) => { const { hacknet: hn } = ns; return { levelCost: hn.getLevelUpgradeCost(currentNode), coreCost: hn.getCoreUpgradeCost(currentNode), ramCost: hn.getRamUpgradeCost(currentNode) }; }; /** @param {NS} ns @param {number} currentNode @param {number} money */ const getCanUpgrade = (ns, currentNode, money) => { let { levelCost, coreCost, ramCost } = getUpgradeCosts(ns, currentNode); const all = (money >= levelCost || levelCost === Infinity) && (money >= coreCost || coreCost === Infinity) && (money >= ramCost || ramCost === Infinity) ? true : false; return { all, level: { isAffordable: money < levelCost && levelCost !== Infinity ? false : true, cost: levelCost }, core: { isAffordable: money < coreCost && coreCost !== Infinity ? false : true, cost: coreCost }, ram: { isAffordable: money < ramCost && ramCost !== Infinity ? false : true, cost: ramCost } } } /** @param {NS} ns @param {number} currentNode */ const upgradeNode = (ns, currentNode) => { const { hacknet: hn } = ns; hn.upgradeLevel(currentNode, 10); hn.upgradeCore(currentNode); hn.upgradeRam(currentNode); }; /** @param {NS} ns */ const getStats = (ns) => { const { hacknet: hn } = ns; return { ownedNodes: hn.numNodes(), maxedNodes: getMaxedNodes(ns), nextNodeCost: hn.getPurchaseNodeCost(), money: ns.getPlayer().money }; }; /** @param {NS} ns @param {number} currentNode @param {number} level @param {number} ram @param {number} cores */ const handleSuccess = (ns, currentNode, level, ram, cores) => { ns.print(`Node #${currentNode} upgraded successfully`); ns.print(`Level: ${level} | RAM: ${ram} | Cores: ${cores}`); }; /** * @param {NS} ns * @param {Function} resolve */ const handleNoKill = async (ns, resolve, money, upgradeType, upgradeCost) => { ns.print('Not enough funds for next upgrade'); ns.print(`Funds needed for ${upgradeType}: ${ns.formatNumber(upgradeCost)}`); ns.print(`Current funds: ${ns.formatNumber(money)}`); ns.print('Running in no-kill mode, restarting in 15 seconds'); await ns.sleep(15000); return resolve(); }; /** @param {NS} ns */ const terminate = (ns) => { ns.tprint('Insufficient funds for automation, terminating manager.js'); return ns.kill(ns.pid); };