From e1b8a23f1e0a40d187e3016fe77b3a17d5842e62 Mon Sep 17 00:00:00 2001 From: danielyxie Date: Sat, 2 Mar 2019 19:15:10 -0800 Subject: [PATCH 01/10] Started server code refactor --- src/Message/Message.ts | 32 ++ src/{Message.js => Message/MessageHelpers.js} | 41 +- src/Script/RunningScript.ts | 155 ++++++ src/Script/Script.ts | 104 ++++ src/{Script.js => Script/ScriptHelpers.js} | 187 +------ src/Server.js | 468 ------------------ src/Server/AllServers.ts | 111 +++++ src/Server/Server.ts | 303 ++++++++++++ src/Server/ServerHelpers.js | 126 +++++ src/{ => Server}/ServerPurchases.js | 0 src/{ => Server}/SpecialServerIps.js | 0 utils/IPAddress.js | 33 -- utils/IPAddress.ts | 25 + 13 files changed, 868 insertions(+), 717 deletions(-) create mode 100644 src/Message/Message.ts rename src/{Message.js => Message/MessageHelpers.js} (86%) create mode 100644 src/Script/RunningScript.ts create mode 100644 src/Script/Script.ts rename src/{Script.js => Script/ScriptHelpers.js} (85%) mode change 100755 => 100644 delete mode 100644 src/Server.js create mode 100644 src/Server/AllServers.ts create mode 100644 src/Server/Server.ts create mode 100644 src/Server/ServerHelpers.js rename src/{ => Server}/ServerPurchases.js (100%) rename src/{ => Server}/SpecialServerIps.js (100%) delete mode 100644 utils/IPAddress.js create mode 100644 utils/IPAddress.ts diff --git a/src/Message/Message.ts b/src/Message/Message.ts new file mode 100644 index 000000000..1c56ecd8b --- /dev/null +++ b/src/Message/Message.ts @@ -0,0 +1,32 @@ +import { Reviver, + Generic_toJSON, + Generic_fromJSON } from "../../utils/JSONReviver"; + +export class Message { + // Initializes a Message Object from a JSON save state + static fromJSON(value: any): Message { + return Generic_fromJSON(Message, value.data); + } + + // Name of Message file + filename: string = ""; + + // The text contains in the Message + msg: string = "": + + // Flag indicating whether this Message has been received by the player + recvd: boolean = false; + + constructor(filename="", msg="") { + this.filename = filename; + this.msg = msg; + this.recvd = false; + } + + // Serialize the current object to a JSON save state + toJSON(): any { + return Generic_toJSON("Message", this); + } +} + +Reviver.constructors.Message = Message; diff --git a/src/Message.js b/src/Message/MessageHelpers.js similarity index 86% rename from src/Message.js rename to src/Message/MessageHelpers.js index 7ebee10fa..8fc056fb9 100644 --- a/src/Message.js +++ b/src/Message/MessageHelpers.js @@ -1,34 +1,15 @@ -import { Augmentatation } from "./Augmentation/Augmentation"; -import { Augmentations } from "./Augmentation/Augmentations"; -import { AugmentationNames } from "./Augmentation/data/AugmentationNames"; -import { Programs } from "./Programs/Programs"; -import { inMission } from "./Missions"; -import { Player } from "./Player"; -import { redPillFlag } from "./RedPill"; -import { GetServerByHostname } from "./Server"; -import { Settings } from "./Settings/Settings"; +import { Message } from "./Message"; +import { Augmentatation } from "../Augmentation/Augmentation"; +import { Augmentations } from "../Augmentation/Augmentations"; +import { AugmentationNames } from "../Augmentation/data/AugmentationNames"; +import { Programs } from "../Programs/Programs"; +import { inMission } from "../Missions"; +import { Player } from "../Player"; +import { redPillFlag } from "../RedPill"; +import { GetServerByHostname } from "../Server"; +import { Settings } from "../Settings/Settings"; import { dialogBoxCreate, - dialogBoxOpened} from "../utils/DialogBox"; -import {Reviver, Generic_toJSON, - Generic_fromJSON} from "../utils/JSONReviver"; - -/* Message.js */ -function Message(filename="", msg="") { - this.filename = filename; - this.msg = msg; - this.recvd = false; -} - -Message.prototype.toJSON = function() { - return Generic_toJSON("Message", this); -} - - -Message.fromJSON = function(value) { - return Generic_fromJSON(Message, value.data); -} - -Reviver.constructors.Message = Message; + dialogBoxOpened} from "../../utils/DialogBox"; //Sends message to player, including a pop up function sendMessage(msg, forced=false) { diff --git a/src/Script/RunningScript.ts b/src/Script/RunningScript.ts new file mode 100644 index 000000000..c20e3e5f5 --- /dev/null +++ b/src/Script/RunningScript.ts @@ -0,0 +1,155 @@ +// Class representing a Script instance that is actively running. +// A Script can have multiple active instances +import { Script } from "./Script"; +import { IMap } from "../types"; +import { Generic_fromJSON, + Generic_toJSON, + Reviver } from "../../utils/JSONReviver"; + +export class RunningScript { + // Initializes a RunningScript Object from a JSON save state + static fromJSON(value: any): RunningScript { + return Generic_fromJSON(RunningScript, value.data); + } + + // Script arguments + args: any[] = []; + + // Holds a map of servers hacked, where server = key and the value for each + // server is an array of four numbers. The four numbers represent: + // [MoneyStolen, NumTimesHacked, NumTimesGrown, NumTimesWeaken] + // This data is used for offline progress + dataMap: IMap = {}; + + // Script filename + filename: string = ""; + + // This script's logs. An array of log entries + logs: string[] = []; + + // Flag indicating whether the logs have been updated since + // the last time the UI was updated + logUpd: boolean = false; + + // Total amount of hacking experience earned from this script when offline + offlineExpGained: number = 0; + + // Total amount of money made by this script when offline + offlineMoneyMade: number = 0; + + // Number of seconds that the script has been running offline + offlineRunningTime: number = 0.01; + + // Total amount of hacking experience earned from this script when online + onlineExpGained: number = 0; + + // Total amount of money made by this script when online + onlineMoneyMade: number = 0; + + // Number of seconds that this script has been running online + onlineRunningTime: number = 0.01; + + // How much RAM this script uses for ONE thread + ramUsage: number = 0; + + // IP of the server on which this script is running + server: string = ""; + + // Number of threads that this script is running with + threads: number = 1; + + constructor(script: Script | null = null, args: any[] = []) { + if (script == null) { return; } + this.filename = script.filename; + this.args = args; + + this.server = script.server; //IP Address only + this.ramUsage = script.ramUsage; + } + + RunningScript.prototype.getCode = function() { + const server = AllServers[this.server]; + if (server == null) { return ""; } + for (let i = 0; i < server.scripts.length; ++i) { + if (server.scripts[i].filename === this.filename) { + return server.scripts[i].code; + } + } + + return ""; + } + + RunningScript.prototype.getRamUsage = function() { + if (this.ramUsage != null && this.ramUsage > 0) { return this.ramUsage; } // Use cached value + + const server = AllServers[this.server]; + if (server == null) { return 0; } + for (let i = 0; i < server.scripts.length; ++i) { + if (server.scripts[i].filename === this.filename) { + // Cache the ram usage for the next call + this.ramUsage = server.scripts[i].ramUsage; + return this.ramUsage; + } + } + + + return 0; + } + + RunningScript.prototype.log = function(txt) { + if (this.logs.length > Settings.MaxLogCapacity) { + //Delete first element and add new log entry to the end. + //TODO Eventually it might be better to replace this with circular array + //to improve performance + this.logs.shift(); + } + let logEntry = txt; + if (FconfSettings.ENABLE_TIMESTAMPS) { + logEntry = "[" + getTimestamp() + "] " + logEntry; + } + this.logs.push(logEntry); + this.logUpd = true; + } + + RunningScript.prototype.displayLog = function() { + for (var i = 0; i < this.logs.length; ++i) { + post(this.logs[i]); + } + } + + RunningScript.prototype.clearLog = function() { + this.logs.length = 0; + } + + //Update the moneyStolen and numTimesHack maps when hacking + RunningScript.prototype.recordHack = function(serverIp, moneyGained, n=1) { + if (this.dataMap[serverIp] == null || this.dataMap[serverIp].constructor !== Array) { + this.dataMap[serverIp] = [0, 0, 0, 0]; + } + this.dataMap[serverIp][0] += moneyGained; + this.dataMap[serverIp][1] += n; + } + + //Update the grow map when calling grow() + RunningScript.prototype.recordGrow = function(serverIp, n=1) { + if (this.dataMap[serverIp] == null || this.dataMap[serverIp].constructor !== Array) { + this.dataMap[serverIp] = [0, 0, 0, 0]; + } + this.dataMap[serverIp][2] += n; + } + + //Update the weaken map when calling weaken() { + RunningScript.prototype.recordWeaken = function(serverIp, n=1) { + if (this.dataMap[serverIp] == null || this.dataMap[serverIp].constructor !== Array) { + this.dataMap[serverIp] = [0, 0, 0, 0]; + } + this.dataMap[serverIp][3] += n; + } + + // Serialize the current object to a JSON save state + toJSON(): any { + return Generic_toJSON("RunningScript", this); + } +} + +Reviver.constructors.RunningScript = RunningScript; diff --git a/src/Script/Script.ts b/src/Script/Script.ts new file mode 100644 index 000000000..fa64ed0d8 --- /dev/null +++ b/src/Script/Script.ts @@ -0,0 +1,104 @@ +// Class representing a script file +// This does NOT represent a script that is actively running and +// being evaluated. See RunningScript for that +import { Page, + routing } from "../ui/navigationTracking"; +import { setTimeoutRef } from "../utils/SetTimeoutRef"; +import { Generic_fromJSON, + Generic_toJSON, + Reviver } from "../../utils/JSONReviver"; +import { roundToTwo } from "../../utils/helpers/roundToTwo"; + +export class Script { + // Initializes a Script Object from a JSON save state + static fromJSON(value: any): Script { + return Generic_fromJSON(Script, value.data); + } + + // Code for this script + code: string = ""; + + // Filename for the script file + filename: string = ""; + + // The dynamic module generated for this script when it is run. + // This is only applicable for NetscriptJS + module: any = ""; + + // Amount of RAM this Script requres to run + ramUsage: number = 0; + + // IP of server that this script is on. + server: string = ""; + + + constructor(fn: string = "", code: string = "", server: string = "") { + this.filename = fn; + this.code = code; + this.ramUsage = 0; + this.server = server; // IP of server this script is on + this.module = ""; + if (this.code !== "") {this.updateRamUsage();} + }; + + download(): void { + const filename = this.filename + ".js"; + const file = new Blob([this.code], {type: 'text/plain'}); + if (window.navigator.msSaveOrOpenBlob) {// IE10+ + window.navigator.msSaveOrOpenBlob(file, filename); + } else { // Others + var a = document.createElement("a"), + url = URL.createObjectURL(file); + a.href = url; + a.download = filename; + document.body.appendChild(a); + a.click(); + setTimeoutRef(function() { + document.body.removeChild(a); + window.URL.revokeObjectURL(url); + }, 0); + } + } + + // Save a script FROM THE SCRIPT EDITOR + saveScript(): void { + if (routing.isOn(Page.ScriptEditor)) { + //Update code and filename + const code = getCurrentEditor().getCode(); + this.code = code.replace(/^\s+|\s+$/g, ''); + + const filenameElem: HTMLInputElement | null = document.getElementById("script-editor-filename") as HTMLInputElement; + if (filenameElem == null) { + console.error(`Failed to get Script filename DOM element`); + return; + } + this.filename = filenameElem!.value; + + // Server + this.server = Player.currentServer; + + //Calculate/update ram usage, execution time, etc. + this.updateRamUsage(); + + this.module = ""; + } + } + + // Updates the script's RAM usage based on its code + async updateRamUsage(): void { + // TODO Commented this out because I think its unnecessary + // DOuble check/Test + // var codeCopy = this.code.repeat(1); + var res = await calculateRamUsage(this.code); + if (res !== -1) { + this.ramUsage = roundToTwo(res); + } + } + + // Serialize the current object to a JSON save state + toJSON(): any { + return Generic_toJSON("Script", this); + } +} + +Reviver.constructors.Script = Script; diff --git a/src/Script.js b/src/Script/ScriptHelpers.js old mode 100755 new mode 100644 similarity index 85% rename from src/Script.js rename to src/Script/ScriptHelpers.js index 5733ab404..45a6ebb99 --- a/src/Script.js +++ b/src/Script/ScriptHelpers.js @@ -308,44 +308,6 @@ function checkValidFilename(filename) { return false; } -function Script(fn = "", code = "", server = "") { - this.filename = fn; - this.code = code; - this.ramUsage = 0; - this.server = server; //IP of server this script is on - this.module = ""; - if (this.code !== "") {this.updateRamUsage();} -}; - -//Get the script data from the Script Editor and save it to the object -Script.prototype.saveScript = function() { - if (routing.isOn(Page.ScriptEditor)) { - //Update code and filename - const code = getCurrentEditor().getCode(); - this.code = code.replace(/^\s+|\s+$/g, ''); - - var filename = document.getElementById("script-editor-filename").value; - this.filename = filename; - - //Server - this.server = Player.currentServer; - - //Calculate/update ram usage, execution time, etc. - this.updateRamUsage(); - - this.module = ""; - } -} - -//Updates how much RAM the script uses when it is running. -Script.prototype.updateRamUsage = async function() { - var codeCopy = this.code.repeat(1); - var res = await calculateRamUsage(codeCopy); - if (res !== -1) { - this.ramUsage = roundToTwo(res); - } -} - // These special strings are used to reference the presence of a given logical // construct within a user script. const specialReferenceIF = "__SPECIAL_referenceIf"; @@ -747,36 +709,6 @@ async function calculateRamUsage(codeCopy) { return ramUsage; } -Script.prototype.download = function() { - var filename = this.filename + ".js"; - var file = new Blob([this.code], {type: 'text/plain'}); - if (window.navigator.msSaveOrOpenBlob) {// IE10+ - window.navigator.msSaveOrOpenBlob(file, filename); - } else { // Others - var a = document.createElement("a"), - url = URL.createObjectURL(file); - a.href = url; - a.download = filename; - document.body.appendChild(a); - a.click(); - setTimeoutRef(function() { - document.body.removeChild(a); - window.URL.revokeObjectURL(url); - }, 0); - } -} - -Script.prototype.toJSON = function() { - return Generic_toJSON("Script", this); -} - - -Script.fromJSON = function(value) { - return Generic_fromJSON(Script, value.data); -} - -Reviver.constructors.Script = Script; - //Called when the game is loaded. Loads all running scripts (from all servers) //into worker scripts so that they will start running function loadAllRunningScripts() { @@ -916,122 +848,5 @@ function findRunningScript(filename, args, server) { return null; } -function RunningScript(script, args) { - if (script == null || script == undefined) { return; } - this.filename = script.filename; - this.args = args; - this.server = script.server; //IP Address only - this.ramUsage = script.ramUsage; - - this.logs = []; //Script logging. Array of strings, with each element being a log entry - this.logUpd = false; - - //Stats to display on the Scripts menu, and used to determine offline progress - this.offlineRunningTime = 0.01; //Seconds - this.offlineMoneyMade = 0; - this.offlineExpGained = 0; - this.onlineRunningTime = 0.01; //Seconds - this.onlineMoneyMade = 0; - this.onlineExpGained = 0; - - this.threads = 1; - - // Holds a map of all servers, where server = key and the value for each - // server is an array of four numbers. The four numbers represent: - // [MoneyStolen, NumTimesHacked, NumTimesGrown, NumTimesWeaken] - // This data is used for offline progress - this.dataMap = {}; -} - -RunningScript.prototype.getCode = function() { - const server = AllServers[this.server]; - if (server == null) { return ""; } - for (let i = 0; i < server.scripts.length; ++i) { - if (server.scripts[i].filename === this.filename) { - return server.scripts[i].code; - } - } - - return ""; -} - -RunningScript.prototype.getRamUsage = function() { - if (this.ramUsage != null && this.ramUsage > 0) { return this.ramUsage; } // Use cached value - - const server = AllServers[this.server]; - if (server == null) { return 0; } - for (let i = 0; i < server.scripts.length; ++i) { - if (server.scripts[i].filename === this.filename) { - // Cache the ram usage for the next call - this.ramUsage = server.scripts[i].ramUsage; - return this.ramUsage; - } - } - - - return 0; -} - -RunningScript.prototype.log = function(txt) { - if (this.logs.length > Settings.MaxLogCapacity) { - //Delete first element and add new log entry to the end. - //TODO Eventually it might be better to replace this with circular array - //to improve performance - this.logs.shift(); - } - let logEntry = txt; - if (FconfSettings.ENABLE_TIMESTAMPS) { - logEntry = "[" + getTimestamp() + "] " + logEntry; - } - this.logs.push(logEntry); - this.logUpd = true; -} - -RunningScript.prototype.displayLog = function() { - for (var i = 0; i < this.logs.length; ++i) { - post(this.logs[i]); - } -} - -RunningScript.prototype.clearLog = function() { - this.logs.length = 0; -} - -//Update the moneyStolen and numTimesHack maps when hacking -RunningScript.prototype.recordHack = function(serverIp, moneyGained, n=1) { - if (this.dataMap[serverIp] == null || this.dataMap[serverIp].constructor !== Array) { - this.dataMap[serverIp] = [0, 0, 0, 0]; - } - this.dataMap[serverIp][0] += moneyGained; - this.dataMap[serverIp][1] += n; -} - -//Update the grow map when calling grow() -RunningScript.prototype.recordGrow = function(serverIp, n=1) { - if (this.dataMap[serverIp] == null || this.dataMap[serverIp].constructor !== Array) { - this.dataMap[serverIp] = [0, 0, 0, 0]; - } - this.dataMap[serverIp][2] += n; -} - -//Update the weaken map when calling weaken() { -RunningScript.prototype.recordWeaken = function(serverIp, n=1) { - if (this.dataMap[serverIp] == null || this.dataMap[serverIp].constructor !== Array) { - this.dataMap[serverIp] = [0, 0, 0, 0]; - } - this.dataMap[serverIp][3] += n; -} - -RunningScript.prototype.toJSON = function() { - return Generic_toJSON("RunningScript", this); -} - - -RunningScript.fromJSON = function(value) { - return Generic_fromJSON(RunningScript, value.data); -} - -Reviver.constructors.RunningScript = RunningScript; - export {loadAllRunningScripts, findRunningScript, - RunningScript, Script, scriptEditorInit, isScriptFilename}; + scriptEditorInit, isScriptFilename}; diff --git a/src/Server.js b/src/Server.js deleted file mode 100644 index 6aa27b6f4..000000000 --- a/src/Server.js +++ /dev/null @@ -1,468 +0,0 @@ -import { BitNodeMultipliers } from "./BitNode/BitNodeMultipliers"; -import { CodingContract, - ContractTypes } from "./CodingContracts"; -import { CONSTANTS } from "./Constants"; -import { Script, - isScriptFilename } from "./Script"; -import { Player } from "./Player"; -import { Programs } from "./Programs/Programs"; -import { SpecialServerIps } from "./SpecialServerIps"; -import { TextFile } from "./TextFile"; -import { getRandomInt } from "../utils/helpers/getRandomInt"; -import { createRandomIp, - ipExists } from "../utils/IPAddress"; -import { serverMetadata } from "./data/servers"; -import { Reviver, - Generic_toJSON, - Generic_fromJSON} from "../utils/JSONReviver"; -import {isValidIPAddress} from "../utils/helpers/isValidIPAddress"; - -function Server(params={ip:createRandomIp(), hostname:""}) { - /* Properties */ - //Connection information - this.ip = params.ip ? params.ip : createRandomIp(); - - var hostname = params.hostname; - var i = 0; - var suffix = ""; - while (GetServerByHostname(hostname+suffix) != null) { - //Server already exists - suffix = "-" + i; - ++i; - } - this.hostname = hostname + suffix; - this.organizationName = params.organizationName != null ? params.organizationName : ""; - this.isConnectedTo = params.isConnectedTo != null ? params.isConnectedTo : false; - - //Access information - this.hasAdminRights = params.adminRights != null ? params.adminRights : false; - this.purchasedByPlayer = params.purchasedByPlayer != null ? params.purchasedByPlayer : false; - this.manuallyHacked = false; //Flag that tracks whether or not the server has been hacked at least once - - //RAM, CPU speed and Scripts - this.maxRam = params.maxRam != null ? params.maxRam : 0; //GB - this.ramUsed = 0; - this.cpuCores = 1; //Max of 8, affects hacking times and Hacking Mission starting Cores - - this.scripts = []; - this.runningScripts = []; //Stores RunningScript objects - this.programs = []; - this.messages = []; - this.textFiles = []; - this.contracts = []; - this.dir = 0; //new Directory(this, null, ""); TODO - - /* Hacking information (only valid for "foreign" aka non-purchased servers) */ - this.requiredHackingSkill = params.requiredHackingSkill != null ? params.requiredHackingSkill : 1; - this.moneyAvailable = params.moneyAvailable != null ? params.moneyAvailable * BitNodeMultipliers.ServerStartingMoney : 0; - this.moneyMax = 25 * this.moneyAvailable * BitNodeMultipliers.ServerMaxMoney; - - //Hack Difficulty is synonymous with server security. Base Difficulty = Starting difficulty - this.hackDifficulty = params.hackDifficulty != null ? params.hackDifficulty * BitNodeMultipliers.ServerStartingSecurity : 1; - this.baseDifficulty = this.hackDifficulty; - this.minDifficulty = Math.max(1, Math.round(this.hackDifficulty / 3)); - this.serverGrowth = params.serverGrowth != null ? params.serverGrowth : 1; //Integer from 0 to 100. Affects money increase from grow() - - //The IP's of all servers reachable from this one (what shows up if you run scan/netstat) - // NOTE: Only contains IP and not the Server objects themselves - this.serversOnNetwork = []; - - //Port information, required for porthacking servers to get admin rights - this.numOpenPortsRequired = params.numOpenPortsRequired != null ? params.numOpenPortsRequired : 5; - this.sshPortOpen = false; //Port 22 - this.ftpPortOpen = false; //Port 21 - this.smtpPortOpen = false; //Port 25 - this.httpPortOpen = false; //Port 80 - this.sqlPortOpen = false; //Port 1433 - this.openPortCount = 0; -}; - -Server.prototype.setMaxRam = function(ram) { - this.maxRam = ram; -} - -//The serverOnNetwork array holds the IP of all the servers. This function -//returns the actual Server objects -Server.prototype.getServerOnNetwork = function(i) { - if (i > this.serversOnNetwork.length) { - console.log("Tried to get server on network that was out of range"); - return; - } - return AllServers[this.serversOnNetwork[i]]; -} - -//Given the name of the script, returns the corresponding -//script object on the server (if it exists) -Server.prototype.getScript = function(scriptName) { - for (var i = 0; i < this.scripts.length; i++) { - if (this.scripts[i].filename == scriptName) { - return this.scripts[i]; - } - } - return null; -} - -Server.prototype.capDifficulty = function() { - if (this.hackDifficulty < this.minDifficulty) {this.hackDifficulty = this.minDifficulty;} - if (this.hackDifficulty < 1) {this.hackDifficulty = 1;} - //Place some arbitrarily limit that realistically should never happen unless someone is - //screwing around with the game - if (this.hackDifficulty > 1000000) {this.hackDifficulty = 1000000;} -} - -//Strengthens a server's security level (difficulty) by the specified amount -Server.prototype.fortify = function(amt) { - this.hackDifficulty += amt; - this.capDifficulty(); -} - -Server.prototype.weaken = function(amt) { - this.hackDifficulty -= (amt * BitNodeMultipliers.ServerWeakenRate); - this.capDifficulty(); -} - -// Write to a script file -// Overwrites existing files. Creates new files if the script does not eixst -Server.prototype.writeToScriptFile = function(fn, code) { - var ret = {success: false, overwritten: false}; - if (!isScriptFilename(fn)) { return ret; } - - //Check if the script already exists, and overwrite it if it does - for (let i = 0; i < this.scripts.length; ++i) { - if (fn === this.scripts[i].filename) { - let script = this.scripts[i]; - script.code = code; - script.updateRamUsage(); - script.module = ""; - ret.overwritten = true; - ret.success = true; - return ret; - } - } - - //Otherwise, create a new script - var newScript = new Script(); - newScript.filename = fn; - newScript.code = code; - newScript.updateRamUsage(); - newScript.server = this.ip; - this.scripts.push(newScript); - ret.success = true; - return ret; -} - -// Write to a text file -// Overwrites existing files. Creates new files if the text file does not exist -Server.prototype.writeToTextFile = function(fn, txt) { - var ret = {success: false, overwritten: false}; - if (!fn.endsWith("txt")) { return ret; } - - //Check if the text file already exists, and overwrite if it does - for (let i = 0; i < this.textFiles.length; ++i) { - if (this.textFiles[i].fn === fn) { - ret.overwritten = true; - this.textFiles[i].text = txt; - ret.success = true; - return ret; - } - } - - //Otherwise create a new text file - var newFile = new TextFile(fn, txt); - this.textFiles.push(newFile); - ret.success = true; - return ret; -} - -Server.prototype.addContract = function(contract) { - this.contracts.push(contract); -} - -Server.prototype.removeContract = function(contract) { - if (contract instanceof CodingContract) { - this.contracts = this.contracts.filter((c) => { - return c.fn !== contract.fn; - }); - } else { - this.contracts = this.contracts.filter((c) => { - return c.fn !== contract; - }); - } -} - -Server.prototype.getContract = function(contractName) { - for (const contract of this.contracts) { - if (contract.fn === contractName) { - return contract; - } - } - return null; -} - -//Functions for loading and saving a Server -Server.prototype.toJSON = function() { - return Generic_toJSON("Server", this); -} - -Server.fromJSON = function(value) { - return Generic_fromJSON(Server, value.data); -} - -Reviver.constructors.Server = Server; - -export function initForeignServers() { - /* Create a randomized network for all the foreign servers */ - //Groupings for creating a randomized network - const networkLayers = []; - for (let i = 0; i < 15; i++) { - networkLayers.push([]); - } - - // Essentially any property that is of type 'number | IMinMaxRange' - const propertiesToPatternMatch = [ - "hackDifficulty", - "moneyAvailable", - "requiredHackingSkill", - "serverGrowth" - ]; - - const toNumber = (value) => { - switch (typeof value) { - case 'number': - return value; - case 'object': - return getRandomInt(value.min, value.max); - default: - throw Error(`Do not know how to convert the type '${typeof value}' to a number`); - } - } - - for (const metadata of serverMetadata) { - const serverParams = { - hostname: metadata.hostname, - ip: createRandomIp(), - numOpenPortsRequired: metadata.numOpenPortsRequired, - organizationName: metadata.organizationName - }; - - if (metadata.maxRamExponent !== undefined) { - serverParams.maxRam = Math.pow(2, toNumber(metadata.maxRamExponent)); - } - - for (const prop of propertiesToPatternMatch) { - if (metadata[prop] !== undefined) { - serverParams[prop] = toNumber(metadata[prop]); - } - } - - const server = new Server(serverParams); - for (const filename of (metadata.literature || [])) { - server.messages.push(filename); - } - - if (metadata.specialName !== undefined) { - SpecialServerIps.addIp(metadata.specialName, server.ip); - } - - AddToAllServers(server); - if (metadata.networkLayer !== undefined) { - networkLayers[toNumber(metadata.networkLayer) - 1].push(server); - } - } - - /* Create a randomized network for all the foreign servers */ - const linkComputers = (server1, server2) => { - server1.serversOnNetwork.push(server2.ip); - server2.serversOnNetwork.push(server1.ip); - }; - - const getRandomArrayItem = (arr) => arr[Math.floor(Math.random() * arr.length)]; - - const linkNetworkLayers = (network1, selectServer) => { - for (const server of network1) { - linkComputers(server, selectServer()); - } - }; - - // Connect the first tier of servers to the player's home computer - linkNetworkLayers(networkLayers[0], () => Player.getHomeComputer()); - for (let i = 1; i < networkLayers.length; i++) { - linkNetworkLayers(networkLayers[i], () => getRandomArrayItem(networkLayers[i - 1])); - } -} - -// Returns the number of cycles needed to grow the specified server by the -// specified amount. 'growth' parameter is in decimal form, not percentage -export function numCycleForGrowth(server, growth) { - let ajdGrowthRate = 1 + (CONSTANTS.ServerBaseGrowthRate - 1) / server.hackDifficulty; - if(ajdGrowthRate > CONSTANTS.ServerMaxGrowthRate) { - ajdGrowthRate = CONSTANTS.ServerMaxGrowthRate; - } - - const serverGrowthPercentage = server.serverGrowth / 100; - - const cycles = Math.log(growth)/(Math.log(ajdGrowthRate)*Player.hacking_grow_mult*serverGrowthPercentage); - return cycles; -} - -//Applied server growth for a single server. Returns the percentage growth -export function processSingleServerGrowth(server, numCycles) { - //Server growth processed once every 450 game cycles - const numServerGrowthCycles = Math.max(Math.floor(numCycles / 450), 0); - - //Get adjusted growth rate, which accounts for server security - const growthRate = CONSTANTS.ServerBaseGrowthRate; - var adjGrowthRate = 1 + (growthRate - 1) / server.hackDifficulty; - if (adjGrowthRate > CONSTANTS.ServerMaxGrowthRate) {adjGrowthRate = CONSTANTS.ServerMaxGrowthRate;} - - //Calculate adjusted server growth rate based on parameters - const serverGrowthPercentage = server.serverGrowth / 100; - const numServerGrowthCyclesAdjusted = numServerGrowthCycles * serverGrowthPercentage * BitNodeMultipliers.ServerGrowthRate; - - //Apply serverGrowth for the calculated number of growth cycles - var serverGrowth = Math.pow(adjGrowthRate, numServerGrowthCyclesAdjusted * Player.hacking_grow_mult); - if (serverGrowth < 1) { - console.log("WARN: serverGrowth calculated to be less than 1"); - serverGrowth = 1; - } - - const oldMoneyAvailable = server.moneyAvailable; - server.moneyAvailable *= serverGrowth; - - // in case of data corruption - if (server.moneyMax && isNaN(server.moneyAvailable)) { - server.moneyAvailable = server.moneyMax; - } - - // cap at max - if (server.moneyMax && server.moneyAvailable > server.moneyMax) { - server.moneyAvailable = server.moneyMax; - } - - // if there was any growth at all, increase security - if (oldMoneyAvailable !== server.moneyAvailable) { - //Growing increases server security twice as much as hacking - let usedCycles = numCycleForGrowth(server, server.moneyAvailable / oldMoneyAvailable); - usedCycles = Math.max(0, usedCycles); - server.fortify(2 * CONSTANTS.ServerFortifyAmount * Math.ceil(usedCycles)); - } - return server.moneyAvailable / oldMoneyAvailable; -} - -export function prestigeHomeComputer(homeComp) { - const hasBitflume = homeComp.programs.includes(Programs.BitFlume.name); - - homeComp.programs.length = 0; //Remove programs - homeComp.runningScripts = []; - homeComp.serversOnNetwork = []; - homeComp.isConnectedTo = true; - homeComp.ramUsed = 0; - homeComp.programs.push(Programs.NukeProgram.name); - if (hasBitflume) { homeComp.programs.push(Programs.BitFlume.name); } - - //Update RAM usage on all scripts - homeComp.scripts.forEach(function(script) { - script.updateRamUsage(); - }); - - homeComp.messages.length = 0; //Remove .lit and .msg files - homeComp.messages.push("hackers-starting-handbook.lit"); -} - -//List of all servers that exist in the game, indexed by their ip -let AllServers = {}; - -export function prestigeAllServers() { - for (var member in AllServers) { - delete AllServers[member]; - } - AllServers = {}; -} - -export function loadAllServers(saveString) { - AllServers = JSON.parse(saveString, Reviver); -} - -function SizeOfAllServers() { - var size = 0, key; - for (key in AllServers) { - if (AllServers.hasOwnProperty(key)) size++; - } - return size; -} - -//Add a server onto the map of all servers in the game -export function AddToAllServers(server) { - var serverIp = server.ip; - if (ipExists(serverIp)) { - console.log("IP of server that's being added: " + serverIp); - console.log("Hostname of the server thats being added: " + server.hostname); - console.log("The server that already has this IP is: " + AllServers[serverIp].hostname); - throw new Error("Error: Trying to add a server with an existing IP"); - return; - } - AllServers[serverIp] = server; -} - -//Returns server object with corresponding hostname -// Relatively slow, would rather not use this a lot -export function GetServerByHostname(hostname) { - for (var ip in AllServers) { - if (AllServers.hasOwnProperty(ip)) { - if (AllServers[ip].hostname == hostname) { - return AllServers[ip]; - } - } - } - return null; -} - -//Get server by IP or hostname. Returns null if invalid -export function getServer(s) { - if (!isValidIPAddress(s)) { - return GetServerByHostname(s); - } - if(AllServers[s] !== undefined) { - return AllServers[s]; - } - return null; -} - -//Debugging tool -function PrintAllServers() { - for (var ip in AllServers) { - if (AllServers.hasOwnProperty(ip)) { - console.log("Ip: " + ip + ", hostname: " + AllServers[ip].hostname); - } - } -} - -// Directory object (folders) -function Directory(server, parent, name) { - this.s = server; //Ref to server - this.p = parent; //Ref to parent directory - this.c = []; //Subdirs - this.n = name; - this.d = parent.d + 1; //We'll only have a maximum depth of 3 or something - this.scrs = []; //Holds references to the scripts in server.scripts - this.pgms = []; - this.msgs = []; -} - -Directory.prototype.createSubdir = function(name) { - var subdir = new Directory(this.s, this, name); - -} - -Directory.prototype.getPath = function(name) { - var res = []; - var i = this; - while (i !== null) { - res.unshift(i.n, "/"); - i = i.parent; - } - res.unshift("/"); - return res.join(""); -} - -export {Server, AllServers}; diff --git a/src/Server/AllServers.ts b/src/Server/AllServers.ts new file mode 100644 index 000000000..e04c33fbb --- /dev/null +++ b/src/Server/AllServers.ts @@ -0,0 +1,111 @@ +import { ipExists } from "../../utils/IPAddress"; + +// Map of all Servers that exist in the game +// Key (string) = IP +// Value = Server object +let AllServers = {}; + +// Saftely add a Server to the AllServers map +export function AddToAllServers(server) { + var serverIp = server.ip; + if (ipExists(serverIp)) { + console.log("IP of server that's being added: " + serverIp); + console.log("Hostname of the server thats being added: " + server.hostname); + console.log("The server that already has this IP is: " + AllServers[serverIp].hostname); + throw new Error("Error: Trying to add a server with an existing IP"); + return; + } + AllServers[serverIp] = server; +} + +export function initForeignServers() { + /* Create a randomized network for all the foreign servers */ + //Groupings for creating a randomized network + const networkLayers = []; + for (let i = 0; i < 15; i++) { + networkLayers.push([]); + } + + // Essentially any property that is of type 'number | IMinMaxRange' + const propertiesToPatternMatch = [ + "hackDifficulty", + "moneyAvailable", + "requiredHackingSkill", + "serverGrowth" + ]; + + const toNumber = (value) => { + switch (typeof value) { + case 'number': + return value; + case 'object': + return getRandomInt(value.min, value.max); + default: + throw Error(`Do not know how to convert the type '${typeof value}' to a number`); + } + } + + for (const metadata of serverMetadata) { + const serverParams = { + hostname: metadata.hostname, + ip: createRandomIp(), + numOpenPortsRequired: metadata.numOpenPortsRequired, + organizationName: metadata.organizationName + }; + + if (metadata.maxRamExponent !== undefined) { + serverParams.maxRam = Math.pow(2, toNumber(metadata.maxRamExponent)); + } + + for (const prop of propertiesToPatternMatch) { + if (metadata[prop] !== undefined) { + serverParams[prop] = toNumber(metadata[prop]); + } + } + + const server = new Server(serverParams); + for (const filename of (metadata.literature || [])) { + server.messages.push(filename); + } + + if (metadata.specialName !== undefined) { + SpecialServerIps.addIp(metadata.specialName, server.ip); + } + + AddToAllServers(server); + if (metadata.networkLayer !== undefined) { + networkLayers[toNumber(metadata.networkLayer) - 1].push(server); + } + } + + /* Create a randomized network for all the foreign servers */ + const linkComputers = (server1, server2) => { + server1.serversOnNetwork.push(server2.ip); + server2.serversOnNetwork.push(server1.ip); + }; + + const getRandomArrayItem = (arr) => arr[Math.floor(Math.random() * arr.length)]; + + const linkNetworkLayers = (network1, selectServer) => { + for (const server of network1) { + linkComputers(server, selectServer()); + } + }; + + // Connect the first tier of servers to the player's home computer + linkNetworkLayers(networkLayers[0], () => Player.getHomeComputer()); + for (let i = 1; i < networkLayers.length; i++) { + linkNetworkLayers(networkLayers[i], () => getRandomArrayItem(networkLayers[i - 1])); + } +} + +export function prestigeAllServers() { + for (var member in AllServers) { + delete AllServers[member]; + } + AllServers = {}; +} + +export function loadAllServers(saveString) { + AllServers = JSON.parse(saveString, Reviver); +} diff --git a/src/Server/Server.ts b/src/Server/Server.ts new file mode 100644 index 000000000..b4d1f33dc --- /dev/null +++ b/src/Server/Server.ts @@ -0,0 +1,303 @@ +// Class representing a single generic Server +import { BitNodeMultipliers } from "../BitNode/BitNodeMultipliers"; +import { CodingContract } from "../CodingContracts"; +import { Message } from "../Message/Message"; +import { RunningScript } from "../Script/RunningScript"; +import { Script } from "../Script/Script"; +import { TextFile } from "../TextFile"; + +import { createRandomIp } from "../../utils/IPAddress"; +import { Generic_fromJSON, + Generic_toJSON, + Reviver } from "../../utils/JSONReviver"; + +interface IConstructorParams { + adminRights?: boolean; + hackDifficulty?: number; + hostname: string; + ip?: string; + isConnectedTo?: boolean; + maxRam?: number; + moneyAvailable?: number; + numOpenPortsRequired?: number; + organizationName?: string; + purchasedByPlayer?: boolean; + requiredHackingSkill?: number; + serverGrowth?: number; +} + +export class Server { + // Initial server security level + // (i.e. security level when the server was created) + baseDifficulty: number = 1; + + // Coding Contract files on this server + contracts: CodingContract[] = []; + + // How many CPU cores this server has. Maximum of 8. + // Currently, this only affects hacking missions + cpuCores: number = 1; + + // Flag indicating whether the FTP port is open + ftpPortOpen: boolean = false; + + // Server Security Level + hackDifficulty: number = 1; + + // Flag indicating whether player has admin/root access to this server + hasAdminRights: boolean = false; + + // Hostname. Must be unique + hostname: string = ""; + + // Flag indicating whether HTTP Port is open + httpPortOpen: boolean = false; + + // IP Address. Must be unique + ip: string = ""; + + // Flag indicating whether player is curently connected to this server + isConnectedTo: boolean = false; + + // Flag indicating whether this server has been manually hacked (ie. + // hacked through Terminal) by the player + manuallyHacked: boolean = false; + + // RAM (GB) available on this server + maxRam: number = 0; + + // Message files AND Literature files on this Server + // For Literature files, this array contains only the filename (string) + // For Messages, it contains the actual Message object + // TODO Separate literature files into its own property + messages: (Message | string)[] = []; + + // Minimum server security level that this server can be weakened to + minDifficulty: number = 1; + + // How much money currently resides on the server and can be hacked + moneyAvailable: number = 0; + + // Maximum amount of money that this server can hold + moneyMax: number = 0; + + // Number of open ports required in order to gain admin/root access + numOpenPortsRequired: number = 5; + + // How many ports are currently opened on the server + openPortCount: number = 0; + + // Name of company/faction/etc. that this server belongs to. + // Optional, not applicable to all Servers + organizationName: string = ""; + + // Programs on this servers. Contains only the names of the programs + programs: string[] = []; + + // Flag indicating wehther this is a purchased server + purchasedByPlayer: boolean = false; + + // RAM (GB) used. i.e. unavailable RAM + ramUsed: number = 0; + + // Hacking level required to hack this server + requiredHackingSkill: number = 1; + + // RunningScript files on this server + runningScripts: RunningScript[] = []; + + // Script files on this Server + scripts: Script[] = []; + + // Parameter that affects how effectively this server's money can + // be increased using the grow() Netscript function + serverGrowth: number = 1; + + // Contains the IP Addresses of all servers that are immediately + // reachable from this one + serversOnNetwork: string[] = []; + + // Flag indicating whether SMTP Port is open + smtpPortOpen: boolean = false; + + // Flag indicating whether SQL Port is open + sqlPortOpen: boolean = false; + + // Flag indicating whether the SSH Port is open + sshPortOpen: boolean = false; + + // Text files on this server + textFiles: TextFile[] = []; + + constructor(params: IConstructorParams={hostname: "", ip: createRandomIp() }) { + /* Properties */ + //Connection information + this.ip = params.ip ? params.ip : createRandomIp(); + + var hostname = params.hostname; + var i = 0; + var suffix = ""; + while (GetServerByHostname(hostname+suffix) != null) { + //Server already exists + suffix = "-" + i; + ++i; + } + this.hostname = hostname + suffix; + this.organizationName = params.organizationName != null ? params.organizationName : ""; + this.isConnectedTo = params.isConnectedTo != null ? params.isConnectedTo : false; + + //Access information + this.hasAdminRights = params.adminRights != null ? params.adminRights : false; + this.purchasedByPlayer = params.purchasedByPlayer != null ? params.purchasedByPlayer : false; + + //RAM, CPU speed and Scripts + this.maxRam = params.maxRam != null ? params.maxRam : 0; //GB + + /* Hacking information (only valid for "foreign" aka non-purchased servers) */ + this.requiredHackingSkill = params.requiredHackingSkill != null ? params.requiredHackingSkill : 1; + this.moneyAvailable = params.moneyAvailable != null ? params.moneyAvailable * BitNodeMultipliers.ServerStartingMoney : 0; + this.moneyMax = 25 * this.moneyAvailable * BitNodeMultipliers.ServerMaxMoney; + + //Hack Difficulty is synonymous with server security. Base Difficulty = Starting difficulty + this.hackDifficulty = params.hackDifficulty != null ? params.hackDifficulty * BitNodeMultipliers.ServerStartingSecurity : 1; + this.baseDifficulty = this.hackDifficulty; + this.minDifficulty = Math.max(1, Math.round(this.hackDifficulty / 3)); + this.serverGrowth = params.serverGrowth != null ? params.serverGrowth : 1; //Integer from 0 to 100. Affects money increase from grow() + + //Port information, required for porthacking servers to get admin rights + this.numOpenPortsRequired = params.numOpenPortsRequired != null ? params.numOpenPortsRequired : 5; + }; + + setMaxRam(ram: number): void { + this.maxRam = ram; + } + + //The serverOnNetwork array holds the IP of all the servers. This function + //returns the actual Server objects + Server.prototype.getServerOnNetwork = function(i) { + if (i > this.serversOnNetwork.length) { + console.log("Tried to get server on network that was out of range"); + return; + } + return AllServers[this.serversOnNetwork[i]]; + } + + //Given the name of the script, returns the corresponding + //script object on the server (if it exists) + Server.prototype.getScript = function(scriptName) { + for (var i = 0; i < this.scripts.length; i++) { + if (this.scripts[i].filename == scriptName) { + return this.scripts[i]; + } + } + return null; + } + + Server.prototype.capDifficulty = function() { + if (this.hackDifficulty < this.minDifficulty) {this.hackDifficulty = this.minDifficulty;} + if (this.hackDifficulty < 1) {this.hackDifficulty = 1;} + //Place some arbitrarily limit that realistically should never happen unless someone is + //screwing around with the game + if (this.hackDifficulty > 1000000) {this.hackDifficulty = 1000000;} + } + + //Strengthens a server's security level (difficulty) by the specified amount + Server.prototype.fortify = function(amt) { + this.hackDifficulty += amt; + this.capDifficulty(); + } + + Server.prototype.weaken = function(amt) { + this.hackDifficulty -= (amt * BitNodeMultipliers.ServerWeakenRate); + this.capDifficulty(); + } + + // Write to a script file + // Overwrites existing files. Creates new files if the script does not eixst + Server.prototype.writeToScriptFile = function(fn, code) { + var ret = {success: false, overwritten: false}; + if (!isScriptFilename(fn)) { return ret; } + + //Check if the script already exists, and overwrite it if it does + for (let i = 0; i < this.scripts.length; ++i) { + if (fn === this.scripts[i].filename) { + let script = this.scripts[i]; + script.code = code; + script.updateRamUsage(); + script.module = ""; + ret.overwritten = true; + ret.success = true; + return ret; + } + } + + //Otherwise, create a new script + var newScript = new Script(); + newScript.filename = fn; + newScript.code = code; + newScript.updateRamUsage(); + newScript.server = this.ip; + this.scripts.push(newScript); + ret.success = true; + return ret; + } + + // Write to a text file + // Overwrites existing files. Creates new files if the text file does not exist + Server.prototype.writeToTextFile = function(fn, txt) { + var ret = {success: false, overwritten: false}; + if (!fn.endsWith("txt")) { return ret; } + + //Check if the text file already exists, and overwrite if it does + for (let i = 0; i < this.textFiles.length; ++i) { + if (this.textFiles[i].fn === fn) { + ret.overwritten = true; + this.textFiles[i].text = txt; + ret.success = true; + return ret; + } + } + + //Otherwise create a new text file + var newFile = new TextFile(fn, txt); + this.textFiles.push(newFile); + ret.success = true; + return ret; + } + + Server.prototype.addContract = function(contract) { + this.contracts.push(contract); + } + + Server.prototype.removeContract = function(contract) { + if (contract instanceof CodingContract) { + this.contracts = this.contracts.filter((c) => { + return c.fn !== contract.fn; + }); + } else { + this.contracts = this.contracts.filter((c) => { + return c.fn !== contract; + }); + } + } + + Server.prototype.getContract = function(contractName) { + for (const contract of this.contracts) { + if (contract.fn === contractName) { + return contract; + } + } + return null; + } +} + +//Functions for loading and saving a Server +Server.prototype.toJSON = function() { + return Generic_toJSON("Server", this); +} + +Server.fromJSON = function(value) { + return Generic_fromJSON(Server, value.data); +} + +Reviver.constructors.Server = Server; diff --git a/src/Server/ServerHelpers.js b/src/Server/ServerHelpers.js new file mode 100644 index 000000000..0ab1183f5 --- /dev/null +++ b/src/Server/ServerHelpers.js @@ -0,0 +1,126 @@ +import { BitNodeMultipliers } from "./BitNode/BitNodeMultipliers"; +import { CodingContract, + ContractTypes } from "./CodingContracts"; +import { CONSTANTS } from "./Constants"; +import { Script, + isScriptFilename } from "./Script"; +import { Player } from "./Player"; +import { Programs } from "./Programs/Programs"; +import { SpecialServerIps } from "./SpecialServerIps"; +import { TextFile } from "./TextFile"; +import { getRandomInt } from "../utils/helpers/getRandomInt"; +import { serverMetadata } from "./data/servers"; +import { Reviver, + Generic_toJSON, + Generic_fromJSON} from "../utils/JSONReviver"; +import {isValidIPAddress} from "../utils/helpers/isValidIPAddress"; + +// Returns the number of cycles needed to grow the specified server by the +// specified amount. 'growth' parameter is in decimal form, not percentage +export function numCycleForGrowth(server, growth) { + let ajdGrowthRate = 1 + (CONSTANTS.ServerBaseGrowthRate - 1) / server.hackDifficulty; + if(ajdGrowthRate > CONSTANTS.ServerMaxGrowthRate) { + ajdGrowthRate = CONSTANTS.ServerMaxGrowthRate; + } + + const serverGrowthPercentage = server.serverGrowth / 100; + + const cycles = Math.log(growth)/(Math.log(ajdGrowthRate)*Player.hacking_grow_mult*serverGrowthPercentage); + return cycles; +} + +//Applied server growth for a single server. Returns the percentage growth +export function processSingleServerGrowth(server, numCycles) { + //Server growth processed once every 450 game cycles + const numServerGrowthCycles = Math.max(Math.floor(numCycles / 450), 0); + + //Get adjusted growth rate, which accounts for server security + const growthRate = CONSTANTS.ServerBaseGrowthRate; + var adjGrowthRate = 1 + (growthRate - 1) / server.hackDifficulty; + if (adjGrowthRate > CONSTANTS.ServerMaxGrowthRate) {adjGrowthRate = CONSTANTS.ServerMaxGrowthRate;} + + //Calculate adjusted server growth rate based on parameters + const serverGrowthPercentage = server.serverGrowth / 100; + const numServerGrowthCyclesAdjusted = numServerGrowthCycles * serverGrowthPercentage * BitNodeMultipliers.ServerGrowthRate; + + //Apply serverGrowth for the calculated number of growth cycles + var serverGrowth = Math.pow(adjGrowthRate, numServerGrowthCyclesAdjusted * Player.hacking_grow_mult); + if (serverGrowth < 1) { + console.log("WARN: serverGrowth calculated to be less than 1"); + serverGrowth = 1; + } + + const oldMoneyAvailable = server.moneyAvailable; + server.moneyAvailable *= serverGrowth; + + // in case of data corruption + if (server.moneyMax && isNaN(server.moneyAvailable)) { + server.moneyAvailable = server.moneyMax; + } + + // cap at max + if (server.moneyMax && server.moneyAvailable > server.moneyMax) { + server.moneyAvailable = server.moneyMax; + } + + // if there was any growth at all, increase security + if (oldMoneyAvailable !== server.moneyAvailable) { + //Growing increases server security twice as much as hacking + let usedCycles = numCycleForGrowth(server, server.moneyAvailable / oldMoneyAvailable); + usedCycles = Math.max(0, usedCycles); + server.fortify(2 * CONSTANTS.ServerFortifyAmount * Math.ceil(usedCycles)); + } + return server.moneyAvailable / oldMoneyAvailable; +} + +export function prestigeHomeComputer(homeComp) { + const hasBitflume = homeComp.programs.includes(Programs.BitFlume.name); + + homeComp.programs.length = 0; //Remove programs + homeComp.runningScripts = []; + homeComp.serversOnNetwork = []; + homeComp.isConnectedTo = true; + homeComp.ramUsed = 0; + homeComp.programs.push(Programs.NukeProgram.name); + if (hasBitflume) { homeComp.programs.push(Programs.BitFlume.name); } + + //Update RAM usage on all scripts + homeComp.scripts.forEach(function(script) { + script.updateRamUsage(); + }); + + homeComp.messages.length = 0; //Remove .lit and .msg files + homeComp.messages.push("hackers-starting-handbook.lit"); +} + +function SizeOfAllServers() { + var size = 0, key; + for (key in AllServers) { + if (AllServers.hasOwnProperty(key)) size++; + } + return size; +} + +//Returns server object with corresponding hostname +// Relatively slow, would rather not use this a lot +export function GetServerByHostname(hostname) { + for (var ip in AllServers) { + if (AllServers.hasOwnProperty(ip)) { + if (AllServers[ip].hostname == hostname) { + return AllServers[ip]; + } + } + } + return null; +} + +//Get server by IP or hostname. Returns null if invalid +export function getServer(s) { + if (!isValidIPAddress(s)) { + return GetServerByHostname(s); + } + if(AllServers[s] !== undefined) { + return AllServers[s]; + } + return null; +} diff --git a/src/ServerPurchases.js b/src/Server/ServerPurchases.js similarity index 100% rename from src/ServerPurchases.js rename to src/Server/ServerPurchases.js diff --git a/src/SpecialServerIps.js b/src/Server/SpecialServerIps.js similarity index 100% rename from src/SpecialServerIps.js rename to src/Server/SpecialServerIps.js diff --git a/utils/IPAddress.js b/utils/IPAddress.js deleted file mode 100644 index 44fa97156..000000000 --- a/utils/IPAddress.js +++ /dev/null @@ -1,33 +0,0 @@ -import {AllServers} from "../src/Server"; -import {getRandomByte} from "./helpers/getRandomByte"; - -/* Functions to deal with manipulating IP addresses*/ - -//Generate a random IP address -//Will not return an IP address that already exists in the AllServers array -function createRandomIp() { - var ip = getRandomByte(99) + '.' + - getRandomByte(9) + '.' + - getRandomByte(9) + '.' + - getRandomByte(9); - - //If the Ip already exists, recurse to create a new one - if (ipExists(ip)) { - return createRandomIp(); - } - return ip; -} - -//Returns true if the IP already exists in one of the game's servers -function ipExists(ip) { - for (var property in AllServers) { - if (AllServers.hasOwnProperty(property)) { - if (property == ip) { - return true; - } - } - } - return false; -} - -export {createRandomIp, ipExists}; diff --git a/utils/IPAddress.ts b/utils/IPAddress.ts new file mode 100644 index 000000000..691f27505 --- /dev/null +++ b/utils/IPAddress.ts @@ -0,0 +1,25 @@ +import {AllServers} from "../src/Server"; +import {getRandomByte} from "./helpers/getRandomByte"; + +/* Functions to deal with manipulating IP addresses*/ + +//Generate a random IP address +//Will not return an IP address that already exists in the AllServers array +export function createRandomIp(): string { + const ip: string = getRandomByte(99) + '.' + + getRandomByte(9) + '.' + + getRandomByte(9) + '.' + + getRandomByte(9); + + // If the Ip already exists, recurse to create a new one + if (ipExists(ip)) { + return createRandomIp(); + } + + return ip; +} + +// Returns true if the IP already exists in one of the game's servers +export function ipExists(ip: string) { + return (AllServers[ip] != null); +} From 473f0f14475c40b829fc278dcd07bdd27a5da76f Mon Sep 17 00:00:00 2001 From: danielyxie Date: Mon, 4 Mar 2019 17:40:28 -0800 Subject: [PATCH 02/10] Refactored Server/Script/Files code to TypeScript --- src/ActiveScriptsUI.js | 4 +- src/Augmentation/AugmentationHelpers.js | 6 +- src/CodingContractGenerator.js | 4 +- src/DarkWeb/DarkWeb.js | 2 +- src/DevMenu.js | 2 +- src/{ => Fconf}/Fconf.js | 15 +- src/Fconf/FconfSettings.ts | 10 + src/Hacking.js | 2 +- src/Location.js | 9 +- src/Message/Message.ts | 2 +- src/Message/MessageHelpers.js | 2 +- src/NetscriptEvaluator.js | 8 +- src/NetscriptFunctions.js | 30 +- src/NetscriptWorker.js | 2 +- src/PersonObjects/IPlayer.ts | 1 + src/Player.js | 6 +- src/Prestige.js | 25 +- src/SaveObject.js | 15 +- src/Script/RamCalculations.d.ts | 1 + src/Script/RamCalculations.js | 409 +++++++++++++++ src/Script/RunningScript.ts | 34 +- src/Script/Script.ts | 10 +- src/Script/ScriptHelpers.js | 475 ++---------------- src/Script/ScriptHelpersTS.ts | 4 + src/Server/AllServers.ts | 49 +- src/Server/Server.ts | 72 +-- .../{ServerHelpers.js => ServerHelpers.ts} | 65 ++- src/Server/ServerPurchases.js | 20 +- src/Server/SpecialServerIps.js | 50 -- src/Server/SpecialServerIps.ts | 56 +++ src/{ => Server}/data/servers.ts | 2 + src/Terminal.js | 45 +- src/engine.js | 17 +- utils/IPAddress.ts | 4 +- 34 files changed, 763 insertions(+), 695 deletions(-) rename src/{ => Fconf}/Fconf.js (95%) create mode 100644 src/Fconf/FconfSettings.ts create mode 100644 src/Script/RamCalculations.d.ts create mode 100644 src/Script/RamCalculations.js create mode 100644 src/Script/ScriptHelpersTS.ts rename src/Server/{ServerHelpers.js => ServerHelpers.ts} (67%) delete mode 100644 src/Server/SpecialServerIps.js create mode 100644 src/Server/SpecialServerIps.ts rename src/{ => Server}/data/servers.ts (99%) diff --git a/src/ActiveScriptsUI.js b/src/ActiveScriptsUI.js index cb86c0b1a..1cdc66b26 100644 --- a/src/ActiveScriptsUI.js +++ b/src/ActiveScriptsUI.js @@ -1,7 +1,7 @@ import {workerScripts, killWorkerScript} from "./NetscriptWorker"; -import {Player} from "./Player"; -import {getServer} from "./Server"; +import { Player } from "./Player"; +import { getServer } from "./Server/ServerHelpers"; import {numeralWrapper} from "./ui/numeralFormat"; import {dialogBoxCreate} from "../utils/DialogBox"; import {createAccordionElement} from "../utils/uiHelpers/createAccordionElement"; diff --git a/src/Augmentation/AugmentationHelpers.js b/src/Augmentation/AugmentationHelpers.js index b1cd629df..af405e93e 100644 --- a/src/Augmentation/AugmentationHelpers.js +++ b/src/Augmentation/AugmentationHelpers.js @@ -12,9 +12,9 @@ import { addWorkerScript } from "../NetscriptWorker"; import { Player } from "../Player"; import { prestigeAugmentation } from "../Prestige"; import { saveObject } from "../SaveObject"; -import { Script, - RunningScript} from "../Script"; -import { Server } from "../Server"; +import { RunningScript } from "../Script/RunningScript"; +import { Script } from "../Script/Script"; +import { Server } from "../Server/Server"; import { OwnedAugmentationsOrderSetting } from "../Settings/SettingEnums"; import { Settings } from "../Settings/Settings"; diff --git a/src/CodingContractGenerator.js b/src/CodingContractGenerator.js index 8cf9fdda2..1538703bd 100644 --- a/src/CodingContractGenerator.js +++ b/src/CodingContractGenerator.js @@ -3,8 +3,8 @@ import { CodingContract, CodingContractTypes } from "./CodingContracts"; import { Factions } from "./Faction/Factions"; import { Player } from "./Player"; -import { GetServerByHostname, - AllServers } from "./Server"; +import { AllServers } from "./Server/Server"; +import { GetServerByHostname } from "./Server/ServerHelpers"; import { getRandomInt } from "../utils/helpers/getRandomInt"; diff --git a/src/DarkWeb/DarkWeb.js b/src/DarkWeb/DarkWeb.js index 394e31982..b8a5cdf89 100644 --- a/src/DarkWeb/DarkWeb.js +++ b/src/DarkWeb/DarkWeb.js @@ -1,7 +1,7 @@ import { DarkWebItems } from "./DarkWebItems"; import { Player } from "../Player"; -import { SpecialServerIps } from "../SpecialServerIps"; +import { SpecialServerIps } from "../Server/SpecialServerIps"; import { post } from "../ui/postToTerminal"; import { isValidIPAddress } from "../../utils/helpers/isValidIPAddress"; diff --git a/src/DevMenu.js b/src/DevMenu.js index 03b72f108..95591491c 100644 --- a/src/DevMenu.js +++ b/src/DevMenu.js @@ -8,7 +8,7 @@ import { Company } from "./Company/Company"; import { Programs } from "./Programs/Programs"; import { Factions } from "./Faction/Factions"; import { Player } from "./Player"; -import { AllServers } from "./Server"; +import { AllServers } from "./Server/AllServers"; import { hackWorldDaemon } from "./RedPill"; import { StockMarket, SymbolToStockMap } from "./StockMarket/StockMarket"; diff --git a/src/Fconf.js b/src/Fconf/Fconf.js similarity index 95% rename from src/Fconf.js rename to src/Fconf/Fconf.js index 6f125c26c..f40bcc8ce 100644 --- a/src/Fconf.js +++ b/src/Fconf/Fconf.js @@ -1,16 +1,7 @@ -import {parse, Node} from "../utils/acorn"; -import {dialogBoxCreate} from "../utils/DialogBox"; +import { FconfSettings } from "./FconfSettings"; -var FconfSettings = { - ENABLE_BASH_HOTKEYS: false, - ENABLE_TIMESTAMPS: false, - MAIN_MENU_STYLE: "default", - THEME_BACKGROUND_COLOR: "#000000", - THEME_FONT_COLOR: "#66ff33", - THEME_HIGHLIGHT_COLOR: "#ffffff", - THEME_PROMPT_COLOR: "#f92672", - WRAP_INPUT: false, -} +import { parse, Node } from "../../utils/acorn"; +import { dialogBoxCreate } from "../../utils/DialogBox"; var FconfComments = { ENABLE_BASH_HOTKEYS: "Improved Bash emulation mode. Setting this to 1 enables several\n" + diff --git a/src/Fconf/FconfSettings.ts b/src/Fconf/FconfSettings.ts new file mode 100644 index 000000000..f1c08bc73 --- /dev/null +++ b/src/Fconf/FconfSettings.ts @@ -0,0 +1,10 @@ +export const FconfSettings = { + ENABLE_BASH_HOTKEYS: false, + ENABLE_TIMESTAMPS: false, + MAIN_MENU_STYLE: "default", + THEME_BACKGROUND_COLOR: "#000000", + THEME_FONT_COLOR: "#66ff33", + THEME_HIGHLIGHT_COLOR: "#ffffff", + THEME_PROMPT_COLOR: "#f92672", + WRAP_INPUT: false, +} diff --git a/src/Hacking.js b/src/Hacking.js index 452d38120..893ca3779 100644 --- a/src/Hacking.js +++ b/src/Hacking.js @@ -1,6 +1,6 @@ import { BitNodeMultipliers } from "./BitNode/BitNodeMultipliers"; import { Player } from "./Player"; -import { Server } from "./Server"; +import { Server } from "./Server/Server"; /** * Returns the chance the player has to successfully hack a server diff --git a/src/Location.js b/src/Location.js index b00adc353..42caaa0ba 100644 --- a/src/Location.js +++ b/src/Location.js @@ -11,13 +11,16 @@ import {beginInfiltration} from "./Infiltration"; import {hasBladeburnerSF} from "./NetscriptFunctions"; import {Locations} from "./Locations"; import {Player} from "./Player"; -import {Server, AllServers, AddToAllServers} from "./Server"; +import { AllServers } from "./Server/AllServers"; +import { Server } from "./Server/Server"; +import { AddToAllServers } from "./Server/ServerHelpers"; import { getPurchaseServerCost, purchaseServer, - purchaseRamForHomeComputer} from "./ServerPurchases"; + purchaseRamForHomeComputer } from "./Server/ServerPurchases"; import {Settings} from "./Settings/Settings"; import { SourceFileFlags } from "./SourceFile/SourceFileFlags"; -import {SpecialServerNames, SpecialServerIps} from "./SpecialServerIps"; +import { SpecialServerNames, + SpecialServerIps } from "./Server/SpecialServerIps"; import {numeralWrapper} from "./ui/numeralFormat"; diff --git a/src/Message/Message.ts b/src/Message/Message.ts index 1c56ecd8b..90e24aa2c 100644 --- a/src/Message/Message.ts +++ b/src/Message/Message.ts @@ -12,7 +12,7 @@ export class Message { filename: string = ""; // The text contains in the Message - msg: string = "": + msg: string = ""; // Flag indicating whether this Message has been received by the player recvd: boolean = false; diff --git a/src/Message/MessageHelpers.js b/src/Message/MessageHelpers.js index 8fc056fb9..4845e6a30 100644 --- a/src/Message/MessageHelpers.js +++ b/src/Message/MessageHelpers.js @@ -6,7 +6,7 @@ import { Programs } from "../Programs/Programs"; import { inMission } from "../Missions"; import { Player } from "../Player"; import { redPillFlag } from "../RedPill"; -import { GetServerByHostname } from "../Server"; +import { GetServerByHostname } from "../Server/ServerHelpers"; import { Settings } from "../Settings/Settings"; import { dialogBoxCreate, dialogBoxOpened} from "../../utils/DialogBox"; diff --git a/src/NetscriptEvaluator.js b/src/NetscriptEvaluator.js index 8e144ac15..c301f2b76 100644 --- a/src/NetscriptEvaluator.js +++ b/src/NetscriptEvaluator.js @@ -3,10 +3,12 @@ import { CONSTANTS } from "./Constants"; import { Player } from "./Player"; import { Environment } from "./NetscriptEnvironment"; import { WorkerScript, addWorkerScript} from "./NetscriptWorker"; -import { Server, getServer} from "./Server"; +import { Server } from "./Server/Server"; +import { getServer } from "./Server/ServerHelpers"; import { Settings } from "./Settings/Settings"; -import { Script, findRunningScript, - RunningScript } from "./Script"; +import { RunningScript } from "./Script/RunningScript"; +import { Script } from "./Script/Script"; +import { findRunningScript } from "./Script/ScriptHelpers"; import { setTimeoutRef } from "./utils/SetTimeoutRef"; diff --git a/src/NetscriptFunctions.js b/src/NetscriptFunctions.js index 9b7e6a56a..ddc833cf6 100644 --- a/src/NetscriptFunctions.js +++ b/src/NetscriptFunctions.js @@ -31,20 +31,26 @@ import { joinFaction, import { getCostOfNextHacknetNode, purchaseHacknet } from "./HacknetNode"; import {Locations} from "./Locations"; -import {Message, Messages} from "./Message"; +import { Message } from "./Message/Message"; +import { Messages } from "./Message/MessageHelpers"; import {inMission} from "./Missions"; import {Player} from "./Player"; import { Programs } from "./Programs/Programs"; -import {Script, findRunningScript, RunningScript, - isScriptFilename} from "./Script"; -import {Server, getServer, AddToAllServers, - AllServers, processSingleServerGrowth, - GetServerByHostname, numCycleForGrowth} from "./Server"; +import { Script } from "./Script/Script"; +import { findRunningScript } from "./Script/ScriptHelpers"; +import { isScriptFilename } from "./Script/ScriptHelpersTS"; +import { AllServers, + AddToAllServers } from "./Server/AllServers"; +import { Server } from "./Server/Server"; +import { GetServerByHostname, + getServer, + numCycleForGrowth, + processSingleServerGrowth } from "./Server/ServerHelpers"; import { getPurchaseServerCost, getPurchaseServerLimit, - getPurchaseServerMaxRam } from "./ServerPurchases"; + getPurchaseServerMaxRam } from "./Server/ServerPurchases"; import {Settings} from "./Settings/Settings"; -import {SpecialServerIps} from "./SpecialServerIps"; +import {SpecialServerIps} from "./Server/SpecialServerIps"; import {Stock} from "./StockMarket/Stock"; import {StockMarket, StockSymbols, SymbolToStockMap, initStockMarket, initSymbolToStockMap, buyStock, @@ -305,9 +311,9 @@ function NetscriptFunctions(workerScript) { for (var i = 0; i < server.serversOnNetwork.length; i++) { var entry; if (hostnames) { - entry = server.getServerOnNetwork(i).hostname; + entry = getServerOnNetwork(server, i).hostname; } else { - entry = server.getServerOnNetwork(i).ip; + entry = getServerOnNetwork(server, i).ip; } if (entry == null) { continue; @@ -483,7 +489,7 @@ function NetscriptFunctions(workerScript) { if (workerScript.env.stopFlag) {return Promise.reject(workerScript);} const moneyBefore = server.moneyAvailable <= 0 ? 1 : server.moneyAvailable; server.moneyAvailable += (1 * threads); //It can be grown even if it has no money - var growthPercentage = processSingleServerGrowth(server, 450 * threads); + var growthPercentage = processSingleServerGrowth(server, 450 * threads, Player); const moneyAfter = server.moneyAvailable; workerScript.scriptRef.recordGrow(server.ip, threads); var expGain = calculateHackingExpGain(server) * threads; @@ -512,7 +518,7 @@ function NetscriptFunctions(workerScript) { throw makeRuntimeRejectMsg(workerScript, `Invalid growth argument passed into growthAnalyze: ${growth}. Must be numeric`); } - return numCycleForGrowth(server, Number(growth)); + return numCycleForGrowth(server, Number(growth), Player); }, weaken : function(ip) { if (workerScript.checkingRam) { diff --git a/src/NetscriptWorker.js b/src/NetscriptWorker.js index 936dc82f6..e698166f8 100644 --- a/src/NetscriptWorker.js +++ b/src/NetscriptWorker.js @@ -11,7 +11,7 @@ import {evaluate, isScriptErrorMessage, import {NetscriptFunctions} from "./NetscriptFunctions"; import {executeJSScript} from "./NetscriptJSEvaluator"; import {NetscriptPort} from "./NetscriptPort"; -import {AllServers} from "./Server"; +import { AllServers } from "./Server/AllServers"; import {Settings} from "./Settings/Settings"; import { setTimeoutRef } from "./utils/SetTimeoutRef"; diff --git a/src/PersonObjects/IPlayer.ts b/src/PersonObjects/IPlayer.ts index 948b72e22..c41247a2e 100644 --- a/src/PersonObjects/IPlayer.ts +++ b/src/PersonObjects/IPlayer.ts @@ -20,6 +20,7 @@ export interface IPlayer { city: string; companyName: string; corporation: any; + currentServer: string; factions: string[]; hacknetNodes: any[]; hasWseAccount: boolean; diff --git a/src/Player.js b/src/Player.js index 25c19af5f..a95234937 100644 --- a/src/Player.js +++ b/src/Player.js @@ -24,9 +24,11 @@ import {Gang, resetGangs} from "./Gang"; import {Locations} from "./Locations"; import {hasBn11SF, hasWallStreetSF,hasAISF} from "./NetscriptFunctions"; import { Sleeve } from "./PersonObjects/Sleeve/Sleeve"; -import {AllServers, Server, AddToAllServers} from "./Server"; +import { AllServers } from "./Server/AllServers"; +import { Server } from "./Server/Server"; +import { AddToAllServers } from "./Server/ServerHelpers"; import {Settings} from "./Settings/Settings"; -import {SpecialServerIps, SpecialServerNames} from "./SpecialServerIps"; +import {SpecialServerIps, SpecialServerNames} from "./Server/SpecialServerIps"; import {SourceFiles, applySourceFile} from "./SourceFile"; import { SourceFileFlags } from "./SourceFile/SourceFileFlags"; import Decimal from "decimal.js"; diff --git a/src/Prestige.js b/src/Prestige.js index 3e46c03f1..1b3ba168d 100755 --- a/src/Prestige.js +++ b/src/Prestige.js @@ -16,20 +16,25 @@ import { Factions, import { joinFaction } from "./Faction/FactionHelpers"; import {deleteGangDisplayContent} from "./Gang"; import {Locations} from "./Location"; -import {initMessages, Messages, Message} from "./Message"; +import { initMessages, + Messages, + Message } from "./Message/MessageHelpers"; import {initSingularitySFFlags, hasWallStreetSF}from "./NetscriptFunctions"; import {WorkerScript, workerScripts, prestigeWorkerScripts} from "./NetscriptWorker"; import {Player} from "./Player"; -import {AllServers, AddToAllServers, - initForeignServers, Server, - prestigeAllServers, - prestigeHomeComputer} from "./Server"; +import { AllServers } from "./Server/AllServers"; +import { Server } from "./Server/Server" +import { AddToAllServers, + initForeignServers, + prestigeAllServers, + prestigeHomeComputer } from "./Server/ServerHelpers"; import { updateSourceFileFlags } from "./SourceFile/SourceFileFlags"; -import {SpecialServerIps, SpecialServerIpsMap, - prestigeSpecialServerIps, - SpecialServerNames} from "./SpecialServerIps"; +import { SpecialServerIps, + SpecialServerIpsMap, + prestigeSpecialServerIps, + SpecialServerNames} from "./Server/SpecialServerIps"; import {initStockMarket, initSymbolToStockMap, stockMarketContentCreated, setStockMarketContentCreated} from "./StockMarket/StockMarket"; @@ -89,7 +94,7 @@ function prestigeAugmentation() { } //Re-create foreign servers - initForeignServers(); + initForeignServers(Player.getHomeComputer()); //Darkweb is purchase-able document.getElementById("location-purchase-tor").setAttribute("class", "a-link-button"); @@ -194,7 +199,7 @@ function prestigeSourceFile() { prestigeHomeComputer(homeComp); //Re-create foreign servers - initForeignServers(); + initForeignServers(Player.getHomeComputer()); var srcFile1Owned = false; for (var i = 0; i < Player.sourceFiles.length; ++i) { diff --git a/src/SaveObject.js b/src/SaveObject.js index 5667b1efb..e59c0f777 100755 --- a/src/SaveObject.js +++ b/src/SaveObject.js @@ -7,15 +7,18 @@ import {Engine} from "./engine"; import { Factions, loadFactions } from "./Faction/Factions"; import { processPassiveFactionRepGain } from "./Faction/FactionHelpers"; -import {FconfSettings, loadFconf} from "./Fconf"; +import { loadFconf } from "./Fconf/Fconf"; +import { FconfSettings } from "./Fconf/FconfSettings"; import {loadAllGangs, AllGangs} from "./Gang"; import {processAllHacknetNodeEarnings} from "./HacknetNode"; -import {loadMessages, initMessages, Messages} from "./Message"; +import { loadMessages, initMessages, Messages } from "./Message/MessageHelpers"; import {Player, loadPlayer} from "./Player"; -import {loadAllRunningScripts} from "./Script"; -import {AllServers, loadAllServers} from "./Server"; -import {Settings} from "./Settings/Settings"; -import {loadSpecialServerIps, SpecialServerIps} from "./SpecialServerIps"; +import { loadAllRunningScripts } from "./Script/ScriptHelpers"; +import { AllServers } from "./Server/AllServers"; +import { loadAllServers } from "./Server/ServerHelpers"; +import { Settings } from "./Settings/Settings"; +import { loadSpecialServerIps, + SpecialServerIps } from "./Server/SpecialServerIps"; import {loadStockMarket, StockMarket} from "./StockMarket/StockMarket"; import { setTimeoutRef } from "./utils/SetTimeoutRef"; diff --git a/src/Script/RamCalculations.d.ts b/src/Script/RamCalculations.d.ts new file mode 100644 index 000000000..0a886c842 --- /dev/null +++ b/src/Script/RamCalculations.d.ts @@ -0,0 +1 @@ +export declare function calculateRamUsage(codeCopy: string): number; diff --git a/src/Script/RamCalculations.js b/src/Script/RamCalculations.js new file mode 100644 index 000000000..643845a8e --- /dev/null +++ b/src/Script/RamCalculations.js @@ -0,0 +1,409 @@ +// Calculate a script's RAM usage +const walk = require("acorn/dist/walk"); // Importing this doesn't work for some reason. + +import { CONSTANTS } from "../Constants"; +import {evaluateImport} from "../NetscriptEvaluator"; +import { WorkerScript } from "../NetscriptWorker"; +import { Player } from "../Player"; +import {parse, Node} from "../../utils/acorn"; + +// These special strings are used to reference the presence of a given logical +// construct within a user script. +const specialReferenceIF = "__SPECIAL_referenceIf"; +const specialReferenceFOR = "__SPECIAL_referenceFor"; +const specialReferenceWHILE = "__SPECIAL_referenceWhile"; + +// The global scope of a script is registered under this key during parsing. +const memCheckGlobalKey = ".__GLOBAL__"; + +// Calcluates the amount of RAM a script uses. Uses parsing and AST walking only, +// rather than NetscriptEvaluator. This is useful because NetscriptJS code does +// not work under NetscriptEvaluator. +async function parseOnlyRamCalculate(server, code, workerScript) { + try { + // Maps dependent identifiers to their dependencies. + // + // The initial identifier is __SPECIAL_INITIAL_MODULE__.__GLOBAL__. + // It depends on all the functions declared in the module, all the global scopes + // of its imports, and any identifiers referenced in this global scope. Each + // function depends on all the identifiers referenced internally. + // We walk the dependency graph to calculate RAM usage, given that some identifiers + // reference Netscript functions which have a RAM cost. + let dependencyMap = {}; + + // Scripts we've parsed. + const completedParses = new Set(); + + // Scripts we've discovered that need to be parsed. + const parseQueue = []; + + // Parses a chunk of code with a given module name, and updates parseQueue and dependencyMap. + function parseCode(code, moduleName) { + const result = parseOnlyCalculateDeps(code, moduleName); + completedParses.add(moduleName); + + // Add any additional modules to the parse queue; + for (let i = 0; i < result.additionalModules.length; ++i) { + if (!completedParses.has(result.additionalModules[i])) { + parseQueue.push(result.additionalModules[i]); + } + } + + // Splice all the references in. + //Spread syntax not supported in edge, use Object.assign instead + //dependencyMap = {...dependencyMap, ...result.dependencyMap}; + dependencyMap = Object.assign(dependencyMap, result.dependencyMap); + } + + const initialModule = "__SPECIAL_INITIAL_MODULE__"; + parseCode(code, initialModule); + + while (parseQueue.length > 0) { + // Get the code from the server. + const nextModule = parseQueue.shift(); + + let code; + if (nextModule.startsWith("https://") || nextModule.startsWith("http://")) { + try { + const module = await eval('import(nextModule)'); + code = ""; + for (const prop in module) { + if (typeof module[prop] === 'function') { + code += module[prop].toString() + ";\n"; + } + } + } catch(e) { + console.error(`Error dynamically importing module from ${nextModule} for RAM calculations: ${e}`); + return -1; + } + } else { + const script = server.getScript(nextModule.startsWith("./") ? nextModule.slice(2) : nextModule); + if (!script) { + console.warn("Invalid script"); + return -1; // No such script on the server. + } + code = script.code; + } + + parseCode(code, nextModule); + } + + // Finally, walk the reference map and generate a ram cost. The initial set of keys to scan + // are those that start with __SPECIAL_INITIAL_MODULE__. + let ram = CONSTANTS.ScriptBaseRamCost; + const unresolvedRefs = Object.keys(dependencyMap).filter(s => s.startsWith(initialModule)); + const resolvedRefs = new Set(); + while (unresolvedRefs.length > 0) { + const ref = unresolvedRefs.shift(); + + // Check if this is one of the special keys, and add the appropriate ram cost if so. + if (ref === "hacknet" && !resolvedRefs.has("hacknet")) { + ram += CONSTANTS.ScriptHacknetNodesRamCost; + } + if (ref === "document" && !resolvedRefs.has("document")) { + ram += CONSTANTS.ScriptDomRamCost; + } + if (ref === "window" && !resolvedRefs.has("window")) { + ram += CONSTANTS.ScriptDomRamCost; + } + + resolvedRefs.add(ref); + + if (ref.endsWith(".*")) { + // A prefix reference. We need to find all matching identifiers. + const prefix = ref.slice(0, ref.length - 2); + for (let ident of Object.keys(dependencyMap).filter(k => k.startsWith(prefix))) { + for (let dep of dependencyMap[ident] || []) { + if (!resolvedRefs.has(dep)) unresolvedRefs.push(dep); + } + } + } else { + // An exact reference. Add all dependencies of this ref. + for (let dep of dependencyMap[ref] || []) { + if (!resolvedRefs.has(dep)) unresolvedRefs.push(dep); + } + } + + // Check if this ident is a function in the workerscript env. If it is, then we need to + // get its RAM cost. We do this by calling it, which works because the running script + // is in checkingRam mode. + // + // TODO it would be simpler to just reference a dictionary. + try { + function applyFuncRam(func) { + if (typeof func === "function") { + try { + let res; + if (func.constructor.name === "AsyncFunction") { + res = 0; // Async functions will always be 0 RAM + } else { + res = func.apply(null, []); + } + if (typeof res === "number") { + return res; + } + return 0; + } catch(e) { + console.log("ERROR applying function: " + e); + return 0; + } + } else { + return 0; + } + } + + //Special logic for namespaces (Bladeburner, CodingCOntract) + var func; + if (ref in workerScript.env.vars.bladeburner) { + func = workerScript.env.vars.bladeburner[ref]; + } else if (ref in workerScript.env.vars.codingcontract) { + func = workerScript.env.vars.codingcontract[ref]; + } else if (ref in workerScript.env.vars.gang) { + func = workerScript.env.vars.gang[ref]; + } else { + func = workerScript.env.get(ref); + } + ram += applyFuncRam(func); + } catch (error) {continue;} + } + return ram; + + } catch (error) { + // console.info("parse or eval error: ", error); + // This is not unexpected. The user may be editing a script, and it may be in + // a transitory invalid state. + return -1; + } +} + +// Parses one script and calculates its ram usage, for the global scope and each function. +// Returns a cost map and a dependencyMap for the module. Returns a reference map to be joined +// onto the main reference map, and a list of modules that need to be parsed. +function parseOnlyCalculateDeps(code, currentModule) { + const ast = parse(code, {sourceType:"module", ecmaVersion: 8}); + + // Everything from the global scope goes in ".". Everything else goes in ".function", where only + // the outermost layer of functions counts. + const globalKey = currentModule + memCheckGlobalKey; + const dependencyMap = {}; + dependencyMap[globalKey] = new Set(); + + // If we reference this internal name, we're really referencing that external name. + // Filled when we import names from other modules. + let internalToExternal = {}; + + var additionalModules = []; + + // References get added pessimistically. They are added for thisModule.name, name, and for + // any aliases. + function addRef(key, name) { + const s = dependencyMap[key] || (dependencyMap[key] = new Set()); + if (name in internalToExternal) { + s.add(internalToExternal[name]); + } + s.add(currentModule + "." + name); + s.add(name); // For builtins like hack. + } + + //A list of identifiers that resolve to "native Javascript code" + const objectPrototypeProperties = Object.getOwnPropertyNames(Object.prototype); + + // If we discover a dependency identifier, state.key is the dependent identifier. + // walkDeeper is for doing recursive walks of expressions in composites that we handle. + function commonVisitors() { + return { + Identifier: (node, st, walkDeeper) => { + if (objectPrototypeProperties.includes(node.name)) {return;} + addRef(st.key, node.name); + }, + WhileStatement: (node, st, walkDeeper) => { + addRef(st.key, specialReferenceWHILE); + node.test && walkDeeper(node.test, st); + node.body && walkDeeper(node.body, st); + }, + DoWhileStatement: (node, st, walkDeeper) => { + addRef(st.key, specialReferenceWHILE); + node.test && walkDeeper(node.test, st); + node.body && walkDeeper(node.body, st); + }, + ForStatement: (node, st, walkDeeper) => { + addRef(st.key, specialReferenceFOR); + node.init && walkDeeper(node.init, st); + node.test && walkDeeper(node.test, st); + node.update && walkDeeper(node.update, st); + node.body && walkDeeper(node.body, st); + }, + IfStatement: (node, st, walkDeeper) => { + addRef(st.key, specialReferenceIF); + node.test && walkDeeper(node.test, st); + node.consequent && walkDeeper(node.consequent, st); + node.alternate && walkDeeper(node.alternate, st); + }, + MemberExpression: (node, st, walkDeeper) => { + node.object && walkDeeper(node.object, st); + node.property && walkDeeper(node.property, st); + }, + } + } + + //Spread syntax not supported in Edge yet, use Object.assign + /* + walk.recursive(ast, {key: globalKey}, { + ImportDeclaration: (node, st, walkDeeper) => { + const importModuleName = node.source.value; + additionalModules.push(importModuleName); + + // This module's global scope refers to that module's global scope, no matter how we + // import it. + dependencyMap[st.key].add(importModuleName + memCheckGlobalKey); + + for (let i = 0; i < node.specifiers.length; ++i) { + const spec = node.specifiers[i]; + if (spec.imported !== undefined && spec.local !== undefined) { + // We depend on specific things. + internalToExternal[spec.local.name] = importModuleName + "." + spec.imported.name; + } else { + // We depend on everything. + dependencyMap[st.key].add(importModuleName + ".*"); + } + } + }, + FunctionDeclaration: (node, st, walkDeeper) => { + // Don't use walkDeeper, because we are changing the visitor set. + const key = currentModule + "." + node.id.name; + walk.recursive(node, {key: key}, commonVisitors()); + }, + ...commonVisitors() + }); + */ + walk.recursive(ast, {key: globalKey}, Object.assign({ + ImportDeclaration: (node, st, walkDeeper) => { + const importModuleName = node.source.value; + additionalModules.push(importModuleName); + + // This module's global scope refers to that module's global scope, no matter how we + // import it. + dependencyMap[st.key].add(importModuleName + memCheckGlobalKey); + + for (let i = 0; i < node.specifiers.length; ++i) { + const spec = node.specifiers[i]; + if (spec.imported !== undefined && spec.local !== undefined) { + // We depend on specific things. + internalToExternal[spec.local.name] = importModuleName + "." + spec.imported.name; + } else { + // We depend on everything. + dependencyMap[st.key].add(importModuleName + ".*"); + } + } + }, + FunctionDeclaration: (node, st, walkDeeper) => { + // Don't use walkDeeper, because we are changing the visitor set. + const key = currentModule + "." + node.id.name; + walk.recursive(node, {key: key}, commonVisitors()); + }, + }, commonVisitors())); + + return {dependencyMap: dependencyMap, additionalModules: additionalModules}; +} + +export async function calculateRamUsage(codeCopy) { + //Create a temporary/mock WorkerScript and an AST from the code + var currServ = Player.getCurrentServer(); + var workerScript = new WorkerScript({ + filename:"foo", + scriptRef: {code:""}, + args:[], + getCode: function() { return ""; } + }); + workerScript.checkingRam = true; //Netscript functions will return RAM usage + workerScript.serverIp = currServ.ip; + + try { + return await parseOnlyRamCalculate(currServ, codeCopy, workerScript); + } catch (e) { + console.log("Failed to parse ram using new method. Falling back.", e); + } + + // Try the old way. + + try { + var ast = parse(codeCopy, {sourceType:"module"}); + } catch(e) { + return -1; + } + + //Search through AST, scanning for any 'Identifier' nodes for functions, or While/For/If nodes + var queue = [], ramUsage = CONSTANTS.ScriptBaseRamCost; + var whileUsed = false, forUsed = false, ifUsed = false; + queue.push(ast); + while (queue.length != 0) { + var exp = queue.shift(); + switch (exp.type) { + case "ImportDeclaration": + //Gets an array of all imported functions as AST expressions + //and pushes them on the queue. + var res = evaluateImport(exp, workerScript, true); + for (var i = 0; i < res.length; ++i) { + queue.push(res[i]); + } + break; + case "BlockStatement": + case "Program": + for (var i = 0; i < exp.body.length; ++i) { + if (exp.body[i] instanceof Node) { + queue.push(exp.body[i]); + } + } + break; + case "WhileStatement": + if (!whileUsed) { + ramUsage += CONSTANTS.ScriptWhileRamCost; + whileUsed = true; + } + break; + case "ForStatement": + if (!forUsed) { + ramUsage += CONSTANTS.ScriptForRamCost; + forUsed = true; + } + break; + case "IfStatement": + if (!ifUsed) { + ramUsage += CONSTANTS.ScriptIfRamCost; + ifUsed = true; + } + break; + case "Identifier": + if (exp.name in workerScript.env.vars) { + var func = workerScript.env.get(exp.name); + if (typeof func === "function") { + try { + var res = func.apply(null, []); + if (typeof res === "number") { + ramUsage += res; + } + } catch(e) { + console.log("ERROR applying function: " + e); + } + } + } + break; + default: + break; + } + + for (var prop in exp) { + if (exp.hasOwnProperty(prop)) { + if (exp[prop] instanceof Node) { + queue.push(exp[prop]); + } + } + } + } + + //Special case: hacknetnodes array + if (codeCopy.includes("hacknet")) { + ramUsage += CONSTANTS.ScriptHacknetNodesRamCost; + } + return ramUsage; +} diff --git a/src/Script/RunningScript.ts b/src/Script/RunningScript.ts index c20e3e5f5..9da52d47b 100644 --- a/src/Script/RunningScript.ts +++ b/src/Script/RunningScript.ts @@ -1,10 +1,16 @@ // Class representing a Script instance that is actively running. // A Script can have multiple active instances -import { Script } from "./Script"; -import { IMap } from "../types"; +import { Script } from "./Script"; +import { FconfSettings } from "../Fconf/FconfSettings"; +import { AllServers } from "../Server/AllServers"; +import { Settings } from "../Settings/Settings"; +import { IMap } from "../types"; +import { post } from "../ui/postToTerminal"; + import { Generic_fromJSON, Generic_toJSON, - Reviver } from "../../utils/JSONReviver"; + Reviver } from "../../utils/JSONReviver"; +import { getTimestamp } from "../../utils/helpers/getTimestamp"; export class RunningScript { // Initializes a RunningScript Object from a JSON save state @@ -67,7 +73,7 @@ export class RunningScript { this.ramUsage = script.ramUsage; } - RunningScript.prototype.getCode = function() { + getCode(): string { const server = AllServers[this.server]; if (server == null) { return ""; } for (let i = 0; i < server.scripts.length; ++i) { @@ -79,7 +85,7 @@ export class RunningScript { return ""; } - RunningScript.prototype.getRamUsage = function() { + getRamUsage(): number { if (this.ramUsage != null && this.ramUsage > 0) { return this.ramUsage; } // Use cached value const server = AllServers[this.server]; @@ -96,7 +102,7 @@ export class RunningScript { return 0; } - RunningScript.prototype.log = function(txt) { + log(txt: string): void { if (this.logs.length > Settings.MaxLogCapacity) { //Delete first element and add new log entry to the end. //TODO Eventually it might be better to replace this with circular array @@ -111,18 +117,18 @@ export class RunningScript { this.logUpd = true; } - RunningScript.prototype.displayLog = function() { + displayLog(): void { for (var i = 0; i < this.logs.length; ++i) { post(this.logs[i]); } } - RunningScript.prototype.clearLog = function() { + clearLog(): void { this.logs.length = 0; } - //Update the moneyStolen and numTimesHack maps when hacking - RunningScript.prototype.recordHack = function(serverIp, moneyGained, n=1) { + // Update the moneyStolen and numTimesHack maps when hacking + recordHack(serverIp: string, moneyGained: number, n: number=1) { if (this.dataMap[serverIp] == null || this.dataMap[serverIp].constructor !== Array) { this.dataMap[serverIp] = [0, 0, 0, 0]; } @@ -130,16 +136,16 @@ export class RunningScript { this.dataMap[serverIp][1] += n; } - //Update the grow map when calling grow() - RunningScript.prototype.recordGrow = function(serverIp, n=1) { + // Update the grow map when calling grow() + recordGrow(serverIp: string, n: number=1) { if (this.dataMap[serverIp] == null || this.dataMap[serverIp].constructor !== Array) { this.dataMap[serverIp] = [0, 0, 0, 0]; } this.dataMap[serverIp][2] += n; } - //Update the weaken map when calling weaken() { - RunningScript.prototype.recordWeaken = function(serverIp, n=1) { + // Update the weaken map when calling weaken() { + recordWeaken(serverIp: string, n: number=1) { if (this.dataMap[serverIp] == null || this.dataMap[serverIp].constructor !== Array) { this.dataMap[serverIp] = [0, 0, 0, 0]; } diff --git a/src/Script/Script.ts b/src/Script/Script.ts index fa64ed0d8..92f854f40 100644 --- a/src/Script/Script.ts +++ b/src/Script/Script.ts @@ -1,8 +1,11 @@ // Class representing a script file // This does NOT represent a script that is actively running and // being evaluated. See RunningScript for that +import { calculateRamUsage } from "./RamCalculations"; +import { IPlayer } from "../PersonObjects/IPlayer"; import { Page, routing } from "../ui/navigationTracking"; + import { setTimeoutRef } from "../utils/SetTimeoutRef"; import { Generic_fromJSON, Generic_toJSON, @@ -61,10 +64,9 @@ export class Script { } // Save a script FROM THE SCRIPT EDITOR - saveScript(): void { + saveScript(code: string, p: IPlayer): void { if (routing.isOn(Page.ScriptEditor)) { //Update code and filename - const code = getCurrentEditor().getCode(); this.code = code.replace(/^\s+|\s+$/g, ''); const filenameElem: HTMLInputElement | null = document.getElementById("script-editor-filename") as HTMLInputElement; @@ -75,7 +77,7 @@ export class Script { this.filename = filenameElem!.value; // Server - this.server = Player.currentServer; + this.server = p.currentServer; //Calculate/update ram usage, execution time, etc. this.updateRamUsage(); @@ -85,7 +87,7 @@ export class Script { } // Updates the script's RAM usage based on its code - async updateRamUsage(): void { + async updateRamUsage() { // TODO Commented this out because I think its unnecessary // DOuble check/Test // var codeCopy = this.code.repeat(1); diff --git a/src/Script/ScriptHelpers.js b/src/Script/ScriptHelpers.js index 45a6ebb99..821dce013 100644 --- a/src/Script/ScriptHelpers.js +++ b/src/Script/ScriptHelpers.js @@ -1,40 +1,33 @@ -// Importing this doesn't work for some reason. -const walk = require("acorn/dist/walk"); +import { calculateRamUsage } from "./RamCalculations"; +import { isScriptFilename } from "./ScriptHelpersTS"; -import {CONSTANTS} from "./Constants"; -import {Engine} from "./engine"; -import {FconfSettings, parseFconfSettings} from "./Fconf"; +import {CONSTANTS} from "../Constants"; +import {Engine} from "../engine"; +import { parseFconfSettings } from "../Fconf/Fconf"; +import { FconfSettings } from "../Fconf/FconfSettings"; import {iTutorialSteps, iTutorialNextStep, - ITutorial} from "./InteractiveTutorial"; -import {evaluateImport} from "./NetscriptEvaluator"; -import {NetscriptFunctions} from "./NetscriptFunctions"; -import {addWorkerScript, WorkerScript} from "./NetscriptWorker"; -import {Player} from "./Player"; -import { AceEditor } from "./ScriptEditor/Ace"; -import { CodeMirrorEditor } from "./ScriptEditor/CodeMirror"; -import {AllServers, processSingleServerGrowth} from "./Server"; -import { Settings } from "./Settings/Settings"; -import { EditorSetting } from "./Settings/SettingEnums"; -import {post} from "./ui/postToTerminal"; -import {TextFile} from "./TextFile"; -import {parse, Node} from "../utils/acorn"; -import {Page, routing} from "./ui/navigationTracking"; -import {numeralWrapper} from "./ui/numeralFormat"; -import { setTimeoutRef } from "./utils/SetTimeoutRef"; -import {dialogBoxCreate} from "../utils/DialogBox"; -import {Reviver, Generic_toJSON, - Generic_fromJSON} from "../utils/JSONReviver"; -import {compareArrays} from "../utils/helpers/compareArrays"; -import {createElement} from "../utils/uiHelpers/createElement"; -import {getTimestamp} from "../utils/helpers/getTimestamp"; -import {roundToTwo} from "../utils/helpers/roundToTwo"; + ITutorial} from "../InteractiveTutorial"; +import { addWorkerScript } from "../NetscriptWorker"; +import { Player } from "../Player"; +import { AceEditor } from "../ScriptEditor/Ace"; +import { CodeMirrorEditor } from "../ScriptEditor/CodeMirror"; +import { AllServers } from "../Server/AllServers"; +import { processSingleServerGrowth } from "../Server/ServerHelpers"; +import { Settings } from "../Settings/Settings"; +import { EditorSetting } from "../Settings/SettingEnums"; +import {TextFile} from "../TextFile"; -function isScriptFilename(f) { - return f.endsWith(".js") || f.endsWith(".script") || f.endsWith(".ns"); -} +import {Page, routing} from "../ui/navigationTracking"; +import {numeralWrapper} from "../ui/numeralFormat"; + +import {dialogBoxCreate} from "../../utils/DialogBox"; +import {Reviver, Generic_toJSON, + Generic_fromJSON} from "../../utils/JSONReviver"; +import {compareArrays} from "../../utils/helpers/compareArrays"; +import {createElement} from "../../utils/uiHelpers/createElement"; var scriptEditorRamCheck = null, scriptEditorRamText = null; -function scriptEditorInit() { +export function scriptEditorInit() { // Wrapper container that holds all the buttons below the script editor const wrapper = document.getElementById("script-editor-buttons-wrapper"); if (wrapper == null) { @@ -233,7 +226,7 @@ function saveAndCloseScriptEditor() { let s = Player.getCurrentServer(); for (var i = 0; i < s.scripts.length; i++) { if (filename == s.scripts[i].filename) { - s.scripts[i].saveScript(); + s.scripts[i].saveScript(getCurrentEditor().getCode(), Player); Engine.loadTerminalContent(); return iTutorialNextStep(); } @@ -241,7 +234,7 @@ function saveAndCloseScriptEditor() { //If the current script does NOT exist, create a new one let script = new Script(); - script.saveScript(); + script.saveScript(getCurrentEditor().getCode(), Player); s.scripts.push(script); return iTutorialNextStep(); @@ -269,7 +262,7 @@ function saveAndCloseScriptEditor() { //If the current script already exists on the server, overwrite it for (var i = 0; i < s.scripts.length; i++) { if (filename == s.scripts[i].filename) { - s.scripts[i].saveScript(); + s.scripts[i].saveScript(getCurrentEditor().getCode(), Player); Engine.loadTerminalContent(); return; } @@ -277,7 +270,7 @@ function saveAndCloseScriptEditor() { //If the current script does NOT exist, create a new one var script = new Script(); - script.saveScript(); + script.saveScript(getCurrentEditor().getCode(), Player); s.scripts.push(script); } else if (filename.endsWith(".txt")) { for (var i = 0; i < s.textFiles.length; ++i) { @@ -308,410 +301,9 @@ function checkValidFilename(filename) { return false; } -// These special strings are used to reference the presence of a given logical -// construct within a user script. -const specialReferenceIF = "__SPECIAL_referenceIf"; -const specialReferenceFOR = "__SPECIAL_referenceFor"; -const specialReferenceWHILE = "__SPECIAL_referenceWhile"; - -// The global scope of a script is registered under this key during parsing. -const memCheckGlobalKey = ".__GLOBAL__"; - -// Calcluates the amount of RAM a script uses. Uses parsing and AST walking only, -// rather than NetscriptEvaluator. This is useful because NetscriptJS code does -// not work under NetscriptEvaluator. -async function parseOnlyRamCalculate(server, code, workerScript) { - try { - // Maps dependent identifiers to their dependencies. - // - // The initial identifier is __SPECIAL_INITIAL_MODULE__.__GLOBAL__. - // It depends on all the functions declared in the module, all the global scopes - // of its imports, and any identifiers referenced in this global scope. Each - // function depends on all the identifiers referenced internally. - // We walk the dependency graph to calculate RAM usage, given that some identifiers - // reference Netscript functions which have a RAM cost. - let dependencyMap = {}; - - // Scripts we've parsed. - const completedParses = new Set(); - - // Scripts we've discovered that need to be parsed. - const parseQueue = []; - - // Parses a chunk of code with a given module name, and updates parseQueue and dependencyMap. - function parseCode(code, moduleName) { - const result = parseOnlyCalculateDeps(code, moduleName); - completedParses.add(moduleName); - - // Add any additional modules to the parse queue; - for (let i = 0; i < result.additionalModules.length; ++i) { - if (!completedParses.has(result.additionalModules[i])) { - parseQueue.push(result.additionalModules[i]); - } - } - - // Splice all the references in. - //Spread syntax not supported in edge, use Object.assign instead - //dependencyMap = {...dependencyMap, ...result.dependencyMap}; - dependencyMap = Object.assign(dependencyMap, result.dependencyMap); - } - - const initialModule = "__SPECIAL_INITIAL_MODULE__"; - parseCode(code, initialModule); - - while (parseQueue.length > 0) { - // Get the code from the server. - const nextModule = parseQueue.shift(); - - let code; - if (nextModule.startsWith("https://") || nextModule.startsWith("http://")) { - try { - const module = await eval('import(nextModule)'); - code = ""; - for (const prop in module) { - if (typeof module[prop] === 'function') { - code += module[prop].toString() + ";\n"; - } - } - } catch(e) { - console.error(`Error dynamically importing module from ${nextModule} for RAM calculations: ${e}`); - return -1; - } - } else { - const script = server.getScript(nextModule.startsWith("./") ? nextModule.slice(2) : nextModule); - if (!script) { - console.warn("Invalid script"); - return -1; // No such script on the server. - } - code = script.code; - } - - parseCode(code, nextModule); - } - - // Finally, walk the reference map and generate a ram cost. The initial set of keys to scan - // are those that start with __SPECIAL_INITIAL_MODULE__. - let ram = CONSTANTS.ScriptBaseRamCost; - const unresolvedRefs = Object.keys(dependencyMap).filter(s => s.startsWith(initialModule)); - const resolvedRefs = new Set(); - while (unresolvedRefs.length > 0) { - const ref = unresolvedRefs.shift(); - - // Check if this is one of the special keys, and add the appropriate ram cost if so. - if (ref === "hacknet" && !resolvedRefs.has("hacknet")) { - ram += CONSTANTS.ScriptHacknetNodesRamCost; - } - if (ref === "document" && !resolvedRefs.has("document")) { - ram += CONSTANTS.ScriptDomRamCost; - } - if (ref === "window" && !resolvedRefs.has("window")) { - ram += CONSTANTS.ScriptDomRamCost; - } - - resolvedRefs.add(ref); - - if (ref.endsWith(".*")) { - // A prefix reference. We need to find all matching identifiers. - const prefix = ref.slice(0, ref.length - 2); - for (let ident of Object.keys(dependencyMap).filter(k => k.startsWith(prefix))) { - for (let dep of dependencyMap[ident] || []) { - if (!resolvedRefs.has(dep)) unresolvedRefs.push(dep); - } - } - } else { - // An exact reference. Add all dependencies of this ref. - for (let dep of dependencyMap[ref] || []) { - if (!resolvedRefs.has(dep)) unresolvedRefs.push(dep); - } - } - - // Check if this ident is a function in the workerscript env. If it is, then we need to - // get its RAM cost. We do this by calling it, which works because the running script - // is in checkingRam mode. - // - // TODO it would be simpler to just reference a dictionary. - try { - function applyFuncRam(func) { - if (typeof func === "function") { - try { - let res; - if (func.constructor.name === "AsyncFunction") { - res = 0; // Async functions will always be 0 RAM - } else { - res = func.apply(null, []); - } - if (typeof res === "number") { - return res; - } - return 0; - } catch(e) { - console.log("ERROR applying function: " + e); - return 0; - } - } else { - return 0; - } - } - - //Special logic for namespaces (Bladeburner, CodingCOntract) - var func; - if (ref in workerScript.env.vars.bladeburner) { - func = workerScript.env.vars.bladeburner[ref]; - } else if (ref in workerScript.env.vars.codingcontract) { - func = workerScript.env.vars.codingcontract[ref]; - } else if (ref in workerScript.env.vars.gang) { - func = workerScript.env.vars.gang[ref]; - } else { - func = workerScript.env.get(ref); - } - ram += applyFuncRam(func); - } catch (error) {continue;} - } - return ram; - - } catch (error) { - // console.info("parse or eval error: ", error); - // This is not unexpected. The user may be editing a script, and it may be in - // a transitory invalid state. - return -1; - } -} - -// Parses one script and calculates its ram usage, for the global scope and each function. -// Returns a cost map and a dependencyMap for the module. Returns a reference map to be joined -// onto the main reference map, and a list of modules that need to be parsed. -function parseOnlyCalculateDeps(code, currentModule) { - const ast = parse(code, {sourceType:"module", ecmaVersion: 8}); - - // Everything from the global scope goes in ".". Everything else goes in ".function", where only - // the outermost layer of functions counts. - const globalKey = currentModule + memCheckGlobalKey; - const dependencyMap = {}; - dependencyMap[globalKey] = new Set(); - - // If we reference this internal name, we're really referencing that external name. - // Filled when we import names from other modules. - let internalToExternal = {}; - - var additionalModules = []; - - // References get added pessimistically. They are added for thisModule.name, name, and for - // any aliases. - function addRef(key, name) { - const s = dependencyMap[key] || (dependencyMap[key] = new Set()); - if (name in internalToExternal) { - s.add(internalToExternal[name]); - } - s.add(currentModule + "." + name); - s.add(name); // For builtins like hack. - } - - //A list of identifiers that resolve to "native Javascript code" - const objectPrototypeProperties = Object.getOwnPropertyNames(Object.prototype); - - // If we discover a dependency identifier, state.key is the dependent identifier. - // walkDeeper is for doing recursive walks of expressions in composites that we handle. - function commonVisitors() { - return { - Identifier: (node, st, walkDeeper) => { - if (objectPrototypeProperties.includes(node.name)) {return;} - addRef(st.key, node.name); - }, - WhileStatement: (node, st, walkDeeper) => { - addRef(st.key, specialReferenceWHILE); - node.test && walkDeeper(node.test, st); - node.body && walkDeeper(node.body, st); - }, - DoWhileStatement: (node, st, walkDeeper) => { - addRef(st.key, specialReferenceWHILE); - node.test && walkDeeper(node.test, st); - node.body && walkDeeper(node.body, st); - }, - ForStatement: (node, st, walkDeeper) => { - addRef(st.key, specialReferenceFOR); - node.init && walkDeeper(node.init, st); - node.test && walkDeeper(node.test, st); - node.update && walkDeeper(node.update, st); - node.body && walkDeeper(node.body, st); - }, - IfStatement: (node, st, walkDeeper) => { - addRef(st.key, specialReferenceIF); - node.test && walkDeeper(node.test, st); - node.consequent && walkDeeper(node.consequent, st); - node.alternate && walkDeeper(node.alternate, st); - }, - MemberExpression: (node, st, walkDeeper) => { - node.object && walkDeeper(node.object, st); - node.property && walkDeeper(node.property, st); - }, - } - } - - //Spread syntax not supported in Edge yet, use Object.assign - /* - walk.recursive(ast, {key: globalKey}, { - ImportDeclaration: (node, st, walkDeeper) => { - const importModuleName = node.source.value; - additionalModules.push(importModuleName); - - // This module's global scope refers to that module's global scope, no matter how we - // import it. - dependencyMap[st.key].add(importModuleName + memCheckGlobalKey); - - for (let i = 0; i < node.specifiers.length; ++i) { - const spec = node.specifiers[i]; - if (spec.imported !== undefined && spec.local !== undefined) { - // We depend on specific things. - internalToExternal[spec.local.name] = importModuleName + "." + spec.imported.name; - } else { - // We depend on everything. - dependencyMap[st.key].add(importModuleName + ".*"); - } - } - }, - FunctionDeclaration: (node, st, walkDeeper) => { - // Don't use walkDeeper, because we are changing the visitor set. - const key = currentModule + "." + node.id.name; - walk.recursive(node, {key: key}, commonVisitors()); - }, - ...commonVisitors() - }); - */ - walk.recursive(ast, {key: globalKey}, Object.assign({ - ImportDeclaration: (node, st, walkDeeper) => { - const importModuleName = node.source.value; - additionalModules.push(importModuleName); - - // This module's global scope refers to that module's global scope, no matter how we - // import it. - dependencyMap[st.key].add(importModuleName + memCheckGlobalKey); - - for (let i = 0; i < node.specifiers.length; ++i) { - const spec = node.specifiers[i]; - if (spec.imported !== undefined && spec.local !== undefined) { - // We depend on specific things. - internalToExternal[spec.local.name] = importModuleName + "." + spec.imported.name; - } else { - // We depend on everything. - dependencyMap[st.key].add(importModuleName + ".*"); - } - } - }, - FunctionDeclaration: (node, st, walkDeeper) => { - // Don't use walkDeeper, because we are changing the visitor set. - const key = currentModule + "." + node.id.name; - walk.recursive(node, {key: key}, commonVisitors()); - }, - }, commonVisitors())); - - return {dependencyMap: dependencyMap, additionalModules: additionalModules}; -} - -async function calculateRamUsage(codeCopy) { - //Create a temporary/mock WorkerScript and an AST from the code - var currServ = Player.getCurrentServer(); - var workerScript = new WorkerScript({ - filename:"foo", - scriptRef: {code:""}, - args:[], - getCode: function() { return ""; } - }); - workerScript.checkingRam = true; //Netscript functions will return RAM usage - workerScript.serverIp = currServ.ip; - - try { - return await parseOnlyRamCalculate(currServ, codeCopy, workerScript); - } catch (e) { - console.log("Failed to parse ram using new method. Falling back.", e); - } - - // Try the old way. - - try { - var ast = parse(codeCopy, {sourceType:"module"}); - } catch(e) { - return -1; - } - - //Search through AST, scanning for any 'Identifier' nodes for functions, or While/For/If nodes - var queue = [], ramUsage = CONSTANTS.ScriptBaseRamCost; - var whileUsed = false, forUsed = false, ifUsed = false; - queue.push(ast); - while (queue.length != 0) { - var exp = queue.shift(); - switch (exp.type) { - case "ImportDeclaration": - //Gets an array of all imported functions as AST expressions - //and pushes them on the queue. - var res = evaluateImport(exp, workerScript, true); - for (var i = 0; i < res.length; ++i) { - queue.push(res[i]); - } - break; - case "BlockStatement": - case "Program": - for (var i = 0; i < exp.body.length; ++i) { - if (exp.body[i] instanceof Node) { - queue.push(exp.body[i]); - } - } - break; - case "WhileStatement": - if (!whileUsed) { - ramUsage += CONSTANTS.ScriptWhileRamCost; - whileUsed = true; - } - break; - case "ForStatement": - if (!forUsed) { - ramUsage += CONSTANTS.ScriptForRamCost; - forUsed = true; - } - break; - case "IfStatement": - if (!ifUsed) { - ramUsage += CONSTANTS.ScriptIfRamCost; - ifUsed = true; - } - break; - case "Identifier": - if (exp.name in workerScript.env.vars) { - var func = workerScript.env.get(exp.name); - if (typeof func === "function") { - try { - var res = func.apply(null, []); - if (typeof res === "number") { - ramUsage += res; - } - } catch(e) { - console.log("ERROR applying function: " + e); - } - } - } - break; - default: - break; - } - - for (var prop in exp) { - if (exp.hasOwnProperty(prop)) { - if (exp[prop] instanceof Node) { - queue.push(exp[prop]); - } - } - } - } - - //Special case: hacknetnodes array - if (codeCopy.includes("hacknet")) { - ramUsage += CONSTANTS.ScriptHacknetNodesRamCost; - } - return ramUsage; -} - //Called when the game is loaded. Loads all running scripts (from all servers) //into worker scripts so that they will start running -function loadAllRunningScripts() { +export function loadAllRunningScripts() { var total = 0; let skipScriptLoad = (window.location.href.toLowerCase().indexOf("?noscripts") !== -1); if (skipScriptLoad) { console.info("Skipping the load of any scripts during startup"); } @@ -767,7 +359,7 @@ function scriptCalculateOfflineProduction(runningScriptObj) { var timesGrown = Math.round(0.5 * runningScriptObj.dataMap[ip][2] / runningScriptObj.onlineRunningTime * timePassed); console.log(runningScriptObj.filename + " called grow() on " + serv.hostname + " " + timesGrown + " times while offline"); runningScriptObj.log("Called grow() on " + serv.hostname + " " + timesGrown + " times while offline"); - var growth = processSingleServerGrowth(serv, timesGrown * 450); + var growth = processSingleServerGrowth(serv, timesGrown * 450, Player); runningScriptObj.log(serv.hostname + " grown by " + numeralWrapper.format(growth * 100 - 100, '0.000000%') + " from grow() calls made while offline"); } } @@ -838,7 +430,7 @@ function scriptCalculateOfflineProduction(runningScriptObj) { //Returns a RunningScript object matching the filename and arguments on the //designated server, and false otherwise -function findRunningScript(filename, args, server) { +export function findRunningScript(filename, args, server) { for (var i = 0; i < server.runningScripts.length; ++i) { if (server.runningScripts[i].filename == filename && compareArrays(server.runningScripts[i].args, args)) { @@ -847,6 +439,3 @@ function findRunningScript(filename, args, server) { } return null; } - -export {loadAllRunningScripts, findRunningScript, - scriptEditorInit, isScriptFilename}; diff --git a/src/Script/ScriptHelpersTS.ts b/src/Script/ScriptHelpersTS.ts new file mode 100644 index 000000000..f59b69810 --- /dev/null +++ b/src/Script/ScriptHelpersTS.ts @@ -0,0 +1,4 @@ +// Script helper functions +export function isScriptFilename(f: string) { + return f.endsWith(".js") || f.endsWith(".script") || f.endsWith(".ns"); +} diff --git a/src/Server/AllServers.ts b/src/Server/AllServers.ts index e04c33fbb..6d8d91c0b 100644 --- a/src/Server/AllServers.ts +++ b/src/Server/AllServers.ts @@ -1,40 +1,61 @@ -import { ipExists } from "../../utils/IPAddress"; +import { Server } from "./Server"; +import { SpecialServerIps } from "./SpecialServerIps"; +import { serverMetadata } from "./data/servers"; + +import { IMap } from "../types"; +import { createRandomIp, + ipExists } from "../../utils/IPAddress"; +import { getRandomInt } from "../../utils/helpers/getRandomInt"; +import { Reviver } from "../../utils/JSONReviver"; // Map of all Servers that exist in the game // Key (string) = IP // Value = Server object -let AllServers = {}; +export let AllServers: IMap = {}; // Saftely add a Server to the AllServers map -export function AddToAllServers(server) { +export function AddToAllServers(server: Server): void { var serverIp = server.ip; if (ipExists(serverIp)) { console.log("IP of server that's being added: " + serverIp); console.log("Hostname of the server thats being added: " + server.hostname); console.log("The server that already has this IP is: " + AllServers[serverIp].hostname); throw new Error("Error: Trying to add a server with an existing IP"); - return; } AllServers[serverIp] = server; } -export function initForeignServers() { +interface IServerParams { + hackDifficulty?: number; + hostname: string; + ip: string; + maxRam?: number; + moneyAvailable?: number; + numOpenPortsRequired: number; + organizationName: string; + requiredHackingSkill?: number; + serverGrowth?: number; + + [key: string]: any; +} + +export function initForeignServers(homeComputer: Server) { /* Create a randomized network for all the foreign servers */ //Groupings for creating a randomized network - const networkLayers = []; + const networkLayers: Server[][] = []; for (let i = 0; i < 15; i++) { networkLayers.push([]); } // Essentially any property that is of type 'number | IMinMaxRange' - const propertiesToPatternMatch = [ + const propertiesToPatternMatch: string[] = [ "hackDifficulty", "moneyAvailable", "requiredHackingSkill", "serverGrowth" ]; - const toNumber = (value) => { + const toNumber = (value: any) => { switch (typeof value) { case 'number': return value; @@ -46,7 +67,7 @@ export function initForeignServers() { } for (const metadata of serverMetadata) { - const serverParams = { + const serverParams: IServerParams = { hostname: metadata.hostname, ip: createRandomIp(), numOpenPortsRequired: metadata.numOpenPortsRequired, @@ -79,21 +100,21 @@ export function initForeignServers() { } /* Create a randomized network for all the foreign servers */ - const linkComputers = (server1, server2) => { + const linkComputers = (server1: Server, server2: Server) => { server1.serversOnNetwork.push(server2.ip); server2.serversOnNetwork.push(server1.ip); }; - const getRandomArrayItem = (arr) => arr[Math.floor(Math.random() * arr.length)]; + const getRandomArrayItem = (arr: any[]) => arr[Math.floor(Math.random() * arr.length)]; - const linkNetworkLayers = (network1, selectServer) => { + const linkNetworkLayers = (network1: Server[], selectServer: () => Server) => { for (const server of network1) { linkComputers(server, selectServer()); } }; // Connect the first tier of servers to the player's home computer - linkNetworkLayers(networkLayers[0], () => Player.getHomeComputer()); + linkNetworkLayers(networkLayers[0], () => homeComputer); for (let i = 1; i < networkLayers.length; i++) { linkNetworkLayers(networkLayers[i], () => getRandomArrayItem(networkLayers[i - 1])); } @@ -106,6 +127,6 @@ export function prestigeAllServers() { AllServers = {}; } -export function loadAllServers(saveString) { +export function loadAllServers(saveString: string) { AllServers = JSON.parse(saveString, Reviver); } diff --git a/src/Server/Server.ts b/src/Server/Server.ts index b4d1f33dc..bc3056d14 100644 --- a/src/Server/Server.ts +++ b/src/Server/Server.ts @@ -1,9 +1,14 @@ // Class representing a single generic Server + +// TODO This import is a circular import. Try to fix it in the future +import { GetServerByHostname } from "./ServerHelpers"; + import { BitNodeMultipliers } from "../BitNode/BitNodeMultipliers"; import { CodingContract } from "../CodingContracts"; import { Message } from "../Message/Message"; import { RunningScript } from "../Script/RunningScript"; import { Script } from "../Script/Script"; +import { isScriptFilename } from "../Script/ScriptHelpersTS"; import { TextFile } from "../TextFile"; import { createRandomIp } from "../../utils/IPAddress"; @@ -27,6 +32,11 @@ interface IConstructorParams { } export class Server { + // Initializes a Server Object from a JSON save state + static fromJSON(value: any): Server { + return Generic_fromJSON(Server, value.data); + } + // Initial server security level // (i.e. security level when the server was created) baseDifficulty: number = 1; @@ -172,49 +182,43 @@ export class Server { this.maxRam = ram; } - //The serverOnNetwork array holds the IP of all the servers. This function - //returns the actual Server objects - Server.prototype.getServerOnNetwork = function(i) { - if (i > this.serversOnNetwork.length) { - console.log("Tried to get server on network that was out of range"); - return; - } - return AllServers[this.serversOnNetwork[i]]; - } - - //Given the name of the script, returns the corresponding - //script object on the server (if it exists) - Server.prototype.getScript = function(scriptName) { - for (var i = 0; i < this.scripts.length; i++) { - if (this.scripts[i].filename == scriptName) { + // Given the name of the script, returns the corresponding + // script object on the server (if it exists) + getScript(scriptName: string): Script | null { + for (let i = 0; i < this.scripts.length; i++) { + if (this.scripts[i].filename === scriptName) { return this.scripts[i]; } } + return null; } - Server.prototype.capDifficulty = function() { + // Ensures that the server's difficulty (server security) doesn't get too high + capDifficulty(): void { if (this.hackDifficulty < this.minDifficulty) {this.hackDifficulty = this.minDifficulty;} if (this.hackDifficulty < 1) {this.hackDifficulty = 1;} - //Place some arbitrarily limit that realistically should never happen unless someone is - //screwing around with the game + + // Place some arbitrarily limit that realistically should never happen unless someone is + // screwing around with the game if (this.hackDifficulty > 1000000) {this.hackDifficulty = 1000000;} } - //Strengthens a server's security level (difficulty) by the specified amount - Server.prototype.fortify = function(amt) { + // Strengthens a server's security level (difficulty) by the specified amount + fortify(amt: number): void { this.hackDifficulty += amt; this.capDifficulty(); } - Server.prototype.weaken = function(amt) { + // Lowers the server's security level (difficulty) by the specified amount) + weaken(amt: number): void { this.hackDifficulty -= (amt * BitNodeMultipliers.ServerWeakenRate); this.capDifficulty(); } // Write to a script file // Overwrites existing files. Creates new files if the script does not eixst - Server.prototype.writeToScriptFile = function(fn, code) { + writeToScriptFile(fn: string, code: string) { var ret = {success: false, overwritten: false}; if (!isScriptFilename(fn)) { return ret; } @@ -232,7 +236,7 @@ export class Server { } //Otherwise, create a new script - var newScript = new Script(); + const newScript = new Script(); newScript.filename = fn; newScript.code = code; newScript.updateRamUsage(); @@ -244,8 +248,8 @@ export class Server { // Write to a text file // Overwrites existing files. Creates new files if the text file does not exist - Server.prototype.writeToTextFile = function(fn, txt) { - var ret = {success: false, overwritten: false}; + writeToTextFile(fn: string, txt: string) { + var ret = { success: false, overwritten: false }; if (!fn.endsWith("txt")) { return ret; } //Check if the text file already exists, and overwrite if it does @@ -265,11 +269,11 @@ export class Server { return ret; } - Server.prototype.addContract = function(contract) { + addContract(contract: CodingContract) { this.contracts.push(contract); } - Server.prototype.removeContract = function(contract) { + removeContract(contract: CodingContract) { if (contract instanceof CodingContract) { this.contracts = this.contracts.filter((c) => { return c.fn !== contract.fn; @@ -281,7 +285,7 @@ export class Server { } } - Server.prototype.getContract = function(contractName) { + getContract(contractName: string) { for (const contract of this.contracts) { if (contract.fn === contractName) { return contract; @@ -289,15 +293,11 @@ export class Server { } return null; } -} -//Functions for loading and saving a Server -Server.prototype.toJSON = function() { - return Generic_toJSON("Server", this); -} - -Server.fromJSON = function(value) { - return Generic_fromJSON(Server, value.data); + // Serialize the current object to a JSON save state + toJSON(): any { + return Generic_toJSON("Server", this); + } } Reviver.constructors.Server = Server; diff --git a/src/Server/ServerHelpers.js b/src/Server/ServerHelpers.ts similarity index 67% rename from src/Server/ServerHelpers.js rename to src/Server/ServerHelpers.ts index 0ab1183f5..e443deb0c 100644 --- a/src/Server/ServerHelpers.js +++ b/src/Server/ServerHelpers.ts @@ -1,23 +1,16 @@ -import { BitNodeMultipliers } from "./BitNode/BitNodeMultipliers"; -import { CodingContract, - ContractTypes } from "./CodingContracts"; -import { CONSTANTS } from "./Constants"; -import { Script, - isScriptFilename } from "./Script"; -import { Player } from "./Player"; -import { Programs } from "./Programs/Programs"; -import { SpecialServerIps } from "./SpecialServerIps"; -import { TextFile } from "./TextFile"; -import { getRandomInt } from "../utils/helpers/getRandomInt"; -import { serverMetadata } from "./data/servers"; -import { Reviver, - Generic_toJSON, - Generic_fromJSON} from "../utils/JSONReviver"; -import {isValidIPAddress} from "../utils/helpers/isValidIPAddress"; +import { AllServers } from "./AllServers"; +import { Server } from "./Server"; + +import { BitNodeMultipliers } from "../BitNode/BitNodeMultipliers"; +import { CONSTANTS } from "../Constants"; +import { IPlayer } from "../PersonObjects/IPlayer"; +import { Programs } from "../Programs/Programs"; + +import {isValidIPAddress} from "../../utils/helpers/isValidIPAddress"; // Returns the number of cycles needed to grow the specified server by the // specified amount. 'growth' parameter is in decimal form, not percentage -export function numCycleForGrowth(server, growth) { +export function numCycleForGrowth(server: Server, growth: number, p: IPlayer) { let ajdGrowthRate = 1 + (CONSTANTS.ServerBaseGrowthRate - 1) / server.hackDifficulty; if(ajdGrowthRate > CONSTANTS.ServerMaxGrowthRate) { ajdGrowthRate = CONSTANTS.ServerMaxGrowthRate; @@ -25,12 +18,12 @@ export function numCycleForGrowth(server, growth) { const serverGrowthPercentage = server.serverGrowth / 100; - const cycles = Math.log(growth)/(Math.log(ajdGrowthRate)*Player.hacking_grow_mult*serverGrowthPercentage); + const cycles = Math.log(growth)/(Math.log(ajdGrowthRate) * p.hacking_grow_mult * serverGrowthPercentage); return cycles; } //Applied server growth for a single server. Returns the percentage growth -export function processSingleServerGrowth(server, numCycles) { +export function processSingleServerGrowth(server: Server, numCycles: number, p: IPlayer) { //Server growth processed once every 450 game cycles const numServerGrowthCycles = Math.max(Math.floor(numCycles / 450), 0); @@ -44,7 +37,7 @@ export function processSingleServerGrowth(server, numCycles) { const numServerGrowthCyclesAdjusted = numServerGrowthCycles * serverGrowthPercentage * BitNodeMultipliers.ServerGrowthRate; //Apply serverGrowth for the calculated number of growth cycles - var serverGrowth = Math.pow(adjGrowthRate, numServerGrowthCyclesAdjusted * Player.hacking_grow_mult); + let serverGrowth = Math.pow(adjGrowthRate, numServerGrowthCyclesAdjusted * p.hacking_grow_mult); if (serverGrowth < 1) { console.log("WARN: serverGrowth calculated to be less than 1"); serverGrowth = 1; @@ -66,14 +59,14 @@ export function processSingleServerGrowth(server, numCycles) { // if there was any growth at all, increase security if (oldMoneyAvailable !== server.moneyAvailable) { //Growing increases server security twice as much as hacking - let usedCycles = numCycleForGrowth(server, server.moneyAvailable / oldMoneyAvailable); + let usedCycles = numCycleForGrowth(server, server.moneyAvailable / oldMoneyAvailable, p); usedCycles = Math.max(0, usedCycles); server.fortify(2 * CONSTANTS.ServerFortifyAmount * Math.ceil(usedCycles)); } return server.moneyAvailable / oldMoneyAvailable; } -export function prestigeHomeComputer(homeComp) { +export function prestigeHomeComputer(homeComp: Server) { const hasBitflume = homeComp.programs.includes(Programs.BitFlume.name); homeComp.programs.length = 0; //Remove programs @@ -93,17 +86,9 @@ export function prestigeHomeComputer(homeComp) { homeComp.messages.push("hackers-starting-handbook.lit"); } -function SizeOfAllServers() { - var size = 0, key; - for (key in AllServers) { - if (AllServers.hasOwnProperty(key)) size++; - } - return size; -} - //Returns server object with corresponding hostname // Relatively slow, would rather not use this a lot -export function GetServerByHostname(hostname) { +export function GetServerByHostname(hostname: string): Server | null { for (var ip in AllServers) { if (AllServers.hasOwnProperty(ip)) { if (AllServers[ip].hostname == hostname) { @@ -111,16 +96,30 @@ export function GetServerByHostname(hostname) { } } } + return null; } //Get server by IP or hostname. Returns null if invalid -export function getServer(s) { +export function getServer(s: string): Server | null { if (!isValidIPAddress(s)) { return GetServerByHostname(s); } - if(AllServers[s] !== undefined) { + if (AllServers[s] !== undefined) { return AllServers[s]; } + return null; } + +// Returns the i-th server on the specified server's network +// A Server's serverOnNetwork property holds only the IPs. This function returns +// the actual Server object +export function getServerOnNetwork(server: Server, i: number) { + if (i > server.serversOnNetwork.length) { + console.error("Tried to get server on network that was out of range"); + return; + } + + return AllServers[server.serversOnNetwork[i]]; +} diff --git a/src/Server/ServerPurchases.js b/src/Server/ServerPurchases.js index 00535b997..163a725d1 100644 --- a/src/Server/ServerPurchases.js +++ b/src/Server/ServerPurchases.js @@ -2,16 +2,16 @@ * Implements functions for purchasing servers or purchasing more RAM for * the home computer */ -import { BitNodeMultipliers } from "./BitNode/BitNodeMultipliers"; -import { CONSTANTS } from "./Constants"; -import { Player } from "./Player"; -import { Server, - AllServers, - AddToAllServers} from "./Server"; -import { dialogBoxCreate } from "../utils/DialogBox"; -import { createRandomIp } from "../utils/IPAddress"; -import { yesNoTxtInpBoxGetInput } from "../utils/YesNoBox"; -import { isPowerOfTwo } from "../utils/helpers/isPowerOfTwo"; +import { BitNodeMultipliers } from "../BitNode/BitNodeMultipliers"; +import { CONSTANTS } from "../Constants"; +import { Player } from "../Player"; +import { AllServers } from "../Server/AllServers"; +import { Server } from "../Server/Server"; +import { AddToAllServers } from "../Server/ServerHelpers"; +import { dialogBoxCreate } from "../../utils/DialogBox"; +import { createRandomIp } from "../../utils/IPAddress"; +import { yesNoTxtInpBoxGetInput } from "../../utils/YesNoBox"; +import { isPowerOfTwo } from "../../utils/helpers/isPowerOfTwo"; // Returns the cost of purchasing a server with the given RAM // Returns Infinity for invalid 'ram' arguments diff --git a/src/Server/SpecialServerIps.js b/src/Server/SpecialServerIps.js deleted file mode 100644 index d3175dc01..000000000 --- a/src/Server/SpecialServerIps.js +++ /dev/null @@ -1,50 +0,0 @@ -import {Reviver, Generic_toJSON, - Generic_fromJSON} from "../utils/JSONReviver"; - -/* Holds IP of Special Servers */ -let SpecialServerNames = { - FulcrumSecretTechnologies: "Fulcrum Secret Technologies Server", - CyberSecServer: "CyberSec Server", - NiteSecServer: "NiteSec Server", - TheBlackHandServer: "The Black Hand Server", - BitRunnersServer: "BitRunners Server", - TheDarkArmyServer: "The Dark Army Server", - DaedalusServer: "Daedalus Server", - WorldDaemon: "w0r1d_d43m0n", -} -function SpecialServerIpsMap() {} - -SpecialServerIpsMap.prototype.addIp = function(name, ip) { - this[name] = ip; -} - -SpecialServerIpsMap.prototype.toJSON = function() { - return Generic_toJSON("SpecialServerIpsMap", this); -} - -SpecialServerIpsMap.fromJSON = function(value) { - return Generic_fromJSON(SpecialServerIpsMap, value.data); -} - -Reviver.constructors.SpecialServerIpsMap = SpecialServerIpsMap; - -let SpecialServerIps = new SpecialServerIpsMap(); - -function prestigeSpecialServerIps() { - for (var member in SpecialServerIps) { - delete SpecialServerIps[member]; - } - SpecialServerIps = null; - SpecialServerIps = new SpecialServerIpsMap(); -} - -function loadSpecialServerIps(saveString) { - SpecialServerIps = JSON.parse(saveString, Reviver); -} - -function initSpecialServerIps() { - SpecialServerIps = new SpecialServerIpsMap(); -} - -export {SpecialServerNames, SpecialServerIps, SpecialServerIpsMap, loadSpecialServerIps, - prestigeSpecialServerIps, initSpecialServerIps}; diff --git a/src/Server/SpecialServerIps.ts b/src/Server/SpecialServerIps.ts new file mode 100644 index 000000000..121c377cc --- /dev/null +++ b/src/Server/SpecialServerIps.ts @@ -0,0 +1,56 @@ +import { IMap } from "../types"; +import { Reviver, + Generic_toJSON, + Generic_fromJSON } from "../../utils/JSONReviver"; + +/* Holds IP of Special Servers */ +export let SpecialServerNames: IMap = { + FulcrumSecretTechnologies: "Fulcrum Secret Technologies Server", + CyberSecServer: "CyberSec Server", + NiteSecServer: "NiteSec Server", + TheBlackHandServer: "The Black Hand Server", + BitRunnersServer: "BitRunners Server", + TheDarkArmyServer: "The Dark Army Server", + DaedalusServer: "Daedalus Server", + WorldDaemon: "w0r1d_d43m0n", +} + +export class SpecialServerIpsMap { + // Initializes a SpecialServerIpsMap Object from a JSON save state + static fromJSON(value: any): SpecialServerIpsMap { + return Generic_fromJSON(SpecialServerIpsMap, value.data); + } + + [key: string]: Function | string; + + constructor() {} + + addIp(name:string, ip: string) { + this[name] = ip; + } + + // Serialize the current object to a JSON save state + toJSON(): any { + return Generic_toJSON("SpecialServerIpsMap", this); + } +} + +Reviver.constructors.SpecialServerIpsMap = SpecialServerIpsMap; + +export let SpecialServerIps: SpecialServerIpsMap = new SpecialServerIpsMap(); + +export function prestigeSpecialServerIps() { + for (var member in SpecialServerIps) { + delete SpecialServerIps[member]; + } + + SpecialServerIps = new SpecialServerIpsMap(); +} + +export function loadSpecialServerIps(saveString: string) { + SpecialServerIps = JSON.parse(saveString, Reviver); +} + +export function initSpecialServerIps() { + SpecialServerIps = new SpecialServerIpsMap(); +} diff --git a/src/data/servers.ts b/src/Server/data/servers.ts similarity index 99% rename from src/data/servers.ts rename to src/Server/data/servers.ts index 12698a3b8..ec661c458 100644 --- a/src/data/servers.ts +++ b/src/Server/data/servers.ts @@ -81,6 +81,8 @@ interface IServerMetadata { * A "unique" server that has special implications when the player manually hacks it. */ specialName?: string; + + [key: string]: any; } /** diff --git a/src/Terminal.js b/src/Terminal.js index bb06b6c25..d2744b2c9 100644 --- a/src/Terminal.js +++ b/src/Terminal.js @@ -10,8 +10,9 @@ import { executeDarkwebTerminalCommand, checkIfConnectedToDarkweb } from "./DarkWeb/DarkWeb"; import { DarkWebItems } from "./DarkWeb/DarkWebItems"; import {Engine} from "./engine"; -import {FconfSettings, parseFconfSettings, - createFconf} from "./Fconf"; +import { parseFconfSettings, + createFconf } from "./Fconf/Fconf"; +import { FconfSettings } from "./Fconf/FconfSettings"; import {calculateHackingChance, calculateHackingExpGain, calculatePercentMoneyHacked, @@ -22,18 +23,22 @@ import {TerminalHelpText, HelpTexts} from "./HelpText"; import {iTutorialNextStep, iTutorialSteps, ITutorial} from "./InteractiveTutorial"; import {showLiterature} from "./Literature"; -import {showMessage, Message} from "./Message"; +import { Message } from "./Message/Message"; +import { showMessage } from "./Message/MessageHelpers"; import {killWorkerScript, addWorkerScript} from "./NetscriptWorker"; import {Player} from "./Player"; import {hackWorldDaemon} from "./RedPill"; -import { findRunningScript, - RunningScript, - isScriptFilename } from "./Script"; -import {AllServers, GetServerByHostname, - getServer, Server} from "./Server"; +import { RunningScript } from "./Script/RunningScript"; +import { findRunningScript } from "./Script/ScriptHelpers"; +import { isScriptFilename } from "./Script/ScriptHelpersTS"; +import { AllServers } from "./Server/AllServers"; +import { Server } from "./Server/Server"; +import { GetServerByHostname, + getServer, + getServerOnNetwork } from "./Server/ServerHelpers"; import {Settings} from "./Settings/Settings"; -import {SpecialServerIps, - SpecialServerNames} from "./SpecialServerIps"; +import { SpecialServerIps, + SpecialServerNames } from "./Server/SpecialServerIps"; import {getTextFile} from "./TextFile"; import { setTimeoutRef } from "./utils/SetTimeoutRef"; import {containsAllStrings, @@ -1157,8 +1162,8 @@ let Terminal = { let ip = commandArray[1]; - for (var i = 0; i < Player.getCurrentServer().serversOnNetwork.length; i++) { - if (Player.getCurrentServer().getServerOnNetwork(i).ip == ip || Player.getCurrentServer().getServerOnNetwork(i).hostname == ip) { + for (var i = 0; i < s.serversOnNetwork.length; i++) { + if (getServerOnNetwork(s, i).ip == ip || getServerOnNetwork(s, i).hostname == ip) { Terminal.connectToServer(ip); return; } @@ -1812,11 +1817,13 @@ let Terminal = { postError("Incorrect usage of netstat/scan command. Usage: netstat/scan"); return; } - //Displays available network connections using TCP + + // Displays available network connections using TCP + const currServ = Player.getCurrentServer(); post("Hostname IP Root Access"); - for (let i = 0; i < Player.getCurrentServer().serversOnNetwork.length; i++) { + for (let i = 0; i < currServ.serversOnNetwork.length; i++) { //Add hostname - let entry = Player.getCurrentServer().getServerOnNetwork(i); + let entry = getServerOnNetwork(currServ, i); if (entry == null) { continue; } entry = entry.hostname; @@ -1824,16 +1831,16 @@ let Terminal = { let numSpaces = 21 - entry.length; let spaces = Array(numSpaces+1).join(" "); entry += spaces; - entry += Player.getCurrentServer().getServerOnNetwork(i).ip; + entry += getServerOnNetwork(currServ, i).ip; //Calculate padding and add root access info let hasRoot; - if (Player.getCurrentServer().getServerOnNetwork(i).hasAdminRights) { + if (getServerOnNetwork(currServ, i).hasAdminRights) { hasRoot = 'Y'; } else { hasRoot = 'N'; } - numSpaces = 21 - Player.getCurrentServer().getServerOnNetwork(i).ip.length; + numSpaces = 21 - getServerOnNetwork(currServ, i).ip.length; spaces = Array(numSpaces+1).join(" "); entry += spaces; entry += hasRoot; @@ -1867,7 +1874,7 @@ let Terminal = { visited[s.ip] = 1; } for (var i = s.serversOnNetwork.length-1; i >= 0; --i) { - stack.push(s.getServerOnNetwork(i)); + stack.push(getServerOnNetwork(s, i)); depthQueue.push(d+1); } if (d == 0) {continue;} //Don't print current server diff --git a/src/engine.js b/src/engine.js index 7ad1175f9..5051c707d 100644 --- a/src/engine.js +++ b/src/engine.js @@ -21,14 +21,12 @@ import {CompanyPositions} from "./Company/CompanyP import {initCompanies} from "./Company/Companies"; import { Corporation } from "./Corporation/Corporation"; import {CONSTANTS} from "./Constants"; - - import {createDevMenu, closeDevMenu} from "./DevMenu"; import { Factions, initFactions } from "./Faction/Factions"; import { displayFactionContent, joinFaction, processPassiveFactionRepGain, inviteToFaction } from "./Faction/FactionHelpers"; -import {FconfSettings} from "./Fconf"; +import { FconfSettings } from "./Fconf/FconfSettings"; import {displayLocationContent, initLocationButtons} from "./Location"; import {Locations} from "./Locations"; @@ -36,7 +34,7 @@ import {displayHacknetNodesContent, processAllHacknetNodeEarnings, updateHacknetNodesContent} from "./HacknetNode"; import {iTutorialStart} from "./InteractiveTutorial"; import {initLiterature} from "./Literature"; -import {checkForMessagesToSend, initMessages} from "./Message"; +import { checkForMessagesToSend, initMessages } from "./Message/MessageHelpers"; import {inMission, currMission} from "./Missions"; import {initSingularitySFFlags, hasSingularitySF, hasCorporationSF} from "./NetscriptFunctions"; @@ -54,13 +52,14 @@ import {saveObject, loadGame} from "./SaveObject"; import { getCurrentEditor, loadAllRunningScripts, scriptEditorInit, - updateScriptEditorContent } from "./Script"; -import {AllServers, Server, initForeignServers} from "./Server"; + updateScriptEditorContent } from "./Script/ScriptHelpers"; +import { AllServers } from "./Server/AllServers"; +import { Server } from "./Server/Server"; +import { initForeignServers } from "./Server/ServerHelpers"; import {Settings} from "./Settings/Settings"; import { initSourceFiles, SourceFiles } from "./SourceFile"; import { updateSourceFileFlags } from "./SourceFile/SourceFileFlags"; - -import {SpecialServerIps, initSpecialServerIps} from "./SpecialServerIps"; +import {SpecialServerIps, initSpecialServerIps} from "./Server/SpecialServerIps"; import {StockMarket, StockSymbols, SymbolToStockMap, initStockSymbols, initSymbolToStockMap, stockMarketCycle, @@ -1317,7 +1316,7 @@ const Engine = { Engine.setDisplayElements(); //Sets variables for important DOM elements Engine.start(); //Run main game loop and Scripts loop Player.init(); - initForeignServers(); + initForeignServers(Player.getHomeComputer()); initCompanies(); initFactions(); initAugmentations(); diff --git a/utils/IPAddress.ts b/utils/IPAddress.ts index 691f27505..948b055eb 100644 --- a/utils/IPAddress.ts +++ b/utils/IPAddress.ts @@ -1,5 +1,5 @@ -import {AllServers} from "../src/Server"; -import {getRandomByte} from "./helpers/getRandomByte"; +import { AllServers } from "../src/Server/AllServers"; +import { getRandomByte } from "./helpers/getRandomByte"; /* Functions to deal with manipulating IP addresses*/ From e6a379484950142c74fb0810f14d7e32fbdcc206 Mon Sep 17 00:00:00 2001 From: danielyxie Date: Tue, 26 Feb 2019 00:45:34 -0800 Subject: [PATCH 03/10] Fixed changelog formatting --- doc/source/changelog.rst | 32 ++++++++++++++++---------------- 1 file changed, 16 insertions(+), 16 deletions(-) diff --git a/doc/source/changelog.rst b/doc/source/changelog.rst index 2804ea535..47ee7059c 100644 --- a/doc/source/changelog.rst +++ b/doc/source/changelog.rst @@ -6,22 +6,22 @@ Changelog v0.44.0 - 2/26/2019 ------------------- * Bladeburner Changes: - ** Reduced the amount of rank needed to earn a skill point - ** Reduced the effects of the "Reaper" and "Evasive System" skills - ** Increased the effect of the "Hyperdrive" and "Hands of Midas" skills - ** Slightly increased the rate which the skill point cost rises for almost all skills - ** The "Overlock" Skill now has a maximum level of 90 instead of 95 - ** Money earned from Contracts increased by 400% - ** Changed the way population affects success rate. Extreme populations now have less dramatic effects - ** Added two new General Actions: Diplomacy and Hyperbolic Regeneration Chamber - ** Lowered the rep and money cost of the "Blade's Simulacrum" augmentation - ** Significantly decreased the initial amount of Contracts/Operations (the "Contracts/Operations remaining" value) - ** Decreased the rate at which the amount of Contracts/Operations increases over time - ** Decreased the number of successes you need to increase the max level of a Contract/Operation - ** Increased the average number of Synthoid communities each city has - ** Reduced the amount by which a successful raid will decrease the population of a city - ** The "riots" event will now increase the chaos of a city by a greater amount - ** Significantly increased the effect that Agility and Dexterity have on action time + * Reduced the amount of rank needed to earn a skill point + * Reduced the effects of the "Reaper" and "Evasive System" skills + * Increased the effect of the "Hyperdrive" and "Hands of Midas" skills + * Slightly increased the rate which the skill point cost rises for almost all skills + * The "Overlock" Skill now has a maximum level of 90 instead of 95 + * Money earned from Contracts increased by 400% + * Changed the way population affects success rate. Extreme populations now have less dramatic effects + * Added two new General Actions: Diplomacy and Hyperbolic Regeneration Chamber + * Lowered the rep and money cost of the "Blade's Simulacrum" augmentation + * Significantly decreased the initial amount of Contracts/Operations (the "Contracts/Operations remaining" value) + * Decreased the rate at which the amount of Contracts/Operations increases over time + * Decreased the number of successes you need to increase the max level of a Contract/Operation + * Increased the average number of Synthoid communities each city has + * Reduced the amount by which a successful raid will decrease the population of a city + * The "riots" event will now increase the chaos of a city by a greater amount + * Significantly increased the effect that Agility and Dexterity have on action time * Added new BitNode multipliers: * HomeComputerRamCost - Affects how much it costs to upgrade home computer's RAM * DaedalusAugsRequirement - Affects how many Augmentations you need in order to get invited to Daedalus From 67d083772a97da7dd5ce7123dae318063e10bc43 Mon Sep 17 00:00:00 2001 From: Mason Dechaineux Date: Mon, 25 Feb 2019 07:20:56 +1000 Subject: [PATCH 04/10] Fixed bug with rm terminal and ns commands that made non-exe files with .exe in their name undeletable. --- src/NetscriptFunctions.js | 2 +- src/Terminal.js | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/NetscriptFunctions.js b/src/NetscriptFunctions.js index 439eebfa5..9b7e6a56a 100644 --- a/src/NetscriptFunctions.js +++ b/src/NetscriptFunctions.js @@ -2261,7 +2261,7 @@ function NetscriptFunctions(workerScript) { throw makeRuntimeRejectMsg(workerScript, `Invalid server specified for rm(): ${ip}`); } - if (fn.includes(".exe")) { + if (fn.endsWith(".exe")) { for (var i = 0; i < s.programs.length; ++i) { if (s.programs[i] === fn) { s.programs.splice(i, 1); diff --git a/src/Terminal.js b/src/Terminal.js index 926318047..bb06b6c25 100644 --- a/src/Terminal.js +++ b/src/Terminal.js @@ -1342,7 +1342,7 @@ let Terminal = { //Check programs let delTarget = commandArray[1]; - if (delTarget.includes(".exe")) { + if (delTarget.endsWith(".exe")) { for (let i = 0; i < s.programs.length; ++i) { if (s.programs[i] === delTarget) { s.programs.splice(i, 1); From d54e39c9c65a8ac2ac8f55fd6588f4a7fb64c594 Mon Sep 17 00:00:00 2001 From: danielyxie Date: Tue, 26 Feb 2019 00:51:48 -0800 Subject: [PATCH 05/10] v0.44.1 Duplicate Sleeve updates --- src/DevMenu.js | 26 ++++++ src/PersonObjects/Person.ts | 1 - src/PersonObjects/Sleeve/Sleeve.ts | 44 ++++++++- .../Sleeve/SleeveAugmentationsUI.ts | 91 +++++++++++++++++++ .../Sleeve/SleeveCovenantPurchases.ts | 2 +- src/PersonObjects/Sleeve/SleeveUI.ts | 25 +++-- src/Player.js | 4 +- 7 files changed, 180 insertions(+), 13 deletions(-) create mode 100644 src/PersonObjects/Sleeve/SleeveAugmentationsUI.ts diff --git a/src/DevMenu.js b/src/DevMenu.js index ceadc51cf..03b72f108 100644 --- a/src/DevMenu.js +++ b/src/DevMenu.js @@ -51,6 +51,15 @@ export function createDevMenu() { innerText: "Add $1000t", }); + const addMoney2 = createElement("button", { + class: "std-button", + clickListener: () => { + Player.gainMoney(1e12); + }, + display: "block", + innerText: "Add $1t", + }) + const addRam = createElement("button", { class: "std-button", clickListener: () => { @@ -588,6 +597,20 @@ export function createDevMenu() { innerText: "View Stock Price Caps", }); + // Sleeves + const sleevesHeader = createElement("h2", { innerText: "Sleeves" }); + + const sleevesRemoveAllShockRecovery = createElement("button", { + class: "std-button", + display: "block", + innerText: "Set Shock Recovery of All Sleeves to 0", + clickListener: () => { + for (let i = 0; i < Player.sleeves.length; ++i) { + Player.sleeves[i].shock = 100; + } + } + }); + // Add everything to container, then append to main menu const devMenuContainer = createElement("div", { class: "generic-menupage-container", @@ -597,6 +620,7 @@ export function createDevMenu() { devMenuContainer.appendChild(devMenuText); devMenuContainer.appendChild(genericHeader); devMenuContainer.appendChild(addMoney); + devMenuContainer.appendChild(addMoney2); devMenuContainer.appendChild(addRam); devMenuContainer.appendChild(triggerBitflume); devMenuContainer.appendChild(destroyCurrentBitnode); @@ -674,6 +698,8 @@ export function createDevMenu() { devMenuContainer.appendChild(stockPriceChangeBtn); devMenuContainer.appendChild(createElement("br")); devMenuContainer.appendChild(stockViewPriceCapBtn); + devMenuContainer.appendChild(sleevesHeader); + devMenuContainer.appendChild(sleevesRemoveAllShockRecovery); const entireGameContainer = document.getElementById("entire-game-container"); if (entireGameContainer == null) { diff --git a/src/PersonObjects/Person.ts b/src/PersonObjects/Person.ts index 6a80dce2f..9bc208a0a 100644 --- a/src/PersonObjects/Person.ts +++ b/src/PersonObjects/Person.ts @@ -5,7 +5,6 @@ import { IPlayerOwnedAugmentation } from "../Augmentation/PlayerOwnedAugmentatio import { BitNodeMultipliers } from "../BitNode/BitNodeMultipliers"; import { Cities } from "../Locations/Cities"; import { CONSTANTS } from "../Constants"; -import { IMap } from "../types"; // Interface that defines a generic object used to track experience/money // earnings for tasks diff --git a/src/PersonObjects/Sleeve/Sleeve.ts b/src/PersonObjects/Sleeve/Sleeve.ts index a634a8d2f..a02842560 100644 --- a/src/PersonObjects/Sleeve/Sleeve.ts +++ b/src/PersonObjects/Sleeve/Sleeve.ts @@ -13,6 +13,8 @@ import { Person, ITaskTracker, createTaskTracker } from "../Person"; +import { Augmentation } from "../../Augmentation/Augmentation"; + import { BitNodeMultipliers } from "../../BitNode/BitNodeMultipliers"; import { Crime } from "../../Crime/Crime"; @@ -136,8 +138,11 @@ export class Sleeve extends Person { */ sync: number = 1; - constructor() { + constructor(p: IPlayer | null = null) { super(); + if (p != null) { + this.shockRecovery(p); + } } /** @@ -381,6 +386,17 @@ export class Sleeve extends Person { } } + installAugmentation(aug: Augmentation): void { + this.hacking_exp = 0; + this.strength_exp = 0; + this.defense_exp = 0; + this.dexterity_exp = 0; + this.agility_exp = 0; + this.charisma_exp = 0; + this.applyAugmentation(aug); + this.updateStatLevels(); + } + log(entry: string): void { const MaxLogSize: number = 50; this.logs.push(entry); @@ -452,10 +468,10 @@ export class Sleeve extends Person { company!.playerReputation += (this.getRepGain(p) * cyclesUsed); break; case SleeveTaskType.Recovery: - this.shock = Math.min(100, this.shock + (0.0001 * cyclesUsed)); + this.shock = Math.min(100, this.shock + (0.0002 * cyclesUsed)); break; case SleeveTaskType.Sync: - this.sync = Math.min(100, this.sync + (0.0001 * cyclesUsed)); + this.sync = Math.min(100, this.sync + (0.0002 * cyclesUsed)); break; default: break; @@ -492,6 +508,28 @@ export class Sleeve extends Person { this.gymStatType = ""; } + shockRecovery(p: IPlayer): boolean { + if (this.currentTask !== SleeveTaskType.Idle) { + this.finishTask(p); + } else { + this.resetTaskStatus(); + } + + this.currentTask = SleeveTaskType.Recovery; + return true; + } + + synchronize(p: IPlayer): boolean { + if (this.currentTask !== SleeveTaskType.Idle) { + this.finishTask(p); + } else { + this.resetTaskStatus(); + } + + this.currentTask = SleeveTaskType.Sync; + return true; + } + /** * Take a course at a university */ diff --git a/src/PersonObjects/Sleeve/SleeveAugmentationsUI.ts b/src/PersonObjects/Sleeve/SleeveAugmentationsUI.ts new file mode 100644 index 000000000..280b8b37b --- /dev/null +++ b/src/PersonObjects/Sleeve/SleeveAugmentationsUI.ts @@ -0,0 +1,91 @@ +/** + * Module for handling the UI for purchasing Sleeve Augmentations + * This UI is a popup, not a full page + */ +import { Sleeve } from "./Sleeve"; + +import { IPlayer } from "../IPlayer"; + +import { Augmentation } from "../../Augmentation/Augmentation"; +import { Augmentations } from "../../Augmentation/Augmentations"; + +import { Faction } from "../../Faction/Faction"; +import { Factions } from "../../Faction/Factions"; + +import { numeralWrapper } from "../../ui/numeralFormat"; + +import { dialogBoxCreate } from "../../../utils/DialogBox"; + +import { clearEventListeners } from "../../../utils/uiHelpers/clearEventListeners"; +import { createElement } from "../../../utils/uiHelpers/createElement"; +import { createPopup } from "../../../utils/uiHelpers/createPopup"; +import { createPopupCloseButton } from "../../../utils/uiHelpers/createPopupCloseButton"; +import { getSelectValue } from "../../../utils/uiHelpers/getSelectData"; +import { removeChildrenFromElement } from "../../../utils/uiHelpers/removeChildrenFromElement"; +import { removeElement } from "../../../utils/uiHelpers/removeElement"; +import { removeElementById } from "../../../utils/uiHelpers/removeElementById"; + +export function createSleevePurchaseAugsPopup(sleeve: Sleeve, p: IPlayer) { + // You can only purchase Augmentations that are actually available from + // your factions. I.e. you must be in a faction that has the Augmentation + // and you must also have enough rep in that faction in order to purchase it. + const availableAugs: Augmentation[] = []; + + for (const facName of p.factions) { + const fac: Faction | null = Factions[facName]; + if (fac == null) { continue; } + + for (const augName of fac.augmentations) { + const aug: Augmentation | null = Augmentations[augName]; + + if (fac.playerReputation > aug.baseRepRequirement && !availableAugs.includes(aug)) { + availableAugs.push(aug); + } + } + } + + + // General info about buying Augmentations + const info = createElement("p", { + innerHTML: + [ + `You can purchase Augmentations for your Duplicate Sleeves. These Augmentations`, + `have the same effect as they would for you. You can only purchase Augmentations`, + `that you have unlocked through Factions.

`, + `When purchasing an Augmentation for a Duplicate Sleeve, they are immediately`, + `installed. This means that the Duplicate Sleeve will immediately lose all of`, + `its stat experience.` + ].join(" "), + }); + + const popupId = "purchase-sleeve-augs-popup"; + const popupElems: HTMLElement[] = [info]; + + for (const aug of availableAugs) { + const div = createElement("div", { + class: "cmpy-mgmt-upgrade-div", // We'll reuse this CSS class + }); + + div.appendChild(createElement("p", { + innerHTML: + [ + `

${aug.name}


`, + `Cost: ${numeralWrapper.formatMoney(aug.baseCost)}

`, + `${aug.info}` + ].join(" "), + clickListener: () => { + if (p.canAfford(aug.baseCost)) { + p.loseMoney(aug.baseCost); + sleeve.installAugmentation(aug); + dialogBoxCreate(`Installed ${aug.name} on Duplicate Sleeve!`, false) + } else { + dialogBoxCreate(`You cannot afford ${aug.name}`, false); + } + } + })); + + popupElems.push(div); + } + + createPopup(popupId, popupElems); +} diff --git a/src/PersonObjects/Sleeve/SleeveCovenantPurchases.ts b/src/PersonObjects/Sleeve/SleeveCovenantPurchases.ts index b05eb80bf..496641415 100644 --- a/src/PersonObjects/Sleeve/SleeveCovenantPurchases.ts +++ b/src/PersonObjects/Sleeve/SleeveCovenantPurchases.ts @@ -29,7 +29,7 @@ export function createPurchaseSleevesFromCovenantPopup(p: IPlayer) { if (p.canAfford(cost)) { p.loseMoney(cost); p.sleevesFromCovenant += 1; - p.sleeves.push(new Sleeve()); + p.sleeves.push(new Sleeve(p)); yesNoBoxClose(); } else { dialogBoxCreate("You cannot afford to purchase a Duplicate Sleeve", false); diff --git a/src/PersonObjects/Sleeve/SleeveUI.ts b/src/PersonObjects/Sleeve/SleeveUI.ts index 486e510ea..a9a0197fd 100644 --- a/src/PersonObjects/Sleeve/SleeveUI.ts +++ b/src/PersonObjects/Sleeve/SleeveUI.ts @@ -10,6 +10,8 @@ import { IPlayer } from "../IPlayer"; import { CONSTANTS } from "../../Constants"; import { Locations } from "../../Locations"; +import { Augmentations } from "../../Augmentation/Augmentations"; + import { Faction } from "../../Faction/Faction"; import { Factions } from "../../Faction/Factions"; import { FactionWorkType } from "../../Faction/FactionWorkTypeEnum"; @@ -44,6 +46,7 @@ interface ISleeveUIElems { stats: HTMLElement | null; moreStatsButton: HTMLElement | null; travelButton: HTMLElement | null; + purchaseAugsButton: HTMLElement | null; taskPanel: HTMLElement | null; taskSelector: HTMLSelectElement | null; taskDetailsSelector: HTMLSelectElement | null; @@ -171,6 +174,7 @@ function createSleeveUi(sleeve: Sleeve, allSleeves: Sleeve[]): ISleeveUIElems { stats: null, moreStatsButton: null, travelButton: null, + purchaseAugsButton: null, taskPanel: null, taskSelector: null, taskDetailsSelector: null, @@ -267,10 +271,22 @@ function createSleeveUi(sleeve: Sleeve, allSleeves: Sleeve[]): ISleeveUIElems { createPopup(popupId, popupArguments); } - }) + }); + elems.purchaseAugsButton = createElement("button", { + class: "std-button", + display: "block", + innerText: "Purchase Augmentations", + clickListener: () => { + + } + }); elems.statsPanel.appendChild(elems.stats); elems.statsPanel.appendChild(elems.moreStatsButton); elems.statsPanel.appendChild(elems.travelButton); + if (sleeve.shock >= 100) { + // You can only buy augs when shock recovery is 0 + elems.statsPanel.appendChild(elems.purchaseAugsButton); + } elems.taskPanel = createElement("div", { class: "sleeve-panel", width: "40%" }); elems.taskSelector = createElement("select") as HTMLSelectElement; @@ -655,14 +671,11 @@ function setSleeveTask(sleeve: Sleeve, elems: ISleeveUIElems): boolean { res = sleeve.workoutAtGym(playerRef!, detailValue2, detailValue); break; case "Shock Recovery": - sleeve.finishTask(playerRef!); sleeve.currentTask = SleeveTaskType.Recovery; - res = true; + res = sleeve.shockRecovery(playerRef!); break; case "Synchronize": - sleeve.finishTask(playerRef!); - sleeve.currentTask = SleeveTaskType.Sync; - res = true; + res = sleeve.synchronize(playerRef!); break; default: console.error(`Invalid/Unrecognized taskValue in setSleeveTask(): ${taskValue}`); diff --git a/src/Player.js b/src/Player.js index bdfcd65d4..25c19af5f 100644 --- a/src/Player.js +++ b/src/Player.js @@ -281,7 +281,7 @@ PlayerObject.prototype.prestigeAugmentation = function() { for (let i = 0; i < this.sleeves.length; ++i) { if (this.sleeves[i] instanceof Sleeve) { - this.sleeves[i].resetTaskStatus(); + this.sleeves[i].shockRecovery(this); } } @@ -372,7 +372,7 @@ PlayerObject.prototype.prestigeSourceFile = function() { // Duplicate sleeves are reset to level 1 every Bit Node (but the number of sleeves you have persists) this.sleeves.length = SourceFileFlags[10] + this.sleevesFromCovenant; for (let i = 0; i < this.sleeves.length; ++i) { - this.sleeves[i] = new Sleeve(); + this.sleeves[i] = new Sleeve(this); } this.isWorking = false; From 9879d07d7c637b993a415fa256c253dff69bc805 Mon Sep 17 00:00:00 2001 From: danielyxie Date: Tue, 26 Feb 2019 18:26:29 -0800 Subject: [PATCH 06/10] Fixed a few minor bugs with submitting solutions for Coding Contracts. Started on Getting STarted Guide for wiki --- doc/source/conf.py | 4 +- doc/source/guidesandtips.rst | 8 ++-- ...tingstartedguideforbeginnerprogrammers.rst | 47 +++++++++++++++++++ src/Constants.ts | 45 ++++-------------- src/NetscriptFunctions.js | 24 ++++++++-- .../Sleeve/SleeveAugmentationsUI.ts | 13 +++-- src/PersonObjects/Sleeve/SleeveUI.ts | 3 +- src/data/codingcontracttypes.ts | 3 +- src/ui/displayCharacterInfo.ts | 2 +- src/utils/helpers/is2DArray.ts | 8 ++++ utils/README.md | 4 ++ 11 files changed, 104 insertions(+), 57 deletions(-) create mode 100644 doc/source/guidesandtips/gettingstartedguideforbeginnerprogrammers.rst create mode 100644 src/utils/helpers/is2DArray.ts create mode 100644 utils/README.md diff --git a/doc/source/conf.py b/doc/source/conf.py index 22d8d56b0..46634b60b 100644 --- a/doc/source/conf.py +++ b/doc/source/conf.py @@ -64,9 +64,9 @@ documentation_title = '{0} Documentation'.format(project) # built documents. # # The short X.Y version. -version = '0.43' +version = '0.44' # The full version, including alpha/beta/rc tags. -release = '0.43.0' +release = '0.44.1' # The language for content autogenerated by Sphinx. Refer to documentation # for a list of supported languages. diff --git a/doc/source/guidesandtips.rst b/doc/source/guidesandtips.rst index d01e67cdd..c2f17f202 100644 --- a/doc/source/guidesandtips.rst +++ b/doc/source/guidesandtips.rst @@ -1,9 +1,11 @@ Guides & Tips ============= +Getting Started Guide for Intermediate Programmers +What BitNode should I do? +Beginners FAQ + .. toctree:: :maxdepth: 3 - Getting Started Guide for Beginner Programmers - Getting Started Guide for Intermediate Programmers - What BitNode should I do? + Getting Started Guide for Beginner Programmers diff --git a/doc/source/guidesandtips/gettingstartedguideforbeginnerprogrammers.rst b/doc/source/guidesandtips/gettingstartedguideforbeginnerprogrammers.rst new file mode 100644 index 000000000..7d17e78b5 --- /dev/null +++ b/doc/source/guidesandtips/gettingstartedguideforbeginnerprogrammers.rst @@ -0,0 +1,47 @@ +Getting Started Guide for Beginner Programmers +============================================== + +.. note:: Note that the scripts and strategies given in this guide aren't necessarily + optimal. They're just meant to introduce you to the game and help you get + started. + +This is an introductory guide to getting started with Bitburner. It is not meant to be a +comprehensive guide for the entire game, only the early stages. If you are confused +or overwhelmed by the game, especially the programming and scripting aspects, this +guide is perfect for you! + +Note that this guide is tailored towards those with minimal programming experience. + +Introduction +------------ +Bitburner is a cyberpunk-themed incremental RPG. The player progresses by raising +their :ref:`gameplay_stats`, earning money, and :ref:`climbing the corporate ladder `. +Eventually, after reaching certain criteria, the player will begin receiving invitations +from :ref:`gameplay_factions`. Joining these factions and working for them will unlock +:ref:`gameplay_augmentations`. Purchasing and installing Augmentations provide persistent +upgrades and are necessary for progressing in the game. + +The game has a minimal story/quest-line that can be followed to reach the end of the game. +Since this guide is only about getting started with Bitburner, it will not cover the +entire "quest-line". + +First Steps +----------- +I'm going to assume you followed the introductory tutorial when you first began the game. +In this introductory tutorial you created a script called `foodnstuff.script` and ran it +on the `foodnstuff` server. Right now, we'll kill this script. There are two ways +to do this: + +1. You can go to the Terminal and enter:: + + $ kill foodnstuff.script + +2. You can go to the "Active Scripts" page (:ref:`Keyboard shortcut ` Alt + s) and + press the "Kill Script" button for `foodnstuff.script`. + +If you skipped the introductory tutorial, then ignore the part above. Instead, go to the +"Hacknet Nodes" page (:ref:`Keyboard shortcut ` Alt + h) and purchase a +Hacknet Node to start generating some passive income. + +Creating our First Scripts +-------------------------- diff --git a/src/Constants.ts b/src/Constants.ts index 8ef109394..eb36b0b1b 100644 --- a/src/Constants.ts +++ b/src/Constants.ts @@ -510,44 +510,15 @@ export let CONSTANTS: IMap = { LatestUpdate: ` - v0.44.0 - * Bladeburner Changes: - ** Reduced the amount of rank needed to earn a skill point - ** Reduced the effects of the "Reaper" and "Evasive System" skills - ** Increased the effect of the "Hyperdrive" and "Hands of Midas" skills - ** Slightly increased the rate which the skill point cost rises for almost all skills - ** The "Overlock" Skill now has a maximum level of 90 instead of 95 - ** Money earned from Contracts increased by 400% - ** Changed the way population affects success rate. Extreme populations now have less dramatic effects - ** Added two new General Actions: Diplomacy and Hyperbolic Regeneration Chamber - ** Lowered the rep and money cost of the "Blade's Simulacrum" augmentation - ** Significantly decreased the initial amount of Contracts/Operations (the "Contracts/Operations remaining" value) - ** Decreased the rate at which the amount of Contracts/Operations increases over time - ** Decreased the number of successes you need to increase the max level of a Contract/Operation - ** Increased the average number of Synthoid communities each city has - ** Reduced the amount by which a successful raid will decrease the population of a city - ** The "riots" event will now increase the chaos of a city by a greater amount - ** Significantly increased the effect that Agility and Dexterity have on action time + v0.44.1 + * Duplicate Sleeve changes: + ** You can now purchase Augmentations for your Duplicate Sleeves + ** Sleeves are now assigned to Shock Recovery task by default + ** Shock Recovery and Synchronize tasks are now twice as effective - * Added new BitNode multipliers: - ** HomeComputerRamCost - Affects how much it costs to upgrade home computer's RAM - ** DaedalusAugsRequirement - Affects how many Augmentations you need in order to get invited to Daedalus - ** FourSigmaMarketDataCost - Affects how much it costs to unlock the stock market's 4S Market Data - ** FourSigmaMarketDataApiCost - Affects how much it costs to unlock the stock market's 4S Market Data API - - * A few minor changes to BitNode multipliers across the board (mostly for the new multipliers) - * 'The Covenant' now requires 20 total Augmentations to get invited, rather than 30 - * You can now purchase permanent Duplicate Sleeves from 'The Covenant'. This requires Source-File 10, and you must be in BN-10 or after - * You can now track where all of your money comes from in the 'Stats' page - * Increased the money gained from Coding Contracts by 50% - * getCharacterInformation() function now returns the player's HP and max HP - * Bug Fix: You can no longer disconnect the enemy's connections in Hacking Missions - * Bug Fix: Duplicate Sleeve faction reputation gain is now properly affected by faction favor - * Bug Fix: After installing Augmentations, the Terminal display will now correctly show the current server as "home" - * Bug Fix: Fixed an exploit where you could change the duration of timed functions (e.g. hack, weaken) in NetscriptJS - * Bug Fix: You should now properly be able to use the ServerProfile.exe program - * Bug Fix: Prevented exploit that allowed you to accept faction invites programmatically through NetscriptJS - * Bug Fix: Faction invitations for megacorporations should now work properly + * Bug Fix: 'rm' Terminal and Netscript commands now work on non-program files that have '.exe' in the name (by Github user MasonD) + * Bug Fix: The 'Find All Valid Math Expressions' Coding Contract should now properly ignore whitespace in answers + * Bug Fix: The 'Merge Overlapping Intervals' Coding Contract should now properly accept 2D arrays when being attempted through Netscript ` } diff --git a/src/NetscriptFunctions.js b/src/NetscriptFunctions.js index 9b7e6a56a..e2e585616 100644 --- a/src/NetscriptFunctions.js +++ b/src/NetscriptFunctions.js @@ -53,9 +53,6 @@ import {StockMarket, StockSymbols, SymbolToStockMap, PositionTypes, placeOrder, cancelOrder} from "./StockMarket/StockMarket"; import { getStockmarket4SDataCost, getStockMarket4STixApiCost } from "./StockMarket/StockMarketCosts"; -import {numeralWrapper} from "./ui/numeralFormat"; -import {post} from "./ui/postToTerminal"; -import { setTimeoutRef } from "./utils/SetTimeoutRef"; import {TextFile, getTextFile, createTextFile} from "./TextFile"; import {unknownBladeburnerActionErrorMessage, @@ -68,8 +65,12 @@ import {makeRuntimeRejectMsg, netscriptDelay, runScriptFromScript} from "./NetscriptEvaluator"; import {NetscriptPort} from "./NetscriptPort"; -import Decimal from "decimal.js"; import {Page, routing} from "./ui/navigationTracking"; +import {numeralWrapper} from "./ui/numeralFormat"; +import {post} from "./ui/postToTerminal"; +import { setTimeoutRef } from "./utils/SetTimeoutRef"; +import { is2DArray } from "./utils/helpers/is2DArray"; + import {dialogBoxCreate} from "../utils/DialogBox"; import {isPowerOfTwo} from "../utils/helpers/isPowerOfTwo"; import {arrayToString} from "../utils/helpers/arrayToString"; @@ -4686,7 +4687,20 @@ function NetscriptFunctions(workerScript) { workerScript.log(`ERROR: codingcontract.getData() failed because it could find the specified contract ${fn} on server ${ip}`); return false; } - answer = String(answer); + + // Convert answer to string. If the answer is a 2D array, then we have to + // manually add brackets for the inner arrays + if (is2DArray(answer)) { + let answerComponents = []; + for (let i = 0; i < answer.length; ++i) { + answerComponents.push(["[", answer[i].toString(), "]"].join("")); + } + + answer = answerComponents.join(","); + } else { + answer = String(answer); + } + const serv = safeGetServer(ip, "codingcontract.attempt()"); if (contract.isSolution(answer)) { const reward = Player.gainCodingContractReward(contract.reward, contract.getDifficulty()); diff --git a/src/PersonObjects/Sleeve/SleeveAugmentationsUI.ts b/src/PersonObjects/Sleeve/SleeveAugmentationsUI.ts index 280b8b37b..3fc5b76a7 100644 --- a/src/PersonObjects/Sleeve/SleeveAugmentationsUI.ts +++ b/src/PersonObjects/Sleeve/SleeveAugmentationsUI.ts @@ -16,14 +16,9 @@ import { numeralWrapper } from "../../ui/numeralFormat"; import { dialogBoxCreate } from "../../../utils/DialogBox"; -import { clearEventListeners } from "../../../utils/uiHelpers/clearEventListeners"; import { createElement } from "../../../utils/uiHelpers/createElement"; import { createPopup } from "../../../utils/uiHelpers/createPopup"; import { createPopupCloseButton } from "../../../utils/uiHelpers/createPopupCloseButton"; -import { getSelectValue } from "../../../utils/uiHelpers/getSelectData"; -import { removeChildrenFromElement } from "../../../utils/uiHelpers/removeChildrenFromElement"; -import { removeElement } from "../../../utils/uiHelpers/removeElement"; -import { removeElementById } from "../../../utils/uiHelpers/removeElementById"; export function createSleevePurchaseAugsPopup(sleeve: Sleeve, p: IPlayer) { // You can only purchase Augmentations that are actually available from @@ -44,6 +39,8 @@ export function createSleevePurchaseAugsPopup(sleeve: Sleeve, p: IPlayer) { } } + // Create popup + const popupId = "purchase-sleeve-augs-popup"; // General info about buying Augmentations const info = createElement("p", { @@ -58,8 +55,10 @@ export function createSleevePurchaseAugsPopup(sleeve: Sleeve, p: IPlayer) { ].join(" "), }); - const popupId = "purchase-sleeve-augs-popup"; - const popupElems: HTMLElement[] = [info]; + // Close popup + const closeBtn = createPopupCloseButton(popupId, { innerText: "Cancel" }); + + const popupElems: HTMLElement[] = [closeBtn, info]; for (const aug of availableAugs) { const div = createElement("div", { diff --git a/src/PersonObjects/Sleeve/SleeveUI.ts b/src/PersonObjects/Sleeve/SleeveUI.ts index a9a0197fd..b8ee967a0 100644 --- a/src/PersonObjects/Sleeve/SleeveUI.ts +++ b/src/PersonObjects/Sleeve/SleeveUI.ts @@ -1,6 +1,7 @@ /** * Module for handling the Sleeve UI */ +import { createSleevePurchaseAugsPopup } from "./SleeveAugmentationsUI"; import { Sleeve } from "./Sleeve"; import { SleeveTaskType } from "./SleeveTaskTypesEnum"; import { SleeveFaq } from "./data/SleeveFaq"; @@ -277,7 +278,7 @@ function createSleeveUi(sleeve: Sleeve, allSleeves: Sleeve[]): ISleeveUIElems { display: "block", innerText: "Purchase Augmentations", clickListener: () => { - + createSleevePurchaseAugsPopup(sleeve, playerRef!); } }); elems.statsPanel.appendChild(elems.stats); diff --git a/src/data/codingcontracttypes.ts b/src/data/codingcontracttypes.ts index 3e13799db..f687d688a 100644 --- a/src/data/codingcontracttypes.ts +++ b/src/data/codingcontracttypes.ts @@ -898,7 +898,8 @@ export const codingContractTypesMetadata: ICodingContractTypeMetadata[] = [ const sanitizedPlayerAns: string = removeBracketsFromArrayString(ans); const sanitizedPlayerAnsArr: string[] = sanitizedPlayerAns.split(","); for (let i = 0; i < sanitizedPlayerAnsArr.length; ++i) { - sanitizedPlayerAnsArr[i] = removeQuotesFromString(sanitizedPlayerAnsArr[i]); + sanitizedPlayerAnsArr[i] = removeQuotesFromString(sanitizedPlayerAnsArr[i]) + .replace(/\s/g, "");; } if (num == null || num.length === 0) { diff --git a/src/ui/displayCharacterInfo.ts b/src/ui/displayCharacterInfo.ts index 4328329c0..0eb464032 100644 --- a/src/ui/displayCharacterInfo.ts +++ b/src/ui/displayCharacterInfo.ts @@ -55,7 +55,7 @@ export function displayCharacterInfo(elem: HTMLElement, p: IPlayer) { if (src.bladeburner) { parts.push(`Bladeburner: ${numeralWrapper.formatMoney(src.bladeburner)}`) }; if (src.codingcontract) { parts.push(`Coding Contracts: ${numeralWrapper.formatMoney(src.codingcontract)}`) }; if (src.work) { parts.push(`Company Work: ${numeralWrapper.formatMoney(src.work)}`) }; - if (src.corporation) { parts.push(`Corporation: ${numeralWrapper.formatMoney(src.corporation)}}`) }; + if (src.corporation) { parts.push(`Corporation: ${numeralWrapper.formatMoney(src.corporation)}`) }; if (src.crime) { parts.push(`Crimes: ${numeralWrapper.formatMoney(src.crime)}`) }; if (src.gang) { parts.push(`Gang: ${numeralWrapper.formatMoney(src.gang)}`) }; if (src.hacking) { parts.push(`Hacking: ${numeralWrapper.formatMoney(src.hacking)}`) }; diff --git a/src/utils/helpers/is2DArray.ts b/src/utils/helpers/is2DArray.ts new file mode 100644 index 000000000..f8763921b --- /dev/null +++ b/src/utils/helpers/is2DArray.ts @@ -0,0 +1,8 @@ +// Checks whether an array is a 2D array. +// For this, a 2D array is an array which contains only other arrays. +// If one element in the array is a number or string, it is NOT a 2D array +export function is2DArray(arr: any[]): boolean { + if (arr.constructor !== Array) { return false; } + + return arr.every((e) => { return e.constructor === Array; }); +} diff --git a/utils/README.md b/utils/README.md new file mode 100644 index 000000000..7a8bd8365 --- /dev/null +++ b/utils/README.md @@ -0,0 +1,4 @@ +This directory contains anything that does not represent an actual component of +the game. Instead, it contains utility classes, functions, etc. + +The `/helper` directory contains helper functions. From 18d8b2ecd43d0576e8c90b0daed7e7ddf0ea59d6 Mon Sep 17 00:00:00 2001 From: danielyxie Date: Sat, 2 Mar 2019 19:08:54 -0800 Subject: [PATCH 07/10] v0.44.1 Minor Update - Added Augs to Duplicate Sleeves and updated documentation --- doc/source/advancedgameplay/sleeves.rst | 12 +- doc/source/basicgameplay/augmentations.rst | 4 + doc/source/basicgameplay/hacking.rst | 4 + doc/source/basicgameplay/scripts.rst | 2 + doc/source/basicgameplay/terminal.rst | 10 +- ...tingstartedguideforbeginnerprogrammers.rst | 842 +++++++++- .../getBitNodeMultipliers.rst | 20 + .../getHackGrowWeakenTimes.rst | 15 + .../netscript/basicfunctions/brutessh.rst | 13 + doc/source/netscript/basicfunctions/clear.rst | 13 + .../netscript/basicfunctions/clearLog.rst | 8 + .../netscript/basicfunctions/deleteServer.rst | 14 + .../netscript/basicfunctions/disableLog.rst | 16 + .../netscript/basicfunctions/enableLog.rst | 10 + doc/source/netscript/basicfunctions/exec.rst | 34 + doc/source/netscript/basicfunctions/exit.rst | 8 + .../netscript/basicfunctions/fileExists.rst | 25 + .../netscript/basicfunctions/ftpcrack.rst | 13 + .../basicfunctions/getFavorToDonate.rst | 8 + .../netscript/basicfunctions/getGrowTime.rst | 13 + .../netscript/basicfunctions/getHackTime.rst | 13 + .../basicfunctions/getHackingLevel.rst | 8 + .../basicfunctions/getHackingMultipliers.rst | 22 + .../basicfunctions/getHacknetMultipliers.rst | 23 + .../netscript/basicfunctions/getHostname.rst | 8 + .../basicfunctions/getPortHandle.rst | 11 + .../basicfunctions/getPurchasedServerCost.rst | 16 + .../getPurchasedServerLimit.rst | 8 + .../getPurchasedServerMaxRam.rst | 8 + .../basicfunctions/getPurchasedServers.rst | 11 + .../basicfunctions/getScriptExpGain.rst | 14 + .../basicfunctions/getScriptIncome.rst | 18 + .../basicfunctions/getScriptLogs.rst | 33 + .../basicfunctions/getScriptName.rst | 8 + .../netscript/basicfunctions/getScriptRam.rst | 11 + .../getServerBaseSecurityLevel.rst | 12 + .../basicfunctions/getServerGrowth.rst | 12 + .../basicfunctions/getServerMaxMoney.rst | 9 + .../getServerMinSecurityLevel.rst | 9 + .../getServerMoneyAvailable.rst | 15 + .../getServerNumPortsRequired.rst | 9 + .../netscript/basicfunctions/getServerRam.rst | 17 + .../getServerRequiredHackingLevel.rst | 9 + .../basicfunctions/getServerSecurityLevel.rst | 10 + .../basicfunctions/getTimeSinceLastAug.rst | 8 + .../basicfunctions/getWeakenTime.rst | 13 + doc/source/netscript/basicfunctions/grow.rst | 21 + .../basicfunctions/growthAnalyze.rst | 24 + doc/source/netscript/basicfunctions/hack.rst | 22 + .../basicfunctions/hackAnalyzePercent.rst | 20 + .../basicfunctions/hackAnalyzeThreads.rst | 24 + .../netscript/basicfunctions/hackChance.rst | 11 + .../basicfunctions/hasRootAccess.rst | 15 + .../netscript/basicfunctions/httpworm.rst | 13 + .../netscript/basicfunctions/isLogEnabled.rst | 10 + .../netscript/basicfunctions/isRunning.rst | 29 + doc/source/netscript/basicfunctions/kill.rst | 29 + .../netscript/basicfunctions/killall.rst | 10 + doc/source/netscript/basicfunctions/ls.rst | 11 + .../netscript/basicfunctions/nFormat.rst | 19 + doc/source/netscript/basicfunctions/nuke.rst | 13 + doc/source/netscript/basicfunctions/peek.rst | 12 + doc/source/netscript/basicfunctions/print.rst | 9 + .../netscript/basicfunctions/prompt.rst | 10 + doc/source/netscript/basicfunctions/ps.rst | 28 + .../basicfunctions/purchaseServer.rst | 30 + doc/source/netscript/basicfunctions/read.rst | 16 + .../netscript/basicfunctions/relaysmtp.rst | 13 + doc/source/netscript/basicfunctions/rm.rst | 11 + doc/source/netscript/basicfunctions/run.rst | 33 + doc/source/netscript/basicfunctions/scan.rst | 11 + doc/source/netscript/basicfunctions/scp.rst | 29 + .../netscript/basicfunctions/scriptKill.rst | 11 + .../basicfunctions/scriptRunning.rst | 24 + .../netscript/basicfunctions/serverExists.rst | 9 + doc/source/netscript/basicfunctions/sleep.rst | 9 + doc/source/netscript/basicfunctions/spawn.rst | 20 + .../netscript/basicfunctions/sprintf.rst | 8 + .../netscript/basicfunctions/sqlinject.rst | 13 + .../netscript/basicfunctions/tprint.rst | 9 + .../netscript/basicfunctions/tryWrite.rst | 12 + .../netscript/basicfunctions/vsprintf.rst | 8 + .../netscript/basicfunctions/weaken.rst | 20 + doc/source/netscript/basicfunctions/wget.rst | 31 + doc/source/netscript/basicfunctions/write.rst | 20 + .../bladeburnerapi/getActionAutolevel.rst | 11 + .../getActionCountRemaining.rst | 12 + .../bladeburnerapi/getActionCurrentLevel.rst | 11 + .../getActionEstimatedSuccessChance.rst | 11 + .../bladeburnerapi/getActionMaxLevel.rst | 11 + .../bladeburnerapi/getActionRepGain.rst | 12 + .../bladeburnerapi/getActionTime.rst | 9 + .../bladeburnerapi/getBlackOpNames.rst | 6 + .../netscript/bladeburnerapi/getBonusTime.rst | 14 + .../netscript/bladeburnerapi/getCity.rst | 6 + .../netscript/bladeburnerapi/getCityChaos.rst | 8 + .../getCityEstimatedCommunities.rst | 9 + .../getCityEstimatedPopulation.rst | 9 + .../bladeburnerapi/getContractNames.rst | 6 + .../bladeburnerapi/getCurrentAction.rst | 14 + .../bladeburnerapi/getGeneralActionNames.rst | 6 + .../bladeburnerapi/getOperationNames.rst | 6 + .../netscript/bladeburnerapi/getRank.rst | 6 + .../bladeburnerapi/getSkillLevel.rst | 10 + .../bladeburnerapi/getSkillNames.rst | 6 + .../bladeburnerapi/getSkillPoints.rst | 6 + .../bladeburnerapi/getSkillUpgradeCost.rst | 11 + .../netscript/bladeburnerapi/getStamina.rst | 15 + .../netscript/bladeburnerapi/getTeamSize.rst | 13 + .../joinBladeburnerDivision.rst | 11 + .../bladeburnerapi/joinBladeburnerFaction.rst | 11 + .../bladeburnerapi/setActionAutolevel.rst | 10 + .../bladeburnerapi/setActionLevel.rst | 10 + .../netscript/bladeburnerapi/setTeamSize.rst | 12 + .../netscript/bladeburnerapi/startAction.rst | 10 + .../bladeburnerapi/stopBladeburnerAction.rst | 6 + .../netscript/bladeburnerapi/switchCity.rst | 10 + .../netscript/bladeburnerapi/upgradeSkill.rst | 9 + .../netscript/codingcontractapi/attempt.rst | 13 + .../codingcontractapi/getContractType.rst | 13 + .../netscript/codingcontractapi/getData.rst | 14 + .../codingcontractapi/getDescription.rst | 12 + .../getNumTriesRemaining.rst | 13 + doc/source/netscript/gangapi/ascendMember.rst | 22 + .../netscript/gangapi/canRecruitMember.rst | 6 + doc/source/netscript/gangapi/getBonusTime.rst | 13 + .../netscript/gangapi/getChanceToWinClash.rst | 9 + .../netscript/gangapi/getEquipmentCost.rst | 12 + .../netscript/gangapi/getEquipmentNames.rst | 9 + .../netscript/gangapi/getEquipmentType.rst | 16 + .../netscript/gangapi/getGangInformation.rst | 23 + .../gangapi/getMemberInformation.rst | 36 + .../netscript/gangapi/getMemberNames.rst | 8 + .../gangapi/getOtherGangInformation.rst | 26 + doc/source/netscript/gangapi/getTaskNames.rst | 8 + .../netscript/gangapi/purchaseEquipment.rst | 12 + .../netscript/gangapi/recruitMember.rst | 14 + .../netscript/gangapi/setMemberTask.rst | 12 + .../netscript/gangapi/setTerritoryWarfare.rst | 8 + .../hacknetnodeapi/getCoreUpgradeCost.rst | 12 + .../hacknetnodeapi/getLevelUpgradeCost.rst | 12 + .../netscript/hacknetnodeapi/getNodeStats.rst | 18 + .../hacknetnodeapi/getPurchaseNodeCost.rst | 6 + .../hacknetnodeapi/getRamUpgradeCost.rst | 12 + .../netscript/hacknetnodeapi/numNodes.rst | 6 + .../netscript/hacknetnodeapi/purchaseNode.rst | 11 + .../netscript/hacknetnodeapi/upgradeCore.rst | 14 + .../netscript/hacknetnodeapi/upgradeLevel.rst | 14 + .../netscript/hacknetnodeapi/upgradeRam.rst | 16 + .../netscript/netscriptadvancedfunctions.rst | 38 +- .../netscript/netscriptbladeburnerapi.rst | 388 +---- .../netscript/netscriptcodingcontractapi.rst | 74 +- doc/source/netscript/netscriptfunctions.rst | 1354 +---------------- doc/source/netscript/netscriptgangapi.rst | 271 +--- .../netscript/netscripthacknetnodeapi.rst | 168 +- doc/source/netscript/netscriptixapi.rst | 343 +---- .../netscriptsingularityfunctions.rst | 635 +------- .../singularityfunctions/applyToCompany.rst | 31 + .../checkFactionInvitations.rst | 8 + .../singularityfunctions/commitCrime.rst | 34 + .../singularityfunctions/createProgram.rst | 30 + .../singularityfunctions/donateToFaction.rst | 12 + .../getAugmentationCost.rst | 13 + .../getAugmentationPrereq.rst | 13 + .../getAugmentationsFromFaction.rst | 10 + .../getCharacterInformation.rst | 50 + .../singularityfunctions/getCompanyFavor.rst | 11 + .../getCompanyFavorGain.rst | 11 + .../singularityfunctions/getCompanyRep.rst | 11 + .../singularityfunctions/getCrimeChance.rst | 12 + .../singularityfunctions/getFactionFavor.rst | 10 + .../getFactionFavorGain.rst | 10 + .../singularityfunctions/getFactionRep.rst | 10 + .../getOwnedAugmentations.rst | 12 + .../getOwnedSourceFiles.rst | 9 + .../singularityfunctions/getStats.rst | 23 + .../getUpgradeHomeRamCost.rst | 8 + .../singularityfunctions/gymWorkout.rst | 30 + .../installAugmentations.rst | 14 + .../netscript/singularityfunctions/isBusy.rst | 9 + .../singularityfunctions/joinFaction.rst | 10 + .../purchaseAugmentation.rst | 14 + .../singularityfunctions/purchaseProgram.rst | 18 + .../singularityfunctions/purchaseTor.rst | 11 + .../singularityfunctions/stopAction.rst | 18 + .../singularityfunctions/travelToCity.rst | 20 + .../singularityfunctions/universityCourse.rst | 30 + .../singularityfunctions/upgradeHomeRam.rst | 10 + .../singularityfunctions/workForCompany.rst | 24 + .../singularityfunctions/workForFaction.rst | 32 + doc/source/netscript/tixapi/buyStock.rst | 16 + doc/source/netscript/tixapi/cancelOrder.rst | 24 + doc/source/netscript/tixapi/getOrders.rst | 68 + .../netscript/tixapi/getStockForecast.rst | 21 + .../netscript/tixapi/getStockMaxShares.rst | 11 + .../netscript/tixapi/getStockPosition.rst | 27 + doc/source/netscript/tixapi/getStockPrice.rst | 14 + .../netscript/tixapi/getStockSymbols.rst | 8 + .../netscript/tixapi/getStockVolatility.rst | 17 + doc/source/netscript/tixapi/placeOrder.rst | 26 + .../netscript/tixapi/purchase4SMarketData.rst | 11 + .../tixapi/purchase4SMarketDataTixApi.rst | 11 + doc/source/netscript/tixapi/sellShort.rst | 18 + doc/source/netscript/tixapi/sellStock.rst | 20 + doc/source/netscript/tixapi/shortStock.rst | 18 + src/Constants.ts | 231 +-- src/InteractiveTutorial.js | 3 +- src/PersonObjects/Sleeve/Sleeve.ts | 1 + .../Sleeve/SleeveAugmentationsUI.ts | 47 +- src/PersonObjects/Sleeve/SleeveUI.ts | 4 +- src/PersonObjects/Sleeve/data/SleeveFaq.ts | 17 +- src/engine.js | 92 -- src/index.html | 62 +- 213 files changed, 4024 insertions(+), 3405 deletions(-) create mode 100644 doc/source/netscript/advancedfunctions/getBitNodeMultipliers.rst create mode 100644 doc/source/netscript/advancedfunctions/getHackGrowWeakenTimes.rst create mode 100644 doc/source/netscript/basicfunctions/brutessh.rst create mode 100644 doc/source/netscript/basicfunctions/clear.rst create mode 100644 doc/source/netscript/basicfunctions/clearLog.rst create mode 100644 doc/source/netscript/basicfunctions/deleteServer.rst create mode 100644 doc/source/netscript/basicfunctions/disableLog.rst create mode 100644 doc/source/netscript/basicfunctions/enableLog.rst create mode 100644 doc/source/netscript/basicfunctions/exec.rst create mode 100644 doc/source/netscript/basicfunctions/exit.rst create mode 100644 doc/source/netscript/basicfunctions/fileExists.rst create mode 100644 doc/source/netscript/basicfunctions/ftpcrack.rst create mode 100644 doc/source/netscript/basicfunctions/getFavorToDonate.rst create mode 100644 doc/source/netscript/basicfunctions/getGrowTime.rst create mode 100644 doc/source/netscript/basicfunctions/getHackTime.rst create mode 100644 doc/source/netscript/basicfunctions/getHackingLevel.rst create mode 100644 doc/source/netscript/basicfunctions/getHackingMultipliers.rst create mode 100644 doc/source/netscript/basicfunctions/getHacknetMultipliers.rst create mode 100644 doc/source/netscript/basicfunctions/getHostname.rst create mode 100644 doc/source/netscript/basicfunctions/getPortHandle.rst create mode 100644 doc/source/netscript/basicfunctions/getPurchasedServerCost.rst create mode 100644 doc/source/netscript/basicfunctions/getPurchasedServerLimit.rst create mode 100644 doc/source/netscript/basicfunctions/getPurchasedServerMaxRam.rst create mode 100644 doc/source/netscript/basicfunctions/getPurchasedServers.rst create mode 100644 doc/source/netscript/basicfunctions/getScriptExpGain.rst create mode 100644 doc/source/netscript/basicfunctions/getScriptIncome.rst create mode 100644 doc/source/netscript/basicfunctions/getScriptLogs.rst create mode 100644 doc/source/netscript/basicfunctions/getScriptName.rst create mode 100644 doc/source/netscript/basicfunctions/getScriptRam.rst create mode 100644 doc/source/netscript/basicfunctions/getServerBaseSecurityLevel.rst create mode 100644 doc/source/netscript/basicfunctions/getServerGrowth.rst create mode 100644 doc/source/netscript/basicfunctions/getServerMaxMoney.rst create mode 100644 doc/source/netscript/basicfunctions/getServerMinSecurityLevel.rst create mode 100644 doc/source/netscript/basicfunctions/getServerMoneyAvailable.rst create mode 100644 doc/source/netscript/basicfunctions/getServerNumPortsRequired.rst create mode 100644 doc/source/netscript/basicfunctions/getServerRam.rst create mode 100644 doc/source/netscript/basicfunctions/getServerRequiredHackingLevel.rst create mode 100644 doc/source/netscript/basicfunctions/getServerSecurityLevel.rst create mode 100644 doc/source/netscript/basicfunctions/getTimeSinceLastAug.rst create mode 100644 doc/source/netscript/basicfunctions/getWeakenTime.rst create mode 100644 doc/source/netscript/basicfunctions/grow.rst create mode 100644 doc/source/netscript/basicfunctions/growthAnalyze.rst create mode 100644 doc/source/netscript/basicfunctions/hack.rst create mode 100644 doc/source/netscript/basicfunctions/hackAnalyzePercent.rst create mode 100644 doc/source/netscript/basicfunctions/hackAnalyzeThreads.rst create mode 100644 doc/source/netscript/basicfunctions/hackChance.rst create mode 100644 doc/source/netscript/basicfunctions/hasRootAccess.rst create mode 100644 doc/source/netscript/basicfunctions/httpworm.rst create mode 100644 doc/source/netscript/basicfunctions/isLogEnabled.rst create mode 100644 doc/source/netscript/basicfunctions/isRunning.rst create mode 100644 doc/source/netscript/basicfunctions/kill.rst create mode 100644 doc/source/netscript/basicfunctions/killall.rst create mode 100644 doc/source/netscript/basicfunctions/ls.rst create mode 100644 doc/source/netscript/basicfunctions/nFormat.rst create mode 100644 doc/source/netscript/basicfunctions/nuke.rst create mode 100644 doc/source/netscript/basicfunctions/peek.rst create mode 100644 doc/source/netscript/basicfunctions/print.rst create mode 100644 doc/source/netscript/basicfunctions/prompt.rst create mode 100644 doc/source/netscript/basicfunctions/ps.rst create mode 100644 doc/source/netscript/basicfunctions/purchaseServer.rst create mode 100644 doc/source/netscript/basicfunctions/read.rst create mode 100644 doc/source/netscript/basicfunctions/relaysmtp.rst create mode 100644 doc/source/netscript/basicfunctions/rm.rst create mode 100644 doc/source/netscript/basicfunctions/run.rst create mode 100644 doc/source/netscript/basicfunctions/scan.rst create mode 100644 doc/source/netscript/basicfunctions/scp.rst create mode 100644 doc/source/netscript/basicfunctions/scriptKill.rst create mode 100644 doc/source/netscript/basicfunctions/scriptRunning.rst create mode 100644 doc/source/netscript/basicfunctions/serverExists.rst create mode 100644 doc/source/netscript/basicfunctions/sleep.rst create mode 100644 doc/source/netscript/basicfunctions/spawn.rst create mode 100644 doc/source/netscript/basicfunctions/sprintf.rst create mode 100644 doc/source/netscript/basicfunctions/sqlinject.rst create mode 100644 doc/source/netscript/basicfunctions/tprint.rst create mode 100644 doc/source/netscript/basicfunctions/tryWrite.rst create mode 100644 doc/source/netscript/basicfunctions/vsprintf.rst create mode 100644 doc/source/netscript/basicfunctions/weaken.rst create mode 100644 doc/source/netscript/basicfunctions/wget.rst create mode 100644 doc/source/netscript/basicfunctions/write.rst create mode 100644 doc/source/netscript/bladeburnerapi/getActionAutolevel.rst create mode 100644 doc/source/netscript/bladeburnerapi/getActionCountRemaining.rst create mode 100644 doc/source/netscript/bladeburnerapi/getActionCurrentLevel.rst create mode 100644 doc/source/netscript/bladeburnerapi/getActionEstimatedSuccessChance.rst create mode 100644 doc/source/netscript/bladeburnerapi/getActionMaxLevel.rst create mode 100644 doc/source/netscript/bladeburnerapi/getActionRepGain.rst create mode 100644 doc/source/netscript/bladeburnerapi/getActionTime.rst create mode 100644 doc/source/netscript/bladeburnerapi/getBlackOpNames.rst create mode 100644 doc/source/netscript/bladeburnerapi/getBonusTime.rst create mode 100644 doc/source/netscript/bladeburnerapi/getCity.rst create mode 100644 doc/source/netscript/bladeburnerapi/getCityChaos.rst create mode 100644 doc/source/netscript/bladeburnerapi/getCityEstimatedCommunities.rst create mode 100644 doc/source/netscript/bladeburnerapi/getCityEstimatedPopulation.rst create mode 100644 doc/source/netscript/bladeburnerapi/getContractNames.rst create mode 100644 doc/source/netscript/bladeburnerapi/getCurrentAction.rst create mode 100644 doc/source/netscript/bladeburnerapi/getGeneralActionNames.rst create mode 100644 doc/source/netscript/bladeburnerapi/getOperationNames.rst create mode 100644 doc/source/netscript/bladeburnerapi/getRank.rst create mode 100644 doc/source/netscript/bladeburnerapi/getSkillLevel.rst create mode 100644 doc/source/netscript/bladeburnerapi/getSkillNames.rst create mode 100644 doc/source/netscript/bladeburnerapi/getSkillPoints.rst create mode 100644 doc/source/netscript/bladeburnerapi/getSkillUpgradeCost.rst create mode 100644 doc/source/netscript/bladeburnerapi/getStamina.rst create mode 100644 doc/source/netscript/bladeburnerapi/getTeamSize.rst create mode 100644 doc/source/netscript/bladeburnerapi/joinBladeburnerDivision.rst create mode 100644 doc/source/netscript/bladeburnerapi/joinBladeburnerFaction.rst create mode 100644 doc/source/netscript/bladeburnerapi/setActionAutolevel.rst create mode 100644 doc/source/netscript/bladeburnerapi/setActionLevel.rst create mode 100644 doc/source/netscript/bladeburnerapi/setTeamSize.rst create mode 100644 doc/source/netscript/bladeburnerapi/startAction.rst create mode 100644 doc/source/netscript/bladeburnerapi/stopBladeburnerAction.rst create mode 100644 doc/source/netscript/bladeburnerapi/switchCity.rst create mode 100644 doc/source/netscript/bladeburnerapi/upgradeSkill.rst create mode 100644 doc/source/netscript/codingcontractapi/attempt.rst create mode 100644 doc/source/netscript/codingcontractapi/getContractType.rst create mode 100644 doc/source/netscript/codingcontractapi/getData.rst create mode 100644 doc/source/netscript/codingcontractapi/getDescription.rst create mode 100644 doc/source/netscript/codingcontractapi/getNumTriesRemaining.rst create mode 100644 doc/source/netscript/gangapi/ascendMember.rst create mode 100644 doc/source/netscript/gangapi/canRecruitMember.rst create mode 100644 doc/source/netscript/gangapi/getBonusTime.rst create mode 100644 doc/source/netscript/gangapi/getChanceToWinClash.rst create mode 100644 doc/source/netscript/gangapi/getEquipmentCost.rst create mode 100644 doc/source/netscript/gangapi/getEquipmentNames.rst create mode 100644 doc/source/netscript/gangapi/getEquipmentType.rst create mode 100644 doc/source/netscript/gangapi/getGangInformation.rst create mode 100644 doc/source/netscript/gangapi/getMemberInformation.rst create mode 100644 doc/source/netscript/gangapi/getMemberNames.rst create mode 100644 doc/source/netscript/gangapi/getOtherGangInformation.rst create mode 100644 doc/source/netscript/gangapi/getTaskNames.rst create mode 100644 doc/source/netscript/gangapi/purchaseEquipment.rst create mode 100644 doc/source/netscript/gangapi/recruitMember.rst create mode 100644 doc/source/netscript/gangapi/setMemberTask.rst create mode 100644 doc/source/netscript/gangapi/setTerritoryWarfare.rst create mode 100644 doc/source/netscript/hacknetnodeapi/getCoreUpgradeCost.rst create mode 100644 doc/source/netscript/hacknetnodeapi/getLevelUpgradeCost.rst create mode 100644 doc/source/netscript/hacknetnodeapi/getNodeStats.rst create mode 100644 doc/source/netscript/hacknetnodeapi/getPurchaseNodeCost.rst create mode 100644 doc/source/netscript/hacknetnodeapi/getRamUpgradeCost.rst create mode 100644 doc/source/netscript/hacknetnodeapi/numNodes.rst create mode 100644 doc/source/netscript/hacknetnodeapi/purchaseNode.rst create mode 100644 doc/source/netscript/hacknetnodeapi/upgradeCore.rst create mode 100644 doc/source/netscript/hacknetnodeapi/upgradeLevel.rst create mode 100644 doc/source/netscript/hacknetnodeapi/upgradeRam.rst create mode 100644 doc/source/netscript/singularityfunctions/applyToCompany.rst create mode 100644 doc/source/netscript/singularityfunctions/checkFactionInvitations.rst create mode 100644 doc/source/netscript/singularityfunctions/commitCrime.rst create mode 100644 doc/source/netscript/singularityfunctions/createProgram.rst create mode 100644 doc/source/netscript/singularityfunctions/donateToFaction.rst create mode 100644 doc/source/netscript/singularityfunctions/getAugmentationCost.rst create mode 100644 doc/source/netscript/singularityfunctions/getAugmentationPrereq.rst create mode 100644 doc/source/netscript/singularityfunctions/getAugmentationsFromFaction.rst create mode 100644 doc/source/netscript/singularityfunctions/getCharacterInformation.rst create mode 100644 doc/source/netscript/singularityfunctions/getCompanyFavor.rst create mode 100644 doc/source/netscript/singularityfunctions/getCompanyFavorGain.rst create mode 100644 doc/source/netscript/singularityfunctions/getCompanyRep.rst create mode 100644 doc/source/netscript/singularityfunctions/getCrimeChance.rst create mode 100644 doc/source/netscript/singularityfunctions/getFactionFavor.rst create mode 100644 doc/source/netscript/singularityfunctions/getFactionFavorGain.rst create mode 100644 doc/source/netscript/singularityfunctions/getFactionRep.rst create mode 100644 doc/source/netscript/singularityfunctions/getOwnedAugmentations.rst create mode 100644 doc/source/netscript/singularityfunctions/getOwnedSourceFiles.rst create mode 100644 doc/source/netscript/singularityfunctions/getStats.rst create mode 100644 doc/source/netscript/singularityfunctions/getUpgradeHomeRamCost.rst create mode 100644 doc/source/netscript/singularityfunctions/gymWorkout.rst create mode 100644 doc/source/netscript/singularityfunctions/installAugmentations.rst create mode 100644 doc/source/netscript/singularityfunctions/isBusy.rst create mode 100644 doc/source/netscript/singularityfunctions/joinFaction.rst create mode 100644 doc/source/netscript/singularityfunctions/purchaseAugmentation.rst create mode 100644 doc/source/netscript/singularityfunctions/purchaseProgram.rst create mode 100644 doc/source/netscript/singularityfunctions/purchaseTor.rst create mode 100644 doc/source/netscript/singularityfunctions/stopAction.rst create mode 100644 doc/source/netscript/singularityfunctions/travelToCity.rst create mode 100644 doc/source/netscript/singularityfunctions/universityCourse.rst create mode 100644 doc/source/netscript/singularityfunctions/upgradeHomeRam.rst create mode 100644 doc/source/netscript/singularityfunctions/workForCompany.rst create mode 100644 doc/source/netscript/singularityfunctions/workForFaction.rst create mode 100644 doc/source/netscript/tixapi/buyStock.rst create mode 100644 doc/source/netscript/tixapi/cancelOrder.rst create mode 100644 doc/source/netscript/tixapi/getOrders.rst create mode 100644 doc/source/netscript/tixapi/getStockForecast.rst create mode 100644 doc/source/netscript/tixapi/getStockMaxShares.rst create mode 100644 doc/source/netscript/tixapi/getStockPosition.rst create mode 100644 doc/source/netscript/tixapi/getStockPrice.rst create mode 100644 doc/source/netscript/tixapi/getStockSymbols.rst create mode 100644 doc/source/netscript/tixapi/getStockVolatility.rst create mode 100644 doc/source/netscript/tixapi/placeOrder.rst create mode 100644 doc/source/netscript/tixapi/purchase4SMarketData.rst create mode 100644 doc/source/netscript/tixapi/purchase4SMarketDataTixApi.rst create mode 100644 doc/source/netscript/tixapi/sellShort.rst create mode 100644 doc/source/netscript/tixapi/sellStock.rst create mode 100644 doc/source/netscript/tixapi/shortStock.rst diff --git a/doc/source/advancedgameplay/sleeves.rst b/doc/source/advancedgameplay/sleeves.rst index 8bc9cfedd..a3db25f26 100644 --- a/doc/source/advancedgameplay/sleeves.rst +++ b/doc/source/advancedgameplay/sleeves.rst @@ -50,6 +50,16 @@ no shock. Shock affects the amount of experience earned by the sleeve. Sleeve shock slowly decreases over time. You can further increase the rate at which it decreases by assigning sleeves to the 'Shock Recovery' task. +Obtaining Duplicate Sleeves +~~~~~~~~~~~~~~~~~~~~~~~~~~~ +There are two methods of obtaining Duplicate Sleeves: + +1. Destroy BitNode-10. Each completion give you one additional Duplicate Sleeve +2. Purchase Duplicate Sleeves from :ref:`the faction The Covenant `. + This is only available in BitNodes-10 and above, and is only available after defeating + BitNode-10 at least once. Sleeves purchased this way are permanent. You can purchase + up to 5 Duplicate Sleeves from The Covenant. + Re-sleeving ^^^^^^^^^^^ Re-sleeving is the process of digitizing and transferring your consciousness into a @@ -64,4 +74,4 @@ Note that resleeving **REMOVES** all of your currently-installed Augmentations, and replaces them with the ones provided by the purchased sleeve. However, Augmentations that are purchased but not installed will **not** be removed. If you have purchased an Augmentation and then re-sleeve into a body which already has that Augmentation, -it will be removed since you cannot have duplicate Augmentations. +it will be removed since you cannot have duplicate Augmentations. diff --git a/doc/source/basicgameplay/augmentations.rst b/doc/source/basicgameplay/augmentations.rst index 483bb5168..a79dce68f 100644 --- a/doc/source/basicgameplay/augmentations.rst +++ b/doc/source/basicgameplay/augmentations.rst @@ -23,6 +23,8 @@ enough reputation in it, you will be able to purchase its Augmentations. Different Factions offer different Augmentations. Augmentations must be purchased in order to be installed, and they are fairly expensive. +.. _gameplay_augmentations_installing: + Installing Augmentations ^^^^^^^^^^^^^^^^^^^^^^^^ You will not gain the benefits of your purchased Augmentations until you @@ -57,6 +59,8 @@ Here is everything you will KEEP when you install an Augmentation: * RAM Upgrades on your home computer * World Stock Exchange account and TIX API Access +.. _gameplay_augmentations_purchasingmultiple: + Purchasing Multiple Augmentations ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ You do not have to install an Augmentation right after you purchase it. diff --git a/doc/source/basicgameplay/hacking.rst b/doc/source/basicgameplay/hacking.rst index cf7f837b8..6673ed695 100644 --- a/doc/source/basicgameplay/hacking.rst +++ b/doc/source/basicgameplay/hacking.rst @@ -49,6 +49,8 @@ on a server in order to successfully NUKE it:** Once you have enough ports opened on a server and have ran the NUKE virus to gain root access, you will be able to hack it. +.. _gameplay_hacking_generalhackingmechanics: + General Hacking Mechanics ^^^^^^^^^^^^^^^^^^^^^^^^^ When you execute the hack command, either manually through the terminal @@ -73,6 +75,8 @@ in your hacking (since you are only hacking a certain percentage). You can increase the amount of money on a server using a script and the :js:func:`grow` function in Netscript. +.. _gameplay_hacking_serversecurity: + Server Security ^^^^^^^^^^^^^^^ Each server has a security level, typically between 1 and 100. diff --git a/doc/source/basicgameplay/scripts.rst b/doc/source/basicgameplay/scripts.rst index f44919429..2eec791a5 100644 --- a/doc/source/basicgameplay/scripts.rst +++ b/doc/source/basicgameplay/scripts.rst @@ -37,6 +37,8 @@ name and the arguments that it was run with.** The arguments must be an **exact** match. This means that both the order and type of the arguments matter. +.. _gameplay_scripts_multithreadingscripts: + Multithreading scripts ^^^^^^^^^^^^^^^^^^^^^^ A script can be run with multiple threads. This is also called multithreading. diff --git a/doc/source/basicgameplay/terminal.rst b/doc/source/basicgameplay/terminal.rst index da1da41c3..c2f06d8b0 100644 --- a/doc/source/basicgameplay/terminal.rst +++ b/doc/source/basicgameplay/terminal.rst @@ -126,6 +126,8 @@ Both 'clear' and 'cls' do the same thing:: $ clear $ cls +.. _connect_terminal_command: + connect ^^^^^^^ @@ -165,8 +167,6 @@ Examples:: $ expr 3 ** 3 -Evalutes a - free ^^^^ @@ -193,6 +193,7 @@ detailed information about the Terminal command. Examples:: $ help alias $ help scan-analyze +.. _home_terminal_command: home ^^^^ @@ -346,6 +347,9 @@ scan Prints all immediately-available network connections. This will print a list of all servers that you can currently connect to using the 'connect' Terminal command. + +.. _scan_analyze_terminal_command: + scan-analyze ^^^^^^^^^^^^ @@ -364,6 +368,8 @@ The information 'scan-analyze' displays about each server includes whether or not you have root access to it, its required hacking level, the number of open ports required to run NUKE.exe on it, and how much RAM it has. +.. _scp_terminal_command: + scp ^^^ diff --git a/doc/source/guidesandtips/gettingstartedguideforbeginnerprogrammers.rst b/doc/source/guidesandtips/gettingstartedguideforbeginnerprogrammers.rst index 7d17e78b5..b2a1df285 100644 --- a/doc/source/guidesandtips/gettingstartedguideforbeginnerprogrammers.rst +++ b/doc/source/guidesandtips/gettingstartedguideforbeginnerprogrammers.rst @@ -28,20 +28,850 @@ entire "quest-line". First Steps ----------- I'm going to assume you followed the introductory tutorial when you first began the game. -In this introductory tutorial you created a script called `foodnstuff.script` and ran it -on the `foodnstuff` server. Right now, we'll kill this script. There are two ways +In this introductory tutorial you created a script called :code:`foodnstuff.script` and ran it +on the :code:`foodnstuff` server. Right now, we'll kill this script. There are two ways to do this: 1. You can go to the Terminal and enter:: $ kill foodnstuff.script -2. You can go to the "Active Scripts" page (:ref:`Keyboard shortcut ` Alt + s) and - press the "Kill Script" button for `foodnstuff.script`. +2. You can go to the :code:`Active Scripts` page (|Keyboard shortcut| Alt + s) and + press the "Kill Script" button for :code:`foodnstuff.script`. If you skipped the introductory tutorial, then ignore the part above. Instead, go to the -"Hacknet Nodes" page (:ref:`Keyboard shortcut ` Alt + h) and purchase a +:code:`Hacknet Nodes` page (|Keyboard shortcut| Alt + h) and purchase a Hacknet Node to start generating some passive income. -Creating our First Scripts +Creating our First Script +------------------------- +Now, we'll create a generic hacking script that can be used early on in the game (or throughout the +entire game, if you want). + +Before we write the script, here are some things you'll want to familiarize yourself with: + +* :ref:`gameplay_hacking_generalhackingmechanics` +* :ref:`gameplay_hacking_serversecurity` +* :js:func:`hack` +* :js:func:`grow` +* :js:func:`weaken` +* :js:func:`brutessh` +* :js:func:`nuke` + +To briefly summarize the information from the links above: Each server has a +security level that affects how difficult it is to hack. Each server also has a +certain amount of money, as well as a maximum amount of money it can hold. Hacking a +server steals a percentage of that server's money. The :js:func:`hack` Netscript function +is used to hack server. The :js:func:`grow` Netscript function is used to increase +the amount of money available on a server. The :js:func:`weaken` Netscript function is +used to decrease a server's security level. + +Now let's move on to actually creating the script. +Go to your home computer and then create a script called :code:`early-hack-template.script` by +going to Terminal and entering the following two commands:: + + $ home + $ nano early-hack-template.script + +This will take you to the script editor, which you can use to code and create +:ref:`gameplay_scripts`. It will be helpful to consult the :ref:`netscript` documentation. +Specifically, you'll want to take a look at :ref:`netscriptfunctions`. + +Enter the following code in the script editor: + +.. code:: javascript + + // Defines the "target server", which is the server + // that we're going to hack. In this case, it's "foodnstuff" + var target = "foodnstuff"; + + // Defines how much money a server should have before we hack it + // In this case, it is set to 75% of the server's max money + var moneyThresh = getServerMaxMoney(target) * 0.75; + + // Defines the maximum security level the target server can + // have. If the target's security level is higher than this, + // we'll weaken it before doing anything else + var securityThresh = getServerMinSecurityLevel(target) + 5; + + // If we have the BruteSSH.exe program, use it to open the SSH Port + // on the target server + if (fileExists("BruteSSH.exe", "home")) { + brutessh(target); + } + + // Get root access to target server + nuke(target); + + // Infinite loop that continously hacks/grows/weakens the target server + while(true) { + if (getServerSecurityLevel(target) > securityThresh) { + // If the server's security level is above our threshold, weaken it + weaken(target); + } else if (getServerMoneyAvailable(target) < moneyThresh) { + // If the server's money is less than our threshold, grow it + grow(target); + } else { + // Otherwise, hack it + hack(target); + } + } + +The script above contains comments that document what it does, but let's go through it +step-by-step anyways. + +.. code:: javascript + + var target = "foodnstuff"; + +This first command defines a string which contains our target server. That's the server +that we're going to hack. For now, it's set to `foodnstuff` because that's the only +server with a required hacking level of 1. If you want to hack a different server, +simply change this +variable to be the hostname of another server. + +.. code:: javascript + + var moneyThresh = getServerMaxMoney(target) * 0.75; + +This second command defines a numerical value representing the minimum +amount of money that must be available on the target server in order for our script +to hack it. If the money available on the target server is less than this value, +then our script will :js:func:`grow` the server rather than hacking it. +It is set to 75% of the maximum amount of money that can be available on the server. +The :js:func:`getServerMaxMoney` Netscript function is used to find this value + +.. code:: javascript + + var securityThresh = getServerMinSecurityLevel(target) + 5; + +This third command defines a numerical value representing the maximum security level +the target server can have. If the target server's security level is higher than +this value, then our script will :js:func:`weaken` the script before doing anything else. + +.. code:: javascript + + if (fileExists("BruteSSH.exe", "home")) { + brutessh(target); + } + + nuke(target); + +This section of code is used to gain root access on the target server. This is +necessary for hacking. See :ref:`here for more details `. + +.. code:: javascript + + while (true) { + if (getServerSecurityLevel(target) > securityThresh) { + // If the server's security level is above our threshold, weaken it + weaken(target); + } else if (getServerMoneyAvailable(target) < moneyThresh) { + // Otherwise, if the server's money is less than our threshold, grow it + grow(target); + } else { + // Otherwise, hack it + hack(target); + } + } + +This is the main section that drives our script. It dictates the script's logic +and carries out the hacking operations. The `while (true)` creates an infinite loop +that will continuously run the hacking logic until the the script is killed. + +Running our Scripts +------------------- +Now we want to start running our hacking script so that it can start earning us +money and experience. Our home computer only has 8GB of RAM and we'll be using it for +something else later. So instead, we'll take advantage of the RAM on other machines. + +Go to |Terminal| and enter the following command:: + + $ scan-analyze 2 + +This will show detailed information about some servers on the network. The +**network is randomized so it will be different for every person**. +Here's what mine showed at the time I made this:: + + [home ~]> scan-analyze 2 + ~~~~~~~~~~ Beginning scan-analyze ~~~~~~~~~~ + + >foodnstuff + --Root Access: NO, Required hacking skill: 1 + --Number of open ports required to NUKE: 0 + --RAM: 16 + + >sigma-cosmetics + --Root Access: NO, Required hacking skill: 5 + --Number of open ports required to NUKE: 0 + --RAM: 16 + + >joesguns + --Root Access: NO, Required hacking skill: 10 + --Number of open ports required to NUKE: 0 + --RAM: 16 + + ---->max-hardware + ------Root Access: NO, Required hacking skill: 80 + ------Number of open ports required to NUKE: 1 + ------RAM: 32 + + >hong-fang-tea + --Root Access: NO, Required hacking skill: 30 + --Number of open ports required to NUKE: 0 + --RAM: 16 + + ---->nectar-net + ------Root Access: NO, Required hacking skill: 20 + ------Number of open ports required to NUKE: 0 + ------RAM: 16 + + >harakiri-sushi + --Root Access: NO, Required hacking skill: 40 + --Number of open ports required to NUKE: 0 + --RAM: 16 + + >iron-gym + --Root Access: NO, Required hacking skill: 100 + --Number of open ports required to NUKE: 1 + --RAM: 32 + + ---->zer0 + ------Root Access: NO, Required hacking skill: 75 + ------Number of open ports required to NUKE: 1 + ------RAM: 32 + + ---->CSEC + ------Root Access: NO, Required hacking skill: 54 + ------Number of open ports required to NUKE: 1 + ------RAM: 8 + +Take note of the following servers: + +* |foodnstuff| +* |sigma-cosmetics| +* |joesguns| +* |nectar-net| +* |hong-fang-tea| +* |harakiri-sushi| + +All of these servers have 16GB of RAM. Furthermore, all of these servers do not require +any open ports in order to NUKE. In other words, we can gain root access to all of these +servers and then run scripts on them. + +First, let's determine how many threads of our hacking script we can run. +:ref:`Read more about multithreading scripts here ` +The script we wrote +uses 2.6GB of RAM. You can check this using the following |Terminal| command:: + + $ mem early-hack-template.script + +This means we can run 6 threads on a 16GB server. Now, to run our scripts on all of these +servers, we have to do the following: + +1. Use the :ref:`scp_terminal_command` |Terminal| command to copy our script to each server. +2. Use the :ref:`connect_terminal_command` |Terminal| command to connect to a server. +3. Use the :ref:`run_terminal_command` |Terminal| command to run the `NUKE.exe` program and + gain root access. +4. Use the :ref:`run_terminal_command` |Terminal| command again to run our script. +5. Repeat steps 2-4 for each server. + +Here's the sequence of |Terminal| commands I used in order to achieve this:: + + $ home + $ scp early-hack-template.script foodnstuff + $ scp early-hack-template.script sigma-cosmetics + $ scp early-hack-template.script joesguns + $ scp early-hack-template.script nectar-net + $ scp early-hack-template.script hong-fang-tea + $ scp early-hack-template.script harakiri-sushi + $ connect foodnstuff + $ run NUKE.exe + $ run early-hack-template.script -t 6 + $ home + $ connect sigma-cosmetics + $ run NUKE.exe + $ run early-hack-template.script -t 6 + $ home + $ connect joesguns + $ run NUKE.exe + $ run early-hack-template.script -t 6 + $ home + $ connect hong-fang-tea + $ run NUKE.exe + $ run early-hack-template.script -t 6 + $ home + $ connect harakiri-sushi + $ run NUKE.exe + $ run early-hack-template.script -t 6 + $ home + $ connect hong-fang-tea + $ connect nectar-net + $ run NUKE.exe + $ run early-hack-template.script -t 6 + +.. note:: + + Pressing the :code:`Tab` key in the middle of a Terminal command will attempt to + auto-complete the command. For example, if you type in :code:`scp ea` and then + hit :code:`Tab`, the rest of the script's name should automatically be filled in. + This works for most commands in the game! + +The :ref:`home_terminal_command` |Terminal| command is used to connect to the home +computer. When running our scripts with the :code:`run early-hack-template.script -t 6` +command, the :code:`-t 6` specifies that the script should be run with 6 threads. + +Note that the |nectar-net| server isn't in the home computer's immediate network. +This means you can't directly connect to it from home. You will have to search for it +inside the network. The results of the `scan-analyze 2` command we ran before +will show where it is. In my case, I could connect to it by going from +`hong-fang-tea -> nectar-net`. However, this will probably be different for you. + +After running all of these |Terminal| commands, our scripts are now up and running. +These will earn money and hacking experience over time. These gains will be +really slow right now, but they will increase once our hacking skill rises and +we start running more scripts. + +Increasing Hacking Level +------------------------ +There are many servers besides |foodnstuff| that can be hacked, but they have +higher required hacking levels. Therefore, we should raise our hacking level. Not only +will this let us hack more servers, but it will also increase the effectiveness of our hacking +against |foodnstuff|. + +The easiest way to train your hacking level is to visit Rothman University. You can do this by +clicking the `City` tab on the left-hand navigation menu, or you can use the +:ref:`keyboard shortcut ` Alt + w. Rothman University should be one of the buttons +near the top. Click the button to go to the location. + +Once you go to Rothman University, you should see a screen with several options. These +options describe different courses you can take. You should click the first button, which +says: "Study Computer Science (free)". + +After you click the button, you will start studying and earning hacking experience. While you +are doing this, you cannot interact with any other part of the game until you click the button +that says "Stop taking course". + +Right now, we want a hacking level of 10. You need approximately 174 hacking experience to reach +level 10. You can check how much hacking experience you have by clicking the `Stats` tab +on the left-hand navigation menu, or by using |Keyboard shortcut| Alt + c. +Since studying at Rothman University earns you 1 experience per second, this will take +174 seconds, or approximately 3 minutes. Feel free to do something in the meantime! + +Editing our Hacking Script -------------------------- +Now that we have a hacking level of 10, we can hack the :code:`joesguns` server. This server +will be slightly more profitable than :code:`foodnstuff`. Therefore, we want to change our hacking +script to target :code:`joesguns` instead of :code:`foodnstuff`. + +Go to |Terminal| and edit the hacking script by entering:: + + $ home + $ nano early-hack-template.script + +At the top of the script, change the `target` variable to be `joesguns`: + +.. code:: javascript + + var target = "joesguns"; + +Note that this will **NOT** affect any instances of the script that are already running. +This will only affect instances of the script that are ran from this point forward. + +Creating a New Script to Purchase New Servers +--------------------------------------------- +Next, we're going to create a script that automatically purchases additional servers. These +servers will be used to run many scripts. Running this script will initially be very +expensive since purchasing a server costs money, but it will pay off in the long run. + +In order to create this script, you should familiarize yourself with the following +Netscript functions: + +* :js:func:`purchaseServer` +* :js:func:`getPurchasedServerCost` +* :js:func:`getPurchasedServerLimit` +* :js:func:`getServerMoneyAvailable` +* :js:func:`scp` +* :js:func:`exec` + +Create the script by going to |Terminal| and typing:: + + $ home + $ nano purchase-server-8gb.script + +Paste the following code into the script editor: + +.. code:: javascript + + // How much RAM each purchased server will have. In this case, it'll + // be 8GB. + var ram = 8; + + // Iterator we'll use for our loop + var i = 0; + + // Continuously try to purchase servers until we've reached the maximum + // amount of servers + while (i < getPurchasedServerLimit()) { + // Check if we have enough money to purchase a server + if (getServerMoneyAvailable("home") > getPurchasedServerCost(ram)) { + // If we have enough money, then: + // 1. Purchase the server + // 2. Copy our hacking script onto the newly-purchased server + // 3. Run our hacking script on the newly-purchased server with 3 threads + // 4. Increment our iterator to indicate that we've bought a new server + var hostname = purchaseServer("pserv-" + i, ram); + scp("early-hack-template.script", hostname); + exec("early-hack-template.script", hostname, 3); + ++i; + } + } + +This code uses a while loop to purchase the maximum amount of servers using the +:js:func:`purchaseServer` Netscript function. Each of these servers will have +8GB of RAM, as defined in the :code:`ram` variable. Note that the script uses the command +:code:`getServerMoneyAvailable("home")` to get the amount of money you currently have. +This is then used to check if you can afford to purchase a server. + +Whenever the script purchases a new server, it uses the :js:func:`scp` function to copy +our script onto that new server, and then it uses the :js:func:`exec` function to +execute it on that server. + +To run this script, go to |Terminal| and type:: + + $ run purchase-server-8gb.script + +This purchase will continuously run until it has purchased the maximum number of servers. +When this happens, it'll mean that you have a bunch of new servers that are all running +hacking scripts against the :code:`joesguns` server! + +.. note:: + + The reason we're using so many scripts to hack :code:`joesguns` instead of targeting other + servers is because it's more effective. This early in the game, we don't have enough RAM + to efficiently hack multiple targets, and trying to do so would be slow as we'd be spread + too thin. You should definitely do this later on, though! + +Note that purchasing a server is fairly expensive, and purchasing the maximum amount of +servers even more so. At the time of writing this guide, the script above requires +$11 million in order to finish purchasing all of the 8GB servers. +Therefore, we need to find additional ways to make money to speed +up the process! These are covered in the next section. + +Additional Sources of Income +---------------------------- +There are other ways to gain money in this game besides scripts & hacking. + +Hacknet Nodes +^^^^^^^^^^^^^ +If you completed the introductory tutorial, you were already introduced to this method: Hacknet Nodes. +Once you have enough money, you can start upgrading your Hacknet Nodes in order to increase +your passive income stream. This is completely optional. Since each Hacknet Node upgrade +takes a certain amount of time to "pay itself off", it may not necessarily be in your best +interest to use these. + +Nonetheless, Hacknet Nodes are a good source of income early in the game, although +their effectiveness tapers off later on. If you do wind up purchasing and upgrading Hacknet Nodes, +I would suggest only upgrading their levels for now. I wouldn't bother with RAM and Core +upgrades until later on. + +Crime +^^^^^ +The best source of income right now is from :ref:`committing crimes `. +This is because it not only gives you a large amount of money, but it also raises your +hacking level. To commit crimes, click on the :code:`City` tab on the left-hand +navigation menu or use the |Keyboard shortcut| Alt + w. +Then, click on the link that says :code:`The Slums`. + +In the Slums, you can attempt to commit a variety of crimes, each of which gives certain +types of experience and money if successful. See :ref:`gameplay_crimes` for more details. + +.. note:: + + You are not always successful when you attempt to commit a crime. Nothing bad happens + if you fail a crime, but you won't earn any money and the experience gained will be + reduced. Raising your stats improves your chance of successfully committing a crime. + +Right now, the best option is the :code:`Rob Store` crime. This takes 60 seconds to attempt +and gives $400k if successful. I suggest this crime because you don't have to click or check +in too often since it takes a whole minute to attempt. Furthermore, it gives hacking experience, +which is very important right now. + +Alternatively, you can also use the :code:`Shoplift` crime. This takes 2 seconds to attempt +and gives $15k if successful. This crime is slightly easier and is more profitable +than :code:`Rob Store`, but it requires constant clicking and it doesn't give +hacking experience. + +Work for a Company +^^^^^^^^^^^^^^^^^^ +If you don't want to constantly check in on the game to commit crimes, there's another option +that's much more passive: working for a :ref:`company `. +This will not be nearly as profitable as crimes, but it's completely passive. + +Go to the :code:`City` tab on the left-hand navigation menu and then go to +:code:`Joe's Guns`. At :code:`Joe's Guns`, there will be an option that says +:code:`Apply to be an Employee`. Click this to get the job. Then, a new option +will appear that simply says :code:`Work`. Click this to start working. +Working at :code:`Joe's Guns` earns $110 per second and also grants some experience +for every stat except hacking. + +Working for a company is completely passive. However, you will not be able to do anything +else in the game while you work. You can cancel working at any time. You'll notice that +cancelling your work early causes you to lose out on some reputation gains, but +you shouldn't worry about this. Company reputation isn't important right now. + +Once your hacking hits level 75, you can visit :code:`Carmichael Security` in the city +and get a software job there. This job offers higher pay and also earns you +hacking experience. + +There are many more companies in the |City tab| that offer more pay and also more gameplay +features. Feel free to explore! + +After you Purchase your New Servers +----------------------------------- +After you've made a total of $11 million, your automatic server-purchasing script should +finish running. This will free up some RAM on your home computer. We don't want this RAM +to go to waste, so we'll make use of it. Go to |Terminal| and enter the following commands:: + + $ home + $ run early-hack-template.script -t 3 + +Reaching a Hacking Level of 50 +------------------------------ +Once you reach a hacking level of 50, two new important parts of the game open up. + +Creating your first program: BruteSSH.exe +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ +On the left-hand navigation menu you will notice a :code:`Create Programs` tab with a +red notification icon. This indicates that there are programs available to be created. +Click on that tab (or use |Keyboard shortcut| Alt + p) and you'll see a +list of all the programs you can currently create. Hovering over a program will give a +brief description of its function. Simply click on a program to start creating it. + +Right now, the program we want to create is :code:`BruteSSH.exe`. This program is used +to open up SSH ports on servers. This will allow you to hack more servers, +as many servers in the game require a certain number of opened ports in order for +:code:`NUKE.exe` to gain root access. + +When you are creating a program, you cannot interact with any other part of the game. +Feel free to cancel your work on creating a program at any time, as your progress will +be saved and can be picked back up later. :code:`BruteSSH.exe` takes about +10 minutes to complete. + +Optional: Create AutoLink.exe +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ +On the :code:`Create Programs` page, you will notice another program you can create +called :code:`AutoLink.exe`. If you don't mind waiting another 10-15 minutes, you should +go ahead and create this program. It makes it much less tedious to connect to other servers, +but it's not necessary for progressing. + +Joining your first faction: CyberSec +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ +Shortly after you reached level 50 hacking, you should have received a message that +said this:: + + Message received from unknown sender: + + We've been watching you. Your skills are very impressive. But you're wasting + your talents. If you join us, you can put your skills to good use and change + the world for the better. If you join us, we can unlock your full potential. + But first, you must pass our test. Find and hack our server using the Terminal. + + -CyberSec + + This message was saved as csec-test.msg onto your home computer. + +If you didn't, or if you accidentally closed it, that's okay! Messages get saved onto +your home computer. Enter the following |Terminal| commands to view the message:: + + $ home + $ cat csec-test.msg + +This message is part of the game's main "quest-line". It is a message from the +|CyberSec faction| that is asking you to pass their test. +Passing their test is simple, you just have to find their server and hack it through +the |Terminal|. Their server is called :code:`CSEC`. +To do this, we'll use the :ref:`scan_analyze_terminal_command` +Terminal command, just like we did before:: + + $ home + $ scan-analyze 2 + +This will show you the network for all servers that are up to 2 "nodes" away from +your home computer. Remember that the network is randomly generated so it'll look +different for everyone. Here's the relevant part of my :code:`scan-analyze` results:: + + >iron-gym + --Root Access: NO, Required hacking skill: 100 + --Number of open ports required to NUKE: 1 + --RAM: 32 + + ---->zer0 + ------Root Access: NO, Required hacking skill: 75 + ------Number of open ports required to NUKE: 1 + ------RAM: 32 + + ---->CSEC + ------Root Access: NO, Required hacking skill: 54 + ------Number of open ports required to NUKE: 1 + ------RAM: 8 + +This tells me that I can reach :code:`CSEC` by going through :code:`iron-gym`:: + + $ connect iron-gym + $ connect CSEC + +.. note:: + + If you created the :code:`AutoLink.exe` program earlier, then there is an easier + method of connecting to :code:`CSEC`. You'll notice that in the :code:`scan-analyze` + results, all of the server hostnames are white and underlined. You can simply + click one of the server hostnames in order to connect to it. So, simply click + :code:`CSEC`! + +.. note:: + + Make sure you notice the required hacking skill for the :code:`CSEC` server. + This is a random value between 51 and 60. Although you receive the message + from CSEC once you hit 50 hacking, you cannot actually pass their test + until your hacking is high enough to hack their server. + +After you are connected to the :code:`CSEC` server, you can hack it. Note that this +server requires one open port in order to gain root access. We can open the SSH port +using the :code:`BruteSSH.exe` program we created earlier. In |Terminal|:: + + $ run BruteSSH.exe + $ run NUKE.exe + $ hack + +Keep hacking the server until you are successful. After you successfully hack it, you should +receive a faction invitation from |CyberSec| shortly afterwards. Accept it. If you accidentally +reject the invitation, that's okay. Just go to the :code:`Factions` tab +(|Keyboard shortcut| Alt + f) and you should see an option that lets you +accept the invitation. + +Congrats! You just joined your first faction. Don't worry about doing anything +with this faction yet, we can come back to it later. + +Using Additional Servers to Hack Joesguns +----------------------------------------- +Once you have the |BruteSSH| program, you will be able to gain root access +to several additional servers. These servers have more RAM that you can use to +run scripts. We'll use the RAM on these servers to run more scripts that target +:code:`joesguns`. + +Copying our Scripts +^^^^^^^^^^^^^^^^^^^ +The server's we'll be using to run our scripts are: + +* :code:`neo-net` +* :code:`zer0` +* :code:`max-hardware` +* :code:`iron-gym` + +All of these servers have 32GB of RAM. You can use the |Terminal| command +:code:`scan-analyze 3` to see for yourself. To copy our hacking scripts onto these servers, +go to |Terminal| and run:: + + $ home + $ scp early-hack-template.script neo-net + $ scp early-hack-template.script zer0 + $ scp early-hack-template.script max-hardware + $ scp early-hack-template.script iron-gym + +Since each of these servers has 32GB of RAM, we can run our hacking script with 12 threads +on each server. By now, you should know how to connect to servers. So find and connect to +each of the servers above using the :code:`scan-analyze 3` |Terminal| command. Then, use +following |Terminal| command to run our hacking +script with 12 threads:: + + $ run early-hack-template.script -t 12 + +Remember that if you have the |AutoLink| program, you can simply click on the hostname of a server +after running :ref:`scan_analyze_terminal_command` to connect to it. + +Profiting from Scripts & Gaining Reputation with CyberSec +--------------------------------------------------------- +Now it's time to play the waiting game. It will take some time for your scripts to start +earning money. Remember that most of your scripts are targeting |joesguns|. It will take a +bit for them to :js:func:`grow` and :js:func:`weaken` the server to the appropriate values +before they start hacking it. Once they do, however, the scripts will be very profitable. + +.. note:: + + For reference, in about two hours after starting my first script, my scripts had a + production rate of $20k per second and had earned a total of $70 million. + (You can see these stats on the :code:`Active Scripts` tab). + + After another 15 minutes, the production rate had increased to $25k per second + and the scripts had made an additional $55 million. + + Your results will vary based on how fast you earned money from crime/working/hacknet nodes, + but this will hopefully give you a good indication of how much the scripts can earn. + +In the meantime, we are going to be gaining reputation with the |CyberSec faction|. +Go to the |Factions tab| on the left-hand +navigation menu, and from there select |CyberSec|. In the middle of +the page there should be a button for :code:`Hacking Contracts`. +Click it to start earning reputation for the |CyberSec| faction (as well +as some hacking experience). The higher your hacking level, the more reputation you +will gain. Note that while you are working for a faction, you cannot interact with +the rest of the game in any way. You can cancel your faction work at any time +with no penalty. + +Purchasing Upgrades and Augmentations +------------------------------------- +As I mentioned before, within 1-2 hours I had earned over $200 million. Now, it's time +to spend all of this money on some persistent upgrades to help progress! + +Upgrading RAM on Home computer +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ +The most important thing to upgrade right now is the RAM on your home computer. This +will allow you to run more scripts. + +To upgrade your RAM, go to the |City tab| and visit the company |Alpha Enterprises|. +There will be an option that says :code:`Purchase additional RAM for Home Computer`. +Click it and follow the dialog box to upgrade your RAM. + +I recommend getting your home computer's RAM to *at least* 128GB. Getting it even +higher would be better. + +Purchasing your First Augmentations +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ +Once you get ~1000 reputation with the |CyberSec faction|, you can purchase +your first :ref:`Augmentation ` from them. + +To do this, go to the |Factions tab| on the left-hand navigation menu +(|Keyboard shortcut| Alt + f) and select |CyberSec|. There is an button +near the bottom that says :code:`Purchase Augmentations`. This will bring up a +page that displays all of the Augmentations available from |CyberSec|. Some of them +may be locked right now. To unlock these, you will need to earn more +reputation with |CyberSec|. + +Augmentations give persistent upgrades in the form of multipliers. These aren't very +powerful early in the game because the multipliers are small. However, the effects +of Augmentations stack multiplicatively **with each other**, so as you continue to install +many Augmentations their effects will increase significantly. + +Because of this, I would recommend investing more in RAM upgrades for your home computer rather +than Augmentations early on. Having enough RAM to run many scripts will allow you to make +much more money, and then you can come back later on and get all these Augmentations. + +Right now, I suggest purchasing at the very least the :code:`Neurotrainer I` Augmentation from +|CyberSec|. If you have the money to spare, I would also suggest getting :code:`BitWire` and +several levels of the :code:`NeuroFlux Governor` Augmentations. Note that each time +you purchase an Augmentation, +:ref:`the price of purchasing another increases by 90% `, +so make sure you buy the most expensive Augmentation first. Don't worry, once you choose to +install Augmentations, their prices will reset back to their original values. + +Next Steps +---------- +That's the end of the walkthrough portion of this guide! You should continue to explore +what the game has to offer. There's quite a few features that aren't covered or mentioned +in this guide, and even more that get unlocked as you continue to play! + +Also, check out the :ref:`netscript` documentation to see what it has to offer. Writing +scripts to perform and automate various tasks is where most of the fun in the game comes +from (in my opinion)! + +The following are a few things you may want to consider doing in the near future. + +Installing Augmentations (and Resetting) +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ +If you've purchased any :ref:`gameplay_augmentations`, you'll need to install them before you +actually gain their effects. Installing Augmentations is the game's "soft-reset" or "prestige" +mechanic. You can :ref:`read more details about it here `. + +To install your Augmentations, click the |Augmentations tab| on the left-hand navigation +menu (|Keyboard shortcut| Alt + a). You will see a list of all of the Augmentations +you have purchased. Below that, you will see a button that says :code:`Install Augmentations`. +Be warned, after clicking this there is no way to undo it (unless you load an earlier save). + +Automating the Script Startup Process +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ +Whenever you install Augmentations, all of your scripts are killed and you'll have to +re-run them. Doing this every time you install Augmentations would be very tedious and annoying, +so you should write a script to automate the process. Here's a simple example for a +startup script. Feel free to adjust it to your liking. + +.. code:: javascript + + // Array of all servers that don't need any ports opened + // to gain root access. These have 16 GB of RAM + var servers0Port = ["foodnstuff", + "sigma-cosmetics", + "joesguns", + "nectar-net", + "hong-fang-tea", + "harakiri-sushi"]; + + // Array of all servers that only need 1 port opened + // to gain root access. These have 32 GB of RAM + var servers1Port = ["neo-net", + "zer0", + "max-hardware", + "iron-gym"]; + + // Copy our scripts onto each server that requires 0 ports + // to gain root access. Then use nuke() to gain admin access and + // run the scripts. + for (var i = 0; i < servers0Port.length; ++i) { + var serv = servers0Port[i]; + + scp("early-hack-template.script", serv); + nuke(serv); + exec("early-hack-template.script", serv, 6); + } + + // Wait until we acquire the "BruteSSH.exe" program + while (!fileExists("BruteSSH.exe")) { + sleep(60000); + } + + // Copy our scripts onto each server that requires 1 port + // to gain root access. Then use brutessh() and nuke() + // to gain admin access and run the scripts. + for (var i = 0; i < servers1Port.length; ++i) { + var serv = servers1Port[i]; + + scp("early-hack-template.script", serv); + brutessh(serv); + nuke(serv); + exec("early-hack-template.script", serv, 12); + } + +Random Tips +----------- +* Early on in the game, it's better to spend your money on upgrading RAM and purchasing + new servers rather than spending it on Augmentations +* The more money available on a server, the more effective the :js:func:`hack` and + :js:func:`grow` Netscript functions will be. This is because both of these functions + use percentages rather than flat values. :js:func:`hack` steals a percentage of a server's + total available money, and :js:func:`grow` increases a server's money by X%. +* There is a limit to how much money can exist on a server. This value is different for each + server. The :js:func:`getServerMaxMoney` function will tell you this maximum value. +* At this stage in the game, your combat stats (strength, defense, etc.) are not nearly + as useful as your hacking stat. Do not invest too much time or money into gaining combat + stat exp. + + + +.. Substitution definitions +.. |Alpha Enterprises| replace:: :code:`Alpha Enterprises` +.. |Augmentations tab| replace:: :code:`Augmentations` tab +.. |AutoLink| replace:: :code:`NUKE.exe` +.. |BruteSSH| replace:: :code:`BruteSSH.exe` +.. |City tab| replace:: :code:`City` tab +.. |CyberSec| replace:: :code:`CyberSec` +.. |CyberSec faction| replace:: :code:`CyberSec` :ref:`faction ` +.. |Factions tab| replace:: :code:`Factions` tab +.. |Keyboard shortcut| replace:: :ref:`Keyboard shortcut ` +.. |NUKE| replace:: :code:`NUKE.exe` +.. |Terminal| replace:: :code:`Terminal` +.. |foodnstuff| replace:: :code:`foodnstuff` +.. |harakiri-sushi| replace:: :code:`harakiri-sushi` +.. |hong-fang-tea| replace:: :code:`hong-fang-tea` +.. |joesguns| replace:: :code:`joesguns` +.. |nectar-net| replace:: :code:`nectar-net` +.. |sigma-cosmetics| replace:: :code:`sigma-cosmetics` diff --git a/doc/source/netscript/advancedfunctions/getBitNodeMultipliers.rst b/doc/source/netscript/advancedfunctions/getBitNodeMultipliers.rst new file mode 100644 index 000000000..9ac808ee1 --- /dev/null +++ b/doc/source/netscript/advancedfunctions/getBitNodeMultipliers.rst @@ -0,0 +1,20 @@ +getBitNodeMultipliers() Netscript Function +========================================== + +.. js:function:: getBitNodeMultipliers() + + Returns an object containing the current BitNode multipliers. This function requires Source-File 5 in order + to run. The multipliers are returned in decimal forms (e.g. 1.5 instead of 150%). The multipliers represent + the difference between the current BitNode and the original BitNode (BitNode-1). For example, if the + *CrimeMoney* multiplier has a value of 0.1, then that means that committing crimes in the current BitNode + will only give 10% of the money you would have received in BitNode-1. + + The structure of the returned object is subject to change as BitNode multipliers get added to the game. + Refer to the `source code here `_ + to see the name of the BitNode multipliers. + + Example:: + + mults = getBitNodeMultipliers(); + print(mults.ServerMaxMoney); + print(mults.HackExpGain); diff --git a/doc/source/netscript/advancedfunctions/getHackGrowWeakenTimes.rst b/doc/source/netscript/advancedfunctions/getHackGrowWeakenTimes.rst new file mode 100644 index 000000000..d07130126 --- /dev/null +++ b/doc/source/netscript/advancedfunctions/getHackGrowWeakenTimes.rst @@ -0,0 +1,15 @@ +getHackTime(), getGrowTime(), & getWeakenTime() +=============================================== + +The :js:func:`getHackTime`, :js:func:`getGrowTime`, and :js:func:`getWeakenTime` +all take an additional third optional parameter for specifying a specific intelligence +level to see how that would affect the hack/grow/weaken times. This parameter +defaults to your current intelligence level. + +(Intelligence is unlocked after obtaining Source-File 5). + +The function signatures are then:: + + getHackTime(hostname/ip[, hackLvl=current level, intLvl=current level]) + getGrowTime(hostname/ip[, hackLvl=current level, intLvl=current level]) + getWeakenTime(hostname/ip[, hackLvl=current level, intLvl=current level]) diff --git a/doc/source/netscript/basicfunctions/brutessh.rst b/doc/source/netscript/basicfunctions/brutessh.rst new file mode 100644 index 000000000..0b90ef061 --- /dev/null +++ b/doc/source/netscript/basicfunctions/brutessh.rst @@ -0,0 +1,13 @@ +brutessh() Netscript Function +============================= + +.. js:function:: brutessh(hostname/ip) + + :param string hostname/ip: IP or hostname of the target server + :RAM cost: 0 GB + + Runs the BruteSSH.exe program on the target server. BruteSSH.exe must exist on your home computer. + + Example:: + + brutessh("foodnstuff"); diff --git a/doc/source/netscript/basicfunctions/clear.rst b/doc/source/netscript/basicfunctions/clear.rst new file mode 100644 index 000000000..13f5393eb --- /dev/null +++ b/doc/source/netscript/basicfunctions/clear.rst @@ -0,0 +1,13 @@ +clear() Netscript Function +========================== + +.. js:function:: clear(port/fn) + + :param string/number port/fn: Port or text file to clear + :RAM cost: 1 GB + + This function is used to clear data in a `Netscript Ports `_ or a text file. + + If the *port/fn* argument is a number between 1 and 20, then it specifies a port and will clear it (deleting all data from the underlying queue). + + If the *port/fn* argument is a string, then it specifies the name of a text file (.txt) and will delete all data from that text file. diff --git a/doc/source/netscript/basicfunctions/clearLog.rst b/doc/source/netscript/basicfunctions/clearLog.rst new file mode 100644 index 000000000..96bcc694f --- /dev/null +++ b/doc/source/netscript/basicfunctions/clearLog.rst @@ -0,0 +1,8 @@ +clearLog() Netscript Function +============================= + +.. js:function:: clearLog() + + :RAM cost: 0 GB + + Clears the script's logs diff --git a/doc/source/netscript/basicfunctions/deleteServer.rst b/doc/source/netscript/basicfunctions/deleteServer.rst new file mode 100644 index 000000000..e1c400f62 --- /dev/null +++ b/doc/source/netscript/basicfunctions/deleteServer.rst @@ -0,0 +1,14 @@ +deleteServer() Netscript Function +================================= + +.. js:function:: deleteServer(hostname) + + :param string hostname: Hostname of the server to delete + :RAM cost: 2.25 GB + + Deletes one of your purchased servers, which is specified by its hostname. + + The *hostname* argument can be any data type, but it will be converted to a string. Whitespace is automatically removed from + the string. This function will not delete a server that still has scripts running on it. + + Returns true if successful, and false otherwise. diff --git a/doc/source/netscript/basicfunctions/disableLog.rst b/doc/source/netscript/basicfunctions/disableLog.rst new file mode 100644 index 000000000..dba73c24d --- /dev/null +++ b/doc/source/netscript/basicfunctions/disableLog.rst @@ -0,0 +1,16 @@ +disableLog() Netscript Function +=============================== + +.. js:function:: disableLog(fn) + + :param string fn: Name of function for which to disable logging + :RAM cost: 0 GB + + Disables logging for the given function. Logging can be disabled for + all functions by passing 'ALL' as the argument. + + Note that this does not completely remove all logging functionality. + This only stops a function from logging + when the function is successful. If the function fails, it will still log the reason for failure. + + Notable functions that cannot have their logs disabled: run, exec, exit diff --git a/doc/source/netscript/basicfunctions/enableLog.rst b/doc/source/netscript/basicfunctions/enableLog.rst new file mode 100644 index 000000000..46b445d53 --- /dev/null +++ b/doc/source/netscript/basicfunctions/enableLog.rst @@ -0,0 +1,10 @@ +enableLog() Netscript Function +============================= + +.. js:function:: enableLog(fn) + + :param string fn: Name of function for which to enable logging + :RAM cost: 0 GB + + Re-enables logging for the given function. If 'ALL' is passed into this function + as an argument, then it will revert the effects of disableLog('ALL') diff --git a/doc/source/netscript/basicfunctions/exec.rst b/doc/source/netscript/basicfunctions/exec.rst new file mode 100644 index 000000000..95967d933 --- /dev/null +++ b/doc/source/netscript/basicfunctions/exec.rst @@ -0,0 +1,34 @@ +exec() Netscript Function +========================= + +.. js:function:: exec(script, hostname/ip, [numThreads=1], [args...]) + + :param string script: Filename of script to execute + :param string hostname/ip: IP or hostname of the 'target server' on which to execute the script + :param number numThreads: Optional thread count for new script. Set to 1 by default. Will be rounded to nearest integer + :param args...: + Additional arguments to pass into the new script that is being run. Note that if any arguments are being + passed into the new script, then the third argument *numThreads* must be filled in with a value. + :RAM cost: 1.3 GB + + Run a script as a separate process on a specified server. This is similar to the *run* function except + that it can be used to run a script on any server, instead of just the current server. + + Returns true if the script is successfully started, and false otherwise. + + Running this function with a *numThreads* argument of 0 will return false without running the script. + However, running this function with a negative *numThreads* argument will cause a runtime error. + + The simplest way to use the *exec* command is to call it with just the script name and the target server. + The following example will try to run *generic-hack.script* on the *foodnstuff* server:: + + exec("generic-hack.script", "foodnstuff"); + + The following example will try to run the script *generic-hack.script* on the *joesguns* server with 10 threads:: + + exec("generic-hack.script", "joesguns", 10); + + This last example will try to run the script *foo.script* on the *foodnstuff* server with 5 threads. It will also pass + the number 1 and the string "test" in as arguments to the script:: + + exec("foo.script", "foodnstuff", 5, 1, "test"); diff --git a/doc/source/netscript/basicfunctions/exit.rst b/doc/source/netscript/basicfunctions/exit.rst new file mode 100644 index 000000000..64b8f4c44 --- /dev/null +++ b/doc/source/netscript/basicfunctions/exit.rst @@ -0,0 +1,8 @@ +exit() Netscript Function +========================= + +.. js:function:: exit() + + :RAM cost: 0 GB + + Terminates the current script immediately diff --git a/doc/source/netscript/basicfunctions/fileExists.rst b/doc/source/netscript/basicfunctions/fileExists.rst new file mode 100644 index 000000000..a14478a94 --- /dev/null +++ b/doc/source/netscript/basicfunctions/fileExists.rst @@ -0,0 +1,25 @@ +fileExists() Netscript Function +=============================== + +.. js:function:: fileExists(filename, [hostname/ip]) + + :param string filename: Filename of file to check + :param string hostname/ip: + Hostname or IP of target server. This is optional. If it is not specified then the + function will use the current server as the target server + :RAM cost: 0.1 GB + + Returns a boolean indicating whether the specified file exists on the target server. The filename + for scripts is case-sensitive, but for other types of files it is not. For example, *fileExists("brutessh.exe")* + will work fine, even though the actual program is named "BruteSSH.exe". + + If the *hostname/ip* argument is omitted, then the function will search through the current server (the server + running the script that calls this function) for the file. + + Examples:: + + fileExists("foo.script", "foodnstuff"); + fileExists("ftpcrack.exe"); + + The first example above will return true if the script named *foo.script* exists on the *foodnstuff* server, and false otherwise. + The second example above will return true if the current server contains the *FTPCrack.exe* program, and false otherwise. diff --git a/doc/source/netscript/basicfunctions/ftpcrack.rst b/doc/source/netscript/basicfunctions/ftpcrack.rst new file mode 100644 index 000000000..c3301a155 --- /dev/null +++ b/doc/source/netscript/basicfunctions/ftpcrack.rst @@ -0,0 +1,13 @@ +ftpcrack() Netscript Function +============================= + +.. js:function:: ftpcrack(hostname/ip) + + :param string hostname/ip: IP or hostname of the target server + :RAM cost: 0 GB + + Runs the FTPCrack.exe program on the target server. FTPCrack.exe must exist on your home computer. + + Example:: + + ftpcrack("foodnstuff"); diff --git a/doc/source/netscript/basicfunctions/getFavorToDonate.rst b/doc/source/netscript/basicfunctions/getFavorToDonate.rst new file mode 100644 index 000000000..8c9feb203 --- /dev/null +++ b/doc/source/netscript/basicfunctions/getFavorToDonate.rst @@ -0,0 +1,8 @@ +getFavorToDonate() Netscript Function +===================================== + +.. js:function:: getFavorToDonate() + + :RAM cost: 0.1 GB + + Returns the amount of Faction favor required to be able to donate to a faction. diff --git a/doc/source/netscript/basicfunctions/getGrowTime.rst b/doc/source/netscript/basicfunctions/getGrowTime.rst new file mode 100644 index 000000000..adaf4f842 --- /dev/null +++ b/doc/source/netscript/basicfunctions/getGrowTime.rst @@ -0,0 +1,13 @@ +getGrowTime() Netscript Function +================================ + +.. js:function:: getGrowTime(hostname/ip[, hackLvl=current level]) + + :param string hostname/ip: Hostname or IP of target server + :param number hackLvl: Optional hacking level for the calculation. Defaults to player's current hacking level + :RAM cost: 0.05 GB + + Returns the amount of time in seconds it takes to execute the *grow()* Netscript function on the target server. + + The function takes in an optional *hackLvl* parameter that can be specified + to see what the grow time would be at different hacking levels. diff --git a/doc/source/netscript/basicfunctions/getHackTime.rst b/doc/source/netscript/basicfunctions/getHackTime.rst new file mode 100644 index 000000000..1372111e3 --- /dev/null +++ b/doc/source/netscript/basicfunctions/getHackTime.rst @@ -0,0 +1,13 @@ +getHackTime() Netscript Function +================================ + +.. js:function:: getHackTime(hostname/ip[, hackLvl=current level]) + + :param string hostname/ip: Hostname or IP of target server + :param number hackLvl: Optional hacking level for the calculation. Defaults to player's current hacking level + :RAM cost: 0.05 GB + + Returns the amount of time in seconds it takes to execute the *hack()* Netscript function on the target server. + + The function takes in an optional *hackLvl* parameter that can be specified + to see what the hack time would be at different hacking levels. diff --git a/doc/source/netscript/basicfunctions/getHackingLevel.rst b/doc/source/netscript/basicfunctions/getHackingLevel.rst new file mode 100644 index 000000000..f87e324b6 --- /dev/null +++ b/doc/source/netscript/basicfunctions/getHackingLevel.rst @@ -0,0 +1,8 @@ +getHackingLevel() Netscript Function +==================================== + +.. js:function:: getHackingLevel() + + :RAM cost: 0.05 GB + + Returns the player's current hacking level diff --git a/doc/source/netscript/basicfunctions/getHackingMultipliers.rst b/doc/source/netscript/basicfunctions/getHackingMultipliers.rst new file mode 100644 index 000000000..37032122e --- /dev/null +++ b/doc/source/netscript/basicfunctions/getHackingMultipliers.rst @@ -0,0 +1,22 @@ +getHackingMultipliers() Netscript Function +========================================== + +.. js:function:: getHackingMultipliers() + + :RAM cost: 4 GB + + Returns an object containing the Player's hacking related multipliers. These multipliers are + returned in decimal forms, not percentages (e.g. 1.5 instead of 150%). The object has the following structure:: + + { + chance: Player's hacking chance multiplier, + speed: Player's hacking speed multiplier, + money: Player's hacking money stolen multiplier, + growth: Player's hacking growth multiplier + } + + Example of how this can be used:: + + mults = getHackingMultipliers(); + print(mults.chance); + print(mults.growth); diff --git a/doc/source/netscript/basicfunctions/getHacknetMultipliers.rst b/doc/source/netscript/basicfunctions/getHacknetMultipliers.rst new file mode 100644 index 000000000..d37eda610 --- /dev/null +++ b/doc/source/netscript/basicfunctions/getHacknetMultipliers.rst @@ -0,0 +1,23 @@ +getHacknetMultipliers() Netscript Function +========================================== + +.. js:function:: getHacknetMultipliers() + + :RAM cost: 4 GB + + Returns an object containing the Player's hacknet related multipliers. These multipliers are + returned in decimal forms, not percentages (e.g. 1.5 instead of 150%). The object has the following structure:: + + { + production: Player's hacknet production multiplier, + purchaseCost: Player's hacknet purchase cost multiplier, + ramCost: Player's hacknet ram cost multiplier, + coreCost: Player's hacknet core cost multiplier, + levelCost: Player's hacknet level cost multiplier + } + + Example of how this can be used:: + + mults = getHacknetMultipliers(); + print(mults.production); + print(mults.purchaseCost); diff --git a/doc/source/netscript/basicfunctions/getHostname.rst b/doc/source/netscript/basicfunctions/getHostname.rst new file mode 100644 index 000000000..6e14224cd --- /dev/null +++ b/doc/source/netscript/basicfunctions/getHostname.rst @@ -0,0 +1,8 @@ +getHostname() Netscript Function +================================ + +.. js:function:: getHostname() + + :RAM cost: 0.05 GB + + Returns a string with the hostname of the server that the script is running on diff --git a/doc/source/netscript/basicfunctions/getPortHandle.rst b/doc/source/netscript/basicfunctions/getPortHandle.rst new file mode 100644 index 000000000..a0a6ed05c --- /dev/null +++ b/doc/source/netscript/basicfunctions/getPortHandle.rst @@ -0,0 +1,11 @@ +getPortHandle() Netscript Function +================================== + +.. js:function:: getPortHandle(port) + + :param number port: Port number + :RAM cost: 10 GB + + Get a handle to a Netscript Port. See more details here: :ref:`netscript_ports` + + **WARNING:** Port Handles only work in :ref:`netscriptjs`. They will not work in :ref:`netscript1`. diff --git a/doc/source/netscript/basicfunctions/getPurchasedServerCost.rst b/doc/source/netscript/basicfunctions/getPurchasedServerCost.rst new file mode 100644 index 000000000..764baa309 --- /dev/null +++ b/doc/source/netscript/basicfunctions/getPurchasedServerCost.rst @@ -0,0 +1,16 @@ +getPurchasedServerCost() Netscript Function +=========================================== + +.. js:function:: getPurchasedServerCost(ram) + + :RAM cost: 0.25 GB + + :param number ram: Amount of RAM of a potential purchased server. Must be a power of 2 (2, 4, 8, 16, etc.). Maximum value of 1048576 (2^20) + + Returns the cost to purchase a server with the specified amount of *ram*. + + Examples:: + + for (i = 1; i <= 20; i++) { + tprint(i + " -- " + getPurchasedServerCost(Math.pow(2, i))); + } diff --git a/doc/source/netscript/basicfunctions/getPurchasedServerLimit.rst b/doc/source/netscript/basicfunctions/getPurchasedServerLimit.rst new file mode 100644 index 000000000..7784203dc --- /dev/null +++ b/doc/source/netscript/basicfunctions/getPurchasedServerLimit.rst @@ -0,0 +1,8 @@ +getPurchasedServerLimit() Netscript Function +============================================ + +.. js:function:: getPurchasedServerLimit() + + :RAM cost: 0.05 GB + + Returns the maximum number of servers you can purchase diff --git a/doc/source/netscript/basicfunctions/getPurchasedServerMaxRam.rst b/doc/source/netscript/basicfunctions/getPurchasedServerMaxRam.rst new file mode 100644 index 000000000..844e7ba18 --- /dev/null +++ b/doc/source/netscript/basicfunctions/getPurchasedServerMaxRam.rst @@ -0,0 +1,8 @@ +getPurchasedServerMaxRam() Netscript Function +============================================= + +.. js:function:: getPurchasedServerMaxRam() + + :RAM cost: 0.05 GB + + Returns the maximum RAM that a purchased server can have diff --git a/doc/source/netscript/basicfunctions/getPurchasedServers.rst b/doc/source/netscript/basicfunctions/getPurchasedServers.rst new file mode 100644 index 000000000..89b3cfedc --- /dev/null +++ b/doc/source/netscript/basicfunctions/getPurchasedServers.rst @@ -0,0 +1,11 @@ +getPurchasedServers() Netscript Function +======================================== + +.. js:function:: getPurchasedServers([hostname=true]) + + :param boolean hostname: + Specifies whether hostnames or IP addresses should be returned. If it's true then hostnames will be returned, and if false + then IPs will be returned. If this argument is omitted then it is true by default + :RAM cost: 2.25 GB + + Returns an array with either the hostnames or IPs of all of the servers you have purchased. diff --git a/doc/source/netscript/basicfunctions/getScriptExpGain.rst b/doc/source/netscript/basicfunctions/getScriptExpGain.rst new file mode 100644 index 000000000..65f6f9c86 --- /dev/null +++ b/doc/source/netscript/basicfunctions/getScriptExpGain.rst @@ -0,0 +1,14 @@ +getScriptExpGain() Netscript Function +===================================== + +.. js:function:: getScriptExpGain([scriptname], [hostname/ip], [args...]) + + :param string scriptname: Filename of script + :param string hostname/ip: Server on which script is running + :param args...: Arguments that the script is running with + :RAM cost: 0.1 GB + + Returns the amount of hacking experience the specified script generates while online (when the game is open, does not apply for offline experience gains). + Remember that a script is uniquely identified by both its name and its arguments. + + This function can also return the total experience gain rate of all of your active scripts by running the function with no arguments. diff --git a/doc/source/netscript/basicfunctions/getScriptIncome.rst b/doc/source/netscript/basicfunctions/getScriptIncome.rst new file mode 100644 index 000000000..e3496c015 --- /dev/null +++ b/doc/source/netscript/basicfunctions/getScriptIncome.rst @@ -0,0 +1,18 @@ +getScriptIncome() Netscript Function +==================================== + +.. js:function:: getScriptIncome([scriptname], [hostname/ip], [args...]) + + :param string scriptname: Filename of script + :param string hostname/ip: Server on which script is running + :param args...: Arguments that the script is running with + :RAM cost: 0.1 GB + + Returns the amount of income the specified script generates while online (when the game is open, does not apply for offline income). + Remember that a script is uniquely identified by both its name and its arguments. So for example if you ran a script with the arguments + "foodnstuff" and "5" then in order to use this function to get that script's income you must specify those same arguments in the same order + in this function call. + + This function can also be called with no arguments. If called with no arguments, then this function will return an array of two values. The + first value is the total income ($ / second) of all of your active scripts (scripts that are currently running on any server). The second value + is the total income ($ / second) that you've earned from scripts since you last installed Augmentations. diff --git a/doc/source/netscript/basicfunctions/getScriptLogs.rst b/doc/source/netscript/basicfunctions/getScriptLogs.rst new file mode 100644 index 000000000..ac721c4d3 --- /dev/null +++ b/doc/source/netscript/basicfunctions/getScriptLogs.rst @@ -0,0 +1,33 @@ +getScriptLogs() Netscript Function +================================== + +.. js:function:: getScriptLogs([fn], [hostname/ip=current ip], [args...]) + + :param string fn: Optional. Filename of script to get logs from. + :param string ip: Optional. IP or hostname of the server that the script is on + :param args...: Arguments to identify which scripts to get logs for + :RAM cost: 0 GB + + Returns a script's logs. The logs are returned as an array, where each + line is an element in the array. The most recently logged line is at the + end of the array. + + Note that there is a maximum number of lines that a script stores in its logs. + This is configurable in the game's options. + + If the function is called with no arguments, it will return the current script's logs. + + Otherwise, the `fn`, `hostname/ip,` and `args...` arguments can be used to get the logs + from another script. Remember that scripts are uniquely identified by both + their names and arguments. + + Examples:: + + // Get logs from foo.script on the current server that was run with no args + getScriptLogs("foo.script"); + + // Get logs from foo.script on the foodnstuff server that was run with no args + getScriptLogs("foo.script", "foodnstuff"); + + // Get logs from foo.script on the foodnstuff server that was run with the arguments [1, "test"] + getScriptLogs("foo.script", "foodnstuff", 1, "test"); diff --git a/doc/source/netscript/basicfunctions/getScriptName.rst b/doc/source/netscript/basicfunctions/getScriptName.rst new file mode 100644 index 000000000..cc7de660b --- /dev/null +++ b/doc/source/netscript/basicfunctions/getScriptName.rst @@ -0,0 +1,8 @@ +getScriptName() Netscript Function +================================== + +.. js:function:: getScriptName() + + :RAM cost: 0 GB + + Returns the current script name diff --git a/doc/source/netscript/basicfunctions/getScriptRam.rst b/doc/source/netscript/basicfunctions/getScriptRam.rst new file mode 100644 index 000000000..373599c37 --- /dev/null +++ b/doc/source/netscript/basicfunctions/getScriptRam.rst @@ -0,0 +1,11 @@ +getScriptRam() Netscript Function +=========================== + +.. js:function:: getScriptRam(scriptname[, hostname/ip]) + + :param string scriptname: Filename of script. This is case-sensitive. + :param string hostname/ip: Hostname or IP of target server the script is located on. This is optional, If it is not specified then the function will set the current server as the target server. + :RAM cost: 0.1 GB + + Returns the amount of RAM required to run the specified script on the target server. Returns + 0 if the script does not exist. diff --git a/doc/source/netscript/basicfunctions/getServerBaseSecurityLevel.rst b/doc/source/netscript/basicfunctions/getServerBaseSecurityLevel.rst new file mode 100644 index 000000000..162ee4c38 --- /dev/null +++ b/doc/source/netscript/basicfunctions/getServerBaseSecurityLevel.rst @@ -0,0 +1,12 @@ +getServerBaseSecurityLevel() Netscript Function +=============================================== + +.. js:function:: getServerBaseSecurityLevel(hostname/ip) + + :param string hostname/ip: Hostname or IP of target server + :RAM cost: 0.1 GB + + Returns the base security level of the target server. This is the security level that the server starts out with. + This is different than *getServerSecurityLevel()* because *getServerSecurityLevel()* returns the current + security level of a server, which can constantly change due to *hack()*, *grow()*, and *weaken()*, calls on that + server. The base security level will stay the same until you reset by installing an Augmentation(s). diff --git a/doc/source/netscript/basicfunctions/getServerGrowth.rst b/doc/source/netscript/basicfunctions/getServerGrowth.rst new file mode 100644 index 000000000..8fbdd3d45 --- /dev/null +++ b/doc/source/netscript/basicfunctions/getServerGrowth.rst @@ -0,0 +1,12 @@ +getServerGrowth() Netscript Function +==================================== + +.. js:function:: getServerGrowth(hostname/ip) + + :param string hostname/ip: Hostname or IP of target server + :RAM cost: 0.1 GB + + Returns the server's instrinsic "growth parameter". This growth parameter is a number + between 1 and 100 that represents how quickly the server's money grows. This parameter affects the + percentage by which the server's money is increased when using the *grow()* function. A higher + growth parameter will result in a higher percentage increase from *grow()*. diff --git a/doc/source/netscript/basicfunctions/getServerMaxMoney.rst b/doc/source/netscript/basicfunctions/getServerMaxMoney.rst new file mode 100644 index 000000000..c9feeb8be --- /dev/null +++ b/doc/source/netscript/basicfunctions/getServerMaxMoney.rst @@ -0,0 +1,9 @@ +getServerMaxMoney() Netscript Function +====================================== + +.. js:function:: getServerMaxMoney(hostname/ip) + + :param string hostname/ip: Hostname or IP of target server + :RAM cost: 0.1 GB + + Returns the maximum amount of money that can be available on a server diff --git a/doc/source/netscript/basicfunctions/getServerMinSecurityLevel.rst b/doc/source/netscript/basicfunctions/getServerMinSecurityLevel.rst new file mode 100644 index 000000000..7036c0f7e --- /dev/null +++ b/doc/source/netscript/basicfunctions/getServerMinSecurityLevel.rst @@ -0,0 +1,9 @@ +getServerMinSecurityLevel() Netscript Function +============================================== + +.. js:function:: getServerMinSecurityLevel(hostname/ip) + + :param string hostname/ip: Hostname or IP of target server + :RAM cost: 0.1 GB + + Returns the minimum security level of the target server diff --git a/doc/source/netscript/basicfunctions/getServerMoneyAvailable.rst b/doc/source/netscript/basicfunctions/getServerMoneyAvailable.rst new file mode 100644 index 000000000..55b909396 --- /dev/null +++ b/doc/source/netscript/basicfunctions/getServerMoneyAvailable.rst @@ -0,0 +1,15 @@ +getServerMoneyAvailable() Netscript Function +============================================ + +.. js:function:: getServerMoneyAvailable(hostname/ip) + + :param string hostname/ip: Hostname or IP of target server + :RAM cost: 0.1 GB + + Returns the amount of money available on a server. **Running this function on the home computer will return + the player's money.** + + Example:: + + getServerMoneyAvailable("foodnstuff"); + getServerMoneyAvailable("home"); //Returns player's money diff --git a/doc/source/netscript/basicfunctions/getServerNumPortsRequired.rst b/doc/source/netscript/basicfunctions/getServerNumPortsRequired.rst new file mode 100644 index 000000000..1f05a6141 --- /dev/null +++ b/doc/source/netscript/basicfunctions/getServerNumPortsRequired.rst @@ -0,0 +1,9 @@ +getServerNumPortsRequired() Netscript Function +============================================== + +.. js:function:: getServerNumPortsRequired(hostname/ip) + + :param string hostname/ip: Hostname or IP of target server + :RAM cost: 0.1 GB + + Returns the number of open ports required to successfully run NUKE.exe on the specified server. diff --git a/doc/source/netscript/basicfunctions/getServerRam.rst b/doc/source/netscript/basicfunctions/getServerRam.rst new file mode 100644 index 000000000..a8a8cb200 --- /dev/null +++ b/doc/source/netscript/basicfunctions/getServerRam.rst @@ -0,0 +1,17 @@ +getServerRam() Netscript Function +================================= + +.. js:function:: getServerRam(hostname/ip) + + :param string hostname/ip: Hostname or IP of target server + :RAM cost: 0.1 GB + + Returns an array with two elements that gives information about a server's memory (RAM). The first + element in the array is the amount of RAM that the server has total (in GB). The second element in + the array is the amount of RAM that is currently being used on the server (in GB). + + Example:: + + res = getServerRam("helios"); + totalRam = res[0]; + ramUsed = res[1]; diff --git a/doc/source/netscript/basicfunctions/getServerRequiredHackingLevel.rst b/doc/source/netscript/basicfunctions/getServerRequiredHackingLevel.rst new file mode 100644 index 000000000..122c5b41e --- /dev/null +++ b/doc/source/netscript/basicfunctions/getServerRequiredHackingLevel.rst @@ -0,0 +1,9 @@ +getServerRequiredHackingLevel() Netscript Function +================================================== + +.. js:function:: getServerRequiredHackingLevel(hostname/ip) + + :param string hostname/ip: Hostname or IP of target server + :RAM cost: 0.1 GB + + Returns the required hacking level of the target server diff --git a/doc/source/netscript/basicfunctions/getServerSecurityLevel.rst b/doc/source/netscript/basicfunctions/getServerSecurityLevel.rst new file mode 100644 index 000000000..a2c612170 --- /dev/null +++ b/doc/source/netscript/basicfunctions/getServerSecurityLevel.rst @@ -0,0 +1,10 @@ +getServerSecurityLevel() Netscript Function +=========================================== + +.. js:function:: getServerSecurityLevel(hostname/ip) + + :param string hostname/ip: Hostname or IP of target server + :RAM cost: 0.1 GB + + Returns the security level of the target server. A server's security level is denoted by a number, typically + between 1 and 100 (but it can go above 100). diff --git a/doc/source/netscript/basicfunctions/getTimeSinceLastAug.rst b/doc/source/netscript/basicfunctions/getTimeSinceLastAug.rst new file mode 100644 index 000000000..b81d48482 --- /dev/null +++ b/doc/source/netscript/basicfunctions/getTimeSinceLastAug.rst @@ -0,0 +1,8 @@ +getTimeSinceLastAug() Netscript Function +======================================== + +.. js:function:: getTimeSinceLastAug() + + :RAM cost: 0.05 GB + + Returns the amount of time in milliseconds that have passed since you last installed Augmentations diff --git a/doc/source/netscript/basicfunctions/getWeakenTime.rst b/doc/source/netscript/basicfunctions/getWeakenTime.rst new file mode 100644 index 000000000..e13e7408e --- /dev/null +++ b/doc/source/netscript/basicfunctions/getWeakenTime.rst @@ -0,0 +1,13 @@ +getWeakenTime() Netscript Function +================================== + +.. js:function:: getWeakenTime(hostname/ip[, hackLvl=current level]) + + :param string hostname/ip: Hostname or IP of target server + :param number hackLvl: Optional hacking level for the calculation. Defaults to player's current hacking level + :RAM cost: 0.05 GB + + Returns the amount of time in seconds it takes to execute the *weaken()* Netscript function on the target server. + + The function takes in an optional *hackLvl* parameter that can be specified + to see what the weaken time would be at different hacking levels. diff --git a/doc/source/netscript/basicfunctions/grow.rst b/doc/source/netscript/basicfunctions/grow.rst new file mode 100644 index 000000000..9ab819462 --- /dev/null +++ b/doc/source/netscript/basicfunctions/grow.rst @@ -0,0 +1,21 @@ +grow() Netscript Function +========================= + +.. js:function:: grow(hostname/ip) + + :param string hostname/ip: IP or hostname of the target server to grow + :returns: The number by which the money on the server was multiplied for the growth + :RAM cost: 0.15 GB + + Use your hacking skills to increase the amount of money available on a server. The runtime for this command depends on your hacking + level and the target server's security level. When grow() completes, the money available on a target server will be increased by a + certain, fixed percentage. This percentage is determined by the target server's growth rate (which varies between servers) and security level. + Generally, higher-level servers have higher growth rates. The getServerGrowth() function can be used to obtain a server's growth rate. + + Like hack(), grow() can be called on any server, regardless of where the script is running. The grow() command requires + root access to the target server, but there is no required hacking level to run the command. It also raises the security level + of the target server by 0.004. + + Example:: + + grow("foodnstuff"); diff --git a/doc/source/netscript/basicfunctions/growthAnalyze.rst b/doc/source/netscript/basicfunctions/growthAnalyze.rst new file mode 100644 index 000000000..4414dbcfa --- /dev/null +++ b/doc/source/netscript/basicfunctions/growthAnalyze.rst @@ -0,0 +1,24 @@ +growthAnalyze() Netscript Function +================================== + +.. js:function:: growthAnalyze(hostname/ip, growthAmount) + + :param string hostname/ip: IP or hostname of server to analyze + :param number growthAmount: Multiplicative factor by which the server is grown. Decimal form. + :returns: The amount of grow() calls needed to grow the specified server by the specified amount + :RAM cost: 1 GB + + This function returns the number of "growths" needed in order to increase the amount + of money available on the specified server by the specified amount. + + The specified amount is multiplicative and is in decimal form, not percentage. + + For example, if you want to determine how many `grow()` calls you need + to double the amount of money on `foodnstuff`, you would use:: + + growthAnalyze("foodnstuff", 2); + + If this returns 100, then this means you need to call `grow()` 100 times + in order to double the money (or once with 100 threads). + + **Warning**: The value returned by this function isn't necessarily a whole number. diff --git a/doc/source/netscript/basicfunctions/hack.rst b/doc/source/netscript/basicfunctions/hack.rst new file mode 100644 index 000000000..a62c85a3d --- /dev/null +++ b/doc/source/netscript/basicfunctions/hack.rst @@ -0,0 +1,22 @@ +hack() Netscript Function +========================= + +.. js:function:: hack(hostname/ip) + + :param string hostname/ip: IP or hostname of the target server to hack + :returns: The amount of money stolen if the hack is successful, and zero otherwise + :RAM cost: 0.1 GB + + Function that is used to try and hack servers to steal money and gain hacking experience. The runtime for this command depends + on your hacking level and the target server's security level. In order to hack a server you must first gain root access + to that server and also have the required hacking level. + + A script can hack a server from anywhere. It does not need to be running on the same server to hack that server. For example, + you can create a script that hacks the 'foodnstuff' server and run that script on any server in the game. + + A successful hack() on a server will raise that server's security level by 0.002. + + Example:: + + hack("foodnstuff"); + hack("10.1.2.3"); diff --git a/doc/source/netscript/basicfunctions/hackAnalyzePercent.rst b/doc/source/netscript/basicfunctions/hackAnalyzePercent.rst new file mode 100644 index 000000000..b0a64236c --- /dev/null +++ b/doc/source/netscript/basicfunctions/hackAnalyzePercent.rst @@ -0,0 +1,20 @@ +hackAnalyzePercent() Netscript Function +======================================= + +.. js:function:: hackAnalyzePercent(hostname/ip) + + :param string hostname/ip: IP or hostname of target server + :returns: The percentage of money you will steal from the target server with a single hack + :RAM cost: 1 GB + + Returns the percentage of the specified server's money you will steal with a + single hack. This value is returned in **percentage form, not decimal (Netscript + functions typically return in decimal form, but not this one).** + + For example, assume the following returns 1:: + + hackAnalyzePercent("foodnstuff"); + + This means that if hack the `foodnstuff` server, then you will steal 1% of its + total money. If you `hack()` using N threads, then you will steal N% of its total + money. diff --git a/doc/source/netscript/basicfunctions/hackAnalyzeThreads.rst b/doc/source/netscript/basicfunctions/hackAnalyzeThreads.rst new file mode 100644 index 000000000..97ae1803c --- /dev/null +++ b/doc/source/netscript/basicfunctions/hackAnalyzeThreads.rst @@ -0,0 +1,24 @@ +hackAnalyzeThreads() Netscript Function +======================================= + +.. js:function:: hackAnalyzeThreads(hostname/ip, hackAmount) + + :param string hostname/ip: IP or hostname of server to analyze + :param number hackAmount: Amount of money you want to hack from the server + :returns: The number of threads needed to hack() the server for *hackAmount* money + :RAM cost: 1 GB + + This function returns the number of script threads you need when running + the `hack()` command to steal the specified amount of money from the target server. + + If `hackAmount` is less than zero or greater than the amount of money available + on the server, then this function returns -1. + + For example, let's say the `foodnstuff` server has $10m and you run:: + + hackAnalyzeThreads("foodnstuff", 1e6); + + If this function returns 50, this means that if your next `hack()` call + is run on a script with 50 threads, it will steal $1m from the `foodnstuff` server. + + **Warning**: The value returned by this function isn't necessarily a whole number. diff --git a/doc/source/netscript/basicfunctions/hackChance.rst b/doc/source/netscript/basicfunctions/hackChance.rst new file mode 100644 index 000000000..714b8ef44 --- /dev/null +++ b/doc/source/netscript/basicfunctions/hackChance.rst @@ -0,0 +1,11 @@ +hackChance() Netscript Function +=============================== + +.. js:function:: hackChance(hostname/ip) + + :param string hostname/ip: IP or hostname of target server + :returns: The chance you have of successfully hacking the target server + :RAM cost: 1 GB + + Returns the chance you have of successfully hacking the specified server. This + returned value is in decimal form, not percentage. diff --git a/doc/source/netscript/basicfunctions/hasRootAccess.rst b/doc/source/netscript/basicfunctions/hasRootAccess.rst new file mode 100644 index 000000000..ee561eb7b --- /dev/null +++ b/doc/source/netscript/basicfunctions/hasRootAccess.rst @@ -0,0 +1,15 @@ +hasRootAccess() Netscript Function +================================== + +.. js:function:: hasRootAccess(hostname/ip) + + :param string hostname/ip: Hostname or IP of the target server + :RAM cost: 0.05 GB + + Returns a boolean indicating whether or not the player has root access to the specified target server. + + Example:: + + if (hasRootAccess("foodnstuff") == false) { + nuke("foodnstuff"); + } diff --git a/doc/source/netscript/basicfunctions/httpworm.rst b/doc/source/netscript/basicfunctions/httpworm.rst new file mode 100644 index 000000000..599c25ca8 --- /dev/null +++ b/doc/source/netscript/basicfunctions/httpworm.rst @@ -0,0 +1,13 @@ +httpworm() Netscript Function +============================= + +.. js:function:: httpworm(hostname/ip) + + :param string hostname/ip: IP or hostname of the target server + :RAM cost: 0 GB + + Runs the HTTPWorm.exe program on the target server. HTTPWorm.exe must exist on your home computer. + + Example:: + + httpworm("foodnstuff"); diff --git a/doc/source/netscript/basicfunctions/isLogEnabled.rst b/doc/source/netscript/basicfunctions/isLogEnabled.rst new file mode 100644 index 000000000..8967b4cf7 --- /dev/null +++ b/doc/source/netscript/basicfunctions/isLogEnabled.rst @@ -0,0 +1,10 @@ +isLogEnabled() Netscript Function +================================= + +.. js:function:: isLogEnabled(fn) + + :param string fn: Name of function to check + :RAM cost: 0 GB + + Returns a boolean indicating whether or not logging is enabled for that + function (or 'ALL') diff --git a/doc/source/netscript/basicfunctions/isRunning.rst b/doc/source/netscript/basicfunctions/isRunning.rst new file mode 100644 index 000000000..41afdd46a --- /dev/null +++ b/doc/source/netscript/basicfunctions/isRunning.rst @@ -0,0 +1,29 @@ +isRunning() Netscript Function +============================== + +.. js:function:: isRunning(filename, hostname/ip, [args...]) + + :param string filename: Filename of script to check. This is case-sensitive. + :param string hostname/ip: Hostname or IP of target server + :param args...: Arguments to specify/identify which scripts to search for + :RAM cost: 0.1 GB + + Returns a boolean indicating whether the specified script is running on the target server. Remember that a script is + uniquely identified by both its name and its arguments. + + **Examples:** + + In this first example below, the function call will return true if there is a script named *foo.script* with no arguments + running on the *foodnstuff* server, and false otherwise:: + + isRunning("foo.script", "foodnstuff"); + + In this second example below, the function call will return true if there is a script named *foo.script* with no arguments + running on the current server, and false otherwise:: + + isRunning("foo.script", getHostname()); + + In this next example below, the function call will return true if there is a script named *foo.script* running with the arguments + 1, 5, and "test" (in that order) on the *joesguns* server, and false otherwise:: + + isRunning("foo.script", "joesguns", 1, 5, "test"); diff --git a/doc/source/netscript/basicfunctions/kill.rst b/doc/source/netscript/basicfunctions/kill.rst new file mode 100644 index 000000000..3c3579836 --- /dev/null +++ b/doc/source/netscript/basicfunctions/kill.rst @@ -0,0 +1,29 @@ +kill() Netscript Function +========================= + +.. js:function:: kill(script, hostname/ip, [args...]) + + :param string script: Filename of the script to kill + :param string hostname/ip: IP or hostname of the server on which to kill the script + :param args...: Arguments to identify which script to kill + :RAM cost: 0.5 GB + + Kills the script on the target server specified by the script's name and arguments. Remember that scripts + are uniquely identified by both their name and arguments. For example, if *foo.script* is run with the argument 1, then this + is not the same as *foo.script* run with the argument 2, even though they have the same code. + + If this function successfully kills the specified script, then it will return true. Otherwise, it will return false. + + Examples: + + The following example will try to kill a script named *foo.script* on the *foodnstuff* server that was ran with no arguments:: + + kill("foo.script", "foodnstuff"); + + The following will try to kill a script named *foo.script* on the current server that was ran with no arguments:: + + kill("foo.script", getHostname()); + + The following will try to kill a script named *foo.script* on the current server that was ran with the arguments 1 and "foodnstuff":: + + kill("foo.script", getHostname(), 1, "foodnstuff"); diff --git a/doc/source/netscript/basicfunctions/killall.rst b/doc/source/netscript/basicfunctions/killall.rst new file mode 100644 index 000000000..5e1cd87a0 --- /dev/null +++ b/doc/source/netscript/basicfunctions/killall.rst @@ -0,0 +1,10 @@ +killall() Netscript Function +============================ + +.. js:function:: killall(hostname/ip) + + :param string hostname/ip: IP or hostname of the server on which to kill all scripts + :RAM cost: 0.5 GB + + Kills all running scripts on the specified server. This function returns true if any scripts were killed, and + false otherwise. In other words, it will return true if there are any scripts running on the target server. diff --git a/doc/source/netscript/basicfunctions/ls.rst b/doc/source/netscript/basicfunctions/ls.rst new file mode 100644 index 000000000..6055dece3 --- /dev/null +++ b/doc/source/netscript/basicfunctions/ls.rst @@ -0,0 +1,11 @@ +ls() Netscript Function +======================= + +.. js:function:: ls(hostname/ip, [grep]) + + :param string hostname/ip: Hostname or IP of the target server + :param string grep: a substring to search for in the filename + :RAM cost: 0 GB + + Returns an array with the filenames of all files on the specified server (as strings). The returned array + is sorted in alphabetic order diff --git a/doc/source/netscript/basicfunctions/nFormat.rst b/doc/source/netscript/basicfunctions/nFormat.rst new file mode 100644 index 000000000..280d1ca80 --- /dev/null +++ b/doc/source/netscript/basicfunctions/nFormat.rst @@ -0,0 +1,19 @@ +nFormat() Netscript Function +============================ + +.. js:function:: nFormat(n, format) + + :param number n: Number to format + :param string format: Formatter + + Converts a number into a string with the specified formatter. This uses the + `numeraljs `_ library, so the formatters must be compatible + with that. + + This is the same function that the game itself uses to display numbers. + + Examples:: + + nFormat(1.23e9, "$0.000a"); // Returns "$1.230b" + nFormat(12345.678, "0,0"); // Returns "12,346" + nFormat(0.84, "0.0%"); // Returns "84.0% diff --git a/doc/source/netscript/basicfunctions/nuke.rst b/doc/source/netscript/basicfunctions/nuke.rst new file mode 100644 index 000000000..ec951bcc1 --- /dev/null +++ b/doc/source/netscript/basicfunctions/nuke.rst @@ -0,0 +1,13 @@ +nuke() Netscript Function +========================= + +.. js:function:: nuke(hostname/ip) + + :param string hostname/ip: IP or hostname of the target server + :RAM cost: 0 GB + + Runs the NUKE.exe program on the target server. NUKE.exe must exist on your home computer. + + Example:: + + nuke("foodnstuff"); diff --git a/doc/source/netscript/basicfunctions/peek.rst b/doc/source/netscript/basicfunctions/peek.rst new file mode 100644 index 000000000..c118f4307 --- /dev/null +++ b/doc/source/netscript/basicfunctions/peek.rst @@ -0,0 +1,12 @@ +peek() Netscript Function +========================= + +.. js:function:: peek(port) + + :param number port: Port to peek. Must be an integer between 1 and 20 + :RAM cost: 1 GB + + This function is used to peek at the data from a port. It returns the first element in the specified port + without removing that element. If the port is empty, the string "NULL PORT DATA" will be returned. + + Read about how :ref:`netscript_ports` work here diff --git a/doc/source/netscript/basicfunctions/print.rst b/doc/source/netscript/basicfunctions/print.rst new file mode 100644 index 000000000..b50ec4666 --- /dev/null +++ b/doc/source/netscript/basicfunctions/print.rst @@ -0,0 +1,9 @@ +print() Netscript Function +=========================== + +.. js:function:: print(x) + + :param x: Value to be printed + :RAM cost: 0 GB + + Prints a value or a variable to the script's logs. diff --git a/doc/source/netscript/basicfunctions/prompt.rst b/doc/source/netscript/basicfunctions/prompt.rst new file mode 100644 index 000000000..24c2cc7cf --- /dev/null +++ b/doc/source/netscript/basicfunctions/prompt.rst @@ -0,0 +1,10 @@ +prompt() Netscript Function +=========================== + +.. js:function:: prompt(txt) + + :param string txt: Text to appear in the prompt dialog box + :RAM cost: 0 GB + + Prompts the player with a dialog box with two options: "Yes" and "No". This function will return true if the player click "Yes" and + false if the player clicks "No". The script's execution is halted until the player selects one of the options. diff --git a/doc/source/netscript/basicfunctions/ps.rst b/doc/source/netscript/basicfunctions/ps.rst new file mode 100644 index 000000000..fcdf2583a --- /dev/null +++ b/doc/source/netscript/basicfunctions/ps.rst @@ -0,0 +1,28 @@ +ps() Netscript Function +======================= + +.. js:function:: ps(hostname/ip=current ip) + + :param string ip: Hostname or IP address of the target server. + If not specified, it will be the current server's IP by default + :RAM cost: 0.2 GB + + Returns an array with general information about all scripts running on the specified + target server. The information for each server is given in an object with + the following structure:: + + { + filename: Script name, + threads: Number of threads script is running with, + args: Script's arguments + } + + Example usage (using :ref:`netscriptjs`):: + + export async function main(ns) { + const ps = ns.ps("home"); + for (let i = 0; i < ps.length; ++i) { + ns.tprint(ps[i].filename + ' ' + ps[i].threads); + ns.tprint(ps[i].args); + } + } diff --git a/doc/source/netscript/basicfunctions/purchaseServer.rst b/doc/source/netscript/basicfunctions/purchaseServer.rst new file mode 100644 index 000000000..caa316550 --- /dev/null +++ b/doc/source/netscript/basicfunctions/purchaseServer.rst @@ -0,0 +1,30 @@ +purchaseServer() Netscript Function +=================================== + +.. js:function:: purchaseServer(hostname, ram) + + :param string hostname: Hostname of the purchased server + :param number ram: Amount of RAM of the purchased server. Must be a power of 2 (2, 4, 8, 16, etc.). Maximum value of 1048576 (2^20) + :RAM cost: 2.25 GB + + Purchased a server with the specified hostname and amount of RAM. + + The *hostname* argument can be any data type, but it will be converted to a string and have whitespace removed. Anything that resolves to an empty string will + cause the function to fail. If there is already a server with the specified hostname, then the function will automatically append + a number at the end of the *hostname* argument value until it finds a unique hostname. For example, if the script calls + *purchaseServer("foo", 4)* but a server named "foo" already exists, the it will automatically change the hostname to "foo-0". If there is already + a server with the hostname "foo-0", then it will change the hostname to "foo-1", and so on. + + Note that there is a maximum limit to the amount of servers you can purchase. + + Returns the hostname of the newly purchased server as a string. If the function fails to purchase a server, then it will return an + empty string. The function will fail if the arguments passed in are invalid, if the player does not have enough money to purchase + the specified server, or if the player has exceeded the maximum amount of servers. + + Example:: + + ram = 64; + hn = "pserv-"; + for (i = 0; i < 5; ++i) { + purchaseServer(hn + i, ram); + } diff --git a/doc/source/netscript/basicfunctions/read.rst b/doc/source/netscript/basicfunctions/read.rst new file mode 100644 index 000000000..21f1cd693 --- /dev/null +++ b/doc/source/netscript/basicfunctions/read.rst @@ -0,0 +1,16 @@ +read() Netscript Function +========================= + +.. js:function:: read(port/fn) + + :param string/number port/fn: Port or text file to read from + :RAM cost: 1 GB + + This function is used to read data from a port, a text file (.txt), or a script (.script, .js, .ns) + + If the argument *port/fn* is a number between 1 and 20, then it specifies a port and it will read data from that port. Read + about how :ref:`netscript_ports` work here. A port is a serialized queue. This function + will remove the first element from that queue and return it. If the queue is empty, then the string "NULL PORT DATA" will be returned. + + If the argument *port/fn* is a string, then it specifies the name of a text file or script and this function will return the data in the specified text file/script. If + the text file does not exist, an empty string will be returned. diff --git a/doc/source/netscript/basicfunctions/relaysmtp.rst b/doc/source/netscript/basicfunctions/relaysmtp.rst new file mode 100644 index 000000000..b3f4a95f2 --- /dev/null +++ b/doc/source/netscript/basicfunctions/relaysmtp.rst @@ -0,0 +1,13 @@ +relaysmtp() Netscript Function +============================== + +.. js:function:: relaysmtp(hostname/ip) + + :param string hostname/ip: IP or hostname of the target server + :RAM cost: 0 GB + + Runs the relaySMTP.exe program on the target server. relaySMTP.exe must exist on your home computer. + + Example:: + + relaysmtp("foodnstuff"); diff --git a/doc/source/netscript/basicfunctions/rm.rst b/doc/source/netscript/basicfunctions/rm.rst new file mode 100644 index 000000000..e5c64bdfc --- /dev/null +++ b/doc/source/netscript/basicfunctions/rm.rst @@ -0,0 +1,11 @@ +rm() Netscript Function +======================= + +.. js:function:: rm(fn[, hostname/ip=current server]) + + :param string fn: Filename of file to remove. Must include the extension + :param string hostname/ip: Hostname or IP Address of the server on which to delete the file. Optional. Defaults to current server + :returns: True if it successfully deletes the file, and false otherwise + :RAM cost: 1 GB + + Removes the specified file from the current server. This function works for every file type except message (.msg) files. diff --git a/doc/source/netscript/basicfunctions/run.rst b/doc/source/netscript/basicfunctions/run.rst new file mode 100644 index 000000000..b4b9a414e --- /dev/null +++ b/doc/source/netscript/basicfunctions/run.rst @@ -0,0 +1,33 @@ +run() Netscript Function +======================== + +.. js:function:: run(script, [numThreads=1], [args...]) + + :param string script: Filename of script to run + :param number numThreads: Optional thread count for new script. Set to 1 by default. Will be rounded to nearest integer + :param args...: + Additional arguments to pass into the new script that is being run. Note that if any arguments are being + passed into the new script, then the second argument *numThreads* must be filled in with a value. + :RAM cost: 1 GB + + Run a script as a separate process. This function can only be used to run scripts located on the current server (the server + running the script that calls this function). + + Returns true if the script is successfully started, and false otherwise. + + Running this function with a *numThreads* argument of 0 will return false without running the script. + However, running this function with a negative *numThreads* argument will cause a runtime error. + + The simplest way to use the *run* command is to call it with just the script name. The following example will run + 'foo.script' single-threaded with no arguments:: + + run("foo.script"); + + The following example will run 'foo.script' but with 5 threads instead of single-threaded:: + + run("foo.script", 5); + + This next example will run 'foo.script' single-threaded, and will pass the string 'foodnstuff' into the script + as an argument:: + + run("foo.script", 1, 'foodnstuff'); diff --git a/doc/source/netscript/basicfunctions/scan.rst b/doc/source/netscript/basicfunctions/scan.rst new file mode 100644 index 000000000..a3412d260 --- /dev/null +++ b/doc/source/netscript/basicfunctions/scan.rst @@ -0,0 +1,11 @@ +scan() Netscript Function +========================= + +.. js:function:: scan(hostname/ip=current ip[, hostnames=true]) + + :param string hostname/ip: IP or hostname of the server to scan + :param boolean: Optional boolean specifying whether the function should output hostnames (if true) or IP addresses (if false) + :RAM cost: 0.2 GB + + Returns an array containing the hostnames or IPs of all servers that are one node way from the specified target server. The + hostnames/IPs in the returned array are strings. diff --git a/doc/source/netscript/basicfunctions/scp.rst b/doc/source/netscript/basicfunctions/scp.rst new file mode 100644 index 000000000..a290083fa --- /dev/null +++ b/doc/source/netscript/basicfunctions/scp.rst @@ -0,0 +1,29 @@ +scp() Netscript Function +======================== + +.. js:function:: scp(files, [source], destination) + + :param string/array files: Filename or an array of filenames of script/literature files to copy + :param string source: + Hostname or IP of the source server, which is the server from which the file will be copied. + This argument is optional and if it's omitted the source will be the current server. + :param string destination: Hostname or IP of the destination server, which is the server to which the file will be copied. + :RAM cost: 0.6 GB + + Copies a script or literature (.lit) file(s) to another server. The *files* argument can be either a string specifying a + single file to copy, or an array of strings specifying multiple files to copy. + + Returns true if the script/literature file is successfully copied over and false otherwise. If the *files* argument is an array + then this function will return true if at least one of the files in the array is successfully copied. + + Examples:: + + //Copies hack-template.script from the current server to foodnstuff + scp("hack-template.script", "foodnstuff"); + + //Copies foo.lit from the helios server to the home computer + scp("foo.lit", "helios", "home"); + + //Tries to copy three files from rothman-uni to home computer + files = ["foo1.lit", "foo2.script", "foo3.script"]; + scp(files, "rothman-uni", "home"); diff --git a/doc/source/netscript/basicfunctions/scriptKill.rst b/doc/source/netscript/basicfunctions/scriptKill.rst new file mode 100644 index 000000000..767e48322 --- /dev/null +++ b/doc/source/netscript/basicfunctions/scriptKill.rst @@ -0,0 +1,11 @@ +scriptKill() Netscript Function +=============================== + +.. js:function:: scriptKill(scriptname, hostname/ip) + + :param string scriptname: Filename of script to kill. This is case-sensitive. + :param string hostname/ip: Hostname or IP of target server + :RAM cost: 1 GB + + Kills all scripts with the specified filename on the target server specified by *hostname/ip*, regardless of arguments. Returns + true if one or more scripts were successfully killed, and false if none were. diff --git a/doc/source/netscript/basicfunctions/scriptRunning.rst b/doc/source/netscript/basicfunctions/scriptRunning.rst new file mode 100644 index 000000000..a384ea35e --- /dev/null +++ b/doc/source/netscript/basicfunctions/scriptRunning.rst @@ -0,0 +1,24 @@ +scriptRunning() Netscript Function +================================== + +.. js:function:: scriptRunning(scriptname, hostname/ip) + + :param string scriptname: Filename of script to check. This is case-sensitive. + :param string hostname/ip: Hostname or IP of target server + :RAM cost: 1 GB + + Returns a boolean indicating whether any instance of the specified script is running on the target server, regardless of + its arguments. + + This is different than the *isRunning()* function because it does not try to identify a specific instance of a running script + by its arguments. + + **Examples:** + + The example below will return true if there is any script named *foo.script* running on the *foodnstuff* server, and false otherwise:: + + scriptRunning("foo.script", "foodnstuff"); + + The example below will return true if there is any script named "foo.script" running on the current server, and false otherwise:: + + scriptRunning("foo.script", getHostname()); diff --git a/doc/source/netscript/basicfunctions/serverExists.rst b/doc/source/netscript/basicfunctions/serverExists.rst new file mode 100644 index 000000000..c32dab411 --- /dev/null +++ b/doc/source/netscript/basicfunctions/serverExists.rst @@ -0,0 +1,9 @@ +serverExists() Netscript Function +================================= + +.. js:function:: serverExists(hostname/ip) + + :param string hostname/ip: Hostname or IP of target server + :RAM cost: 0.1 GB + + Returns a boolean denoting whether or not the specified server exists diff --git a/doc/source/netscript/basicfunctions/sleep.rst b/doc/source/netscript/basicfunctions/sleep.rst new file mode 100644 index 000000000..b9109a9cb --- /dev/null +++ b/doc/source/netscript/basicfunctions/sleep.rst @@ -0,0 +1,9 @@ +sleep() Netscript Function +========================== + +.. js:function:: sleep(n) + + :param number n: Number of milliseconds to sleep + :RAM cost: 0 GB + + Suspends the script for n milliseconds. diff --git a/doc/source/netscript/basicfunctions/spawn.rst b/doc/source/netscript/basicfunctions/spawn.rst new file mode 100644 index 000000000..c0fe044ce --- /dev/null +++ b/doc/source/netscript/basicfunctions/spawn.rst @@ -0,0 +1,20 @@ +spawn() Netscript Function +========================== + +.. js:function:: spawn(script, numThreads, [args...]) + + :param string script: Filename of script to execute + :param number numThreads: Number of threads to spawn new script with. Will be rounded to nearest integer + :param args...: + Additional arguments to pass into the new script that is being run. + :RAM cost: 2 GB + + Terminates the current script, and then after a delay of about 20 seconds it will execute the newly-specified script. + The purpose of this function is to execute a new script without being constrained by the RAM usage of the current one. + This function can only be used to run scripts on the local server. + + Because this function immediately terminates the script, it does not have a return value. + + The following example will execute the script 'foo.script' with 10 threads and the arguments 'foodnstuff' and 90:: + + spawn('foo.script', 10, 'foodnstuff', 90); diff --git a/doc/source/netscript/basicfunctions/sprintf.rst b/doc/source/netscript/basicfunctions/sprintf.rst new file mode 100644 index 000000000..2ea752682 --- /dev/null +++ b/doc/source/netscript/basicfunctions/sprintf.rst @@ -0,0 +1,8 @@ +sprintf() Netscript Function +============================ + +.. js:function:: sprintf() + + :RAM cost: 0 GB + + See `this link `_ for details. diff --git a/doc/source/netscript/basicfunctions/sqlinject.rst b/doc/source/netscript/basicfunctions/sqlinject.rst new file mode 100644 index 000000000..4eab8dd6a --- /dev/null +++ b/doc/source/netscript/basicfunctions/sqlinject.rst @@ -0,0 +1,13 @@ +sqlinject() Netscript Function +============================== + +.. js:function:: sqlinject(hostname/ip) + + :param string hostname/ip: IP or hostname of the target server + :RAM cost: 0 GB + + Runs the SQLInject.exe program on the target server. SQLInject.exe must exist on your home computer. + + Example:: + + sqlinject("foodnstuff"); diff --git a/doc/source/netscript/basicfunctions/tprint.rst b/doc/source/netscript/basicfunctions/tprint.rst new file mode 100644 index 000000000..271879c30 --- /dev/null +++ b/doc/source/netscript/basicfunctions/tprint.rst @@ -0,0 +1,9 @@ +tprint() Netscript Function +=========================== + +.. js:function:: tprint(x) + + :param x: Value to be printed + :RAM cost: 0 GB + + Prints a value or a variable to the Terminal diff --git a/doc/source/netscript/basicfunctions/tryWrite.rst b/doc/source/netscript/basicfunctions/tryWrite.rst new file mode 100644 index 000000000..3c9a54e84 --- /dev/null +++ b/doc/source/netscript/basicfunctions/tryWrite.rst @@ -0,0 +1,12 @@ +tryWrite() Netscript Function +============================= + +.. js:function:: tryWrite(port, data="") + + :param number port: Port to be written to + :param string data: Data to try to write + :returns: True if the data is successfully written to the port, and false otherwise + :RAM cost: 1 GB + + Attempts to write data to the specified Netscript Port. If the port is full, the data will + not be written. Otherwise, the data will be written normally diff --git a/doc/source/netscript/basicfunctions/vsprintf.rst b/doc/source/netscript/basicfunctions/vsprintf.rst new file mode 100644 index 000000000..1b7431de0 --- /dev/null +++ b/doc/source/netscript/basicfunctions/vsprintf.rst @@ -0,0 +1,8 @@ +vsprintf() Netscript Function +============================= + +.. js:function:: vsprintf() + + :RAM cost: 0 GB + + See `this link `_ for details. diff --git a/doc/source/netscript/basicfunctions/weaken.rst b/doc/source/netscript/basicfunctions/weaken.rst new file mode 100644 index 000000000..e1eebc1db --- /dev/null +++ b/doc/source/netscript/basicfunctions/weaken.rst @@ -0,0 +1,20 @@ +weaken() Netscript Function +=========================== + +.. js:function:: weaken(hostname/ip) + + :param string hostname/ip: IP or hostname of the target server to weaken + :returns: The amount by which the target server's security level was decreased. This is equivalent to 0.05 multiplied + by the number of script threads + :RAM cost: 0.15 GB + + Use your hacking skills to attack a server's security, lowering the server's security level. The runtime for this command + depends on your hacking level and the target server's security level. This function lowers the security level of the target + server by 0.05. + + Like hack() and grow(), weaken() can be called on any server, regardless of where the script is running. This command requires + root access to the target server, but there is no required hacking level to run the command. + + Example:: + + weaken("foodnstuff"); diff --git a/doc/source/netscript/basicfunctions/wget.rst b/doc/source/netscript/basicfunctions/wget.rst new file mode 100644 index 000000000..78db9950e --- /dev/null +++ b/doc/source/netscript/basicfunctions/wget.rst @@ -0,0 +1,31 @@ +wget() Netscript Function +========================= + +.. js:function:: wget(url, target[, hostname/ip=current ip]) + + :param string url: URL to pull data from + :param string target: Filename to write data to. Must be script or text file + :param string ip: Optional hostname/ip of server for target file. + :RAM cost: 0 GB + + Retrieves data from a URL and downloads it to a file on the specified server. The data can only + be downloaded to a script (.script, .ns, .js) or a text file (.txt). If the file already exists, + it will be overwritten by this command. + + Note that it will not be possible to download data from many websites because they do not allow + cross-origin resource sharing (CORS). Example:: + + wget("https://raw.githubusercontent.com/danielyxie/bitburner/master/README.md", "game_readme.txt"); + + **IMPORTANT:** This is an asynchronous function that returns a Promise. The Promise's resolved + value will be a boolean indicating whether or not the data was successfully + retrieved from the URL. Because the function is async and returns a Promise, + it is recommended you use :code:`wget` in :ref:`netscriptjs`. + + In NetscriptJS, you must preface any call to + :code:`wget` with the :code:`await` keyword (like you would :code:`hack` or :code:`sleep`). + + :code:`wget` will still work in :ref:`netscript1`, but the functions execution will not + be synchronous (i.e. it may not execute when you expect/want it to). Furthermore, since Promises are not + supported in ES5, you will not be able to process the returned value of :code:`wget` in + Netscript 1.0. diff --git a/doc/source/netscript/basicfunctions/write.rst b/doc/source/netscript/basicfunctions/write.rst new file mode 100644 index 000000000..c492ea51d --- /dev/null +++ b/doc/source/netscript/basicfunctions/write.rst @@ -0,0 +1,20 @@ +write() Netscript Function +=========================== + +.. js:function:: write(port/fn, data="", mode="a") + + :param string/number port/fn: Port or text file/script that will be written to + :param string data: Data to write + :param string mode: Defines the write mode. Only valid when writing to text files or scripts. + :RAM cost: 1 GB + + This function can be used to either write data to a port, a text file (.txt), or a script (.script, .js, .ns) + + If the first argument is a number between 1 and 20, then it specifies a port and this function will write *data* to that port. Read + about how :ref:`netscript_ports` work here. The third argument, *mode*, is not used + when writing to a port. + + If the first argument is a string, then it specifies the name of a text file or script and this function will write *data* to that text file/script. If the + specified text file/script does not exist, then it will be created. The third argument *mode*, defines how the data will be written. If *mode* + is set to "w", then the data is written in "write" mode which means that it will overwrite all existing data on the text file/script. If *mode* is set to + any other value then the data will be written in "append" mode which means that the data will be added at the end of the file. diff --git a/doc/source/netscript/bladeburnerapi/getActionAutolevel.rst b/doc/source/netscript/bladeburnerapi/getActionAutolevel.rst new file mode 100644 index 000000000..77c504d67 --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/getActionAutolevel.rst @@ -0,0 +1,11 @@ +getActionAutolevel() Netscript Function +======================================= + +.. js:function:: getActionAutolevel(type, name) + + :param string type: Type of action. See :ref:`bladeburner_action_types` + :param string name: Name of action. Must be an exact match + + Return a boolean indicating whether or not this action is currently set to autolevel. + + Returns false if an invalid action is specified. diff --git a/doc/source/netscript/bladeburnerapi/getActionCountRemaining.rst b/doc/source/netscript/bladeburnerapi/getActionCountRemaining.rst new file mode 100644 index 000000000..af50a19d5 --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/getActionCountRemaining.rst @@ -0,0 +1,12 @@ +getActionCountRemaining() Netscript Function +============================================ + +.. js:function:: getActionCountRemaining(type, name) + + :param string type: Type of action. See :ref:`bladeburner_action_types` + :param string name: Name of action. Must be an exact match + + Returns the remaining count of the specified action. + + Note that this is meant to be used for Contracts and Operations. + This function will return 'Infinity' for actions such as Training and Field Analysis. diff --git a/doc/source/netscript/bladeburnerapi/getActionCurrentLevel.rst b/doc/source/netscript/bladeburnerapi/getActionCurrentLevel.rst new file mode 100644 index 000000000..c3476096d --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/getActionCurrentLevel.rst @@ -0,0 +1,11 @@ +getActionCurrentLevel() Netscript Function +========================================== + +.. js:function:: getActionCurrentLevel(type, name) + + :param string type: Type of action. See :ref:`bladeburner_action_types` + :param string name: Name of action. Must be an exact match + + Returns the current level of this action. + + Returns -1 if an invalid action is specified. diff --git a/doc/source/netscript/bladeburnerapi/getActionEstimatedSuccessChance.rst b/doc/source/netscript/bladeburnerapi/getActionEstimatedSuccessChance.rst new file mode 100644 index 000000000..26c1b610a --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/getActionEstimatedSuccessChance.rst @@ -0,0 +1,11 @@ +getActionEstimatedSuccessChance() Netscript Function +==================================================== + +.. js:function:: getActionEstimatedSuccessChance(type, name) + + :param string type: Type of action. See :ref:`bladeburner_action_types` + :param string name: Name of action. Must be an exact match + + Returns the estimated success chance for the specified action. This chance + is returned as a decimal value, NOT a percentage (e.g. if you have an estimated + success chance of 80%, then this function will return 0.80, NOT 80). diff --git a/doc/source/netscript/bladeburnerapi/getActionMaxLevel.rst b/doc/source/netscript/bladeburnerapi/getActionMaxLevel.rst new file mode 100644 index 000000000..dec0076fb --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/getActionMaxLevel.rst @@ -0,0 +1,11 @@ +getActionMaxLevel() Netscript Function +====================================== + +.. js:function:: getActionMaxLevel(type, name) + + :param string type: Type of action. See :ref:`bladeburner_action_types` + :param string name: Name of action. Must be an exact match + + Returns the maximum level for this action. + + Returns -1 if an invalid action is specified. diff --git a/doc/source/netscript/bladeburnerapi/getActionRepGain.rst b/doc/source/netscript/bladeburnerapi/getActionRepGain.rst new file mode 100644 index 000000000..3846acec8 --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/getActionRepGain.rst @@ -0,0 +1,12 @@ +getActionRepGain() Netscript Function +===================================== + +.. js:function:: getActionRepGain(type, name[, level=current level]) + + :param string type: Type of action. See :ref:`bladeburner_action_types` + :param string name: Name of action. Must be an exact match + :param number level: Optional action level at which to calculate the gain + + Returns the average Bladeburner reputation gain for successfully completing + the specified action. Note that this value is an 'average' and the real + reputation gain may vary slightly from this value. diff --git a/doc/source/netscript/bladeburnerapi/getActionTime.rst b/doc/source/netscript/bladeburnerapi/getActionTime.rst new file mode 100644 index 000000000..8189edf79 --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/getActionTime.rst @@ -0,0 +1,9 @@ +getActionTime() Netscript Function +================================== + +.. js:function:: getActionTime(type, name) + + :param string type: Type of action. See :ref:`bladeburner_action_types` + :param string name: Name of action. Must be an exact match + + Returns the number of seconds it takes to complete the specified action diff --git a/doc/source/netscript/bladeburnerapi/getBlackOpNames.rst b/doc/source/netscript/bladeburnerapi/getBlackOpNames.rst new file mode 100644 index 000000000..164904a23 --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/getBlackOpNames.rst @@ -0,0 +1,6 @@ +getBlackOpNames() Netscript Function +==================================== + +.. js:function:: getBlackOpNames() + + Returns an array of strings containing the names of all Bladeburner Black Ops diff --git a/doc/source/netscript/bladeburnerapi/getBonusTime.rst b/doc/source/netscript/bladeburnerapi/getBonusTime.rst new file mode 100644 index 000000000..c6aa0e088 --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/getBonusTime.rst @@ -0,0 +1,14 @@ +getBonusTime() Netscript Function +================================= + +.. js:function:: getBonusTime() + + Returns the amount of accumulated "bonus time" (seconds) for the Bladeburner mechanic. + + "Bonus time" is accumulated when the game is offline or if the game is + inactive in the browser. + + "Bonus time" makes the game progress faster, up to 5x the normal speed. + For example, if an action takes 30 seconds to complete but you've accumulated + over 30 seconds in bonus time, then the action will only take 6 seconds + in real life to complete. diff --git a/doc/source/netscript/bladeburnerapi/getCity.rst b/doc/source/netscript/bladeburnerapi/getCity.rst new file mode 100644 index 000000000..265ee604d --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/getCity.rst @@ -0,0 +1,6 @@ +getCity() Netscript Function +============================ + +.. js:function:: getCity() + + Returns the city that the player is currently in (for Bladeburner). diff --git a/doc/source/netscript/bladeburnerapi/getCityChaos.rst b/doc/source/netscript/bladeburnerapi/getCityChaos.rst new file mode 100644 index 000000000..65df242ca --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/getCityChaos.rst @@ -0,0 +1,8 @@ +getCityChaos() Netscript Function +================================= + +.. js:function:: getCityChaos(cityName) + + :param string cityName: Name of city. Case-sensitive + + Returns the chaos in the specified city, or -1 if an invalid city was specified diff --git a/doc/source/netscript/bladeburnerapi/getCityEstimatedCommunities.rst b/doc/source/netscript/bladeburnerapi/getCityEstimatedCommunities.rst new file mode 100644 index 000000000..c9862fbeb --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/getCityEstimatedCommunities.rst @@ -0,0 +1,9 @@ +getCityEstimatedCommunities() Netscript Function +================================================ + +.. js:function:: getCityEstimatedCommunities(cityName) + + :param string cityName: Name of city. Case-sensitive + + Returns the estimated number of Synthoid communities in the specified city, + or -1 if an invalid city was specified. diff --git a/doc/source/netscript/bladeburnerapi/getCityEstimatedPopulation.rst b/doc/source/netscript/bladeburnerapi/getCityEstimatedPopulation.rst new file mode 100644 index 000000000..d796feebf --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/getCityEstimatedPopulation.rst @@ -0,0 +1,9 @@ +getCityEstimatedPopulation() Netscript Function +=============================================== + +.. js:function:: getCityEstimatedPopulation(cityName) + + :param string cityName: Name of city. Case-sensitive + + Returns the estimated number of Synthoids in the specified city, or -1 + if an invalid city was specified. diff --git a/doc/source/netscript/bladeburnerapi/getContractNames.rst b/doc/source/netscript/bladeburnerapi/getContractNames.rst new file mode 100644 index 000000000..7b94af4ab --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/getContractNames.rst @@ -0,0 +1,6 @@ +getContractNames() Netscript Function +===================================== + +.. js:function:: getContractNames() + + Returns an array of strings containing the names of all Bladeburner contracts diff --git a/doc/source/netscript/bladeburnerapi/getCurrentAction.rst b/doc/source/netscript/bladeburnerapi/getCurrentAction.rst new file mode 100644 index 000000000..bfb16be69 --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/getCurrentAction.rst @@ -0,0 +1,14 @@ +getCurrentAction() Netscript Function +===================================== + +.. js:function:: getCurrentAction() + + Returns an object that represents the player's current Bladeburner action:: + + { + type: Type of Action + name: Name of Action + } + + If the player is not performing an action, the function will return an object + with the 'type' property set to "Idle". diff --git a/doc/source/netscript/bladeburnerapi/getGeneralActionNames.rst b/doc/source/netscript/bladeburnerapi/getGeneralActionNames.rst new file mode 100644 index 000000000..96556b738 --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/getGeneralActionNames.rst @@ -0,0 +1,6 @@ +getGeneralActionNames() Netscript Function +========================================== + +.. js:function:: getGeneralActionNames() + + Returns an array of strings containing the names of all general Bladeburner actions diff --git a/doc/source/netscript/bladeburnerapi/getOperationNames.rst b/doc/source/netscript/bladeburnerapi/getOperationNames.rst new file mode 100644 index 000000000..e740505f8 --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/getOperationNames.rst @@ -0,0 +1,6 @@ +getOperationNames() Netscript Function +====================================== + +.. js:function:: getOperationNames() + + Returns an array of strings containing the names of all Bladeburner operations diff --git a/doc/source/netscript/bladeburnerapi/getRank.rst b/doc/source/netscript/bladeburnerapi/getRank.rst new file mode 100644 index 000000000..f5745801a --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/getRank.rst @@ -0,0 +1,6 @@ +getRank() Netscript Function +============================ + +.. js:function:: getRank() + + Returns the player's Bladeburner Rank diff --git a/doc/source/netscript/bladeburnerapi/getSkillLevel.rst b/doc/source/netscript/bladeburnerapi/getSkillLevel.rst new file mode 100644 index 000000000..78000b893 --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/getSkillLevel.rst @@ -0,0 +1,10 @@ +getSkillLevel() Netscript Function +================================== + +.. js:function:: getSkillLevel(skillName="") + + :param string skillName: Name of skill. Case-sensitive and must be an exact match + + This function returns your level in the specified skill. + + The function returns -1 if an invalid skill name is passed in diff --git a/doc/source/netscript/bladeburnerapi/getSkillNames.rst b/doc/source/netscript/bladeburnerapi/getSkillNames.rst new file mode 100644 index 000000000..e6893d2a9 --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/getSkillNames.rst @@ -0,0 +1,6 @@ +getSkillNames() Netscript Function +================================== + +.. js:function:: getSkillNames() + + Returns an array of strings containing the names of all Bladeburner skills diff --git a/doc/source/netscript/bladeburnerapi/getSkillPoints.rst b/doc/source/netscript/bladeburnerapi/getSkillPoints.rst new file mode 100644 index 000000000..18ea7f16b --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/getSkillPoints.rst @@ -0,0 +1,6 @@ +getSkillPoints() Netscript Function +=================================== + +.. js:function:: getSkillPoints() + + Returns the number of Bladeburner skill points you have diff --git a/doc/source/netscript/bladeburnerapi/getSkillUpgradeCost.rst b/doc/source/netscript/bladeburnerapi/getSkillUpgradeCost.rst new file mode 100644 index 000000000..6216d4e7a --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/getSkillUpgradeCost.rst @@ -0,0 +1,11 @@ +getSkillUpgradeCost() Netscript Function +======================================== + +.. js:function:: getSkillUpgradeCost(skillName="") + + :param string skillName: Name of skill. Case-sensitive and must be an exact match + + This function returns the number of skill points needed to upgrade the + specified skill. + + The function returns -1 if an invalid skill name is passed in. diff --git a/doc/source/netscript/bladeburnerapi/getStamina.rst b/doc/source/netscript/bladeburnerapi/getStamina.rst new file mode 100644 index 000000000..c2d1f8a05 --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/getStamina.rst @@ -0,0 +1,15 @@ +getStamina() Netscript Function +=============================== + +.. js:function:: getStamina() + + Returns an array with two elements: + + [Current stamina, Max stamina] + + Example usage:: + + function getStaminaPercentage() { + let res = bladeburner.getStamina(); + return res[0] / res[1]; + } diff --git a/doc/source/netscript/bladeburnerapi/getTeamSize.rst b/doc/source/netscript/bladeburnerapi/getTeamSize.rst new file mode 100644 index 000000000..617b0488f --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/getTeamSize.rst @@ -0,0 +1,13 @@ +getTeamSize() Netscript Function +================================ + +.. js:function:: getTeamSize(type, name) + + :param string type: Type of action. See :ref:`bladeburner_action_types` + :param string name: Name of action. Must be an exact match + + Returns the number of Bladeburner team members you have assigned to the + specified action. + + Setting a team is only applicable for Operations and BlackOps. This function + will return 0 for other action types. diff --git a/doc/source/netscript/bladeburnerapi/joinBladeburnerDivision.rst b/doc/source/netscript/bladeburnerapi/joinBladeburnerDivision.rst new file mode 100644 index 000000000..ff524da8d --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/joinBladeburnerDivision.rst @@ -0,0 +1,11 @@ +joinBladeburnerDivision() Netscript Function +============================================ + +.. js:function:: joinBladeburnerDivision() + + Attempts to join the Bladeburner division. + + Returns true if you successfully join the Bladeburner division, or if you + are already a member. + + Returns false otherwise diff --git a/doc/source/netscript/bladeburnerapi/joinBladeburnerFaction.rst b/doc/source/netscript/bladeburnerapi/joinBladeburnerFaction.rst new file mode 100644 index 000000000..58fc32786 --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/joinBladeburnerFaction.rst @@ -0,0 +1,11 @@ +joinBladeburnerFaction() Netscript Function +=========================================== + +.. js:function:: joinBladeburnerFaction() + + Attempts to join the Bladeburner faction. + + Returns true if you successfully join the Bladeburner faction, or if + you are already a member. + + Returns false otherwise. diff --git a/doc/source/netscript/bladeburnerapi/setActionAutolevel.rst b/doc/source/netscript/bladeburnerapi/setActionAutolevel.rst new file mode 100644 index 000000000..d650c551c --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/setActionAutolevel.rst @@ -0,0 +1,10 @@ +setActionAutolevel() Netscript Function +======================================= + +.. js:function:: setActionAutolevel(type, name, autoLevel) + + :param string type: Type of action. See :ref:`bladeburner_action_types` + :param string name: Name of action. Must be an exact match + :param boolean autoLevel: Whether or not to autolevel this action + + Enable/disable autoleveling for the specified action. diff --git a/doc/source/netscript/bladeburnerapi/setActionLevel.rst b/doc/source/netscript/bladeburnerapi/setActionLevel.rst new file mode 100644 index 000000000..42bafa9eb --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/setActionLevel.rst @@ -0,0 +1,10 @@ +setActionLevel() Netscript Function +=================================== + +.. js:function:: setActionLevel(type, name, level) + + :param string type: Type of action. See :ref:`bladeburner_action_types` + :param string name: Name of action. Must be an exact match + :param level int: Level to set this action to + + Set the level for the specified action. diff --git a/doc/source/netscript/bladeburnerapi/setTeamSize.rst b/doc/source/netscript/bladeburnerapi/setTeamSize.rst new file mode 100644 index 000000000..c40e33f8d --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/setTeamSize.rst @@ -0,0 +1,12 @@ +setTeamSize() Netscript Function +================================ + +.. js:function:: setTeamSize(type, name, size) + + :param string type: Type of action. See :ref:`bladeburner_action_types` + :param string name: Name of action. Must be an exact match + :param int size: Number of team members to set. Will be converted using Math.round() + + Set the team size for the specified Bladeburner action. + + Returns the team size that was set, or -1 if the function failed. diff --git a/doc/source/netscript/bladeburnerapi/startAction.rst b/doc/source/netscript/bladeburnerapi/startAction.rst new file mode 100644 index 000000000..c3cb330a9 --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/startAction.rst @@ -0,0 +1,10 @@ +startAction() Netscript Function +================================ + +.. js:function:: startAction(type, name) + + :param string type: Type of action. See :ref:`bladeburner_action_types` + :param string name: Name of action. Must be an exact match + + Attempts to start the specified Bladeburner action. Returns true if the action + was started successfully, and false otherwise. diff --git a/doc/source/netscript/bladeburnerapi/stopBladeburnerAction.rst b/doc/source/netscript/bladeburnerapi/stopBladeburnerAction.rst new file mode 100644 index 000000000..3ff0489a9 --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/stopBladeburnerAction.rst @@ -0,0 +1,6 @@ +stopBladeburnerAction() Netscript Function +========================================== + +.. js:function:: stopBladeburnerAction() + + Stops the current Bladeburner action diff --git a/doc/source/netscript/bladeburnerapi/switchCity.rst b/doc/source/netscript/bladeburnerapi/switchCity.rst new file mode 100644 index 000000000..b0ff6ed53 --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/switchCity.rst @@ -0,0 +1,10 @@ +switchCity() Netscript Function +=============================== + +.. js:function:: switchCity(cityName) + + :param string cityName: Name of city + + Attempts to switch to the specified city (for Bladeburner only). + + Returns true if successful, and false otherwise diff --git a/doc/source/netscript/bladeburnerapi/upgradeSkill.rst b/doc/source/netscript/bladeburnerapi/upgradeSkill.rst new file mode 100644 index 000000000..06d6d6bd7 --- /dev/null +++ b/doc/source/netscript/bladeburnerapi/upgradeSkill.rst @@ -0,0 +1,9 @@ +upgradeSkill() Netscript Function +================================= + +.. js:function:: upgradeSkill(skillName) + + :param string skillName: Name of Skill to be upgraded. Case-sensitive and must be an exact match + + Attempts to upgrade the specified Bladeburner skill. Returns true if the + skill is successfully upgraded, and false otherwise diff --git a/doc/source/netscript/codingcontractapi/attempt.rst b/doc/source/netscript/codingcontractapi/attempt.rst new file mode 100644 index 000000000..342349909 --- /dev/null +++ b/doc/source/netscript/codingcontractapi/attempt.rst @@ -0,0 +1,13 @@ +attempt() Netscript Function +============================ + +.. js:function:: attempt(answer, fn[, hostname/ip=current ip]) + + :param answer: Solution for the contract + :param string fn: Filename of the contract + :param string hostname/ip: Hostname or IP of the server containing the contract. + Optional. Defaults to current server if not provided + + Attempts to solve the Coding Contract with the provided solution. + + :returns: Boolean indicating whether the solution was correct diff --git a/doc/source/netscript/codingcontractapi/getContractType.rst b/doc/source/netscript/codingcontractapi/getContractType.rst new file mode 100644 index 000000000..5713d4405 --- /dev/null +++ b/doc/source/netscript/codingcontractapi/getContractType.rst @@ -0,0 +1,13 @@ +getContractType() Netscript Function +==================================== + +.. js:function:: getContractType(fn[, hostname/ip=current ip]) + + :param string fn: Filename of the contract + :param string hostname/ip: Hostname or IP of the server containing the contract. + Optional. Defaults to current server if not provided + + Returns a name describing the type of problem posed by the Coding Contract. + (e.g. Find Largest Prime Factor, Total Ways to Sum, etc.) + + :returns: A string with the contract's problem type diff --git a/doc/source/netscript/codingcontractapi/getData.rst b/doc/source/netscript/codingcontractapi/getData.rst new file mode 100644 index 000000000..0efe2d224 --- /dev/null +++ b/doc/source/netscript/codingcontractapi/getData.rst @@ -0,0 +1,14 @@ +getData() Netscript Function +============================ + +.. js:function:: getData(fn[, hostname/ip=current ip]) + + :param string fn: Filename of the contract + :param string hostname/ip: Hostname or IP of the server containing the contract. + Optional. Defaults to current server if not provided + + Get the data associated with the specific Coding Contract. Note that this is + not the same as the contract's description. This is just the data that + the contract wants you to act on in order to solve + + :returns: The specified contract's data diff --git a/doc/source/netscript/codingcontractapi/getDescription.rst b/doc/source/netscript/codingcontractapi/getDescription.rst new file mode 100644 index 000000000..08c11c356 --- /dev/null +++ b/doc/source/netscript/codingcontractapi/getDescription.rst @@ -0,0 +1,12 @@ +getDescription() Netscript Function +=================================== + +.. js:function:: getDescription(fn[, hostname/ip=current ip]) + + :param string fn: Filename of the contract + :param string hostname/ip: Hostname or IP of the server containing the contract. + Optional. Defaults to current server if not provided + + Get the full text description for the problem posed by the Coding Contract + + :returns: A string with the contract's text description diff --git a/doc/source/netscript/codingcontractapi/getNumTriesRemaining.rst b/doc/source/netscript/codingcontractapi/getNumTriesRemaining.rst new file mode 100644 index 000000000..5e8620398 --- /dev/null +++ b/doc/source/netscript/codingcontractapi/getNumTriesRemaining.rst @@ -0,0 +1,13 @@ +getNumTriesRemaining() Netscript Function +========================================= + +.. js:function:: getNumTriesRemaining(fn[, hostname/ip=current ip]) + + :param string fn: Filename of the contract + :param string hostname/ip: Hostname or IP of the server containing the contract. + Optional. Defaults to current server if not provided + + Get the number of tries remaining on the contract before it + self-destructs. + + :returns: Number indicating how many attempts are remaining diff --git a/doc/source/netscript/gangapi/ascendMember.rst b/doc/source/netscript/gangapi/ascendMember.rst new file mode 100644 index 000000000..204b5408c --- /dev/null +++ b/doc/source/netscript/gangapi/ascendMember.rst @@ -0,0 +1,22 @@ +ascendMember() Netscript Function +================================= + +.. js:function:: ascendMember(name) + + :param string name: Name of member to ascend + + Ascend the specified Gang Member. + + :returns: An object with info about the ascension results. + + The object has the following structure:: + + { + respect: Amount of respect lost from ascending + hack: Hacking multiplier gained from ascending. Decimal form + str: Strength multiplier gained from ascending. Decimal form + def: Defense multiplier gained from ascending. Decimal form + dex: Dexterity multiplier gained from ascending. Decimal form + agi: Agility multiplier gained from ascending. Decimal form + cha: Charisma multiplier gained from ascending. Decimal form + } diff --git a/doc/source/netscript/gangapi/canRecruitMember.rst b/doc/source/netscript/gangapi/canRecruitMember.rst new file mode 100644 index 000000000..2f0af2017 --- /dev/null +++ b/doc/source/netscript/gangapi/canRecruitMember.rst @@ -0,0 +1,6 @@ +canRecruitMember() Netscript Function +===================================== + +.. js:function:: canRecruitMember() + + :returns: Boolean indicating whether a member can currently be recruited diff --git a/doc/source/netscript/gangapi/getBonusTime.rst b/doc/source/netscript/gangapi/getBonusTime.rst new file mode 100644 index 000000000..2163cc4db --- /dev/null +++ b/doc/source/netscript/gangapi/getBonusTime.rst @@ -0,0 +1,13 @@ +getBonusTime() Netscript Function +================================= + +.. js:function:: getBonusTime() + + Returns the amount of accumulated "bonus time" (seconds) for the Gang mechanic. + + "Bonus time" is accumulated when the game is offline or if the game is + inactive in the browser. + + "Bonus time" makes the game progress faster, up to 10x the normal speed. + + :returns: Bonus time for the Gang mechanic in seconds diff --git a/doc/source/netscript/gangapi/getChanceToWinClash.rst b/doc/source/netscript/gangapi/getChanceToWinClash.rst new file mode 100644 index 000000000..501139d49 --- /dev/null +++ b/doc/source/netscript/gangapi/getChanceToWinClash.rst @@ -0,0 +1,9 @@ +getChanceToWinClash() Netscript Function +======================================== + +.. js:function:: getChanceToWinClash(gangName) + + :param string gangName: Target gang + + Returns the chance you have to win a clash with the specified gang. The chance + is returned in decimal form, not percentage diff --git a/doc/source/netscript/gangapi/getEquipmentCost.rst b/doc/source/netscript/gangapi/getEquipmentCost.rst new file mode 100644 index 000000000..dab58abca --- /dev/null +++ b/doc/source/netscript/gangapi/getEquipmentCost.rst @@ -0,0 +1,12 @@ +getEquipmentCost() Netscript Function +===================================== + +.. js:function:: getEquipmentCost(equipName) + + :param string equipName: Name of equipment + + Get the amount of money it takes to purchase a piece of Equipment or an Augmentation. + If an invalid Equipment/Augmentation is specified, this function will return Infinity. + + :returns: Cost to purchase the specified Equipment/Augmentation (number). Infinity + for invalid arguments diff --git a/doc/source/netscript/gangapi/getEquipmentNames.rst b/doc/source/netscript/gangapi/getEquipmentNames.rst new file mode 100644 index 000000000..ec5f74c30 --- /dev/null +++ b/doc/source/netscript/gangapi/getEquipmentNames.rst @@ -0,0 +1,9 @@ +getEquipmentNames() Netscript Function +====================================== + +.. js:function:: getEquipmentNames() + + Get the name of all possible equipment/upgrades you can purchase for your + Gang Members. This includes Augmentations. + + :returns: Array of strings of the names of all Equpiment/Augmentations diff --git a/doc/source/netscript/gangapi/getEquipmentType.rst b/doc/source/netscript/gangapi/getEquipmentType.rst new file mode 100644 index 000000000..bec9f9fdf --- /dev/null +++ b/doc/source/netscript/gangapi/getEquipmentType.rst @@ -0,0 +1,16 @@ +getEquipmentType() Netscript Function +===================================== + +.. js:function:: getEquipmentType(equipName) + + :param string equipName: Name of equipment + + Get the specified equipment type, which can be one of the following: + + * Weapon + * Armor + * Vehicle + * Rootkit + * Augmentation + + :returns: A string stating the type of the equipment diff --git a/doc/source/netscript/gangapi/getGangInformation.rst b/doc/source/netscript/gangapi/getGangInformation.rst new file mode 100644 index 000000000..008853d1d --- /dev/null +++ b/doc/source/netscript/gangapi/getGangInformation.rst @@ -0,0 +1,23 @@ +getGangInformation() Netscript Function +======================================== + +.. js:function:: getGangInformation() + + Get general information about the gang + + :returns: An object with the gang information. + + The object has the following structure:: + + { + faction: Name of faction that the gang belongs to ("Slum Snakes", etc.) + isHacking: Boolean indicating whether or not its a hacking gang + moneyGainRate: Money earned per second + power: Gang's power for territory warfare + respect: Gang's respect + respectGainRate: Respect earned per second + territory: Amount of territory held. Returned in decimal form, not percentage + territoryClashChance: Clash chance. Returned in decimal form, not percentage + wantedLevel: Gang's wanted level + wantedLevelGainRate: Wanted level gained/lost per second (negative for losses) + } diff --git a/doc/source/netscript/gangapi/getMemberInformation.rst b/doc/source/netscript/gangapi/getMemberInformation.rst new file mode 100644 index 000000000..356913237 --- /dev/null +++ b/doc/source/netscript/gangapi/getMemberInformation.rst @@ -0,0 +1,36 @@ +getMemberInformation() Netscript Function +========================================= + +.. js:function:: getMemberInformation(name) + + :param string name: Name of member + + Get stat and equipment-related information about a Gang Member + + :returns: An object with the gang member information. + + The object has the following structure:: + + { + agility: Agility stat + agilityEquipMult: Agility multiplier from equipment. Decimal form + agilityAscensionMult: Agility multiplier from ascension. Decimal form + augmentations: Array of names of all owned Augmentations + charisma: Charisma stat + charismaEquipMult: Charisma multiplier from equipment. Decimal form + charismaAscensionMult: Charisma multiplier from ascension. Decimal form + defense: Defense stat + defenseEquipMult: Defense multiplier from equipment. Decimal form + defenseAscensionMult: Defense multiplier from ascension. Decimal form + dexterity: Dexterity stat + dexterityEquipMult: Dexterity multiplier from equipment. Decimal form + dexterityAscensionMult: Dexterity multiplier from ascension. Decimal form + equipment: Array of names of all owned Non-Augmentation Equipment + hacking: Hacking stat + hackingEquipMult: Hacking multiplier from equipment. Decimal form + hackingAscensionMult: Hacking multiplier from ascension. Decimal form + strength: Strength stat + strengthEquipMult: Strength multiplier from equipment. Decimal form + strengthAscensionMult: Strength multiplier from ascension. Decimal form + task: Name of currently assigned task + } diff --git a/doc/source/netscript/gangapi/getMemberNames.rst b/doc/source/netscript/gangapi/getMemberNames.rst new file mode 100644 index 000000000..628df081e --- /dev/null +++ b/doc/source/netscript/gangapi/getMemberNames.rst @@ -0,0 +1,8 @@ +getMemberNames() Netscript Function +=================================== + +.. js:function:: getMemberNames() + + Get the names of all Gang members + + :returns: An array of the names of all Gang members as strings diff --git a/doc/source/netscript/gangapi/getOtherGangInformation.rst b/doc/source/netscript/gangapi/getOtherGangInformation.rst new file mode 100644 index 000000000..ffb2c913f --- /dev/null +++ b/doc/source/netscript/gangapi/getOtherGangInformation.rst @@ -0,0 +1,26 @@ +getOtherGangInformation() Netscript Function +============================================ + +.. js:function:: getOtherGangInformation() + + Get territory and power information about all gangs + + :returns: An object with information about all gangs + + The object has the following structure:: + + { + "Slum Snakes" : { + power: Slum Snakes' power + territory: Slum Snakes' territory, in decimal form + }, + "Tetrads" : { + power: ... + territory: ... + }, + "The Syndicate" : { + power: ... + territory: ... + }, + ... (for all six gangs) + } diff --git a/doc/source/netscript/gangapi/getTaskNames.rst b/doc/source/netscript/gangapi/getTaskNames.rst new file mode 100644 index 000000000..06adfe599 --- /dev/null +++ b/doc/source/netscript/gangapi/getTaskNames.rst @@ -0,0 +1,8 @@ +getTaskNames() Netscript Function +================================= + +.. js:function:: getTaskNames() + + Get the name of all valid tasks that Gang members can be assigned to + + :returns: Array of strings of all task names diff --git a/doc/source/netscript/gangapi/purchaseEquipment.rst b/doc/source/netscript/gangapi/purchaseEquipment.rst new file mode 100644 index 000000000..6e985589e --- /dev/null +++ b/doc/source/netscript/gangapi/purchaseEquipment.rst @@ -0,0 +1,12 @@ +purchaseEquipment() Netscript Function +====================================== + +.. js:function:: purchaseEquipment(memberName, equipName) + + :param string memberName: Name of Gang member to purchase the equipment for + :param string equipName: Name of Equipment/Augmentation to purchase + + Attempt to purchase the specified Equipment/Augmentation for the specified + Gang member. + + :returns: True if the equipment was successfully purchased. False otherwise diff --git a/doc/source/netscript/gangapi/recruitMember.rst b/doc/source/netscript/gangapi/recruitMember.rst new file mode 100644 index 000000000..71153a12e --- /dev/null +++ b/doc/source/netscript/gangapi/recruitMember.rst @@ -0,0 +1,14 @@ +recruitMember() Netscript Function +================================== + +.. js:function:: recruitMember(name) + + :param string name: Name of member to recruit + + Attempt to recruit a new gang member. + + Possible reasons for failure: + * Cannot currently recruit a new member + * There already exists a member with the specified name + + :returns: True if the member was successfully recruited. False otherwise diff --git a/doc/source/netscript/gangapi/setMemberTask.rst b/doc/source/netscript/gangapi/setMemberTask.rst new file mode 100644 index 000000000..c75eaefba --- /dev/null +++ b/doc/source/netscript/gangapi/setMemberTask.rst @@ -0,0 +1,12 @@ +setMemberTask() Netscript Function +================================== + +.. js:function:: setMemberTask(memberName, taskName) + + :param string memberName: Name of Gang member to assign + :param string taskName: Task to assign + + Attempts to assign the specified Gang Member to the specified task. + If an invalid task is specified, the Gang member will be set to idle ("Unassigned") + + :returns: True if the Gang Member was successfully assigned to the task. False otherwise diff --git a/doc/source/netscript/gangapi/setTerritoryWarfare.rst b/doc/source/netscript/gangapi/setTerritoryWarfare.rst new file mode 100644 index 000000000..607c8137c --- /dev/null +++ b/doc/source/netscript/gangapi/setTerritoryWarfare.rst @@ -0,0 +1,8 @@ +setTerritoryWarfare() Netscript Function +======================================== + +.. js:function:: setTerritoryWarfare(engage) + + :param bool engage: Whether or not to engage in territory warfare + + Set whether or not the gang should engage in territory warfare diff --git a/doc/source/netscript/hacknetnodeapi/getCoreUpgradeCost.rst b/doc/source/netscript/hacknetnodeapi/getCoreUpgradeCost.rst new file mode 100644 index 000000000..536d1a9ee --- /dev/null +++ b/doc/source/netscript/hacknetnodeapi/getCoreUpgradeCost.rst @@ -0,0 +1,12 @@ +getCoreUpgradeCost() Netscript Function +======================================= + +.. js:function:: getCoreUpgradeCost(i, n) + + :param number i: Index/Identifier of Hacknet Node. :ref:`See here for details ` + :param number n: Number of times to upgrade cores. Must be positive. Rounded to nearest integer + + Returns the cost of upgrading the number of cores of the specified Hacknet Node by *n*. + + If an invalid value for *n* is provided, then this function returns 0. If the + specified Hacknet Node is already at the max number of cores, then Infinity is returned. diff --git a/doc/source/netscript/hacknetnodeapi/getLevelUpgradeCost.rst b/doc/source/netscript/hacknetnodeapi/getLevelUpgradeCost.rst new file mode 100644 index 000000000..d2a933c38 --- /dev/null +++ b/doc/source/netscript/hacknetnodeapi/getLevelUpgradeCost.rst @@ -0,0 +1,12 @@ +getLevelUpgradeCost() Netscript Function +======================================== + +.. js:function:: getLevelUpgradeCost(i, n) + + :param number i: Index/Identifier of Hacknet Node. :ref:`See here for details ` + :param number n: Number of levels to upgrade. Must be positive. Rounded to nearest integer + + Returns the cost of upgrading the specified Hacknet Node by *n* levels. + + If an invalid value for *n* is provided, then this function returns 0. If the + specified Hacknet Node is already at max level, then Infinity is returned. diff --git a/doc/source/netscript/hacknetnodeapi/getNodeStats.rst b/doc/source/netscript/hacknetnodeapi/getNodeStats.rst new file mode 100644 index 000000000..74b7de5ad --- /dev/null +++ b/doc/source/netscript/hacknetnodeapi/getNodeStats.rst @@ -0,0 +1,18 @@ +getNodeStats() Netscript Function +================================= + +.. js:function:: getNodeStats(i) + + :param number i: Index/Identifier of Hacknet Node. :ref:`See here for details ` + + Returns an object containing a variety of stats about the specified Hacknet Node:: + + { + name: Node's name ("hacknet-node-5"), + level: Node's level, + ram: Node's RAM, + cores: Node's number of cores, + production: Node's money earned per second, + timeOnline: Number of seconds since Node has been purchased, + totalProduction: Total number of money Node has produced + } diff --git a/doc/source/netscript/hacknetnodeapi/getPurchaseNodeCost.rst b/doc/source/netscript/hacknetnodeapi/getPurchaseNodeCost.rst new file mode 100644 index 000000000..459c82957 --- /dev/null +++ b/doc/source/netscript/hacknetnodeapi/getPurchaseNodeCost.rst @@ -0,0 +1,6 @@ +getPurchaseNodeCost() Netscript Function +======================================== + +.. js:function:: getPurchaseNodeCost() + + Returns the cost of purchasing a new Hacknet Node diff --git a/doc/source/netscript/hacknetnodeapi/getRamUpgradeCost.rst b/doc/source/netscript/hacknetnodeapi/getRamUpgradeCost.rst new file mode 100644 index 000000000..2ba4974aa --- /dev/null +++ b/doc/source/netscript/hacknetnodeapi/getRamUpgradeCost.rst @@ -0,0 +1,12 @@ +getRamUpgradeCost() Netscript Function +====================================== + +.. js:function:: getRamUpgradeCost(i, n) + + :param number i: Index/Identifier of Hacknet Node. :ref:`See here for details ` + :param number n: Number of times to upgrade RAM. Must be positive. Rounded to nearest integer + + Returns the cost of upgrading the RAM of the specified Hacknet Node *n* times. + + If an invalid value for *n* is provided, then this function returns 0. If the + specified Hacknet Node is already at max RAM, then Infinity is returned. diff --git a/doc/source/netscript/hacknetnodeapi/numNodes.rst b/doc/source/netscript/hacknetnodeapi/numNodes.rst new file mode 100644 index 000000000..0e4c49188 --- /dev/null +++ b/doc/source/netscript/hacknetnodeapi/numNodes.rst @@ -0,0 +1,6 @@ +numNodes() Netscript Function +============================= + +.. js:function:: numNodes() + + Returns the number of Hacknet Nodes you own. diff --git a/doc/source/netscript/hacknetnodeapi/purchaseNode.rst b/doc/source/netscript/hacknetnodeapi/purchaseNode.rst new file mode 100644 index 000000000..7bac278dc --- /dev/null +++ b/doc/source/netscript/hacknetnodeapi/purchaseNode.rst @@ -0,0 +1,11 @@ +purchaseNode() Netscript Function +================================= + +.. js:function:: purchaseNode() + + Purchases a new Hacknet Node. Returns a number with the index of the Hacknet Node. + This index is equivalent to the number at the end of the Hacknet Node's name + (e.g The Hacknet Node named 'hacknet-node-4' will have an index of 4). + + If the player cannot afford to purchase a new Hacknet Node then the function + will return -1. diff --git a/doc/source/netscript/hacknetnodeapi/upgradeCore.rst b/doc/source/netscript/hacknetnodeapi/upgradeCore.rst new file mode 100644 index 000000000..372b4da45 --- /dev/null +++ b/doc/source/netscript/hacknetnodeapi/upgradeCore.rst @@ -0,0 +1,14 @@ +upgradeCore() Netscript Function +================================ + +.. js:function:: upgradeCore(i, n) + + :param number i: Index/Identifier of Hacknet Node. :ref:`See here for details ` + :param number n: Number of cores to purchase. Must be positive. Rounded to nearest integer + + Tries to purchase *n* cores for the specified Hacknet Node. + + Returns true if it successfully purchases *n* cores for the Hacknet Node or if + it purchases some positive amount and the Node reaches its max number of cores. + + Returns false otherwise. diff --git a/doc/source/netscript/hacknetnodeapi/upgradeLevel.rst b/doc/source/netscript/hacknetnodeapi/upgradeLevel.rst new file mode 100644 index 000000000..243d58ca6 --- /dev/null +++ b/doc/source/netscript/hacknetnodeapi/upgradeLevel.rst @@ -0,0 +1,14 @@ +upgradeLevel() Netscript Function +================================= + +.. js:function:: upgradeLevel(i, n) + + :param number i: Index/Identifier of Hacknet Node. :ref:`See here for details ` + :param number n: Number of levels to purchase. Must be positive. Rounded to nearest integer + + Tries to upgrade the level of the specified Hacknet Node by *n*. + + Returns true if the Hacknet Node's level is successfully upgraded by *n* or + if it is upgraded by some positive amount and the Node reaches its max level. + + Returns false otherwise. diff --git a/doc/source/netscript/hacknetnodeapi/upgradeRam.rst b/doc/source/netscript/hacknetnodeapi/upgradeRam.rst new file mode 100644 index 000000000..319981e06 --- /dev/null +++ b/doc/source/netscript/hacknetnodeapi/upgradeRam.rst @@ -0,0 +1,16 @@ +upgradeRam() Netscript Function +=============================== + +.. js:function:: upgradeRam(i, n) + + :param number i: Index/Identifier of Hacknet Node. :ref:`See here for details ` + :param number n: Number of times to upgrade RAM. Must be positive. Rounded to nearest integer + + Tries to upgrade the specified Hacknet Node's RAM *n* times. Note that each upgrade + doubles the Node's RAM. So this is equivalent to multiplying the Node's RAM by + 2 :sup:`n`. + + Returns true if the Hacknet Node's RAM is successfully upgraded *n* times or if + it is upgraded some positive number of times and the Node reaches it max RAM. + + Returns false otherwise. diff --git a/doc/source/netscript/netscriptadvancedfunctions.rst b/doc/source/netscript/netscriptadvancedfunctions.rst index d41cc060b..60400e1f7 100644 --- a/doc/source/netscript/netscriptadvancedfunctions.rst +++ b/doc/source/netscript/netscriptadvancedfunctions.rst @@ -4,39 +4,7 @@ Netscript Advanced Functions These Netscript functions become relevant later on in the game. They are put on a separate page because they contain spoilers for the game. -getBitNodeMultipliers -^^^^^^^^^^^^^^^^^^^^^ +.. toctree:: -.. js:function:: getBitNodeMultipliers() - - Returns an object containing the current BitNode multipliers. This function requires Source-File 5 in order - to run. The multipliers are returned in decimal forms (e.g. 1.5 instead of 150%). The multipliers represent - the difference between the current BitNode and the original BitNode (BitNode-1). For example, if the - *CrimeMoney* multiplier has a value of 0.1, then that means that committing crimes in the current BitNode - will only give 10% of the money you would have received in BitNode-1. - - The structure of the returned object is subject to change as BitNode multipliers get added to the game. - Refer to the `source code here `_ - to see the name of the BitNode multipliers. - - Example:: - - mults = getBitNodeMultipliers(); - print(mults.ServerMaxMoney); - print(mults.HackExpGain); - -getHackTime, getGrowTime, & getWeakenTime -^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ - -The :js:func:`getHackTime`, :js:func:`getGrowTime`, and :js:func:`getWeakenTime` -all take an additional third optional parameter for specifying a specific intelligence -level to see how that would affect the hack/grow/weaken times. This parameter -defaults to your current intelligence level. - -(Intelligence is unlocked after obtaining Source-File 5). - -The function signatures are then:: - - getHackTime(hostname/ip[, hackLvl=current level, intLvl=current level]) - getGrowTime(hostname/ip[, hackLvl=current level, intLvl=current level]) - getWeakenTime(hostname/ip[, hackLvl=current level, intLvl=current level]) + getBitNodeMultipliers() + getHackTime(), getGrowTime(), & getWeakenTime() diff --git a/doc/source/netscript/netscriptbladeburnerapi.rst b/doc/source/netscript/netscriptbladeburnerapi.rst index 23c3584cf..4ff1db46a 100644 --- a/doc/source/netscript/netscriptbladeburnerapi.rst +++ b/doc/source/netscript/netscriptbladeburnerapi.rst @@ -25,6 +25,44 @@ In :ref:`netscriptjs`:: ns.bladeburner.getContractNames(); ns.bladeburner.startAction("general", "Training"); +.. toctree:: + :caption: Functions: + + getContractNames() + getOperationNames() + getBlackOpNames() + getGeneralActionNames() + getSkillNames() + startAction() + stopBladeburnerAction() + getCurrentAction() + getActionTime() + getActionEstimatedSuccessChance() + getActionRepGain() + getActionCountRemaining() + getActionMaxLevel() + getActionCurrentLevel() + getActionAutolevel() + setActionAutolevel() + setActionLevel() + getRank() + getSkillPoints() + getSkillLevel() + getSkillUpgradeCost() + upgradeSkill() + getTeamSize() + setTeamSize() + getCityEstimatedPopulation() + getCityEstimatedCommunities() + getCityChaos() + getCity() + switchCity() + getStamina() + joinBladeburnerFaction() + joinBladeburnerDivision() + getBonusTime() + + .. _bladeburner_action_types: Bladeburner Action Types @@ -58,356 +96,6 @@ its type and name. The following are valid values when specifying the action's t * general action * gen -getContractNames ----------------- - -.. js:function:: getContractNames() - - Returns an array of strings containing the names of all Bladeburner contracts - -getOperationNames ------------------ - -.. js:function:: getOperationNames() - - Returns an array of strings containing the names of all Bladeburner operations - -getBlackOpNames ---------------- - -.. js:function:: getBlackOpNames() - - Returns an array of strings containing the names of all Bladeburner Black Ops - -getGeneralActionNames ---------------------- - -.. js:function:: getGeneralActionNames() - - Returns an array of strings containing the names of all general Bladeburner actions - -getSkillNames -------------- - -.. js:function:: getSkillNames() - - Returns an array of strings containing the names of all Bladeburner skills - -startAction ------------ - -.. js:function:: startAction(type, name) - - :param string type: Type of action. See :ref:`bladeburner_action_types` - :param string name: Name of action. Must be an exact match - - Attempts to start the specified Bladeburner action. Returns true if the action - was started successfully, and false otherwise. - -stopBladeburnerAction ---------------------- - -.. js:function:: stopBladeburnerAction() - - Stops the current Bladeburner action - -getCurrentAction ----------------- - -.. js:function:: getCurrentAction() - - Returns an object that represents the player's current Bladeburner action:: - - { - type: Type of Action - name: Name of Action - } - - If the player is not performing an action, the function will return an object - with the 'type' property set to "Idle". - -getActionTime -------------- - -.. js:function:: getActionTime(type, name) - - :param string type: Type of action. See :ref:`bladeburner_action_types` - :param string name: Name of action. Must be an exact match - - Returns the number of seconds it takes to complete the specified action - -getActionEstimatedSuccessChance -------------------------------- - -.. js:function:: getActionEstimatedSuccessChance(type, name) - - :param string type: Type of action. See :ref:`bladeburner_action_types` - :param string name: Name of action. Must be an exact match - - Returns the estimated success chance for the specified action. This chance - is returned as a decimal value, NOT a percentage (e.g. if you have an estimated - success chance of 80%, then this function will return 0.80, NOT 80). - -getActionRepGain ----------------- - -.. js:function:: getActionRepGain(type, name[, level=current level]) - - :param string type: Type of action. See :ref:`bladeburner_action_types` - :param string name: Name of action. Must be an exact match - :param number level: Optional action level at which to calculate the gain - - Returns the average Bladeburner reputation gain for successfully completing - the specified action. Note that this value is an 'average' and the real - reputation gain may vary slightly from this value. - -getActionCountRemaining ------------------------ - -.. js:function:: getActionCountRemaining(type, name) - - :param string type: Type of action. See :ref:`bladeburner_action_types` - :param string name: Name of action. Must be an exact match - - Returns the remaining count of the specified action. - - Note that this is meant to be used for Contracts and Operations. - This function will return 'Infinity' for actions such as Training and Field Analysis. - -getActionMaxLevel ------------------ - -.. js:function:: getActionMaxLevel(type, name) - - :param string type: Type of action. See :ref:`bladeburner_action_types` - :param string name: Name of action. Must be an exact match - - Returns the maximum level for this action. - - Returns -1 if an invalid action is specified. - -getActionCurrentLevel ---------------------- - -.. js:function:: getActionCurrentLevel(type, name) - - :param string type: Type of action. See :ref:`bladeburner_action_types` - :param string name: Name of action. Must be an exact match - - Returns the current level of this action. - - Returns -1 if an invalid action is specified. - -getActionAutolevel ------------------- - -.. js:function:: getActionAutolevel(type, name) - - :param string type: Type of action. See :ref:`bladeburner_action_types` - :param string name: Name of action. Must be an exact match - - Return a boolean indicating whether or not this action is currently set to autolevel. - - Returns false if an invalid action is specified. - -setActionAutolevel ------------------- - -.. js:function:: setActionAutolevel(type, name, autoLevel) - - :param string type: Type of action. See :ref:`bladeburner_action_types` - :param string name: Name of action. Must be an exact match - :param boolean autoLevel: Whether or not to autolevel this action - - Enable/disable autoleveling for the specified action. - -setActionLevel --------------- - -.. js:function:: setActionLevel(type, name, level) - - :param string type: Type of action. See :ref:`bladeburner_action_types` - :param string name: Name of action. Must be an exact match - :param level int: Level to set this action to - - Set the level for the specified action. - -getRank -------- - -.. js:function:: getRank() - - Returns the player's Bladeburner Rank - -getSkillPoints --------------- - -.. js:function:: getSkillPoints() - - Returns the number of Bladeburner skill points you have - -getSkillLevel -------------- - -.. js:function:: getSkillLevel(skillName="") - - :param string skillName: Name of skill. Case-sensitive and must be an exact match - - This function returns your level in the specified skill. - - The function returns -1 if an invalid skill name is passed in - -getSkillUpgradeCost -------------------- - -.. js:function:: getSkillUpgradeCost(skillName="") - - :param string skillName: Name of skill. Case-sensitive and must be an exact match - - This function returns the number of skill points needed to upgrade the - specified skill. - - The function returns -1 if an invalid skill name is passed in. - -upgradeSkill ------------- - -.. js:function:: upgradeSkill(skillName) - - :param string skillName: Name of Skill to be upgraded. Case-sensitive and must be an exact match - - Attempts to upgrade the specified Bladeburner skill. Returns true if the - skill is successfully upgraded, and false otherwise - -getTeamSize ------------ - -.. js:function:: getTeamSize(type, name) - - :param string type: Type of action. See :ref:`bladeburner_action_types` - :param string name: Name of action. Must be an exact match - - Returns the number of Bladeburner team members you have assigned to the - specified action. - - Setting a team is only applicable for Operations and BlackOps. This function - will return 0 for other action types. - -setTeamSize ------------ - -.. js:function:: setTeamSize(type, name, size) - - :param string type: Type of action. See :ref:`bladeburner_action_types` - :param string name: Name of action. Must be an exact match - :param int size: Number of team members to set. Will be converted using Math.round() - - Set the team size for the specified Bladeburner action. - - Returns the team size that was set, or -1 if the function failed. - -getCityEstimatedPopulation --------------------------- - -.. js:function:: getCityEstimatedPopulation(cityName) - - :param string cityName: Name of city. Case-sensitive - - Returns the estimated number of Synthoids in the specified city, or -1 - if an invalid city was specified. - -getCityEstimatedCommunities ---------------------------- - -.. js:function:: getCityEstimatedCommunities(cityName) - - :param string cityName: Name of city. Case-sensitive - - Returns the estimated number of Synthoid communities in the specified city, - or -1 if an invalid city was specified. - -getCityChaos ------------- - -.. js:function:: getCityChaos(cityName) - - :param string cityName: Name of city. Case-sensitive - - Returns the chaos in the specified city, or -1 if an invalid city was specified - -getCity -------- - -.. js:function:: getCity() - - Returns the city that the player is currently in (for Bladeburner). - -switchCity ----------- - -.. js:function:: switchCity(cityName) - - :param string cityName: Name of city - - Attempts to switch to the specified city (for Bladeburner only). - - Returns true if successful, and false otherwise - -getStamina ----------- - -.. js:function:: getStamina() - - Returns an array with two elements: - - [Current stamina, Max stamina] - - Example usage:: - - function getStaminaPercentage() { - let res = bladeburner.getStamina(); - return res[0] / res[1]; - } - -joinBladeburnerFaction ----------------------- - -.. js:function:: joinBladeburnerFaction() - - Attempts to join the Bladeburner faction. - - Returns true if you successfully join the Bladeburner faction, or if - you are already a member. - - Returns false otherwise. - -joinBladeburnerDivision ------------------------ - -.. js:function:: joinBladeburnerDivision() - - Attempts to join the Bladeburner division. - - Returns true if you successfully join the Bladeburner division, or if you - are already a member. - - Returns false otherwise - -getBonusTime ------------- - -.. js:function:: getBonusTime() - - Returns the amount of accumulated "bonus time" (seconds) for the Bladeburner mechanic. - - "Bonus time" is accumulated when the game is offline or if the game is - inactive in the browser. - - "Bonus time" makes the game progress faster, up to 5x the normal speed. - For example, if an action takes 30 seconds to complete but you've accumulated - over 30 seconds in bonus time, then the action will only take 6 seconds - in real life to complete. - Examples -------- diff --git a/doc/source/netscript/netscriptcodingcontractapi.rst b/doc/source/netscript/netscriptcodingcontractapi.rst index 44db5e6f2..4b2e7ef3d 100644 --- a/doc/source/netscript/netscriptcodingcontractapi.rst +++ b/doc/source/netscript/netscriptcodingcontractapi.rst @@ -17,72 +17,10 @@ In :ref:`netscriptjs`:: ns.codingcontract.getDescription("foo.cct", "home"); ns.codingcontract.attempt(1, "foo.cct", "foodnstuff"); -attempt -------- +.. toctree:: -.. js:function:: attempt(answer, fn[, hostname/ip=current ip]) - - :param answer: Solution for the contract - :param string fn: Filename of the contract - :param string hostname/ip: Hostname or IP of the server containing the contract. - Optional. Defaults to current server if not provided - - Attempts to solve the Coding Contract with the provided solution. - - :returns: Boolean indicating whether the solution was correct - -getContractType ---------------- - -.. js:function:: getContractType(fn[, hostname/ip=current ip]) - - :param string fn: Filename of the contract - :param string hostname/ip: Hostname or IP of the server containing the contract. - Optional. Defaults to current server if not provided - - Returns a name describing the type of problem posed by the Coding Contract. - (e.g. Find Largest Prime Factor, Total Ways to Sum, etc.) - - :returns: A string with the contract's problem type - -getDescription --------------- - -.. js:function:: getDescription(fn[, hostname/ip=current ip]) - - :param string fn: Filename of the contract - :param string hostname/ip: Hostname or IP of the server containing the contract. - Optional. Defaults to current server if not provided - - Get the full text description for the problem posed by the Coding Contract - - :returns: A string with the contract's text description - -getData -------- - -.. js:function:: getData(fn[, hostname/ip=current ip]) - - :param string fn: Filename of the contract - :param string hostname/ip: Hostname or IP of the server containing the contract. - Optional. Defaults to current server if not provided - - Get the data associated with the specific Coding Contract. Note that this is - not the same as the contract's description. This is just the data that - the contract wants you to act on in order to solve - - :returns: The specified contract's data - -getNumTriesRemaining --------------------- - -.. js:function:: getNumTriesRemaining(fn[, hostname/ip=current ip]) - - :param string fn: Filename of the contract - :param string hostname/ip: Hostname or IP of the server containing the contract. - Optional. Defaults to current server if not provided - - Get the number of tries remaining on the contract before it - self-destructs. - - :returns: Number indicating how many attempts are remaining + attempt() + getContractType() + getDescription() + getData() + getNumTriesRemaining() diff --git a/doc/source/netscript/netscriptfunctions.rst b/doc/source/netscript/netscriptfunctions.rst index 5eac68ac5..58067d09d 100644 --- a/doc/source/netscript/netscriptfunctions.rst +++ b/doc/source/netscript/netscriptfunctions.rst @@ -6,1277 +6,83 @@ Netscript Basic Functions This page contains the complete documentation for all functions that are available in Netscript. This includes information such as function signatures, what they do, and their return values. -At the end is also a section that describes how to define your own functions in Netscript. - -hack -^^^^ - -.. js:function:: hack(hostname/ip) - - :param string hostname/ip: IP or hostname of the target server to hack - :returns: The amount of money stolen if the hack is successful, and zero otherwise - :RAM cost: 0.1 GB - - Function that is used to try and hack servers to steal money and gain hacking experience. The runtime for this command depends - on your hacking level and the target server's security level. In order to hack a server you must first gain root access - to that server and also have the required hacking level. - - A script can hack a server from anywhere. It does not need to be running on the same server to hack that server. For example, - you can create a script that hacks the 'foodnstuff' server and run that script on any server in the game. - - A successful hack() on a server will raise that server's security level by 0.002. - - Example:: - - hack("foodnstuff"); - hack("10.1.2.3"); - -grow -^^^^ - -.. js:function:: grow(hostname/ip) - - :param string hostname/ip: IP or hostname of the target server to grow - :returns: The number by which the money on the server was multiplied for the growth - :RAM cost: 0.15 GB - - Use your hacking skills to increase the amount of money available on a server. The runtime for this command depends on your hacking - level and the target server's security level. When grow() completes, the money available on a target server will be increased by a - certain, fixed percentage. This percentage is determined by the target server's growth rate (which varies between servers) and security level. - Generally, higher-level servers have higher growth rates. The getServerGrowth() function can be used to obtain a server's growth rate. - - Like hack(), grow() can be called on any server, regardless of where the script is running. The grow() command requires - root access to the target server, but there is no required hacking level to run the command. It also raises the security level - of the target server by 0.004. - - Example:: - - grow("foodnstuff"); - -weaken -^^^^^^ - -.. js:function:: weaken(hostname/ip) - - :param string hostname/ip: IP or hostname of the target server to weaken - :returns: The amount by which the target server's security level was decreased. This is equivalent to 0.05 multiplied - by the number of script threads - :RAM cost: 0.15 GB - - Use your hacking skills to attack a server's security, lowering the server's security level. The runtime for this command - depends on your hacking level and the target server's security level. This function lowers the security level of the target - server by 0.05. - - Like hack() and grow(), weaken() can be called on any server, regardless of where the script is running. This command requires - root access to the target server, but there is no required hacking level to run the command. - - Example:: - - weaken("foodnstuff"); - -hackAnalyzeThreads -^^^^^^^^^^^^^^^^^^ - -.. js:function:: hackAnalyzeThreads(hostname/ip, hackAmount) - - :param string hostname/ip: IP or hostname of server to analyze - :param number hackAmount: Amount of money you want to hack from the server - :returns: The number of threads needed to hack() the server for *hackAmount* money - :RAM cost: 1 GB - - This function returns the number of script threads you need when running - the `hack()` command to steal the specified amount of money from the target server. - - If `hackAmount` is less than zero or greater than the amount of money available - on the server, then this function returns -1. - - For example, let's say the `foodnstuff` server has $10m and you run:: - - hackAnalyzeThreads("foodnstuff", 1e6); - - If this function returns 50, this means that if your next `hack()` call - is run on a script with 50 threads, it will steal $1m from the `foodnstuff` server. - - **Warning**: The value returned by this function isn't necessarily a whole number. - -hackAnalyzePercent -^^^^^^^^^^^^^^^^^^ - -.. js:function:: hackAnalyzePercent(hostname/ip) - - :param string hostname/ip: IP or hostname of target server - :returns: The percentage of money you will steal from the target server with a single hack - :RAM cost: 1 GB - - Returns the percentage of the specified server's money you will steal with a - single hack. This value is returned in **percentage form, not decimal (Netscript - functions typically return in decimal form, but not this one).** - - For example, assume the following returns 1:: - - hackAnalyzePercent("foodnstuff"); - - This means that if hack the `foodnstuff` server, then you will steal 1% of its - total money. If you `hack()` using N threads, then you will steal N% of its total - money. - -hackChance -^^^^^^^^^^ - -.. js:function:: hackChance(hostname/ip) - - :param string hostname/ip: IP or hostname of target server - :returns: The chance you have of successfully hacking the target server - :RAM cost: 1 GB - - Returns the chance you have of successfully hacking the specified server. This - returned value is in decimal form, not percentage. - -growthAnalyze -^^^^^^^^^^^^^ - -.. js:function:: growthAnalyze(hostname/ip, growthAmount) - - :param string hostname/ip: IP or hostname of server to analyze - :param number growthAmount: Multiplicative factor by which the server is grown. Decimal form. - :returns: The amount of grow() calls needed to grow the specified server by the specified amount - :RAM cost: 1 GB - - This function returns the number of "growths" needed in order to increase the amount - of money available on the specified server by the specified amount. - - The specified amount is multiplicative and is in decimal form, not percentage. - - For example, if you want to determine how many `grow()` calls you need - to double the amount of money on `foodnstuff`, you would use:: - - growthAnalyze("foodnstuff", 2); - - If this returns 100, then this means you need to call `grow()` 100 times - in order to double the money (or once with 100 threads). - - **Warning**: The value returned by this function isn't necessarily a whole number. - -sleep -^^^^^ - -.. js:function:: sleep(n) - - :param number n: Number of milliseconds to sleep - :RAM cost: 0 GB - - Suspends the script for n milliseconds. - -print -^^^^^ - -.. js:function:: print(x) - - :param x: Value to be printed - :RAM cost: 0 GB - - Prints a value or a variable to the script's logs. - -tprint -^^^^^^ - -.. js:function:: tprint(x) - - :param x: Value to be printed - :RAM cost: 0 GB - - Prints a value or a variable to the Terminal - -clearLog -^^^^^^^^ - -.. js:function:: clearLog() - - :RAM cost: 0 GB - - Clears the script's logs - -disableLog -^^^^^^^^^^ - -.. js:function:: disableLog(fn) - - :param string fn: Name of function for which to disable logging - :RAM cost: 0 GB - - Disables logging for the given function. Logging can be disabled for - all functions by passing 'ALL' as the argument. - - Note that this does not completely remove all logging functionality. - This only stops a function from logging - when the function is successful. If the function fails, it will still log the reason for failure. - - Notable functions that cannot have their logs disabled: run, exec, exit - -enableLog -^^^^^^^^^ - -.. js:function:: enableLog(fn) - - :param string fn: Name of function for which to enable logging - :RAM cost: 0 GB - - Re-enables logging for the given function. If 'ALL' is passed into this function - as an argument, then it will revert the effects of disableLog('ALL') - -isLogEnabled -^^^^^^^^^^^^ - -.. js:function:: isLogEnabled(fn) - - :param string fn: Name of function to check - :RAM cost: 0 GB - - Returns a boolean indicating whether or not logging is enabled for that - function (or 'ALL') - -getScriptLogs -^^^^^^^^^^^^^ - -.. js:function:: getScriptLogs([fn], [hostname/ip=current ip], [args...]) - - :param string fn: Optional. Filename of script to get logs from. - :param string ip: Optional. IP or hostname of the server that the script is on - :param args...: Arguments to identify which scripts to get logs for - :RAM cost: 0 GB - - Returns a script's logs. The logs are returned as an array, where each - line is an element in the array. The most recently logged line is at the - end of the array. - - Note that there is a maximum number of lines that a script stores in its logs. - This is configurable in the game's options. - - If the function is called with no arguments, it will return the current script's logs. - - Otherwise, the `fn`, `hostname/ip,` and `args...` arguments can be used to get the logs - from another script. Remember that scripts are uniquely identified by both - their names and arguments. - - Examples:: - - // Get logs from foo.script on the current server that was run with no args - getScriptLogs("foo.script"); - - // Get logs from foo.script on the foodnstuff server that was run with no args - getScriptLogs("foo.script", "foodnstuff"); - - // Get logs from foo.script on the foodnstuff server that was run with the arguments [1, "test"] - getScriptLogs("foo.script", "foodnstuff", 1, "test"); - -scan -^^^^ - -.. js:function:: scan(hostname/ip=current ip[, hostnames=true]) - - :param string hostname/ip: IP or hostname of the server to scan - :param boolean: Optional boolean specifying whether the function should output hostnames (if true) or IP addresses (if false) - :RAM cost: 0.2 GB - - Returns an array containing the hostnames or IPs of all servers that are one node way from the specified target server. The - hostnames/IPs in the returned array are strings. - -nuke -^^^^ - -.. js:function:: nuke(hostname/ip) - - :param string hostname/ip: IP or hostname of the target server - :RAM cost: 0 GB - - Runs the NUKE.exe program on the target server. NUKE.exe must exist on your home computer. - - Example:: - - nuke("foodnstuff"); - -brutessh -^^^^^^^^ - -.. js:function:: brutessh(hostname/ip) - - :param string hostname/ip: IP or hostname of the target server - :RAM cost: 0 GB - - Runs the BruteSSH.exe program on the target server. BruteSSH.exe must exist on your home computer. - - Example:: - - brutessh("foodnstuff"); - -ftpcrack -^^^^^^^^ - -.. js:function:: ftpcrack(hostname/ip) - - :param string hostname/ip: IP or hostname of the target server - :RAM cost: 0 GB - - Runs the FTPCrack.exe program on the target server. FTPCrack.exe must exist on your home computer. - - Example:: - - ftpcrack("foodnstuff"); - -relaysmtp -^^^^^^^^^ - -.. js:function:: relaysmtp(hostname/ip) - - :param string hostname/ip: IP or hostname of the target server - :RAM cost: 0 GB - - Runs the relaySMTP.exe program on the target server. relaySMTP.exe must exist on your home computer. - - Example:: - - relaysmtp("foodnstuff"); - -httpworm -^^^^^^^^ - -.. js:function:: httpworm(hostname/ip) - - :param string hostname/ip: IP or hostname of the target server - :RAM cost: 0 GB - - Runs the HTTPWorm.exe program on the target server. HTTPWorm.exe must exist on your home computer. - - Example:: - - httpworm("foodnstuff"); - -sqlinject -^^^^^^^^^ - -.. js:function:: sqlinject(hostname/ip) - - :param string hostname/ip: IP or hostname of the target server - :RAM cost: 0 GB - - Runs the SQLInject.exe program on the target server. SQLInject.exe must exist on your home computer. - - Example:: - - sqlinject("foodnstuff"); - -run -^^^ - -.. js:function:: run(script, [numThreads=1], [args...]) - - :param string script: Filename of script to run - :param number numThreads: Optional thread count for new script. Set to 1 by default. Will be rounded to nearest integer - :param args...: - Additional arguments to pass into the new script that is being run. Note that if any arguments are being - passed into the new script, then the second argument *numThreads* must be filled in with a value. - :RAM cost: 1 GB - - Run a script as a separate process. This function can only be used to run scripts located on the current server (the server - running the script that calls this function). - - Returns true if the script is successfully started, and false otherwise. - - Running this function with a *numThreads* argument of 0 will return false without running the script. - However, running this function with a negative *numThreads* argument will cause a runtime error. - - The simplest way to use the *run* command is to call it with just the script name. The following example will run - 'foo.script' single-threaded with no arguments:: - - run("foo.script"); - - The following example will run 'foo.script' but with 5 threads instead of single-threaded:: - - run("foo.script", 5); - - This next example will run 'foo.script' single-threaded, and will pass the string 'foodnstuff' into the script - as an argument:: - - run("foo.script", 1, 'foodnstuff'); - -exec -^^^^ - -.. js:function:: exec(script, hostname/ip, [numThreads=1], [args...]) - - :param string script: Filename of script to execute - :param string hostname/ip: IP or hostname of the 'target server' on which to execute the script - :param number numThreads: Optional thread count for new script. Set to 1 by default. Will be rounded to nearest integer - :param args...: - Additional arguments to pass into the new script that is being run. Note that if any arguments are being - passed into the new script, then the third argument *numThreads* must be filled in with a value. - :RAM cost: 1.3 GB - - Run a script as a separate process on a specified server. This is similar to the *run* function except - that it can be used to run a script on any server, instead of just the current server. - - Returns true if the script is successfully started, and false otherwise. - - Running this function with a *numThreads* argument of 0 will return false without running the script. - However, running this function with a negative *numThreads* argument will cause a runtime error. - - The simplest way to use the *exec* command is to call it with just the script name and the target server. - The following example will try to run *generic-hack.script* on the *foodnstuff* server:: - - exec("generic-hack.script", "foodnstuff"); - - The following example will try to run the script *generic-hack.script* on the *joesguns* server with 10 threads:: - - exec("generic-hack.script", "joesguns", 10); - - This last example will try to run the script *foo.script* on the *foodnstuff* server with 5 threads. It will also pass - the number 1 and the string "test" in as arguments to the script:: - - exec("foo.script", "foodnstuff", 5, 1, "test"); - -spawn -^^^^^ - -.. js:function:: spawn(script, numThreads, [args...]) - - :param string script: Filename of script to execute - :param number numThreads: Number of threads to spawn new script with. Will be rounded to nearest integer - :param args...: - Additional arguments to pass into the new script that is being run. - :RAM cost: 2 GB - - Terminates the current script, and then after a delay of about 20 seconds it will execute the newly-specified script. - The purpose of this function is to execute a new script without being constrained by the RAM usage of the current one. - This function can only be used to run scripts on the local server. - - Because this function immediately terminates the script, it does not have a return value. - - The following example will execute the script 'foo.script' with 10 threads and the arguments 'foodnstuff' and 90:: - - spawn('foo.script', 10, 'foodnstuff', 90); - -kill -^^^^ - -.. js:function:: kill(script, hostname/ip, [args...]) - - :param string script: Filename of the script to kill - :param string hostname/ip: IP or hostname of the server on which to kill the script - :param args...: Arguments to identify which script to kill - :RAM cost: 0.5 GB - - Kills the script on the target server specified by the script's name and arguments. Remember that scripts - are uniquely identified by both their name and arguments. For example, if *foo.script* is run with the argument 1, then this - is not the same as *foo.script* run with the argument 2, even though they have the same code. - - If this function successfully kills the specified script, then it will return true. Otherwise, it will return false. - - Examples: - - The following example will try to kill a script named *foo.script* on the *foodnstuff* server that was ran with no arguments:: - - kill("foo.script", "foodnstuff"); - - The following will try to kill a script named *foo.script* on the current server that was ran with no arguments:: - - kill("foo.script", getHostname()); - - The following will try to kill a script named *foo.script* on the current server that was ran with the arguments 1 and "foodnstuff":: - - kill("foo.script", getHostname(), 1, "foodnstuff"); - -killall -^^^^^^^ - -.. js:function:: killall(hostname/ip) - - :param string hostname/ip: IP or hostname of the server on which to kill all scripts - :RAM cost: 0.5 GB - - Kills all running scripts on the specified server. This function returns true if any scripts were killed, and - false otherwise. In other words, it will return true if there are any scripts running on the target server. - - -exit -^^^^ - -.. js:function:: exit() - - :RAM cost: 0 GB - - Terminates the current script immediately - -scp -^^^ - -.. js:function:: scp(files, [source], destination) - - :param string/array files: Filename or an array of filenames of script/literature files to copy - :param string source: - Hostname or IP of the source server, which is the server from which the file will be copied. - This argument is optional and if it's omitted the source will be the current server. - :param string destination: Hostname or IP of the destination server, which is the server to which the file will be copied. - :RAM cost: 0.6 GB - - Copies a script or literature (.lit) file(s) to another server. The *files* argument can be either a string specifying a - single file to copy, or an array of strings specifying multiple files to copy. - - Returns true if the script/literature file is successfully copied over and false otherwise. If the *files* argument is an array - then this function will return true if at least one of the files in the array is successfully copied. - - Examples:: - - //Copies hack-template.script from the current server to foodnstuff - scp("hack-template.script", "foodnstuff"); - - //Copies foo.lit from the helios server to the home computer - scp("foo.lit", "helios", "home"); - - //Tries to copy three files from rothman-uni to home computer - files = ["foo1.lit", "foo2.script", "foo3.script"]; - scp(files, "rothman-uni", "home"); - -ls -^^ - -.. js:function:: ls(hostname/ip, [grep]) - - :param string hostname/ip: Hostname or IP of the target server - :param string grep: a substring to search for in the filename - :RAM cost: 0 GB - - Returns an array with the filenames of all files on the specified server (as strings). The returned array - is sorted in alphabetic order - -ps -^^ - -.. js:function:: ps(hostname/ip=current ip) - - :param string ip: Hostname or IP address of the target server. - If not specified, it will be the current server's IP by default - :RAM cost: 0.2 GB - - Returns an array with general information about all scripts running on the specified - target server. The information for each server is given in an object with - the following structure:: - - { - filename: Script name, - threads: Number of threads script is running with, - args: Script's arguments - } - - Example usage (using :doc:`netscriptjs`):: - - export async function main(ns) { - const ps = ns.ps("home"); - for (let i = 0; i < ps.length; ++i) { - ns.tprint(ps[i].filename + ' ' + ps[i].threads); - ns.tprint(ps[i].args); - } - } - -hasRootAccess -^^^^^^^^^^^^^ - -.. js:function:: hasRootAccess(hostname/ip) - - :param string hostname/ip: Hostname or IP of the target server - :RAM cost: 0.05 GB - - Returns a boolean indicating whether or not the player has root access to the specified target server. - - Example:: - - if (hasRootAccess("foodnstuff") == false) { - nuke("foodnstuff"); - } - -getHostname -^^^^^^^^^^^ - -.. js:function:: getHostname() - - :RAM cost: 0.05 GB - - Returns a string with the hostname of the server that the script is running on - -getHackingLevel -^^^^^^^^^^^^^^^ - -.. js:function:: getHackingLevel() - - :RAM cost: 0.05 GB - - Returns the player's current hacking level - -getHackingMultipliers -^^^^^^^^^^^^^^^^^^^^^ - -.. js:function:: getHackingMultipliers() - - :RAM cost: 4 GB - - Returns an object containing the Player's hacking related multipliers. These multipliers are - returned in decimal forms, not percentages (e.g. 1.5 instead of 150%). The object has the following structure:: - - { - chance: Player's hacking chance multiplier, - speed: Player's hacking speed multiplier, - money: Player's hacking money stolen multiplier, - growth: Player's hacking growth multiplier - } - - Example of how this can be used:: - - mults = getHackingMultipliers(); - print(mults.chance); - print(mults.growth); - -getHacknetMultipliers -^^^^^^^^^^^^^^^^^^^^^ - -.. js:function:: getHacknetMultipliers() - - :RAM cost: 4 GB - - Returns an object containing the Player's hacknet related multipliers. These multipliers are - returned in decimal forms, not percentages (e.g. 1.5 instead of 150%). The object has the following structure:: - - { - production: Player's hacknet production multiplier, - purchaseCost: Player's hacknet purchase cost multiplier, - ramCost: Player's hacknet ram cost multiplier, - coreCost: Player's hacknet core cost multiplier, - levelCost: Player's hacknet level cost multiplier - } - - Example of how this can be used:: - - mults = getHacknetMultipliers(); - print(mults.production); - print(mults.purchaseCost); - - - -getServerMoneyAvailable -^^^^^^^^^^^^^^^^^^^^^^^ - -.. js:function:: getServerMoneyAvailable(hostname/ip) - - :param string hostname/ip: Hostname or IP of target server - :RAM cost: 0.1 GB - - Returns the amount of money available on a server. **Running this function on the home computer will return - the player's money.** - - Example:: - - getServerMoneyAvailable("foodnstuff"); - getServerMoneyAvailable("home"); //Returns player's money - -getServerMaxMoney -^^^^^^^^^^^^^^^^^ - -.. js:function:: getServerMaxMoney(hostname/ip) - - :param string hostname/ip: Hostname or IP of target server - :RAM cost: 0.1 GB - - Returns the maximum amount of money that can be available on a server - -getServerGrowth -^^^^^^^^^^^^^^^ - -.. js:function:: getServerGrowth(hostname/ip) - - :param string hostname/ip: Hostname or IP of target server - :RAM cost: 0.1 GB - - Returns the server's instrinsic "growth parameter". This growth parameter is a number - between 1 and 100 that represents how quickly the server's money grows. This parameter affects the - percentage by which the server's money is increased when using the *grow()* function. A higher - growth parameter will result in a higher percentage increase from *grow()*. - -getServerSecurityLevel -^^^^^^^^^^^^^^^^^^^^^^ - -.. js:function:: getServerSecurityLevel(hostname/ip) - - :param string hostname/ip: Hostname or IP of target server - :RAM cost: 0.1 GB - - Returns the security level of the target server. A server's security level is denoted by a number, typically - between 1 and 100 (but it can go above 100). - -getServerBaseSecurityLevel -^^^^^^^^^^^^^^^^^^^^^^^^^^ - -.. js:function:: getServerBaseSecurityLevel(hostname/ip) - - :param string hostname/ip: Hostname or IP of target server - :RAM cost: 0.1 GB - - Returns the base security level of the target server. This is the security level that the server starts out with. - This is different than *getServerSecurityLevel()* because *getServerSecurityLevel()* returns the current - security level of a server, which can constantly change due to *hack()*, *grow()*, and *weaken()*, calls on that - server. The base security level will stay the same until you reset by installing an Augmentation(s). - -getServerMinSecurityLevel -^^^^^^^^^^^^^^^^^^^^^^^^^ - -.. js:function:: getServerMinSecurityLevel(hostname/ip) - - :param string hostname/ip: Hostname or IP of target server - :RAM cost: 0.1 GB - - Returns the minimum security level of the target server - -getServerRequiredHackingLevel -^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ - -.. js:function:: getServerRequiredHackingLevel(hostname/ip) - - :param string hostname/ip: Hostname or IP of target server - :RAM cost: 0.1 GB - - Returns the required hacking level of the target server - -getServerNumPortsRequired -^^^^^^^^^^^^^^^^^^^^^^^^^ - -.. js:function:: getServerNumPortsRequired(hostname/ip) - - :param string hostname/ip: Hostname or IP of target server - :RAM cost: 0.1 GB - - Returns the number of open ports required to successfully run NUKE.exe on the specified server. - -getServerRam -^^^^^^^^^^^^ - -.. js:function:: getServerRam(hostname/ip) - - :param string hostname/ip: Hostname or IP of target server - :RAM cost: 0.1 GB - - Returns an array with two elements that gives information about a server's memory (RAM). The first - element in the array is the amount of RAM that the server has total (in GB). The second element in - the array is the amount of RAM that is currently being used on the server (in GB). - - Example:: - - res = getServerRam("helios"); - totalRam = res[0]; - ramUsed = res[1]; - -serverExists -^^^^^^^^^^^^ - -.. js:function:: serverExists(hostname/ip) - - :param string hostname/ip: Hostname or IP of target server - :RAM cost: 0.1 GB - - Returns a boolean denoting whether or not the specified server exists - -fileExists -^^^^^^^^^^ - -.. js:function:: fileExists(filename, [hostname/ip]) - - :param string filename: Filename of file to check - :param string hostname/ip: - Hostname or IP of target server. This is optional. If it is not specified then the - function will use the current server as the target server - :RAM cost: 0.1 GB - - Returns a boolean indicating whether the specified file exists on the target server. The filename - for scripts is case-sensitive, but for other types of files it is not. For example, *fileExists("brutessh.exe")* - will work fine, even though the actual program is named "BruteSSH.exe". - - If the *hostname/ip* argument is omitted, then the function will search through the current server (the server - running the script that calls this function) for the file. - - Examples:: - - fileExists("foo.script", "foodnstuff"); - fileExists("ftpcrack.exe"); - - The first example above will return true if the script named *foo.script* exists on the *foodnstuff* server, and false otherwise. - The second example above will return true if the current server contains the *FTPCrack.exe* program, and false otherwise. - -isRunning -^^^^^^^^^ - -.. js:function:: isRunning(filename, hostname/ip, [args...]) - - :param string filename: Filename of script to check. This is case-sensitive. - :param string hostname/ip: Hostname or IP of target server - :param args...: Arguments to specify/identify which scripts to search for - :RAM cost: 0.1 GB - - Returns a boolean indicating whether the specified script is running on the target server. Remember that a script is - uniquely identified by both its name and its arguments. - - **Examples:** - - In this first example below, the function call will return true if there is a script named *foo.script* with no arguments - running on the *foodnstuff* server, and false otherwise:: - - isRunning("foo.script", "foodnstuff"); - - In this second example below, the function call will return true if there is a script named *foo.script* with no arguments - running on the current server, and false otherwise:: - - isRunning("foo.script", getHostname()); - - In this next example below, the function call will return true if there is a script named *foo.script* running with the arguments - 1, 5, and "test" (in that order) on the *joesguns* server, and false otherwise:: - - isRunning("foo.script", "joesguns", 1, 5, "test"); - -getNextHacknetNodeCost -^^^^^^^^^^^^^^^^^^^^^^ - -.. js:function:: getNextHacknetNodeCost() - - :RAM cost: 0 GB - - Deprecated (no longer usable). See :doc:`netscripthacknetnodeapi` - -purchaseHacknetNode -^^^^^^^^^^^^^^^^^^^ - -.. js:function:: purchaseHacknetNode() - - :RAM cost: 0 GB - - Deprecated (no longer usable). See :doc:`netscripthacknetnodeapi` - -getPurchasedServerCost -^^^^^^^^^^^^^^^^^^^^^^ - -.. js:function:: getPurchasedServerCost(ram) - - :RAM cost: 0.25 GB - - :param number ram: Amount of RAM of a potential purchased server. Must be a power of 2 (2, 4, 8, 16, etc.). Maximum value of 1048576 (2^20) - - Returns the cost to purchase a server with the specified amount of *ram*. - - Examples:: - - for (i = 1; i <= 20; i++) { - tprint(i + " -- " + getPurchasedServerCost(Math.pow(2, i))); - } - -purchaseServer -^^^^^^^^^^^^^^ - -.. js:function:: purchaseServer(hostname, ram) - - :param string hostname: Hostname of the purchased server - :param number ram: Amount of RAM of the purchased server. Must be a power of 2 (2, 4, 8, 16, etc.). Maximum value of 1048576 (2^20) - :RAM cost: 2.25 GB - - Purchased a server with the specified hostname and amount of RAM. - - The *hostname* argument can be any data type, but it will be converted to a string and have whitespace removed. Anything that resolves to an empty string will - cause the function to fail. If there is already a server with the specified hostname, then the function will automatically append - a number at the end of the *hostname* argument value until it finds a unique hostname. For example, if the script calls - *purchaseServer("foo", 4)* but a server named "foo" already exists, the it will automatically change the hostname to "foo-0". If there is already - a server with the hostname "foo-0", then it will change the hostname to "foo-1", and so on. - - Note that there is a maximum limit to the amount of servers you can purchase. - - Returns the hostname of the newly purchased server as a string. If the function fails to purchase a server, then it will return an - empty string. The function will fail if the arguments passed in are invalid, if the player does not have enough money to purchase - the specified server, or if the player has exceeded the maximum amount of servers. - - Example:: - - ram = 64; - hn = "pserv-"; - for (i = 0; i < 5; ++i) { - purchaseServer(hn + i, ram); - } - -deleteServer -^^^^^^^^^^^^ - -.. js:function:: deleteServer(hostname) - - :param string hostname: Hostname of the server to delete - :RAM cost: 2.25 GB - - Deletes one of your purchased servers, which is specified by its hostname. - - The *hostname* argument can be any data type, but it will be converted to a string. Whitespace is automatically removed from - the string. This function will not delete a server that still has scripts running on it. - - Returns true if successful, and false otherwise. - -getPurchasedServers -^^^^^^^^^^^^^^^^^^^ - -.. js:function:: getPurchasedServers([hostname=true]) - - :param boolean hostname: - Specifies whether hostnames or IP addresses should be returned. If it's true then hostnames will be returned, and if false - then IPs will be returned. If this argument is omitted then it is true by default - :RAM cost: 2.25 GB - - Returns an array with either the hostnames or IPs of all of the servers you have purchased. - -getPurchasedServerLimit -^^^^^^^^^^^^^^^^^^^^^^^ - -.. js:function:: getPurchasedServerLimit() - - :RAM cost: 0.05 GB - - Returns the maximum number of servers you can purchase - -getPurchasedServerMaxRam -^^^^^^^^^^^^^^^^^^^^^^^^ - -.. js:function:: getPurchasedServerMaxRam() - - :RAM cost: 0.05 GB - - Returns the maximum RAM that a purchased server can have - -write -^^^^^ - -.. js:function:: write(port/fn, data="", mode="a") - - :param string/number port/fn: Port or text file/script that will be written to - :param string data: Data to write - :param string mode: Defines the write mode. Only valid when writing to text files or scripts. - :RAM cost: 1 GB - - This function can be used to either write data to a port, a text file (.txt), or a script (.script, .js, .ns) - - If the first argument is a number between 1 and 20, then it specifies a port and this function will write *data* to that port. Read - about how :ref:`netscript_ports` work here. The third argument, *mode*, is not used - when writing to a port. - - If the first argument is a string, then it specifies the name of a text file or script and this function will write *data* to that text file/script. If the - specified text file/script does not exist, then it will be created. The third argument *mode*, defines how the data will be written. If *mode* - is set to "w", then the data is written in "write" mode which means that it will overwrite all existing data on the text file/script. If *mode* is set to - any other value then the data will be written in "append" mode which means that the data will be added at the end of the file. - -tryWrite -^^^^^^^^ - -.. js:function:: tryWrite(port, data="") - - :param number port: Port to be written to - :param string data: Data to try to write - :returns: True if the data is successfully written to the port, and false otherwise - :RAM cost: 1 GB - - Attempts to write data to the specified Netscript Port. If the port is full, the data will - not be written. Otherwise, the data will be written normally - -read -^^^^ - -.. js:function:: read(port/fn) - - :param string/number port/fn: Port or text file to read from - :RAM cost: 1 GB - - This function is used to read data from a port, a text file (.txt), or a script (.script, .js, .ns) - - If the argument *port/fn* is a number between 1 and 20, then it specifies a port and it will read data from that port. Read - about how :ref:`netscript_ports` work here. A port is a serialized queue. This function - will remove the first element from that queue and return it. If the queue is empty, then the string "NULL PORT DATA" will be returned. - - If the argument *port/fn* is a string, then it specifies the name of a text file or script and this function will return the data in the specified text file/script. If - the text file does not exist, an empty string will be returned. - -peek -^^^^ - -.. js:function:: peek(port) - - :param number port: Port to peek. Must be an integer between 1 and 20 - :RAM cost: 1 GB - - This function is used to peek at the data from a port. It returns the first element in the specified port - without removing that element. If the port is empty, the string "NULL PORT DATA" will be returned. - - Read about how :ref:`netscript_ports` work here - -clear -^^^^^ - -.. js:function:: clear(port/fn) - - :param string/number port/fn: Port or text file to clear - :RAM cost: 1 GB - - This function is used to clear data in a `Netscript Ports `_ or a text file. - - If the *port/fn* argument is a number between 1 and 20, then it specifies a port and will clear it (deleting all data from the underlying queue). - - If the *port/fn* argument is a string, then it specifies the name of a text file (.txt) and will delete all data from that text file. - -getPortHandle -^^^^^^^^^^^^^ - -.. js:function:: getPortHandle(port) - - :param number port: Port number - :RAM cost: 10 GB - - Get a handle to a Netscript Port. See more details here: :ref:`netscript_ports` - - **WARNING:** Port Handles only work in :ref:`netscriptjs`. They will not work in :ref:`netscript1`. - -rm -^^ - -.. js:function:: rm(fn[, hostname/ip=current server]) - - :param string fn: Filename of file to remove. Must include the extension - :param string hostname/ip: Hostname or IP Address of the server on which to delete the file. Optional. Defaults to current server - :returns: True if it successfully deletes the file, and false otherwise - :RAM cost: 1 GB - - Removes the specified file from the current server. This function works for every file type except message (.msg) files. - -scriptRunning -^^^^^^^^^^^^^ - -.. js:function:: scriptRunning(scriptname, hostname/ip) - - :param string scriptname: Filename of script to check. This is case-sensitive. - :param string hostname/ip: Hostname or IP of target server - :RAM cost: 1 GB - - Returns a boolean indicating whether any instance of the specified script is running on the target server, regardless of - its arguments. - - This is different than the *isRunning()* function because it does not try to identify a specific instance of a running script - by its arguments. - - **Examples:** - - The example below will return true if there is any script named *foo.script* running on the *foodnstuff* server, and false otherwise:: - - scriptRunning("foo.script", "foodnstuff"); - - The example below will return true if there is any script named "foo.script" running on the current server, and false otherwise:: - - scriptRunning("foo.script", getHostname()); - -scriptKill -^^^^^^^^^^ - -.. js:function:: scriptKill(scriptname, hostname/ip) - - :param string scriptname: Filename of script to kill. This is case-sensitive. - :param string hostname/ip: Hostname or IP of target server - :RAM cost: 1 GB - - Kills all scripts with the specified filename on the target server specified by *hostname/ip*, regardless of arguments. Returns - true if one or more scripts were successfully killed, and false if none were. - -getScriptName -^^^^^^^^^^^^^ - -.. js:function:: getScriptName() - - :RAM cost: 0 GB - - Returns the current script name - -getScriptRam -^^^^^^^^^^^^ - -.. js:function:: getScriptRam(scriptname[, hostname/ip]) - - :param string scriptname: Filename of script. This is case-sensitive. - :param string hostname/ip: Hostname or IP of target server the script is located on. This is optional, If it is not specified then the function will set the current server as the target server. - :RAM cost: 0.1 GB - - Returns the amount of RAM required to run the specified script on the target server. Returns - 0 if the script does not exist. - -getHackTime -^^^^^^^^^^^ - -.. js:function:: getHackTime(hostname/ip[, hackLvl=current level]) - - :param string hostname/ip: Hostname or IP of target server - :param number hackLvl: Optional hacking level for the calculation. Defaults to player's current hacking level - :RAM cost: 0.05 GB - - Returns the amount of time in seconds it takes to execute the *hack()* Netscript function on the target server. - - The function takes in an optional *hackLvl* parameter that can be specified - to see what the hack time would be at different hacking levels. - -getGrowTime -^^^^^^^^^^^ - -.. js:function:: getGrowTime(hostname/ip[, hackLvl=current level]) - - :param string hostname/ip: Hostname or IP of target server - :param number hackLvl: Optional hacking level for the calculation. Defaults to player's current hacking level - :RAM cost: 0.05 GB - - Returns the amount of time in seconds it takes to execute the *grow()* Netscript function on the target server. - - The function takes in an optional *hackLvl* parameter that can be specified - to see what the grow time would be at different hacking levels. - -getWeakenTime -^^^^^^^^^^^^^ - -.. js:function:: getWeakenTime(hostname/ip[, hackLvl=current level]) - - :param string hostname/ip: Hostname or IP of target server - :param number hackLvl: Optional hacking level for the calculation. Defaults to player's current hacking level - :RAM cost: 0.05 GB - - Returns the amount of time in seconds it takes to execute the *weaken()* Netscript function on the target server. - - The function takes in an optional *hackLvl* parameter that can be specified - to see what the weaken time would be at different hacking levels. - -getScriptIncome -^^^^^^^^^^^^^^^ - -.. js:function:: getScriptIncome([scriptname], [hostname/ip], [args...]) - - :param string scriptname: Filename of script - :param string hostname/ip: Server on which script is running - :param args...: Arguments that the script is running with - :RAM cost: 0.1 GB - - Returns the amount of income the specified script generates while online (when the game is open, does not apply for offline income). - Remember that a script is uniquely identified by both its name and its arguments. So for example if you ran a script with the arguments - "foodnstuff" and "5" then in order to use this function to get that script's income you must specify those same arguments in the same order - in this function call. - - This function can also be called with no arguments. If called with no arguments, then this function will return an array of two values. The - first value is the total income ($ / second) of all of your active scripts (scripts that are currently running on any server). The second value - is the total income ($ / second) that you've earned from scripts since you last installed Augmentations. - -getScriptExpGain -^^^^^^^^^^^^^^^^ - -.. js:function:: getScriptExpGain([scriptname], [hostname/ip], [args...]) - - :param string scriptname: Filename of script - :param string hostname/ip: Server on which script is running - :param args...: Arguments that the script is running with - :RAM cost: 0.1 GB - - Returns the amount of hacking experience the specified script generates while online (when the game is open, does not apply for offline experience gains). - Remember that a script is uniquely identified by both its name and its arguments. - - This function can also return the total experience gain rate of all of your active scripts by running the function with no arguments. - -getTimeSinceLastAug -^^^^^^^^^^^^^^^^^^^ - -.. js:function:: getTimeSinceLastAug() - - :RAM cost: 0.05 GB - - Returns the amount of time in milliseconds that have passed since you last installed Augmentations - -sprintf -^^^^^^^ - -.. js:function:: sprintf() - - :RAM cost: 0 GB - - See `this link `_ for details. - -vsprintf -^^^^^^^^ - -.. js:function:: vsprintf() - - :RAM cost: 0 GB - - See `this link `_ for details. - -nFormat -^^^^^^^ - -.. js:function:: nFormat(n, format) - - :param number n: Number to format - :param string format: Formatter - - Converts a number into a string with the specified formatter. This uses the - `numeraljs `_ library, so the formatters must be compatible - with that. - - This is the same function that the game itself uses to display numbers. - - Examples:: - - nFormat(1.23e9, "$0.000a"); // Returns "$1.230b" - nFormat(12345.678, "0,0"); // Returns "12,346" - nFormat(0.84, "0.0%"); // Returns "84.0% - -prompt -^^^^^^ - -.. js:function:: prompt(txt) - - :param string txt: Text to appear in the prompt dialog box - :RAM cost: 0 GB - - Prompts the player with a dialog box with two options: "Yes" and "No". This function will return true if the player click "Yes" and - false if the player clicks "No". The script's execution is halted until the player selects one of the options. - -wget -^^^^ - -.. js:function:: wget(url, target[, hostname/ip=current ip]) - - :param string url: URL to pull data from - :param string target: Filename to write data to. Must be script or text file - :param string ip: Optional hostname/ip of server for target file. - :RAM cost: 0 GB - - Retrieves data from a URL and downloads it to a file on the specified server. The data can only - be downloaded to a script (.script, .ns, .js) or a text file (.txt). If the file already exists, - it will be overwritten by this command. - - Note that it will not be possible to download data from many websites because they do not allow - cross-origin resource sharing (CORS). Example:: - - wget("https://raw.githubusercontent.com/danielyxie/bitburner/master/README.md", "game_readme.txt"); - - **IMPORTANT:** This is an asynchronous function that returns a Promise. The Promise's resolved - value will be a boolean indicating whether or not the data was successfully - retrieved from the URL. Because the function is async and returns a Promise, - it is recommended you use :code:`wget` in :ref:`netscriptjs`. - - In NetscriptJS, you must preface any call to - :code:`wget` with the :code:`await` keyword (like you would :code:`hack` or :code:`sleep`). - - :code:`wget` will still work in :ref:`netscript1`, but the functions execution will not - be synchronous (i.e. it may not execute when you expect/want it to). Furthermore, since Promises are not - supported in ES5, you will not be able to process the returned value of :code:`wget` in - Netscript 1.0. - -getFavorToDonate -^^^^^^^^^^^^^^^^ - - :RAM cost: 0.1 GB - - Returns the amount of Faction favor required to be able to donate to a faction. +.. toctree:: + :caption: Functions: + + hack() + grow() + weaken() + hackAnalyzeThreads() + hackAnalyzePercent() + hackChance() + growthAnalyze() + sleep() + print() + tprint() + clearLog() + disableLog() + enableLog() + isLogEnabled() + getScriptLogs() + scan() + nuke() + brutessh() + ftpcrack() + relaysmtp() + httpworm() + sqlinject() + run() + exec() + spawn() + kill() + killall() + exit() + scp() + ls() + ps() + hasRootAccess() + getHostname() + getHackingLevel() + getHackingMultipliers() + getHacknetMultipliers() + getServerMoneyAvailable() + getServerMaxMoney() + getServerGrowth() + getServerSecurityLevel() + getServerBaseSecurityLevel() + getServerMinSecurityLevel() + getServerRequiredHackingLevel() + getServerNumPortsRequired() + getServerRam() + serverExists() + fileExists() + isRunning() + getPurchasedServerCost() + purchaseServer() + deleteServer() + getPurchasedServers() + getPurchasedServerLimit() + getPurchasedServerMaxRam() + write() + tryWrite() + read() + peek() + clear() + getPortHandle() + rm() + scriptRunning() + scriptKill() + getScriptName() + getScriptRam() + getHackTime() + getGrowTime() + getWeakenTime() + getScriptIncome() + getScriptExpGain() + getTimeSinceLastAug() + sprintf() + vsprintf() + nFormat() + prompt() + wget() + getFavorToDonate() diff --git a/doc/source/netscript/netscriptgangapi.rst b/doc/source/netscript/netscriptgangapi.rst index 392f7192c..95c322845 100644 --- a/doc/source/netscript/netscriptgangapi.rst +++ b/doc/source/netscript/netscriptgangapi.rst @@ -23,255 +23,22 @@ In :ref:`netscriptjs`:: ns.gang.getMemberNames(); ns.gang.recruitMember("Fry"); -getMemberNames --------------- - -.. js:function:: getMemberNames() - - Get the names of all Gang members - - :returns: An array of the names of all Gang members as strings - -getGangInformation ------------------- - -.. js:function:: getGangInformation() - - Get general information about the gang - - :returns: An object with the gang information. - - The object has the following structure:: - - { - faction: Name of faction that the gang belongs to ("Slum Snakes", etc.) - isHacking: Boolean indicating whether or not its a hacking gang - moneyGainRate: Money earned per second - power: Gang's power for territory warfare - respect: Gang's respect - respectGainRate: Respect earned per second - territory: Amount of territory held. Returned in decimal form, not percentage - territoryClashChance: Clash chance. Returned in decimal form, not percentage - wantedLevel: Gang's wanted level - wantedLevelGainRate: Wanted level gained/lost per second (negative for losses) - } - -getOtherGangInformation ------------------------ - -.. js:function:: getOtherGangInformation() - - Get territory and power information about all gangs - - :returns: An object with information about all gangs - - The object has the following structure:: - - { - "Slum Snakes" : { - power: Slum Snakes' power - territory: Slum Snakes' territory, in decimal form - }, - "Tetrads" : { - power: ... - territory: ... - }, - "The Syndicate" : { - power: ... - territory: ... - }, - ... (for all six gangs) - } - -getMemberInformation --------------------- - -.. js:function:: getMemberInformation(name) - - :param string name: Name of member - - Get stat and equipment-related information about a Gang Member - - :returns: An object with the gang member information. - - The object has the following structure:: - - { - agility: Agility stat - agilityEquipMult: Agility multiplier from equipment. Decimal form - agilityAscensionMult: Agility multiplier from ascension. Decimal form - augmentations: Array of names of all owned Augmentations - charisma: Charisma stat - charismaEquipMult: Charisma multiplier from equipment. Decimal form - charismaAscensionMult: Charisma multiplier from ascension. Decimal form - defense: Defense stat - defenseEquipMult: Defense multiplier from equipment. Decimal form - defenseAscensionMult: Defense multiplier from ascension. Decimal form - dexterity: Dexterity stat - dexterityEquipMult: Dexterity multiplier from equipment. Decimal form - dexterityAscensionMult: Dexterity multiplier from ascension. Decimal form - equipment: Array of names of all owned Non-Augmentation Equipment - hacking: Hacking stat - hackingEquipMult: Hacking multiplier from equipment. Decimal form - hackingAscensionMult: Hacking multiplier from ascension. Decimal form - strength: Strength stat - strengthEquipMult: Strength multiplier from equipment. Decimal form - strengthAscensionMult: Strength multiplier from ascension. Decimal form - task: Name of currently assigned task - } - -canRecruitMember ----------------- - -.. js:function:: canRecruitMember() - - :returns: Boolean indicating whether a member can currently be recruited - -recruitMember -------------- - -.. js:function:: recruitMember(name) - - :param string name: Name of member to recruit - - Attempt to recruit a new gang member. - - Possible reasons for failure: - * Cannot currently recruit a new member - * There already exists a member with the specified name - - :returns: True if the member was successfully recruited. False otherwise - -getTaskNames ------------- - -.. js:function:: getTaskNames() - - Get the name of all valid tasks that Gang members can be assigned to - - :returns: Array of strings of all task names - -setMemberTask -------------- - -.. js:function:: setMemberTask(memberName, taskName) - - :param string memberName: Name of Gang member to assign - :param string taskName: Task to assign - - Attempts to assign the specified Gang Member to the specified task. - If an invalid task is specified, the Gang member will be set to idle ("Unassigned") - - :returns: True if the Gang Member was successfully assigned to the task. False otherwise - -getEquipmentNames ------------------ - -.. js:function:: getEquipmentNames() - - Get the name of all possible equipment/upgrades you can purchase for your - Gang Members. This includes Augmentations. - - :returns: Array of strings of the names of all Equpiment/Augmentations - -getEquipmentCost ----------------- - -.. js:function:: getEquipmentCost(equipName) - - :param string equipName: Name of equipment - - Get the amount of money it takes to purchase a piece of Equipment or an Augmentation. - If an invalid Equipment/Augmentation is specified, this function will return Infinity. - - :returns: Cost to purchase the specified Equipment/Augmentation (number). Infinity - for invalid arguments - -getEquipmentType ----------------- - -.. js:function:: getEquipmentType(equipName) - - :param string equipName: Name of equipment - - Get the specified equipment type, which can be one of the following: - - * Weapon - * Armor - * Vehicle - * Rootkit - * Augmentation - - :returns: A string stating the type of the equipment - -purchaseEquipment ------------------ - -.. js:function:: purchaseEquipment(memberName, equipName) - - :param string memberName: Name of Gang member to purchase the equipment for - :param string equipName: Name of Equipment/Augmentation to purchase - - Attempt to purchase the specified Equipment/Augmentation for the specified - Gang member. - - :returns: True if the equipment was successfully purchased. False otherwise - - -ascendMember ------------- - -.. js:function:: ascendMember(name) - - :param string name: Name of member to ascend - - Ascend the specified Gang Member. - - :returns: An object with info about the ascension results. - - The object has the following structure:: - - { - respect: Amount of respect lost from ascending - hack: Hacking multiplier gained from ascending. Decimal form - str: Strength multiplier gained from ascending. Decimal form - def: Defense multiplier gained from ascending. Decimal form - dex: Dexterity multiplier gained from ascending. Decimal form - agi: Agility multiplier gained from ascending. Decimal form - cha: Charisma multiplier gained from ascending. Decimal form - } - - -setTerritoryWarfare -------------------- - -.. js:function:: setTerritoryWarfare(engage) - - :param bool engage: Whether or not to engage in territory warfare - - Set whether or not the gang should engage in territory warfare - -getChanceToWinClash -------------------- - -.. js:function:: getChanceToWinClash(gangName) - - :param string gangName: Target gang - - Returns the chance you have to win a clash with the specified gang. The chance - is returned in decimal form, not percentage - - -getBonusTime ------------- - -.. js:function:: getBonusTime() - - Returns the amount of accumulated "bonus time" (seconds) for the Gang mechanic. - - "Bonus time" is accumulated when the game is offline or if the game is - inactive in the browser. - - "Bonus time" makes the game progress faster, up to 10x the normal speed. - - :returns: Bonus time for the Gang mechanic in seconds +.. toctree:: + :caption: API Functions: + + getMemberNames() + getGangInformation() + getOtherGangInformation() + getMemberInformation() + canRecruitMember() + recruitMember() + getTaskNames() + setMemberTask() + getEquipmentNames() + getEquipmentCost() + getEquipmentType() + purchaseEquipment() + ascendMember() + setTerritoryWarfare() + getChanceToWinClash() + getBonusTime() diff --git a/doc/source/netscript/netscripthacknetnodeapi.rst b/doc/source/netscript/netscripthacknetnodeapi.rst index d1433eb21..fde650836 100644 --- a/doc/source/netscript/netscripthacknetnodeapi.rst +++ b/doc/source/netscript/netscripthacknetnodeapi.rst @@ -1,4 +1,4 @@ -.. _netscripthacknetnodeapi: +.. _netscript_hacknetnodeapi: Netscript Hacknet Node API ========================== @@ -21,6 +21,22 @@ In :ref:`netscriptjs`:: ns.hacknet.purchaseNode(); ns.hacknet.getNodeStats(3).level; +.. toctree:: + :caption: API Functions: + + numNodes() + purchaseNode() + getPurchaseNodeCost() + getNodeStats() + upgradeLevel() + upgradeRam() + upgradeCore() + getLevelUpgradeCost() + getRamUpgradeCost() + getCoreUpgradeCost() + +.. _netscript_hacknetnodeapi_referencingahacknetnode: + Referencing a Hacknet Node -------------------------- Most of the functions in the Hacknet Node API perform an operation on a single @@ -35,128 +51,7 @@ RAM Cost -------- Accessing the `hacknet` namespace incurs a one time cost of 4 GB of RAM. In other words, using multiple Hacknet Node API functions in a script will not cost -more than 4 GB of RAM. - -numNodes --------- -.. js:function:: numNodes() - - Returns the number of Hacknet Nodes you own. - -purchaseNode ------------- -.. js:function:: purchaseNode() - - Purchases a new Hacknet Node. Returns a number with the index of the Hacknet Node. - This index is equivalent to the number at the end of the Hacknet Node's name - (e.g The Hacknet Node named 'hacknet-node-4' will have an index of 4). - - If the player cannot afford to purchase a new Hacknet Node then the function - will return -1. - -getPurchaseNodeCost -------------------- -.. js:function:: getPurchaseNodeCost() - - Returns the cost of purchasing a new Hacknet Node - -getNodeStats ------------- -.. js:function:: getNodeStats(i) - - :param number i: Index/Identifier of Hacknet Node - - Returns an object containing a variety of stats about the specified Hacknet Node:: - - { - name: Node's name ("hacknet-node-5"), - level: Node's level, - ram: Node's RAM, - cores: Node's number of cores, - production: Node's money earned per second, - timeOnline: Number of seconds since Node has been purchased, - totalProduction: Total number of money Node has produced - } - -upgradeLevel ------------- -.. js:function:: upgradeLevel(i, n) - - :param number i: Index/Identifier of Hacknet Node - :param number n: Number of levels to purchase. Must be positive. Rounded to nearest integer - - Tries to upgrade the level of the specified Hacknet Node by *n*. - - Returns true if the Hacknet Node's level is successfully upgraded by *n* or - if it is upgraded by some positive amount and the Node reaches its max level. - - Returns false otherwise. - -upgradeRam ----------- -.. js:function:: upgradeRam(i, n) - - :param number i: Index/Identifier of Hacknet Node - :param number n: Number of times to upgrade RAM. Must be positive. Rounded to nearest integer - - Tries to upgrade the specified Hacknet Node's RAM *n* times. Note that each upgrade - doubles the Node's RAM. So this is equivalent to multiplying the Node's RAM by - 2 :sup:`n`. - - Returns true if the Hacknet Node's RAM is successfully upgraded *n* times or if - it is upgraded some positive number of times and the Node reaches it max RAM. - - Returns false otherwise. - -upgradeCore ------------ -.. js:function:: upgradeCore(i, n) - - :param number i: Index/Identifier of Hacknet Node - :param number n: Number of cores to purchase. Must be positive. Rounded to nearest integer - - Tries to purchase *n* cores for the specified Hacknet Node. - - Returns true if it successfully purchases *n* cores for the Hacknet Node or if - it purchases some positive amount and the Node reaches its max number of cores. - - Returns false otherwise. - -getLevelUpgradeCost -------------------- -.. js:function:: getLevelUpgradeCost(i, n) - - :param number i: Index/Identifier of Hacknet Node - :param number n: Number of levels to upgrade. Must be positive. Rounded to nearest integer - - Returns the cost of upgrading the specified Hacknet Node by *n* levels. - - If an invalid value for *n* is provided, then this function returns 0. If the - specified Hacknet Node is already at max level, then Infinity is returned. - -getRamUpgradeCost ------------------ -.. js:function:: getRamUpgradeCost(i, n) - - :param number i: Index/Identifier of Hacknet Node - :param number n: Number of times to upgrade RAM. Must be positive. Rounded to nearest integer - - Returns the cost of upgrading the RAM of the specified Hacknet Node *n* times. - - If an invalid value for *n* is provided, then this function returns 0. If the - specified Hacknet Node is already at max RAM, then Infinity is returned. - -getCoreUpgradeCost ------------------- -.. js:function:: getCoreUpgradeCost(i, n) - - :param number i: Index/Identifier of Hacknet Node - :param number n: Number of times to upgrade cores. Must be positive. Rounded to nearest integer - - Returns the cost of upgrading the number of cores of the specified Hacknet Node by *n*. - - If an invalid value for *n* is provided, then this function returns 0. If the - specified Hacknet Node is already at the max number of cores, then Infinity is returned. +more than 4 GB of RAM. Utilities --------- @@ -164,28 +59,10 @@ The following functions are not officially part of the Hacknet Node API, but the can be useful when writing Hacknet Node-related scripts. Since they are not part of the API, they do not need to be accessed using the *hacknet* namespace. -.. js:function:: getHacknetMultipliers() - - Returns an object containing the Player's hacknet related multipliers. These multipliers are - returned in integer forms, not percentages (e.g. 1.5 instead of 150%). The object has the following structure:: - - { - production: Player's hacknet production multiplier, - purchaseCost: Player's hacknet purchase cost multiplier, - ramCost: Player's hacknet ram cost multiplier, - coreCost: Player's hacknet core cost multiplier, - levelCost: Player's hacknet level cost multiplier - } - - Example of how this can be used:: - - mults = getHacknetMultipliers(); - print(mults.production); - print(mults.purchaseCost); - +* :js:func:`getHacknetMultipliers` Example(s) -^^^^^^^^^^ +---------- The following is an example of one way a script can be used to automate the purchasing and upgrading of Hacknet Nodes. @@ -194,8 +71,9 @@ This script attempts to purchase Hacknet Nodes until the player has a total of 8 it gradually upgrades those Node's to a minimum of level 140, 64 GB RAM, and 8 cores:: function myMoney() { - return getServerMoneyAvailable("home"); - } + return getServerMoneyAvailable("home");() + } + }() disableLog("getServerMoneyAvailable"); disableLog("sleep"); diff --git a/doc/source/netscript/netscriptixapi.rst b/doc/source/netscript/netscriptixapi.rst index 98bbfcdd8..9a1d11b66 100644 --- a/doc/source/netscript/netscriptixapi.rst +++ b/doc/source/netscript/netscriptixapi.rst @@ -13,328 +13,21 @@ TIX API can be purchased by visiting the World Stock Exchange in-game. Access to the TIX API currently costs $5 billion. After you purchase it, you will retain this access even after you 'reset' by installing Augmentations -getStockSymbols ---------------- - -.. js:function:: getStockSymbols() - - :RAM cost: 2 GB - - Returns an array of the symbols of the tradable stocks - -getStockPrice -------------- - -.. js:function:: getStockPrice(sym) - - :param string sym: Stock symbol - :RAM cost: 2 GB - - Returns the price of a stock, given its symbol (NOT the company name). The symbol is a sequence - of two to four capital letters. - - Example:: - - getStockPrice("FISG"); - -getStockPosition ----------------- - -.. js:function:: getStockPosition(sym) - - :param string sym: Stock symbol - :RAM cost: 2 GB - - Returns an array of four elements that represents the player's position in a stock. - - The first element is the returned array is the number of shares the player owns of the stock in the - `Long position `_. The second - element in the array is the average price of the player's shares in the Long position. - - The third element in the array is the number of shares the player owns of the stock in the - `Short position `_. The fourth - element in the array is the average price of the player's Short position. - - All elements in the returned array are numeric. - - Example:: - - pos = getStockPosition("ECP"); - shares = pos[0]; - avgPx = pos[1]; - sharesShort = pos[2]; - avgPxShort = pos[3]; - -getStockMaxShares ------------------ - - -.. js:function:: getStockMaxShares(sym) - - :param string sym: Stock symbol - :RAM cost: 2 GB - - Returns the maximum number of shares that the stock has. This is the maximum - amount of the stock that can be purchased in both the Long and Short - positions combined - -buyStock --------- - -.. js:function:: buyStock(sym, shares) - - :param string sym: Symbol of stock to purchase - :param number shares: Number of shares to purchased. Must be positive. Will be rounded to nearest integer - :RAM cost: 2.5 GB - - Attempts to purchase shares of a stock using a `Market Order `_. - - If the player does not have enough money to purchase the specified number of shares, then no shares will be purchased. Remember - that every transaction on the stock exchange costs a certain commission fee. - - If this function successfully purchases the shares, it will return the stock price at which each share was purchased. Otherwise, - it will return 0. - -sellStock ---------- - -.. js:function:: sellStock(sym, shares) - - :param string sym: Symbol of stock to sell - :param number shares: Number of shares to sell. Must be positive. Will be rounded to nearest integer - :RAM cost: 2.5 GB - - Attempts to sell shares of a stock using a `Market Order `_. - - If the specified number of shares in the function exceeds the amount that the player actually owns, then this function will - sell all owned shares. Remember that every transaction on the stock exchange costs a certain commission fee. - - The net profit made from selling stocks with this function is reflected in the script's statistics. - This net profit is calculated as:: - - shares * (sell price - average price of purchased shares) - - If the sale is successful, this function will return the stock price at which each share was sold. Otherwise, it will return 0. - -shortStock ----------- - -.. js:function:: shortStock(sym, shares) - - :param string sym: Symbol of stock to short - :param number shares: Number of shares to short. Must be positive. Will be rounded to nearest integer - :RAM cost: 2.5 GB - - Attempts to purchase a `short `_ position of a stock - using a `Market Order `_. - - The ability to short a stock is **not** immediately available to the player and must be unlocked later on in the game. - - If the player does not have enough money to purchase the specified number of shares, then no shares will be purchased. - Remember that every transaction on the stock exchange costs a certain commission fee. - - If the purchase is successful, this function will return the stock price at which each share was purchased. Otherwise, it will return 0. - -sellShort ---------- - -.. js:function:: sellShort(sym, shares) - - :param string sym: Symbol of stock to sell - :param number shares: Number of shares to sell. Must be positive. Will be rounded to nearest integer - :RAM cost: 2.5 GB - - Attempts to sell a `short `_ position of a stock - using a `Market Order `_. - - The ability to short a stock is **not** immediately available to the player and must be unlocked later on in the game. - - If the specified number of shares exceeds the amount that the player actually owns, then this function will sell all owned - shares. Remember that every transaction on the stock exchange costs a certain commission fee. - - If the sale is successful, this function will return the stock price at which each share was sold. Otherwise it will return 0. - -placeOrder ----------- - -.. js:function:: placeOrder(sym, shares, price, type, pos) - - :param string sym: Symbol of stock to player order for - :param number shares: Number of shares for order. Must be positive. Will be rounded to nearest integer - :param number price: Execution price for the order - :param string type: Type of order. It must specify "limit" or "stop", and must also specify "buy" or "sell". This is NOT - case-sensitive. Here are four examples that will work: - - * limitbuy - * limitsell - * stopbuy - * stopsell - - :param string pos: - Specifies whether the order is a "Long" or "Short" position. The Values "L" or "S" can also be used. This is - NOT case-sensitive. - :RAM cost: 2.5 GB - - Places an order on the stock market. This function only works for `Limit and Stop Orders `_. - - The ability to place limit and stop orders is **not** immediately available to the player and must be unlocked later on in the game. - - Returns true if the order is successfully placed, and false otherwise. - -cancelOrder ------------ - -.. js:function:: cancelOrder(sym, shares, price, type, pos) - - :param string sym: Symbol of stock to player order for - :param number shares: Number of shares for order. Must be positive. Will be rounded to nearest integer - :param number price: Execution price for the order - :param string type: Type of order. It must specify "limit" or "stop", and must also specify "buy" or "sell". This is NOT - case-sensitive. Here are four examples that will work: - - * limitbuy - * limitsell - * stopbuy - * stopsell - - :param string pos: - Specifies whether the order is a "Long" or "Short" position. The Values "L" or "S" can also be used. This is - NOT case-sensitive. - :RAM cost: 2.5 GB - - Cancels an oustanding Limit or Stop order on the stock market. - - The ability to use limit and stop orders is **not** immediately available to the player and must be unlocked later on in the game. - -getOrders ---------- - -.. js:function:: getOrders() - - :RAM cost: 2.5 GB - - Returns your order book for the stock market. This is an object containing information - for all the Limit and Stop Orders you have in the stock market. - - The object has the following structure:: - - { - StockSymbol1: [ // Array of orders for this stock - { - shares: Order quantity - price: Order price - type: Order type - position: Either "L" or "S" for Long or Short position - }, - { - ... - }, - ... - ], - StockSymbol2: [ // Array of orders for this stock - ... - ], - ... - } - - The "Order type" property can have one of the following four values: - - * "Limit Buy Order" - * "Limit Sell Order" - * "Stop Buy Order" - * "Stop Sell Order" - - **Note that the order book will only contain information for stocks that you actually - have orders in**. For example, if you do not have orders in Nova Medical (NVMD), then the returned - object will not have a "NVMD" property. - - Example:: - - { - ECP: [ - { - shares: 5, - price: 100,000 - type: "Stop Buy Order", - position: "S", - }, - { - shares: 25, - price: 125,000 - type: "Limit Sell Order", - position: "L", - }, - ], - SYSC: [ - { - shares: 100, - price: 10,000 - type: "Limit Buy Order", - position: "L", - }, - ], - } - -getStockVolatility ------------------- - -.. js:function:: getStockVolatility(sym) - - :param string sym: Symbol of stock - :RAM cost: 2.5 GB - - Returns the volatility of the specified stock. - - Volatility represents the maximum percentage by which a stock's price can - change every tick. The volatility is returned as a decimal value, NOT - a percentage (e.g. if a stock has a volatility of 3%, then this function will - return 0.03, NOT 3). - - In order to use this function, you must first purchase access to the Four Sigma (4S) - Market Data TIX API. - -getStockForecast ----------------- - -.. js:function:: getStockForecast(sym) - - :param string sym: Symbol of stock - :RAM cost: 2.5 GB - - Returns the probability that the specified stock's price will increase - (as opposed to decrease) during the next tick. - - The probability is returned as a decimal value, NOT a percentage (e.g. if a - stock has a 60% chance of increasing, then this function will return 0.6, - NOT 60). - - In other words, if this function returned 0.30 for a stock, then this means - that the stock's price has a 30% chance of increasing and a 70% chance of - decreasing during the next tick. - - In order to use this function, you must first purchase access to the Four Sigma (4S) - Market Data TIX API. - -purchase4SMarketData --------------------- - -.. js:function:: purchase4SMarketData() - - :RAM cost: 2.5 GB - - Purchase 4S Market Data Access. - - Returns true if you successfully purchased it or if you already have access. - Returns false otherwise. - -purchase4SMarketDataTixApi --------------------------- - -.. js:function:: purchase4SMarketDataTixApi() - - :RAM cost: 2.5 GB - - Purchase 4S Market Data TIX API Access. - - Returns true if you successfully purchased it or if you already have access. - Returns false otherwise. +.. toctree:: + :caption: API Functions: + + getStockSymbols() + getStockPrice() + getStockPosition() + getStockMaxShares() + buyStock() + sellStock() + shortStock() + sellShort() + placeOrder() + cancelOrder() + getOrders() + getStockVolatility() + getStockForecast() + purchase4SMarketData() + purchase4SMarketDataTixApi() diff --git a/doc/source/netscript/netscriptsingularityfunctions.rst b/doc/source/netscript/netscriptsingularityfunctions.rst index 9ba8da067..413d44f60 100644 --- a/doc/source/netscript/netscriptsingularityfunctions.rst +++ b/doc/source/netscript/netscriptsingularityfunctions.rst @@ -18,602 +18,39 @@ level 3, then you will be able to access all of the Singularity Functions. Note that Singularity Functions require twice as much RAM outside of BitNode-4 -universityCourse ----------------- - -.. js:function:: universityCourse(universityName, courseName) - - :param string universityName: - Name of university. Not case-sensitive. You must be in the correct city for whatever university you specify. - - * Summit University - * Rothman University - * ZB Institute Of Technology - :param string courseName: - Name of course. Not case-sensitive. - - * Study Computer Science - * Data Strucures - * Networks - * Algorithms - * Management - * Leadership - - If you are not in BitNode-4, then you must have Level 1 of Source-File 4 in order to use this function. - - This function will automatically set you to start taking a course at a university. If you are already in the middle of some - "working" action (such as working at a company, for a faction, or on a program), then running this function will automatically - cancel that action and give you your earnings. - - The cost and experience gains for all of these universities and classes are the same as if you were to manually visit and take these classes. - - This function will return true if you successfully start taking the course, and false otherwise. - -gymWorkout ----------- - -.. js:function:: gymWorkout(gymName, stat) - - :param string gymName: - Name of gym. Not case-sensitive. You must be in the correct city for whatever gym you specify. - - * Crush Fitness Gym - * Snap Fitness Gym - * Iron Gym - * Powerhouse Gym - * Millenium Fitness Gym - :param string stat: - The stat you want to train. Not case-sensitive. - - * strength OR str - * defense OR def - * dexterity OR dex - * agility OR agi - - If you are not in BitNode-4, then you must have Level 1 of Source-File 4 in order to use this function. - - This function will automatically set you to start working out at a gym to train a particular stat. If you are - already in the middle of some "working" action (such as working at a company, for a faction, or on a program), - then running this function will automatically cancel that action and give you your earnings. - - The cost and experience gains for all of these gyms are the same as if you were to manually visit these gyms and train - - This function will return true if you successfully start working out at the gym, and false otherwise. - -travelToCity ------------- - -.. js:function:: travelToCity(cityName) - - :param string cityName: - City to travel to. CASE-SENSITIVE. - - * Aevum - * Chongqing - * Sector-12 - * New Tokyo - * Ishima - * Volhaven - - If you are not in BitNode-4, then you must have Level 1 of Source-File 4 in order to use this function. - - This function allows the player to travel to any city. The cost for using this function is the same as the cost for traveling through the Travel Agency. - - This function will return true if you successfully travel to the specified city and false otherwise. - -purchaseTor ------------ - -.. js:function:: purchaseTor() - - If you are not in BitNode-4, then you must have Level 1 of Source-File 4 in order to use this function. - - This function allows you to automatically purchase a TOR router. The cost for purchasing a TOR router using this - function is the same as if you were to manually purchase one. - - This function will return true if it successfully purchase a TOR router and false otherwise. - -purchaseProgram ---------------- - -.. js:function:: purchaseProgram(programName) - - :param string programName: Name of program to purchase. Must include '.exe' extension. Not case-sensitive. - - If you are not in BitNode-4, then you must have Level 1 of Source-File 4 in order to use this function. - - This function allows you to automatically purchase programs. You MUST have a TOR router in order to use this function. - The cost of purchasing programs using this function is the same as if you were purchasing them through the Dark Web using the - Terminal *buy* command. - - Example:: - - purchaseProgram("brutessh.exe"); - - This function will return true if the specified program is purchased, and false otherwise. - -getStats --------- - -.. js:function:: getStats() - - If you are not in BitNode-4, then you must have Level 1 of Source-File 4 in order to run this function. - - Returns an object with the Player's stats. The object has the following properties:: - - { - hacking - strength - defense - dexterity - agility - charisma - intelligence - } - - Example:: - - res = getStats(); - print('My charisma level is: ' + res.charisma); - -getCharacterInformation ------------------------ - - If you are not in BitNode-4, then you must have Level 1 of Source-File 4 in order to run this function. - - Returns an object with various information about your character. The object has the following properties:: - - { - bitnode: Current BitNode number - city: Name of city you are currently in - factions: Array of factions you are currently a member of - hp: Current health points - jobs: Array of all companies at which you have jobs - jobTitle: Array of job positions for all companies you are employed at. Same order as 'jobs' - maxHp: Maximum health points - tor: Boolean indicating whether or not you have a tor router - - // The following is an object with many of the player's multipliers from Augmentations/Source Files - mult: { - agility: Agility stat - agilityExp: Agility exp - companyRep: Company reputation - crimeMoney: Money earned from crimes - crimeSuccess: Crime success chance - defense: Defense stat - defenseExp: Defense exp - dexterity: Dexterity stat - dexterityExp: Dexterity exp - factionRep: Faction reputation - hacking: Hacking stat - hackingExp: Hacking exp - strength: Strength stat - strengthExp: Strength exp - workMoney: Money earned from jobs - }, - - // The following apply only to when the character is performing - // some type of working action, such as working for a company/faction - timeWorked: Timed worked in ms - workHackExpGain: Hacking experience earned so far from work - workStrExpGain: Str experience earned so far from work - workDefExpGain: Def experience earned so far from work - workDexExpGain: Dex experience earned so far from work - workAgiExpGain: Agi experience earned so far from work - workChaExpGain: Cha experience earned so far from work - workRepGain: Reputation earned so far from work, if applicable - workMoneyGain: Money earned so far from work, if applicable - } - -isBusy ------- - -.. js:function:: isBusy() - - If you are not in BitNode-4, then you must have Level 1 of Source-File 4 in order to run this function. - - Returns a boolean indicating whether or not the player is currently performing an 'action'. These actions include - working for a company/faction, studying at a univeristy, working out at a gym, creating a program, or committing a crime. - -stopAction ----------- - -.. js:function:: stopAction() - - If you are not in BitNode-4, then you must have Level 1 of Source-File 4 in order to run this function. - This function is used to end whatever 'action' the player is currently performing. The player - will receive whatever money/experience/etc. he has earned from that action. - - The actions that can be stopped with this function are: - - * Studying at a university - * Working for a company/faction - * Creating a program - * Committing a Crime - - This function will return true if the player's action was ended. It will return false if the player was not - performing an action when this function was called. - -upgradeHomeRam --------------- - -.. js:function:: upgradeHomeRam() - - If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. - - This function will upgrade amount of RAM on the player's home computer. The cost is the same as if you were to do it manually. - - This function will return true if the player's home computer RAM is successfully upgraded, and false otherwise. - -getUpgradeHomeRamCost ---------------------- - -.. js:function:: getUpgradeHomeRamCost() - - If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. - - Returns the cost of upgrading the player's home computer RAM. - -workForCompany --------------- - -.. js:function:: workForCompany() - - If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. - - This function will automatically set you to start working at the company at which you are employed. - If you are already in the middle of some "working" action (such as working for a faction, training at - a gym, or creating a program), then running this function will automatically cancel that action and give you your earnings. - - This function will return true if the player starts working, and false otherwise. - - Note that when you are working for a company, you will not actually receive your earnings - (reputation, money, experience) until you FINISH the action. This can be an issue if, for example, - you only want to work until you get 100,000 company reputation. One small hack to get around this is to - continuously restart the action to receive your earnings:: - - while (getCompanyRep(COMPANY HERE) < VALUE) { - workForCompany(); - sleep(60000); - } - - This way, your company reputation will be updated every minute. - -applyToCompany --------------- - -.. js:function:: applyToCompany(companyName, field) - - :param string companyName: Name of company to apply to. CASE-SENSITIVE. - :param string field: - Field to which you want to apply. Not case-sensitive - - * software - * software consultant - * it - * security engineer - * network engineer - * business - * business consultant - * security - * agent - * employee - * part-time employee - * waiter - * part-time waiter - - If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. - - This function will automatically try to apply to the specified company for a position in the specified - field. This function can also be used to apply for promotions by specifying the company and field you - are already employed at. - - This function will return true if you successfully get a job/promotion, and false otherwise. Note that - if you are trying to use this function to apply for a promotion and you don't get one, it will return false. - -getCompanyRep -------------- - -.. js:function:: getCompanyRep(companyName) - - :param string companyName: Name of the company. CASE-SENSITIVE - - If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. - - This function will return the amount of reputation you have at the specified company. - If the company passed in as an argument is invalid, -1 will be returned. - -getCompanyFavor ---------------- - -.. js:function:: getCompanyFavor(companyName) - - :param string companyName: Name of the company. CASE-SENSITIVE - - If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. - - This function will return the amount of favor you have at the specified company. - If the company passed in as an argument is invalid, -1 will be returned. - -getCompanyFavorGain -------------------- - -.. js:function:: getCompanyFavorGain(companyName) - - :param string companyName: Name of the company. CASE-SENSITIVE - - If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. - - This function will return the amount of favor you will gain for the specified company - when you reset by installing Augmentations. - -checkFactionInvitations ------------------------ - -.. js:function:: checkFactionInvitations() - - If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. - - Returns an array with the name of all Factions you currently have oustanding invitations from. - -joinFaction ------------ - -.. js:function:: joinFaction(name) - - :param string name: Name of faction to join. CASE-SENSITIVE - - If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. - - This function will automatically accept an invitation from a faction and join it. - -workForFaction --------------- - -.. js:function:: workForFaction(factionName, workType) - - :param string factionName: Name of faction to work for. CASE-SENSITIVE - :param string workType: - Type of work to perform for the faction - - * hacking/hacking contracts/hackingcontracts - * field/fieldwork/field work - * security/securitywork/security work - - If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. - - This function will automatically set you to start working for the specified faction. - Obviously, you must be a member of the faction or else this function will fail. If you are already in - the middle of some "working" action (such as working for a company, training at a gym, or creating a program), - then running this function will automatically cancel that action and give you your earnings. - - This function will return true if you successfully start working for the specified faction, and false otherwise. - - Note that when you are working for a faction, you will not actually receive your earnings (reputation, experience) - until you FINISH the action. This can be an issue if, for example, you only want to work until you get 100,000 faction - reputation. One small hack to get around this is to continuously restart the action to receive your earnings:: - - while (getFactionRep(FACTION NAME) < VALUE) { - workForFaction(FACNAME, WORKTYPE); - sleep(60000); - } - - This way, your faction reputation will be updated every minute. - -getFactionRep -------------- - -.. js:function:: getFactionRep(factionName) - - :param string factionName: Name of faction. CASE-SENSITIVE - - If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. - - This function returns the amount of reputation you have for the specified faction. - -getFactionFavor ---------------- - -.. js:function:: getFactionFavor(factionName) - - :param string factionName: Name of faction. CASE-SENSITIVE - - If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. - - This function returns the amount of favor you have for the specified faction. - -getFactionFavorGain -------------------- - -.. js:function:: getFactionFavorGain(factionName) - - :param string factionName: Name of faction. CASE-SENSITIVE - - If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. - - This function returns the amount of favor you will gain for the specified faction when you reset by installing Augmentations. - -donateToFaction ---------------- - -.. js:function:: donateToFaction(factionName, donateAmt) - - :param string factionName: Name of faction to donate to. CASE-SENSITIVE - :param number donateAmt: Amount of money to donate - - If you are not in BitNode-4, then you must have Level 3 of Source-File 4 in order to use this function. - - Attempts to donate money to the specified faction in exchange for reputation. - Returns true if you successfully donate the money, and false otherwise. - -createProgram -------------- - -.. js:function:: createProgram(programName) - - :param string programName: Name of program to create. Not case-sensitive - - If you are not in BitNode-4, then you must have Level 3 of Source-File 4 in order to use this function. - - This function will automatically set you to start working on creating the specified program. If you are - already in the middle of some "working" action (such as working for a company, training at a gym, or taking a course), - then running this function will automatically cancel that action and give you your earnings. - - Example: - - createProgram("relaysmtp.exe"); - - Note that creating a program using this function has the same hacking level requirements as it normally would. These level requirements are: - - * BruteSSH.exe: 50 - * FTPCrack.exe: 100 - * relaySMTP.exe: 250 - * HTTPWorm.exe: 500 - * SQLInject.exe: 750 - * DeepscanV1.exe: 75 - * DeepscanV2.exe: 400 - * ServerProfiler.exe: 75 - * AutoLink.exe: 25 - - This function returns true if you successfully start working on the specified program, and false otherwise. - -commitCrime ------------ - -.. js:function:: commitCrime(crime) - - :param string crime: - Name of crime to attempt. Not case-sensitive. This argument is fairly lenient in terms of what inputs it accepts. - Here is a list of valid inputs for all of the crimes: - - * shoplift - * rob store - * mug - * larceny - * deal drugs - * bond forgery - * traffick arms - * homicide - * grand theft auto - * kidnap - * assassinate - * heist - - If you are not in BitNode-4, then you must have Level 3 of Source-File 4 in order to use this function. - - This function is used to automatically attempt to commit crimes. If you are already in the middle of some 'working' action - (such as working for a company or training at a gym), then running this function will automatically cancel that action and give you your earnings. - - This function returns the number of seconds it takes to attempt the specified crime (e.g It takes 60 seconds to attempt the 'Rob Store' crime, - so running *commitCrime('rob store')* will return 60). - - Warning: I do not recommend using the time returned from this function to try and schedule your crime attempts. - Instead, I would use the isBusy() Singularity function to check whether you have finished attempting a crime. - This is because although the game sets a certain crime to be X amount of seconds, there is no guarantee that your - browser will follow that time limit. - -getCrimeChance --------------- - -.. js:function:: getCrimeChance(crime) - - :param string crime: - Name of crime. Not case-sensitive. This argument is fairlyn lenient in terms of what inputs it accepts. - Check the documentation for the *commitCrime()* function for a list of example inputs. - - If you are not in BitNode-4, then you must have Level 3 of Source-File 4 in order to use this function. - - This function returns your chance of success at commiting the specified crime. The chance is returned as a decimal (i.e. 60% would be returned as 0.6). - -getOwnedAugmentations ---------------------- - -.. js:function:: getOwnedAugmentations(purchased=false) - - :param boolean purchase: - Specifies whether the returned array should include Augmentations you have purchased but not yet installed. - By default, this argument is false which means that the return value will NOT have the purchased Augmentations. - - If you are not in BitNode-4, then you must have Level 3 of Source-File 4 in order to use this function. - - This function returns an array containing the names (as strings) of all Augmentations you have. - -getOwnedSourceFiles -------------------- - -.. js:function:: getOwnedSourceFiles() - - If you are not in BitNode-4, then you must have Level 3 of Source-File 4 in order to use this function. - - Returns an array of source files - [{n: 1, lvl: 3}, {n: 4, lvl: 3}] - - -getAugmentationsFromFaction ---------------------------- - -.. js:function:: getAugmentationsFromFaction(facName) - - :param string facName: Name of faction. CASE-SENSITIVE - - If you are not in BitNode-4, then you must have Level 3 of Source-File 4 in order to use this function. - - Returns an array containing the names (as strings) of all Augmentations that are available from the specified faction. - -getAugmentationPrereq ---------------------- - -.. js:function:: getAugmentationPrereq(augName) - - :param string augName: Name of Augmentation. CASE-SENSITIVE - - If you are not in BitNode-4, then you must have Level 3 of Source-File 4 in order to use this function. - - This function returns an array with the names of the prerequisite Augmentation(s) for the specified Augmentation. - If there are no prerequisites, a blank array is returned. - - If an invalid Augmentation name is passed in for the *augName* argument, this function will return a blank array. - -getAugmentationCost -------------------- - -.. js:function:: getAugmentationCost(augName) - - :param string augName: Name of Augmentation. CASE-SENSITIVE - - If you are not in BitNode-4, then you must have Level 3 of Source-File 4 in order to use this function. - - This function returns an array with two elements that gives the cost for the specified Augmentation. - The first element in the returned array is the reputation requirement of the Augmentation, and the second element is the money cost. - - If an invalid Augmentation name is passed in for the *augName* argument, this function will return the array [-1, -1]. - -purchaseAugmentation --------------------- - -.. js:function:: purchaseAugmentation(factionName, augName) - - :param string factionName: Name of faction to purchase Augmentation from. CASE-SENSITIVE - :param string augName: Name of Augmentation to purchase. CASE-SENSITIVE - - - If you are not in BitNode-4, then you must have Level 3 of Source-File 4 in order to use this function. - - This function will try to purchase the specified Augmentation through the given Faction. - - This function will return true if the Augmentation is successfully purchased, and false otherwise. - -installAugmentations --------------------- - -.. js:function:: installAugmentations(cbScript) - - :param string cbScript: - Optional callback script. This is a script that will automatically be run after Augmentations are installed (after the reset). - This script will be run with no arguments and 1 thread. It must be located on your home computer. - - If you are not in BitNode-4, then you must have Level 3 of Source-File 4 in order to use this function. - - This function will automatically install your Augmentations, resetting the game as usual. - - It will return true if successful, and false otherwise. +.. toctree:: + :caption: Functions: + + universityCourse() + gymWorkout() + travelToCity() + purchaseTor() + purchaseProgram() + getStats() + getCharacterInformation() + isBusy() + stopAction() + upgradeHomeRam() + getUpgradeHomeRamCost() + workForCompany() + applyToCompany() + getCompanyRep() + getCompanyFavor() + getCompanyFavorGain() + checkFactionInvitations() + joinFaction() + workForFaction() + getFactionRep() + getFactionFavor() + getFactionFavorGain() + donateToFaction() + createProgram() + commitCrime() + getCrimeChance() + getOwnedAugmentations() + getOwnedSourceFiles() + getAugmentationsFromFaction() + getAugmentationPrereq() + getAugmentationCost() + purchaseAugmentation() + installAugmentations() diff --git a/doc/source/netscript/singularityfunctions/applyToCompany.rst b/doc/source/netscript/singularityfunctions/applyToCompany.rst new file mode 100644 index 000000000..b562bf7a7 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/applyToCompany.rst @@ -0,0 +1,31 @@ +applyToCompany() Netscript Function +=================================== + +.. js:function:: applyToCompany(companyName, field) + + :param string companyName: Name of company to apply to. CASE-SENSITIVE. + :param string field: + Field to which you want to apply. Not case-sensitive + + * software + * software consultant + * it + * security engineer + * network engineer + * business + * business consultant + * security + * agent + * employee + * part-time employee + * waiter + * part-time waiter + + If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. + + This function will automatically try to apply to the specified company for a position in the specified + field. This function can also be used to apply for promotions by specifying the company and field you + are already employed at. + + This function will return true if you successfully get a job/promotion, and false otherwise. Note that + if you are trying to use this function to apply for a promotion and you don't get one, it will return false. diff --git a/doc/source/netscript/singularityfunctions/checkFactionInvitations.rst b/doc/source/netscript/singularityfunctions/checkFactionInvitations.rst new file mode 100644 index 000000000..7dd54a2a9 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/checkFactionInvitations.rst @@ -0,0 +1,8 @@ +checkFactionInvitations() Netscript Function +============================================ + +.. js:function:: checkFactionInvitations() + + If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. + + Returns an array with the name of all Factions you currently have oustanding invitations from. diff --git a/doc/source/netscript/singularityfunctions/commitCrime.rst b/doc/source/netscript/singularityfunctions/commitCrime.rst new file mode 100644 index 000000000..226cfa990 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/commitCrime.rst @@ -0,0 +1,34 @@ +commitCrime() Netscript Function +================================ + +.. js:function:: commitCrime(crime) + + :param string crime: + Name of crime to attempt. Not case-sensitive. This argument is fairly lenient in terms of what inputs it accepts. + Here is a list of valid inputs for all of the crimes: + + * shoplift + * rob store + * mug + * larceny + * deal drugs + * bond forgery + * traffick arms + * homicide + * grand theft auto + * kidnap + * assassinate + * heist + + If you are not in BitNode-4, then you must have Level 3 of Source-File 4 in order to use this function. + + This function is used to automatically attempt to commit crimes. If you are already in the middle of some 'working' action + (such as working for a company or training at a gym), then running this function will automatically cancel that action and give you your earnings. + + This function returns the number of seconds it takes to attempt the specified crime (e.g It takes 60 seconds to attempt the 'Rob Store' crime, + so running *commitCrime('rob store')* will return 60). + + Warning: I do not recommend using the time returned from this function to try and schedule your crime attempts. + Instead, I would use the isBusy() Singularity function to check whether you have finished attempting a crime. + This is because although the game sets a certain crime to be X amount of seconds, there is no guarantee that your + browser will follow that time limit. diff --git a/doc/source/netscript/singularityfunctions/createProgram.rst b/doc/source/netscript/singularityfunctions/createProgram.rst new file mode 100644 index 000000000..30d6568ee --- /dev/null +++ b/doc/source/netscript/singularityfunctions/createProgram.rst @@ -0,0 +1,30 @@ +createProgram() Netscript Function +================================== + +.. js:function:: createProgram(programName) + + :param string programName: Name of program to create. Not case-sensitive + + If you are not in BitNode-4, then you must have Level 3 of Source-File 4 in order to use this function. + + This function will automatically set you to start working on creating the specified program. If you are + already in the middle of some "working" action (such as working for a company, training at a gym, or taking a course), + then running this function will automatically cancel that action and give you your earnings. + + Example: + + createProgram("relaysmtp.exe"); + + Note that creating a program using this function has the same hacking level requirements as it normally would. These level requirements are: + + * BruteSSH.exe: 50 + * FTPCrack.exe: 100 + * relaySMTP.exe: 250 + * HTTPWorm.exe: 500 + * SQLInject.exe: 750 + * DeepscanV1.exe: 75 + * DeepscanV2.exe: 400 + * ServerProfiler.exe: 75 + * AutoLink.exe: 25 + + This function returns true if you successfully start working on the specified program, and false otherwise. diff --git a/doc/source/netscript/singularityfunctions/donateToFaction.rst b/doc/source/netscript/singularityfunctions/donateToFaction.rst new file mode 100644 index 000000000..f61cc70e6 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/donateToFaction.rst @@ -0,0 +1,12 @@ +donateToFaction() Netscript Function +==================================== + +.. js:function:: donateToFaction(factionName, donateAmt) + + :param string factionName: Name of faction to donate to. CASE-SENSITIVE + :param number donateAmt: Amount of money to donate + + If you are not in BitNode-4, then you must have Level 3 of Source-File 4 in order to use this function. + + Attempts to donate money to the specified faction in exchange for reputation. + Returns true if you successfully donate the money, and false otherwise. diff --git a/doc/source/netscript/singularityfunctions/getAugmentationCost.rst b/doc/source/netscript/singularityfunctions/getAugmentationCost.rst new file mode 100644 index 000000000..1d7b623a8 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/getAugmentationCost.rst @@ -0,0 +1,13 @@ +getAugmentationCost() Netscript Function +======================================== + +.. js:function:: getAugmentationCost(augName) + + :param string augName: Name of Augmentation. CASE-SENSITIVE + + If you are not in BitNode-4, then you must have Level 3 of Source-File 4 in order to use this function. + + This function returns an array with two elements that gives the cost for the specified Augmentation. + The first element in the returned array is the reputation requirement of the Augmentation, and the second element is the money cost. + + If an invalid Augmentation name is passed in for the *augName* argument, this function will return the array [-1, -1]. diff --git a/doc/source/netscript/singularityfunctions/getAugmentationPrereq.rst b/doc/source/netscript/singularityfunctions/getAugmentationPrereq.rst new file mode 100644 index 000000000..efe5c38f0 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/getAugmentationPrereq.rst @@ -0,0 +1,13 @@ +getAugmentationPrereq() Netscript Function +========================================== + +.. js:function:: getAugmentationPrereq(augName) + + :param string augName: Name of Augmentation. CASE-SENSITIVE + + If you are not in BitNode-4, then you must have Level 3 of Source-File 4 in order to use this function. + + This function returns an array with the names of the prerequisite Augmentation(s) for the specified Augmentation. + If there are no prerequisites, a blank array is returned. + + If an invalid Augmentation name is passed in for the *augName* argument, this function will return a blank array. diff --git a/doc/source/netscript/singularityfunctions/getAugmentationsFromFaction.rst b/doc/source/netscript/singularityfunctions/getAugmentationsFromFaction.rst new file mode 100644 index 000000000..f1101b23c --- /dev/null +++ b/doc/source/netscript/singularityfunctions/getAugmentationsFromFaction.rst @@ -0,0 +1,10 @@ +getAugmentationsFromFaction() Netscript Function +================================================ + +.. js:function:: getAugmentationsFromFaction(facName) + + :param string facName: Name of faction. CASE-SENSITIVE + + If you are not in BitNode-4, then you must have Level 3 of Source-File 4 in order to use this function. + + Returns an array containing the names (as strings) of all Augmentations that are available from the specified faction. diff --git a/doc/source/netscript/singularityfunctions/getCharacterInformation.rst b/doc/source/netscript/singularityfunctions/getCharacterInformation.rst new file mode 100644 index 000000000..ba3ee9696 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/getCharacterInformation.rst @@ -0,0 +1,50 @@ +getCharacterInformation() Netscript Function +============================================ + +.. js:function:: getCharacterInformation() + + If you are not in BitNode-4, then you must have Level 1 of Source-File 4 in order to run this function. + + Returns an object with various information about your character. The object has the following properties:: + + { + bitnode: Current BitNode number + city: Name of city you are currently in + factions: Array of factions you are currently a member of + hp: Current health points + jobs: Array of all companies at which you have jobs + jobTitle: Array of job positions for all companies you are employed at. Same order as 'jobs' + maxHp: Maximum health points + tor: Boolean indicating whether or not you have a tor router + + // The following is an object with many of the player's multipliers from Augmentations/Source Files + mult: { + agility: Agility stat + agilityExp: Agility exp + companyRep: Company reputation + crimeMoney: Money earned from crimes + crimeSuccess: Crime success chance + defense: Defense stat + defenseExp: Defense exp + dexterity: Dexterity stat + dexterityExp: Dexterity exp + factionRep: Faction reputation + hacking: Hacking stat + hackingExp: Hacking exp + strength: Strength stat + strengthExp: Strength exp + workMoney: Money earned from jobs + }, + + // The following apply only to when the character is performing + // some type of working action, such as working for a company/faction + timeWorked: Timed worked in ms + workHackExpGain: Hacking experience earned so far from work + workStrExpGain: Str experience earned so far from work + workDefExpGain: Def experience earned so far from work + workDexExpGain: Dex experience earned so far from work + workAgiExpGain: Agi experience earned so far from work + workChaExpGain: Cha experience earned so far from work + workRepGain: Reputation earned so far from work, if applicable + workMoneyGain: Money earned so far from work, if applicable + } diff --git a/doc/source/netscript/singularityfunctions/getCompanyFavor.rst b/doc/source/netscript/singularityfunctions/getCompanyFavor.rst new file mode 100644 index 000000000..c303593d4 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/getCompanyFavor.rst @@ -0,0 +1,11 @@ +getCompanyFavor() Netscript Function +==================================== + +.. js:function:: getCompanyFavor(companyName) + + :param string companyName: Name of the company. CASE-SENSITIVE + + If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. + + This function will return the amount of favor you have at the specified company. + If the company passed in as an argument is invalid, -1 will be returned. diff --git a/doc/source/netscript/singularityfunctions/getCompanyFavorGain.rst b/doc/source/netscript/singularityfunctions/getCompanyFavorGain.rst new file mode 100644 index 000000000..9df9d2d32 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/getCompanyFavorGain.rst @@ -0,0 +1,11 @@ +getCompanyFavorGain() Netscript Function +======================================== + +.. js:function:: getCompanyFavorGain(companyName) + + :param string companyName: Name of the company. CASE-SENSITIVE + + If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. + + This function will return the amount of favor you will gain for the specified company + when you reset by installing Augmentations. diff --git a/doc/source/netscript/singularityfunctions/getCompanyRep.rst b/doc/source/netscript/singularityfunctions/getCompanyRep.rst new file mode 100644 index 000000000..6a2bc0db0 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/getCompanyRep.rst @@ -0,0 +1,11 @@ +getCompanyRep() Netscript Function +================================== + +.. js:function:: getCompanyRep(companyName) + + :param string companyName: Name of the company. CASE-SENSITIVE + + If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. + + This function will return the amount of reputation you have at the specified company. + If the company passed in as an argument is invalid, -1 will be returned. diff --git a/doc/source/netscript/singularityfunctions/getCrimeChance.rst b/doc/source/netscript/singularityfunctions/getCrimeChance.rst new file mode 100644 index 000000000..9c8587fc5 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/getCrimeChance.rst @@ -0,0 +1,12 @@ +getCrimeChance() Netscript Function +=================================== + +.. js:function:: getCrimeChance(crime) + + :param string crime: + Name of crime. Not case-sensitive. This argument is fairlyn lenient in terms of what inputs it accepts. + Check the documentation for the *commitCrime()* function for a list of example inputs. + + If you are not in BitNode-4, then you must have Level 3 of Source-File 4 in order to use this function. + + This function returns your chance of success at commiting the specified crime. The chance is returned as a decimal (i.e. 60% would be returned as 0.6). diff --git a/doc/source/netscript/singularityfunctions/getFactionFavor.rst b/doc/source/netscript/singularityfunctions/getFactionFavor.rst new file mode 100644 index 000000000..d995ac999 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/getFactionFavor.rst @@ -0,0 +1,10 @@ +getFactionFavor() Netscript Function +==================================== + +.. js:function:: getFactionFavor(factionName) + + :param string factionName: Name of faction. CASE-SENSITIVE + + If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. + + This function returns the amount of favor you have for the specified faction. diff --git a/doc/source/netscript/singularityfunctions/getFactionFavorGain.rst b/doc/source/netscript/singularityfunctions/getFactionFavorGain.rst new file mode 100644 index 000000000..147f0fe52 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/getFactionFavorGain.rst @@ -0,0 +1,10 @@ +getFactionFavorGain() Netscript Function +======================================== + +.. js:function:: getFactionFavorGain(factionName) + + :param string factionName: Name of faction. CASE-SENSITIVE + + If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. + + This function returns the amount of favor you will gain for the specified faction when you reset by installing Augmentations. diff --git a/doc/source/netscript/singularityfunctions/getFactionRep.rst b/doc/source/netscript/singularityfunctions/getFactionRep.rst new file mode 100644 index 000000000..461b2ce1f --- /dev/null +++ b/doc/source/netscript/singularityfunctions/getFactionRep.rst @@ -0,0 +1,10 @@ +getFactionRep() Netscript Function +================================== + +.. js:function:: getFactionRep(factionName) + + :param string factionName: Name of faction. CASE-SENSITIVE + + If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. + + This function returns the amount of reputation you have for the specified faction. diff --git a/doc/source/netscript/singularityfunctions/getOwnedAugmentations.rst b/doc/source/netscript/singularityfunctions/getOwnedAugmentations.rst new file mode 100644 index 000000000..52b4f94e9 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/getOwnedAugmentations.rst @@ -0,0 +1,12 @@ +getOwnedAugmentations() Netscript Function +========================================== + +.. js:function:: getOwnedAugmentations(purchased=false) + + :param boolean purchase: + Specifies whether the returned array should include Augmentations you have purchased but not yet installed. + By default, this argument is false which means that the return value will NOT have the purchased Augmentations. + + If you are not in BitNode-4, then you must have Level 3 of Source-File 4 in order to use this function. + + This function returns an array containing the names (as strings) of all Augmentations you have. diff --git a/doc/source/netscript/singularityfunctions/getOwnedSourceFiles.rst b/doc/source/netscript/singularityfunctions/getOwnedSourceFiles.rst new file mode 100644 index 000000000..04d963092 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/getOwnedSourceFiles.rst @@ -0,0 +1,9 @@ +getOwnedSourceFiles() Netscript Function +======================================== + +.. js:function:: getOwnedSourceFiles() + + If you are not in BitNode-4, then you must have Level 3 of Source-File 4 in order to use this function. + + Returns an array of source files + [{n: 1, lvl: 3}, {n: 4, lvl: 3}] diff --git a/doc/source/netscript/singularityfunctions/getStats.rst b/doc/source/netscript/singularityfunctions/getStats.rst new file mode 100644 index 000000000..74ad4ae17 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/getStats.rst @@ -0,0 +1,23 @@ +getStats() Netscript Function +============================= + +.. js:function:: getStats() + + If you are not in BitNode-4, then you must have Level 1 of Source-File 4 in order to run this function. + + Returns an object with the Player's stats. The object has the following properties:: + + { + hacking + strength + defense + dexterity + agility + charisma + intelligence + } + + Example:: + + res = getStats(); + print('My charisma level is: ' + res.charisma); diff --git a/doc/source/netscript/singularityfunctions/getUpgradeHomeRamCost.rst b/doc/source/netscript/singularityfunctions/getUpgradeHomeRamCost.rst new file mode 100644 index 000000000..90c6c61ae --- /dev/null +++ b/doc/source/netscript/singularityfunctions/getUpgradeHomeRamCost.rst @@ -0,0 +1,8 @@ +getUpgradeHomeRamCost() Netscript Function +========================================== + +.. js:function:: getUpgradeHomeRamCost() + + If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. + + Returns the cost of upgrading the player's home computer RAM. diff --git a/doc/source/netscript/singularityfunctions/gymWorkout.rst b/doc/source/netscript/singularityfunctions/gymWorkout.rst new file mode 100644 index 000000000..32daad2c6 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/gymWorkout.rst @@ -0,0 +1,30 @@ +gymWorkout() Netscript Function +=============================== + +.. js:function:: gymWorkout(gymName, stat) + + :param string gymName: + Name of gym. Not case-sensitive. You must be in the correct city for whatever gym you specify. + + * Crush Fitness Gym + * Snap Fitness Gym + * Iron Gym + * Powerhouse Gym + * Millenium Fitness Gym + :param string stat: + The stat you want to train. Not case-sensitive. + + * strength OR str + * defense OR def + * dexterity OR dex + * agility OR agi + + If you are not in BitNode-4, then you must have Level 1 of Source-File 4 in order to use this function. + + This function will automatically set you to start working out at a gym to train a particular stat. If you are + already in the middle of some "working" action (such as working at a company, for a faction, or on a program), + then running this function will automatically cancel that action and give you your earnings. + + The cost and experience gains for all of these gyms are the same as if you were to manually visit these gyms and train + + This function will return true if you successfully start working out at the gym, and false otherwise. diff --git a/doc/source/netscript/singularityfunctions/installAugmentations.rst b/doc/source/netscript/singularityfunctions/installAugmentations.rst new file mode 100644 index 000000000..38a862674 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/installAugmentations.rst @@ -0,0 +1,14 @@ +installAugmentations() Netscript Function +========================================= + +.. js:function:: installAugmentations(cbScript) + + :param string cbScript: + Optional callback script. This is a script that will automatically be run after Augmentations are installed (after the reset). + This script will be run with no arguments and 1 thread. It must be located on your home computer. + + If you are not in BitNode-4, then you must have Level 3 of Source-File 4 in order to use this function. + + This function will automatically install your Augmentations, resetting the game as usual. + + It will return true if successful, and false otherwise. diff --git a/doc/source/netscript/singularityfunctions/isBusy.rst b/doc/source/netscript/singularityfunctions/isBusy.rst new file mode 100644 index 000000000..1bccaa2fe --- /dev/null +++ b/doc/source/netscript/singularityfunctions/isBusy.rst @@ -0,0 +1,9 @@ +isBusy() Netscript Function +=========================== + +.. js:function:: isBusy() + + If you are not in BitNode-4, then you must have Level 1 of Source-File 4 in order to run this function. + + Returns a boolean indicating whether or not the player is currently performing an 'action'. These actions include + working for a company/faction, studying at a univeristy, working out at a gym, creating a program, or committing a crime. diff --git a/doc/source/netscript/singularityfunctions/joinFaction.rst b/doc/source/netscript/singularityfunctions/joinFaction.rst new file mode 100644 index 000000000..a4f9da010 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/joinFaction.rst @@ -0,0 +1,10 @@ +joinFaction() Netscript Function +================================ + +.. js:function:: joinFaction(name) + + :param string name: Name of faction to join. CASE-SENSITIVE + + If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. + + This function will automatically accept an invitation from a faction and join it. diff --git a/doc/source/netscript/singularityfunctions/purchaseAugmentation.rst b/doc/source/netscript/singularityfunctions/purchaseAugmentation.rst new file mode 100644 index 000000000..28a5dd542 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/purchaseAugmentation.rst @@ -0,0 +1,14 @@ +purchaseAugmentation() Netscript Function +========================================= + +.. js:function:: purchaseAugmentation(factionName, augName) + + :param string factionName: Name of faction to purchase Augmentation from. CASE-SENSITIVE + :param string augName: Name of Augmentation to purchase. CASE-SENSITIVE + + + If you are not in BitNode-4, then you must have Level 3 of Source-File 4 in order to use this function. + + This function will try to purchase the specified Augmentation through the given Faction. + + This function will return true if the Augmentation is successfully purchased, and false otherwise. diff --git a/doc/source/netscript/singularityfunctions/purchaseProgram.rst b/doc/source/netscript/singularityfunctions/purchaseProgram.rst new file mode 100644 index 000000000..c60ad4bd2 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/purchaseProgram.rst @@ -0,0 +1,18 @@ +purchaseProgram() Netscript Function +==================================== + +.. js:function:: purchaseProgram(programName) + + :param string programName: Name of program to purchase. Must include '.exe' extension. Not case-sensitive. + + If you are not in BitNode-4, then you must have Level 1 of Source-File 4 in order to use this function. + + This function allows you to automatically purchase programs. You MUST have a TOR router in order to use this function. + The cost of purchasing programs using this function is the same as if you were purchasing them through the Dark Web using the + Terminal *buy* command. + + Example:: + + purchaseProgram("brutessh.exe"); + + This function will return true if the specified program is purchased, and false otherwise. diff --git a/doc/source/netscript/singularityfunctions/purchaseTor.rst b/doc/source/netscript/singularityfunctions/purchaseTor.rst new file mode 100644 index 000000000..c9a488ef6 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/purchaseTor.rst @@ -0,0 +1,11 @@ +purchaseTor() Netscript Function +================================ + +.. js:function:: purchaseTor() + + If you are not in BitNode-4, then you must have Level 1 of Source-File 4 in order to use this function. + + This function allows you to automatically purchase a TOR router. The cost for purchasing a TOR router using this + function is the same as if you were to manually purchase one. + + This function will return true if it successfully purchase a TOR router and false otherwise. diff --git a/doc/source/netscript/singularityfunctions/stopAction.rst b/doc/source/netscript/singularityfunctions/stopAction.rst new file mode 100644 index 000000000..3406c7176 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/stopAction.rst @@ -0,0 +1,18 @@ +stopAction() Netscript Function +=============================== + +.. js:function:: stopAction() + + If you are not in BitNode-4, then you must have Level 1 of Source-File 4 in order to run this function. + This function is used to end whatever 'action' the player is currently performing. The player + will receive whatever money/experience/etc. he has earned from that action. + + The actions that can be stopped with this function are: + + * Studying at a university + * Working for a company/faction + * Creating a program + * Committing a Crime + + This function will return true if the player's action was ended. It will return false if the player was not + performing an action when this function was called. diff --git a/doc/source/netscript/singularityfunctions/travelToCity.rst b/doc/source/netscript/singularityfunctions/travelToCity.rst new file mode 100644 index 000000000..a8c798016 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/travelToCity.rst @@ -0,0 +1,20 @@ +travelToCity() Netscript Function +================================= + +.. js:function:: travelToCity(cityName) + + :param string cityName: + City to travel to. CASE-SENSITIVE. + + * Aevum + * Chongqing + * Sector-12 + * New Tokyo + * Ishima + * Volhaven + + If you are not in BitNode-4, then you must have Level 1 of Source-File 4 in order to use this function. + + This function allows the player to travel to any city. The cost for using this function is the same as the cost for traveling through the Travel Agency. + + This function will return true if you successfully travel to the specified city and false otherwise. diff --git a/doc/source/netscript/singularityfunctions/universityCourse.rst b/doc/source/netscript/singularityfunctions/universityCourse.rst new file mode 100644 index 000000000..7d273fe65 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/universityCourse.rst @@ -0,0 +1,30 @@ +universityCourse() Netscript Function +===================================== + +.. js:function:: universityCourse(universityName, courseName) + + :param string universityName: + Name of university. Not case-sensitive. You must be in the correct city for whatever university you specify. + + * Summit University + * Rothman University + * ZB Institute Of Technology + :param string courseName: + Name of course. Not case-sensitive. + + * Study Computer Science + * Data Strucures + * Networks + * Algorithms + * Management + * Leadership + + If you are not in BitNode-4, then you must have Level 1 of Source-File 4 in order to use this function. + + This function will automatically set you to start taking a course at a university. If you are already in the middle of some + "working" action (such as working at a company, for a faction, or on a program), then running this function will automatically + cancel that action and give you your earnings. + + The cost and experience gains for all of these universities and classes are the same as if you were to manually visit and take these classes. + + This function will return true if you successfully start taking the course, and false otherwise. diff --git a/doc/source/netscript/singularityfunctions/upgradeHomeRam.rst b/doc/source/netscript/singularityfunctions/upgradeHomeRam.rst new file mode 100644 index 000000000..c3a7f7da2 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/upgradeHomeRam.rst @@ -0,0 +1,10 @@ +upgradeHomeRam() Netscript Function +=================================== + +.. js:function:: upgradeHomeRam() + + If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. + + This function will upgrade amount of RAM on the player's home computer. The cost is the same as if you were to do it manually. + + This function will return true if the player's home computer RAM is successfully upgraded, and false otherwise. diff --git a/doc/source/netscript/singularityfunctions/workForCompany.rst b/doc/source/netscript/singularityfunctions/workForCompany.rst new file mode 100644 index 000000000..fde4d11d7 --- /dev/null +++ b/doc/source/netscript/singularityfunctions/workForCompany.rst @@ -0,0 +1,24 @@ +workForCompany() Netscript Function +=================================== + +.. js:function:: workForCompany() + + If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. + + This function will automatically set you to start working at the company at which you are employed. + If you are already in the middle of some "working" action (such as working for a faction, training at + a gym, or creating a program), then running this function will automatically cancel that action and give you your earnings. + + This function will return true if the player starts working, and false otherwise. + + Note that when you are working for a company, you will not actually receive your earnings + (reputation, money, experience) until you FINISH the action. This can be an issue if, for example, + you only want to work until you get 100,000 company reputation. One small hack to get around this is to + continuously restart the action to receive your earnings:: + + while (getCompanyRep(COMPANY HERE) < VALUE) { + workForCompany(); + sleep(60000); + } + + This way, your company reputation will be updated every minute. diff --git a/doc/source/netscript/singularityfunctions/workForFaction.rst b/doc/source/netscript/singularityfunctions/workForFaction.rst new file mode 100644 index 000000000..e544a78da --- /dev/null +++ b/doc/source/netscript/singularityfunctions/workForFaction.rst @@ -0,0 +1,32 @@ +workForFaction() Netscript Function +=================================== + +.. js:function:: workForFaction(factionName, workType) + + :param string factionName: Name of faction to work for. CASE-SENSITIVE + :param string workType: + Type of work to perform for the faction + + * hacking/hacking contracts/hackingcontracts + * field/fieldwork/field work + * security/securitywork/security work + + If you are not in BitNode-4, then you must have Level 2 of Source-File 4 in order to use this function. + + This function will automatically set you to start working for the specified faction. + Obviously, you must be a member of the faction or else this function will fail. If you are already in + the middle of some "working" action (such as working for a company, training at a gym, or creating a program), + then running this function will automatically cancel that action and give you your earnings. + + This function will return true if you successfully start working for the specified faction, and false otherwise. + + Note that when you are working for a faction, you will not actually receive your earnings (reputation, experience) + until you FINISH the action. This can be an issue if, for example, you only want to work until you get 100,000 faction + reputation. One small hack to get around this is to continuously restart the action to receive your earnings:: + + while (getFactionRep(FACTION NAME) < VALUE) { + workForFaction(FACNAME, WORKTYPE); + sleep(60000); + } + + This way, your faction reputation will be updated every minute. diff --git a/doc/source/netscript/tixapi/buyStock.rst b/doc/source/netscript/tixapi/buyStock.rst new file mode 100644 index 000000000..06e70c891 --- /dev/null +++ b/doc/source/netscript/tixapi/buyStock.rst @@ -0,0 +1,16 @@ +buyStock() Netscript Function +============================= + +.. js:function:: buyStock(sym, shares) + + :param string sym: Symbol of stock to purchase + :param number shares: Number of shares to purchased. Must be positive. Will be rounded to nearest integer + :RAM cost: 2.5 GB + + Attempts to purchase shares of a stock using a `Market Order `_. + + If the player does not have enough money to purchase the specified number of shares, then no shares will be purchased. Remember + that every transaction on the stock exchange costs a certain commission fee. + + If this function successfully purchases the shares, it will return the stock price at which each share was purchased. Otherwise, + it will return 0. diff --git a/doc/source/netscript/tixapi/cancelOrder.rst b/doc/source/netscript/tixapi/cancelOrder.rst new file mode 100644 index 000000000..da9cb18c0 --- /dev/null +++ b/doc/source/netscript/tixapi/cancelOrder.rst @@ -0,0 +1,24 @@ +cancelOrder() Netscript Function +================================ + +.. js:function:: cancelOrder(sym, shares, price, type, pos) + + :param string sym: Symbol of stock to player order for + :param number shares: Number of shares for order. Must be positive. Will be rounded to nearest integer + :param number price: Execution price for the order + :param string type: Type of order. It must specify "limit" or "stop", and must also specify "buy" or "sell". This is NOT + case-sensitive. Here are four examples that will work: + + * limitbuy + * limitsell + * stopbuy + * stopsell + + :param string pos: + Specifies whether the order is a "Long" or "Short" position. The Values "L" or "S" can also be used. This is + NOT case-sensitive. + :RAM cost: 2.5 GB + + Cancels an oustanding Limit or Stop order on the stock market. + + The ability to use limit and stop orders is **not** immediately available to the player and must be unlocked later on in the game. diff --git a/doc/source/netscript/tixapi/getOrders.rst b/doc/source/netscript/tixapi/getOrders.rst new file mode 100644 index 000000000..2cd22dff5 --- /dev/null +++ b/doc/source/netscript/tixapi/getOrders.rst @@ -0,0 +1,68 @@ +getOrders() Netscript Function +============================== + +.. js:function:: getOrders() + + :RAM cost: 2.5 GB + + Returns your order book for the stock market. This is an object containing information + for all the Limit and Stop Orders you have in the stock market. + + The object has the following structure:: + + { + StockSymbol1: [ // Array of orders for this stock + { + shares: Order quantity + price: Order price + type: Order type + position: Either "L" or "S" for Long or Short position + }, + { + ... + }, + ... + ], + StockSymbol2: [ // Array of orders for this stock + ... + ], + ... + } + + The "Order type" property can have one of the following four values: + + * "Limit Buy Order" + * "Limit Sell Order" + * "Stop Buy Order" + * "Stop Sell Order" + + **Note that the order book will only contain information for stocks that you actually + have orders in**. For example, if you do not have orders in Nova Medical (NVMD), then the returned + object will not have a "NVMD" property. + + Example:: + + { + ECP: [ + { + shares: 5, + price: 100,000 + type: "Stop Buy Order", + position: "S", + }, + { + shares: 25, + price: 125,000 + type: "Limit Sell Order", + position: "L", + }, + ], + SYSC: [ + { + shares: 100, + price: 10,000 + type: "Limit Buy Order", + position: "L", + }, + ], + } diff --git a/doc/source/netscript/tixapi/getStockForecast.rst b/doc/source/netscript/tixapi/getStockForecast.rst new file mode 100644 index 000000000..7ce63f319 --- /dev/null +++ b/doc/source/netscript/tixapi/getStockForecast.rst @@ -0,0 +1,21 @@ +getStockForecast() Netscript Function +===================================== + +.. js:function:: getStockForecast(sym) + + :param string sym: Symbol of stock + :RAM cost: 2.5 GB + + Returns the probability that the specified stock's price will increase + (as opposed to decrease) during the next tick. + + The probability is returned as a decimal value, NOT a percentage (e.g. if a + stock has a 60% chance of increasing, then this function will return 0.6, + NOT 60). + + In other words, if this function returned 0.30 for a stock, then this means + that the stock's price has a 30% chance of increasing and a 70% chance of + decreasing during the next tick. + + In order to use this function, you must first purchase access to the Four Sigma (4S) + Market Data TIX API. diff --git a/doc/source/netscript/tixapi/getStockMaxShares.rst b/doc/source/netscript/tixapi/getStockMaxShares.rst new file mode 100644 index 000000000..4c1838183 --- /dev/null +++ b/doc/source/netscript/tixapi/getStockMaxShares.rst @@ -0,0 +1,11 @@ +getStockMaxShares() Netscript Function +====================================== + +.. js:function:: getStockMaxShares(sym) + + :param string sym: Stock symbol + :RAM cost: 2 GB + + Returns the maximum number of shares that the stock has. This is the maximum + amount of the stock that can be purchased in both the Long and Short + positions combined diff --git a/doc/source/netscript/tixapi/getStockPosition.rst b/doc/source/netscript/tixapi/getStockPosition.rst new file mode 100644 index 000000000..048a3bfb3 --- /dev/null +++ b/doc/source/netscript/tixapi/getStockPosition.rst @@ -0,0 +1,27 @@ +getStockPosition() Netscript Function +===================================== + +.. js:function:: getStockPosition(sym) + + :param string sym: Stock symbol + :RAM cost: 2 GB + + Returns an array of four elements that represents the player's position in a stock. + + The first element is the returned array is the number of shares the player owns of the stock in the + `Long position `_. The second + element in the array is the average price of the player's shares in the Long position. + + The third element in the array is the number of shares the player owns of the stock in the + `Short position `_. The fourth + element in the array is the average price of the player's Short position. + + All elements in the returned array are numeric. + + Example:: + + pos = getStockPosition("ECP"); + shares = pos[0]; + avgPx = pos[1]; + sharesShort = pos[2]; + avgPxShort = pos[3]; diff --git a/doc/source/netscript/tixapi/getStockPrice.rst b/doc/source/netscript/tixapi/getStockPrice.rst new file mode 100644 index 000000000..ae88e644f --- /dev/null +++ b/doc/source/netscript/tixapi/getStockPrice.rst @@ -0,0 +1,14 @@ +getStockPrice() Netscript Function +================================== + +.. js:function:: getStockPrice(sym) + + :param string sym: Stock symbol + :RAM cost: 2 GB + + Returns the price of a stock, given its symbol (NOT the company name). The symbol is a sequence + of two to four capital letters. + + Example:: + + getStockPrice("FISG"); diff --git a/doc/source/netscript/tixapi/getStockSymbols.rst b/doc/source/netscript/tixapi/getStockSymbols.rst new file mode 100644 index 000000000..17714156f --- /dev/null +++ b/doc/source/netscript/tixapi/getStockSymbols.rst @@ -0,0 +1,8 @@ +getStockSymbols() Netscript Function() +====================================== + +.. js:function:: getStockSymbols() + + :RAM cost: 2 GB + + Returns an array of the symbols of the tradable stocks diff --git a/doc/source/netscript/tixapi/getStockVolatility.rst b/doc/source/netscript/tixapi/getStockVolatility.rst new file mode 100644 index 000000000..0cc5fccb6 --- /dev/null +++ b/doc/source/netscript/tixapi/getStockVolatility.rst @@ -0,0 +1,17 @@ +getStockVolatility() Netscript Function +======================================= + +.. js:function:: getStockVolatility(sym) + + :param string sym: Symbol of stock + :RAM cost: 2.5 GB + + Returns the volatility of the specified stock. + + Volatility represents the maximum percentage by which a stock's price can + change every tick. The volatility is returned as a decimal value, NOT + a percentage (e.g. if a stock has a volatility of 3%, then this function will + return 0.03, NOT 3). + + In order to use this function, you must first purchase access to the Four Sigma (4S) + Market Data TIX API. diff --git a/doc/source/netscript/tixapi/placeOrder.rst b/doc/source/netscript/tixapi/placeOrder.rst new file mode 100644 index 000000000..8e8de3dda --- /dev/null +++ b/doc/source/netscript/tixapi/placeOrder.rst @@ -0,0 +1,26 @@ +placeOrder() Netscript Function +=============================== + +.. js:function:: placeOrder(sym, shares, price, type, pos) + + :param string sym: Symbol of stock to player order for + :param number shares: Number of shares for order. Must be positive. Will be rounded to nearest integer + :param number price: Execution price for the order + :param string type: Type of order. It must specify "limit" or "stop", and must also specify "buy" or "sell". This is NOT + case-sensitive. Here are four examples that will work: + + * limitbuy + * limitsell + * stopbuy + * stopsell + + :param string pos: + Specifies whether the order is a "Long" or "Short" position. The Values "L" or "S" can also be used. This is + NOT case-sensitive. + :RAM cost: 2.5 GB + + Places an order on the stock market. This function only works for `Limit and Stop Orders `_. + + The ability to place limit and stop orders is **not** immediately available to the player and must be unlocked later on in the game. + + Returns true if the order is successfully placed, and false otherwise. diff --git a/doc/source/netscript/tixapi/purchase4SMarketData.rst b/doc/source/netscript/tixapi/purchase4SMarketData.rst new file mode 100644 index 000000000..841cb4517 --- /dev/null +++ b/doc/source/netscript/tixapi/purchase4SMarketData.rst @@ -0,0 +1,11 @@ +purchase4SMarketData() Netscript Function +========================================= + +.. js:function:: purchase4SMarketData() + + :RAM cost: 2.5 GB + + Purchase 4S Market Data Access. + + Returns true if you successfully purchased it or if you already have access. + Returns false otherwise. diff --git a/doc/source/netscript/tixapi/purchase4SMarketDataTixApi.rst b/doc/source/netscript/tixapi/purchase4SMarketDataTixApi.rst new file mode 100644 index 000000000..119988f05 --- /dev/null +++ b/doc/source/netscript/tixapi/purchase4SMarketDataTixApi.rst @@ -0,0 +1,11 @@ +purchase4SMarketDataTixApi() Netscript Function +=============================================== + +.. js:function:: purchase4SMarketDataTixApi() + + :RAM cost: 2.5 GB + + Purchase 4S Market Data TIX API Access. + + Returns true if you successfully purchased it or if you already have access. + Returns false otherwise. diff --git a/doc/source/netscript/tixapi/sellShort.rst b/doc/source/netscript/tixapi/sellShort.rst new file mode 100644 index 000000000..2d8291fa5 --- /dev/null +++ b/doc/source/netscript/tixapi/sellShort.rst @@ -0,0 +1,18 @@ +sellShort() Netscript Function +============================== + +.. js:function:: sellShort(sym, shares) + + :param string sym: Symbol of stock to sell + :param number shares: Number of shares to sell. Must be positive. Will be rounded to nearest integer + :RAM cost: 2.5 GB + + Attempts to sell a `short `_ position of a stock + using a `Market Order `_. + + The ability to short a stock is **not** immediately available to the player and must be unlocked later on in the game. + + If the specified number of shares exceeds the amount that the player actually owns, then this function will sell all owned + shares. Remember that every transaction on the stock exchange costs a certain commission fee. + + If the sale is successful, this function will return the stock price at which each share was sold. Otherwise it will return 0. diff --git a/doc/source/netscript/tixapi/sellStock.rst b/doc/source/netscript/tixapi/sellStock.rst new file mode 100644 index 000000000..7c5e66723 --- /dev/null +++ b/doc/source/netscript/tixapi/sellStock.rst @@ -0,0 +1,20 @@ +sellStock() Netscript Function +============================== + +.. js:function:: sellStock(sym, shares) + + :param string sym: Symbol of stock to sell + :param number shares: Number of shares to sell. Must be positive. Will be rounded to nearest integer + :RAM cost: 2.5 GB + + Attempts to sell shares of a stock using a `Market Order `_. + + If the specified number of shares in the function exceeds the amount that the player actually owns, then this function will + sell all owned shares. Remember that every transaction on the stock exchange costs a certain commission fee. + + The net profit made from selling stocks with this function is reflected in the script's statistics. + This net profit is calculated as:: + + shares * (sell price - average price of purchased shares) + + If the sale is successful, this function will return the stock price at which each share was sold. Otherwise, it will return 0. diff --git a/doc/source/netscript/tixapi/shortStock.rst b/doc/source/netscript/tixapi/shortStock.rst new file mode 100644 index 000000000..cd319cf88 --- /dev/null +++ b/doc/source/netscript/tixapi/shortStock.rst @@ -0,0 +1,18 @@ +shortStock() Netscript Function +=============================== + +.. js:function:: shortStock(sym, shares) + + :param string sym: Symbol of stock to short + :param number shares: Number of shares to short. Must be positive. Will be rounded to nearest integer + :RAM cost: 2.5 GB + + Attempts to purchase a `short `_ position of a stock + using a `Market Order `_. + + The ability to short a stock is **not** immediately available to the player and must be unlocked later on in the game. + + If the player does not have enough money to purchase the specified number of shares, then no shares will be purchased. + Remember that every transaction on the stock exchange costs a certain commission fee. + + If the purchase is successful, this function will return the stock price at which each share was purchased. Otherwise, it will return 0. diff --git a/src/Constants.ts b/src/Constants.ts index eb36b0b1b..ee2dca54f 100644 --- a/src/Constants.ts +++ b/src/Constants.ts @@ -279,235 +279,6 @@ export let CONSTANTS: IMap = { // BitNode/Source-File related stuff TotalNumBitNodes: 24, - /* Tutorial related things */ - TutorialNetworkingText: "Servers are a central part of the game. You start with a single personal server (your home computer) " + - "and you can purchase additional servers as you progress through the game. Connecting to other servers " + - "and hacking them can be a major source of income and experience. Servers can also be used to run " + - "scripts which can automatically hack servers for you.

" + - "In order to navigate between machines, use the 'scan' or 'scan-analyze' Terminal command to see all servers " + - "that are reachable from your current server. Then, you can use the 'connect [hostname/ip]' " + - "command to connect to one of the available machines.

" + - "The 'hostname' and 'ifconfig' commands can be used to display the hostname/IP of the " + - "server you are currently connected to.", - - TutorialHackingText: "In the year 2077, currency has become digital and decentralized. People and corporations " + - "store their money on servers. By hacking these servers, you can steal their money and gain " + - "experience.

" + - "

Gaining root access


" + - "The key to hacking a server is to gain root access to that server. This can be done using " + - "the NUKE virus (NUKE.exe). You start the game with a copy of the NUKE virus on your home " + - "computer. The NUKE virus attacks the target server's open ports using buffer overflow " + - "exploits. When successful, you are granted root administrative access to the machine.

" + - "Typically, in order for the NUKE virus to succeed, the target server needs to have at least " + - "one of its ports opened. Some servers have no security and will not need any ports opened. Some " + - "will have very high security and will need many ports opened. In order to open ports on another " + - "server, you will need to run programs that attack the server to open specific ports. These programs " + - "can be coded once your hacking skill gets high enough, or they can be purchased if you can find " + - "a seller.

" + - "In order to determine how many ports need to be opened to successfully NUKE a server, connect to " + - "that server and run the 'analyze' command. This will also show you which ports have already been " + - "opened.

" + - "Once you have enough ports opened and have ran the NUKE virus to gain root access, the server " + - "can then be hacked by simply calling the 'hack' command through terminal, or by using a script.

" + - "

Hacking mechanics


" + - "When you execute the hack command, either manually through the terminal or automatically through " + - "a script, you attempt to hack the server. This action takes time. The more advanced a server's " + - "security is, the more time it will take. Your hacking skill level also affects the hacking time, " + - "with a higher hacking skill leading to shorter hacking times. Also, running the hack command " + - "manually through terminal is faster than hacking from a script.

" + - "Your attempt to hack a server will not always succeed. The chance you have to successfully hack a " + - "server is also determined by the server's security and your hacking skill level. Even if your " + - "hacking attempt is unsuccessful, you will still gain experience points.

" + - "When you successfully hack a server. You steal a certain percentage of that server's total money. This " + - "percentage is determined by the server's security and your hacking skill level. The amount of money " + - "on a server is not limitless. So, if you constantly hack a server and deplete its money, then you will " + - "encounter diminishing returns in your hacking (since you are only hacking a certain percentage). You can " + - "increase the amount of money on a server using a script and the grow() function in Netscript.

" + - "

Server Security


" + - "Each server has a security level, typically between 1 and 100. A higher number means the server has stronger security. " + - "It is possible for a server to have a security level of 100 or higher, in which case hacking that server " + - "will become impossible (0% chance to hack).

" + - "As mentioned above, a server's security level is an important factor " + - "to consider when hacking. You can check a server's security level using the 'analyze' command, although this " + - "only gives an estimate (with 5% uncertainty). You can also check a server's security in a script, using the " + - "getServerSecurityLevel(server) function in Netscript. See the Netscript documentation for more details. " + - "This function will give you an exact value for a server's security.

" + - "Whenever a server is hacked manually or through a script, its security level increases by a small amount. Calling " + - "the grow() command in a script will also increase security level of the target server. These actions will " + - "make it harder for you to hack the server, and decrease the amount of money you can steal. You can lower a " + - "server's security level in a script using the weaken(server) function in Netscript. See the Netscript " + - "documentation for more details.

" + - "A server has a minimum security level that is equal to one third of its starting security, rounded to the " + - "nearest integer. To be more precise:

" + - "server.minSecurityLevel = Math.max(1, Math.round(server.startingSecurityLevel / 3))

" + - "This means that a server's security will not fall below this value if you are trying to weaken it.", - - TutorialScriptsText: "Scripts can be used to automate the hacking process. Scripts must be written in the Netscript language. " + - "Documentation about the Netscript language can be found in the 'Netscript Programming Language' " + - "section of this 'Tutorial' page.

" + - "It is highly recommended that you have a basic background in programming to start writing scripts. " + - "You by no means need to be an expert. All you need is some familiarity with basic programming " + - "constructs like for/while loops, if statements, " + - "functions, variables, etc. The Netscript programming language most resembles the Javascript language. " + - "Therefore, a good beginner's programming tutorial to read might be " + - "this one. Note that while the Netscript language is similar to Javascript, it is not the exact same, so the " + - "syntax will vary a little bit.

" + - "Running a script requires RAM. The more complex a script is, the more RAM " + - "it requires to run. Scripts can be run on any server you have root access to.

" + - "Here are some Terminal commands that are useful when working with scripts:

" + - "check [script] [args...]
Prints the logs of the script specified by the name and arguments to Terminal. Arguments should be separated " + - "by a space. Note that scripts are uniquely " + - "identified by their arguments as well as their name. For example, if you ran a script 'foo.script' with the argument 'foodnstuff' then in order to 'check' it you must " + - "also add the 'foodnstuff' argument to the check command as so:
check foo.script foodnstuff

" + - "free
Shows the current server's RAM usage and availability

" + - "kill [script] [args...]
Stops a script that is running with the specified script name and arguments. " + - "Arguments should be separated by a space. Note that " + - "scripts are uniquely identified by their arguments as well as their name. For example, if you ran a script 'foo.script' with the " + - "argument 1 and 2, then just typing 'kill foo.script' will not work. You have to use 'kill foo.script 1 2'.

" + - "mem [script] [-t] [n]
Check how much RAM a script requires to run with n threads

" + - "nano [script]
Create/Edit a script. The name of the script must end with the '.script' extension

" + - "ps
Displays all scripts that are actively running on the current server

" + - "rm [script]
Delete a script

" + - "run [script] [-t] [n] [args...]
Run a script with n threads and the specified arguments. Each argument should be separated by a space. " + - "Both the arguments and thread specification are optional. If neither are specified, then the script will be run single-threaded with no arguments.
" + - "Examples:
run foo.script
The command above will run 'foo.script' single-threaded with no arguments." + - "
run foo.script -t 10
The command above will run 'foo.script' with 10 threads and no arguments." + - "
run foo.script foodnstuff sigma-cosmetics 10
The command above will run 'foo.script' single-threaded with three arguments: [foodnstuff, sigma-cosmetics, 10]" + - "
run foo.script -t 50 foodnstuff
The command above will run 'foo.script' with 50 threads and a single argument: [foodnstuff]

" + - "tail [script] [args...]
Displays the logs of the script specified by the name and arguments. Note that scripts are uniquely " + - "identified by their arguments as well as their name. For example, if you ran a script 'foo.script' with the argument 'foodnstuff' then in order to 'tail' it you must " + - "also add the 'foodnstuff' argument to the tail command as so:
tail foo.script foodnstuff

" + - "top
Displays all active scripts and their RAM usage

" + - "

Multithreading scripts


" + - "Scripts can be multithreaded. A multithreaded script runs the script's code once in each thread. The result is that " + - "every call to the hack(), grow(), and weaken() Netscript functions will have its effect multiplied by the number of threads. " + - "For example, if a normal single-threaded script is able to hack $10,000, then running the same script with 5 threads would " + - "yield $50,000.

" + - "When multithreading a script, the total RAM cost can be calculated by simply multiplying the base RAM cost of the script " + - "with the number of threads, where the base cost refers to the amount of RAM required to run the script single-threaded. " + - "In the terminal, you can run the " + - "'mem [scriptname] -t n' command to see how much RAM a script requires with n threads.

" + - "Every method for running a script has an option for making it multihreaded. To run a script with " + - "n threads from a Terminal:
" + - "run [scriptname] -t n

" + - "Using Netscript commands:
" + - "run('scriptname.script', n);
" + - "exec('scriptname.script, 'targetServer', n);

" + - "

Notes about how scripts work offline


" + - " The scripts that you write and execute are interpreted in Javascript. For this " + - "reason, it is not possible for these scripts to run while offline (when the game is closed). " + - "It is important to note that for this reason, conditionals such as if/else statements and certain " + - "commands such as purchaseHacknetNode() or nuke() will not work while the game is offline.

" + - "However, Scripts WILL continue to generate money and hacking exp for you while the game is offline. This " + - "offline production is based off of the scripts' production while the game is online.

" + - "grow() and weaken() are two Netscript commands that will also be applied when the game is offline, although at a slower rate " + - "compared to if the game was open. This is done by having each script keep track of the " + - "rate at which the grow() and weaken() commands are called when the game is online. These calculated rates are used to determine how many times " + - "these function calls would be made while the game is offline.

" + - "Also, note that because of the way the Netscript interpreter is implemented, " + - "whenever you reload or re-open the game all of the scripts that you are running will " + - "start running from the BEGINNING of the code. The game does not keep track of where exactly " + - "the execution of a script is when it saves/loads.


", - TutorialNetscriptText: "Netscript is a programming language implemented for this game. There are two versions of Netscript: " + - "Netscript 1.0 and Netscript 2.0 (NetscriptJS).

" + - "Click here for Bitburner's official Netscript documentation", - TutorialTravelingText:"There are six major cities in the world that you are able to travel to:

" + - " Aevum
" + - " Chongqing
" + - " Sector-12
" + - " New Tokyo
" + - " Ishima
" + - " Volhaven

" + - "To travel between cities, visit your current city's travel agency through the 'World' page. " + - "From the travel agency you can travel to any other city. Doing so costs money.

" + - "Each city has its own set of companies and unique locations. Also, certain content is only available to you " + - "if you are in certain cities, so get exploring!", - TutorialCompaniesText: "Hacking is not the only way to gain money and experience! Located around the world are many " + - "different companies which you can work for. By working for a company you can earn money, " + - "train your various labor skills, and unlock powerful passive perks.

" + - "To apply for a job, visit the company you want to work for through the 'World' menu. The company " + - "page will have options that let you apply to positions in the company. There might be several different " + - "positions you can apply for, ranging from software engineer to business analyst to security officer.

" + - "When you apply for a job, you will get the offer if your stats are high enough. Your first position at " + - "a company will be an entry-level position such as 'intern'. Once you get the job, an button will appear on " + - "the company page that allows you to work for the company. Click this button to start working.

" + - "Working occurs in 8 hour shifts. Once you start working, you will begin earning money, experience, " + - "and reputation. The rate at which you money and experience depends on the company and your position. " + - "The amount of reputation you gain for your company is based on your job performance, which is affected by " + - "your stats. Different positions value different stats. When you are working, you are unable to perform any " + - "other actions such as using your terminal or visiting other locations (However, note that any scripts you have " + - "running on servers will continue to run as you work!). It is possible to cancel your work shift before the " + - "8 hours is up. However, if you have a full-time job, then cancelling a shift early will result in you gaining " + - "only half of the reputation " + - "that you had earned up to that point. There are also part-time/consultant jobs available where you will not " + - " be penalized if you cancel a work shift early. However, these positions pay less than full-time positions.

" + - "As you continue to work at a company, you will gain more and more reputation at that company. When your stats " + - "and reputation are high enough, you can get a promotion. You can apply for a promotion on the company page, just like " + - "you applied for the job originally. Higher positions at a company provide better salaries and stat gains.

" + - "

Infiltrating Companies


" + - "Many companies have facilities that you can attempt to infiltrate. By infiltrating, you can steal classified company secrets " + - "and then sell these for money or for faction reputation. To try and infiltrate a company, visit a company through the " + - "'World' menu. There will be an option that says 'Infiltrate Company'.

" + - "When infiltrating a company, you must progress through clearance levels in the facility. Every clearance level " + - "has some form of security that you must get past. There are several forms of security, ranging from high-tech security systems to " + - "armed guards. For each form of security, there are a variety of options that you can choose to try and bypass the security. Examples " + - "include hacking the security, engaging in combat, assassination, or sneaking past the security. The chance to succeed for each option " + - "is determined in part by your stats. So, for example, trying to hack the security system relies on your hacking skill, whereas trying to " + - "sneak past the security relies on your agility level.

" + - "The facility has a 'security level' that affects your chance of success when trying to get past a clearance level. " + - "Every time you advance to the next clearance level, the facility's security level will increase by a fixed amount. Furthermore " + - "the options you choose and whether you succeed or fail will affect the security level as well. For example, " + - "if you try to kill a security guard and fail, the security level will increase by a lot. If you choose to sneak past " + - "security and succeed, the security level will not increase at all.

" + - "Every 5 clearance levels, you will steal classified company secrets that can be sold for money or faction reputation. However, " + - "in order to sell these secrets you must successfully escape the facility using the 'Escape' option. Furthermore, companies have " + - "a max clearance level. If you reach the max clearance level you will automatically escape the facility with all of your " + - "stolen secrets.

", - TutorialFactionsText: "Throughout the game you may receive invitations from factions. There are many different factions, and each faction " + - "has different criteria for determining its potential members. Joining a faction and furthering its cause is crucial " + - "to progressing in the game and unlocking endgame content.

" + - "It is possible to join multiple factions if you receive invitations from them. However, note that joining a faction " + - "may prevent you from joining other rival factions.

" + - "The 'Factions' link on the menu brings up a list of all factions that you have joined. " + - "You can select a Faction on this list to go to that Faction page. This page displays general " + - "information about the Faction and also lets you perform work for the faction. " + - "Working for a Faction is similar to working for a company except that you don't get paid a salary. " + - "You will only earn reputation in your Faction and train your stats. Also, cancelling work early " + - "when working for a Faction does NOT result in reduced experience/reputation earnings.

" + - "Earning reputation for a Faction unlocks powerful Augmentations. Purchasing and installing these Augmentations will " + - "upgrade your abilities. The Augmentations that are available to unlock vary from faction to faction.", - TutorialAugmentationsText: "Advances in science and medicine have lead to powerful new technologies that allow people to augment themselves " + - "beyond normal human capabilities. There are many different types of Augmentations, ranging from cybernetic to " + - "genetic to biological. Acquiring these Augmentations enhances the user's physical and mental faculties.

" + - "Because of how powerful these Augmentations are, the technology behind them is kept private and secret by the " + - "corporations and organizations that create them. Therefore, the only way for the player to obtain Augmentations is " + - "through Factions. After joining a Faction and earning enough reputation in it, you will be able to purchase " + - "its Augmentations. Different Factions offer different Augmentations. Augmentations must be purchased in order to be installed, " + - "and they are fairly expensive.

" + - "When you purchase an Augmentation, the price of purchasing another Augmentation increases by 90%. This multiplier stacks for " + - "each Augmentation you purchase. You will not gain the benefits of your purchased Augmentations until you install them. You can " + - "choose to install Augmentations through the 'Augmentations' menu tab. Once you install your purchased Augmentations, " + - "their costs are reset back to the original price.

" + - "Unfortunately, installing Augmentations has side effects. You will lose most of the progress you've made, including your " + - "skills, stats, and money. You will have to start over, but you will have all of the Augmentations you have installed to " + - "help you progress.

" + - "To summarize, here is a list of everything you will LOSE when you install an Augmentation:

" + - "Stats/Skills
" + - "Money
" + - "Scripts on all servers EXCEPT your home computer
" + - "Purchased servers
" + - "Hacknet Nodes
" + - "Company/faction reputation
" + - "Jobs and Faction memberships
" + - "Programs
" + - "Stocks
" + - "TOR router

" + - "Here is everything you will KEEP when you install an Augmentation:

" + - "Every Augmentation you have installed
" + - "Scripts on your home computer
" + - "RAM and CPU Core Upgrades on your home computer
" + - "World Stock Exchange account and TIX API Access
", - LatestUpdate: ` v0.44.1 @@ -516,6 +287,8 @@ export let CONSTANTS: IMap = { ** Sleeves are now assigned to Shock Recovery task by default ** Shock Recovery and Synchronize tasks are now twice as effective + * Changed documentation so that Netscript functions are own their own pages. Sorry if this is annoying, it was necessary for properly cross-referencing + * Officially deprecated the Wiki (the fandom site). Use the 'readthedocs' Documentation instead * Bug Fix: 'rm' Terminal and Netscript commands now work on non-program files that have '.exe' in the name (by Github user MasonD) * Bug Fix: The 'Find All Valid Math Expressions' Coding Contract should now properly ignore whitespace in answers * Bug Fix: The 'Merge Overlapping Intervals' Coding Contract should now properly accept 2D arrays when being attempted through Netscript diff --git a/src/InteractiveTutorial.js b/src/InteractiveTutorial.js index 8a36775ee..a1c604f08 100644 --- a/src/InteractiveTutorial.js +++ b/src/InteractiveTutorial.js @@ -496,8 +496,7 @@ function iTutorialEnd() { var txt = createElement("p", { innerHTML: "If you are new to the game, the following links may be useful for you!

" + - "Getting Started Guide" + - "Wiki" + + "Getting Started Guide" + "Documentation

" + "The Beginner's Guide to Hacking was added to your home computer! It contains some tips/pointers for starting out with the game. " + "To read it, go to Terminal and enter

cat hackers-starting-handbook.lit" diff --git a/src/PersonObjects/Sleeve/Sleeve.ts b/src/PersonObjects/Sleeve/Sleeve.ts index a02842560..8e644e2af 100644 --- a/src/PersonObjects/Sleeve/Sleeve.ts +++ b/src/PersonObjects/Sleeve/Sleeve.ts @@ -394,6 +394,7 @@ export class Sleeve extends Person { this.agility_exp = 0; this.charisma_exp = 0; this.applyAugmentation(aug); + this.augmentations.push({ name: aug.name, level: 1 }); this.updateStatLevels(); } diff --git a/src/PersonObjects/Sleeve/SleeveAugmentationsUI.ts b/src/PersonObjects/Sleeve/SleeveAugmentationsUI.ts index 3fc5b76a7..bfc4fe43b 100644 --- a/src/PersonObjects/Sleeve/SleeveAugmentationsUI.ts +++ b/src/PersonObjects/Sleeve/SleeveAugmentationsUI.ts @@ -8,6 +8,7 @@ import { IPlayer } from "../IPlayer"; import { Augmentation } from "../../Augmentation/Augmentation"; import { Augmentations } from "../../Augmentation/Augmentations"; +import { AugmentationNames } from "../../Augmentation/data/AugmentationNames"; import { Faction } from "../../Faction/Faction"; import { Factions } from "../../Faction/Factions"; @@ -19,18 +20,25 @@ import { dialogBoxCreate } from "../../../utils/DialogBox"; import { createElement } from "../../../utils/uiHelpers/createElement"; import { createPopup } from "../../../utils/uiHelpers/createPopup"; import { createPopupCloseButton } from "../../../utils/uiHelpers/createPopupCloseButton"; +import { removeElementById } from "../../../utils/uiHelpers/removeElementById"; export function createSleevePurchaseAugsPopup(sleeve: Sleeve, p: IPlayer) { + // Array of all owned Augmentations. Names only + const ownedAugNames: string[] = sleeve.augmentations.map((e) => {return e.name}); + // You can only purchase Augmentations that are actually available from // your factions. I.e. you must be in a faction that has the Augmentation // and you must also have enough rep in that faction in order to purchase it. const availableAugs: Augmentation[] = []; for (const facName of p.factions) { + if (facName === "Bladeburners") { continue; } const fac: Faction | null = Factions[facName]; if (fac == null) { continue; } for (const augName of fac.augmentations) { + if (augName === AugmentationNames.NeuroFluxGovernor) { continue; } + if (ownedAugNames.includes(augName)) { continue; } const aug: Augmentation | null = Augmentations[augName]; if (fac.playerReputation > aug.baseRepRequirement && !availableAugs.includes(aug)) { @@ -42,6 +50,36 @@ export function createSleevePurchaseAugsPopup(sleeve: Sleeve, p: IPlayer) { // Create popup const popupId = "purchase-sleeve-augs-popup"; + // Close popup button + const closeBtn = createPopupCloseButton(popupId, { innerText: "Cancel" }); + + // General info about owned Augmentations + const ownedAugsInfo = createElement("p", { + display: "block", + innerHTML: "Owned Augmentations:", + }); + + const popupElems: HTMLElement[] = [closeBtn, ownedAugsInfo]; + + // Show owned augmentations + // First we'll make a div with a reduced width, so the tooltips don't go off + // the edge of the popup + const ownedAugsDiv = createElement("div", { width: "70%" }); + for (const ownedAug of ownedAugNames) { + const aug: Augmentation | null = Augmentations[ownedAug]; + if (aug == null) { + console.warn(`Invalid Augmentation: ${ownedAug}`); + continue; + } + + ownedAugsDiv.appendChild(createElement("div", { + class: "gang-owned-upgrade", // Reusing a class from the Gang UI + innerText: ownedAug, + tooltip: aug.info, + })) + } + popupElems.push(ownedAugsDiv); + // General info about buying Augmentations const info = createElement("p", { innerHTML: @@ -55,10 +93,7 @@ export function createSleevePurchaseAugsPopup(sleeve: Sleeve, p: IPlayer) { ].join(" "), }); - // Close popup - const closeBtn = createPopupCloseButton(popupId, { innerText: "Cancel" }); - - const popupElems: HTMLElement[] = [closeBtn, info]; + popupElems.push(info); for (const aug of availableAugs) { const div = createElement("div", { @@ -66,17 +101,21 @@ export function createSleevePurchaseAugsPopup(sleeve: Sleeve, p: IPlayer) { }); div.appendChild(createElement("p", { + fontSize: "12px", innerHTML: [ `

${aug.name}


`, `Cost: ${numeralWrapper.formatMoney(aug.baseCost)}

`, `${aug.info}` ].join(" "), + padding: "2px", clickListener: () => { if (p.canAfford(aug.baseCost)) { p.loseMoney(aug.baseCost); sleeve.installAugmentation(aug); dialogBoxCreate(`Installed ${aug.name} on Duplicate Sleeve!`, false) + removeElementById(popupId); + createSleevePurchaseAugsPopup(sleeve, p); } else { dialogBoxCreate(`You cannot afford ${aug.name}`, false); } diff --git a/src/PersonObjects/Sleeve/SleeveUI.ts b/src/PersonObjects/Sleeve/SleeveUI.ts index b8ee967a0..cf8ef9cf4 100644 --- a/src/PersonObjects/Sleeve/SleeveUI.ts +++ b/src/PersonObjects/Sleeve/SleeveUI.ts @@ -11,8 +11,6 @@ import { IPlayer } from "../IPlayer"; import { CONSTANTS } from "../../Constants"; import { Locations } from "../../Locations"; -import { Augmentations } from "../../Augmentation/Augmentations"; - import { Faction } from "../../Faction/Faction"; import { Factions } from "../../Faction/Factions"; import { FactionWorkType } from "../../Faction/FactionWorkTypeEnum"; @@ -276,7 +274,7 @@ function createSleeveUi(sleeve: Sleeve, allSleeves: Sleeve[]): ISleeveUIElems { elems.purchaseAugsButton = createElement("button", { class: "std-button", display: "block", - innerText: "Purchase Augmentations", + innerText: "Manage Augmentations", clickListener: () => { createSleevePurchaseAugsPopup(sleeve, playerRef!); } diff --git a/src/PersonObjects/Sleeve/data/SleeveFaq.ts b/src/PersonObjects/Sleeve/data/SleeveFaq.ts index dbd39b511..72d3a9f99 100644 --- a/src/PersonObjects/Sleeve/data/SleeveFaq.ts +++ b/src/PersonObjects/Sleeve/data/SleeveFaq.ts @@ -1,7 +1,7 @@ export const SleeveFaq: string = [ -"How do sleeves work?
", -"Sleeves are essentially clones. You can use them to perform any work type", +"How do Duplicate Sleeves work?
", +"Duplicate Sleeves are essentially clones. You can use them to perform any work type", "action, such as working for a company/faction or committing a crime.", "Having sleeves perform these tasks earns you money, experience, and reputation.

", "Sleeves are their own individuals, which means they each have their own", @@ -31,6 +31,19 @@ export const SleeveFaq: string = "To clarify further, if you have two sleeves they can work for two different", "companies, but they cannot both work for the same company.

", +"Why did my Sleeve stop working?
", +"Sleeves are subject to the same time restrictions as you. This means that", +"they automatically stop working at a company after 8 hours, and stop working", +"for a faction after 20 hours.

", + +"How do I buy Augmentations for my Sleeves?
", +"Your Sleeve needs to have a Shock of 0 in order for you to buy Augmentations", +"for it.

", + +"Why can't I buy the X Augmentation for my sleeve?
", +"Certain Augmentations, like Bladeburner-specific ones and NeuroFlux Governor,", +"are not available for sleeves.

", + "Do sleeves get reset when installing Augmentations or switching BitNodes?
", "Sleeves are reset when switching BitNodes, but not when installing Augmentations." ].join(" "); diff --git a/src/engine.js b/src/engine.js index 7ad1175f9..bbf24a9f9 100644 --- a/src/engine.js +++ b/src/engine.js @@ -209,17 +209,6 @@ const Engine = { //Main menu buttons saveMainMenuButton: null, deleteMainMenuButton: null, - - //Tutorial buttons - tutorialNetworkingButton: null, - tutorialHackingButton: null, - tutorialScriptsButton: null, - tutorialNetscriptButton: null, - tutorialTravelingButton: null, - tutorialCompaniesButton: null, - tutorialFactionsButton: null, - tutorialAugmentationsButton: null, - tutorialBackButton: null, }, //Display objects @@ -348,7 +337,6 @@ const Engine = { loadTutorialContent: function() { Engine.hideAllContent(); Engine.Display.tutorialContent.style.display = "block"; - Engine.displayTutorialContent(); routing.navigateTo(Page.Tutorial); MainMenuLinks.Tutorial.classList.add("active"); }, @@ -769,40 +757,6 @@ const Engine = { Engine.Display.factionsContent.appendChild(invitationsList); }, - displayTutorialContent: function() { - document.getElementById("tutorial-getting-started-link").style.display = "block"; - Engine.Clickables.tutorialNetworkingButton.style.display = "block"; - Engine.Clickables.tutorialHackingButton.style.display = "block"; - Engine.Clickables.tutorialScriptsButton.style.display = "block"; - Engine.Clickables.tutorialNetscriptButton.style.display = "block"; - Engine.Clickables.tutorialTravelingButton.style.display = "block"; - Engine.Clickables.tutorialCompaniesButton.style.display = "block"; - Engine.Clickables.tutorialFactionsButton.style.display = "block"; - Engine.Clickables.tutorialAugmentationsButton.style.display = "block"; - document.getElementById("tutorial-shortcuts-link").style.display = "block"; - - Engine.Clickables.tutorialBackButton.style.display = "none"; - document.getElementById("tutorial-text").style.display = "none"; - }, - - //Displays the text when a section of the Tutorial is opened - displayTutorialPage: function(text) { - document.getElementById("tutorial-getting-started-link").style.display = "none"; - Engine.Clickables.tutorialNetworkingButton.style.display = "none"; - Engine.Clickables.tutorialHackingButton.style.display = "none"; - Engine.Clickables.tutorialScriptsButton.style.display = "none"; - Engine.Clickables.tutorialNetscriptButton.style.display = "none"; - Engine.Clickables.tutorialTravelingButton.style.display = "none"; - Engine.Clickables.tutorialCompaniesButton.style.display = "none"; - Engine.Clickables.tutorialFactionsButton.style.display = "none"; - Engine.Clickables.tutorialAugmentationsButton.style.display = "none"; - document.getElementById("tutorial-shortcuts-link").style.display = "none"; - - Engine.Clickables.tutorialBackButton.style.display = "inline-block"; - document.getElementById("tutorial-text").style.display = "block"; - document.getElementById("tutorial-text").innerHTML = text; - }, - /* Main Event Loop */ idleTimer: function() { //Get time difference @@ -1446,52 +1400,6 @@ const Engine = { //Init Location buttons initLocationButtons(); - //Tutorial buttons - Engine.Clickables.tutorialNetworkingButton = document.getElementById("tutorial-networking-link"); - Engine.Clickables.tutorialNetworkingButton.addEventListener("click", function() { - Engine.displayTutorialPage(CONSTANTS.TutorialNetworkingText); - }); - - Engine.Clickables.tutorialHackingButton = document.getElementById("tutorial-hacking-link"); - Engine.Clickables.tutorialHackingButton.addEventListener("click", function() { - Engine.displayTutorialPage(CONSTANTS.TutorialHackingText); - }); - - Engine.Clickables.tutorialScriptsButton = document.getElementById("tutorial-scripts-link"); - Engine.Clickables.tutorialScriptsButton.addEventListener("click", function() { - Engine.displayTutorialPage(CONSTANTS.TutorialScriptsText); - }); - - Engine.Clickables.tutorialNetscriptButton = document.getElementById("tutorial-netscript-link"); - Engine.Clickables.tutorialNetscriptButton.addEventListener("click", function() { - Engine.displayTutorialPage(CONSTANTS.TutorialNetscriptText); - }); - - Engine.Clickables.tutorialTravelingButton = document.getElementById("tutorial-traveling-link"); - Engine.Clickables.tutorialTravelingButton.addEventListener("click", function() { - Engine.displayTutorialPage(CONSTANTS.TutorialTravelingText); - }); - - Engine.Clickables.tutorialCompaniesButton = document.getElementById("tutorial-jobs-link"); - Engine.Clickables.tutorialCompaniesButton.addEventListener("click", function() { - Engine.displayTutorialPage(CONSTANTS.TutorialCompaniesText); - }); - - Engine.Clickables.tutorialFactionsButton = document.getElementById("tutorial-factions-link"); - Engine.Clickables.tutorialFactionsButton.addEventListener("click", function() { - Engine.displayTutorialPage(CONSTANTS.TutorialFactionsText); - }); - - Engine.Clickables.tutorialAugmentationsButton = document.getElementById("tutorial-augmentations-link"); - Engine.Clickables.tutorialAugmentationsButton.addEventListener("click", function() { - Engine.displayTutorialPage(CONSTANTS.TutorialAugmentationsText); - }); - - Engine.Clickables.tutorialBackButton = document.getElementById("tutorial-back-button"); - Engine.Clickables.tutorialBackButton.addEventListener("click", function() { - Engine.displayTutorialContent(); - }); - // Initialize references to main menu links if (!initializeMainMenuLinks()) { const errorMsg = "Failed to initialize Main Menu Links. Please try refreshing the page. " + diff --git a/src/index.html b/src/index.html index 6745ad9a8..b88c3e468 100644 --- a/src/index.html +++ b/src/index.html @@ -478,21 +478,51 @@ if (htmlWebpackPlugin.options.googleAnalytics.trackingId) { %> @@ -972,7 +1002,7 @@ if (htmlWebpackPlugin.options.googleAnalytics.trackingId) { %>
Changelog - Wiki + Documentation Subreddit From a22bfb50156480ebb303d1dc5d8eb3536680afe2 Mon Sep 17 00:00:00 2001 From: danielyxie Date: Mon, 4 Mar 2019 22:29:27 -0800 Subject: [PATCH 08/10] Updating version to v0.44.1 and Updating docuemtnation changelog --- doc/source/changelog.rst | 13 +++++++++++++ src/Constants.ts | 2 +- 2 files changed, 14 insertions(+), 1 deletion(-) diff --git a/doc/source/changelog.rst b/doc/source/changelog.rst index 47ee7059c..e6c945faf 100644 --- a/doc/source/changelog.rst +++ b/doc/source/changelog.rst @@ -3,6 +3,19 @@ Changelog ========= +v0.44.1 - 3/4/2019 +------------------ +* Duplicate Sleeve changes: +** You can now purchase Augmentations for your Duplicate Sleeves +** Sleeves are now assigned to Shock Recovery task by default +** Shock Recovery and Synchronize tasks are now twice as effective + +* Changed documentation so that Netscript functions are own their own pages. Sorry if this is annoying, it was necessary for properly cross-referencing +* Officially deprecated the Wiki (the fandom site). Use the 'readthedocs' Documentation instead +* Bug Fix: 'rm' Terminal and Netscript commands now work on non-program files that have '.exe' in the name (by Github user MasonD) +* Bug Fix: The 'Find All Valid Math Expressions' Coding Contract should now properly ignore whitespace in answers +* Bug Fix: The 'Merge Overlapping Intervals' Coding Contract should now properly accept 2D arrays when being attempted through Netscript + v0.44.0 - 2/26/2019 ------------------- * Bladeburner Changes: diff --git a/src/Constants.ts b/src/Constants.ts index ee2dca54f..ed1927d30 100644 --- a/src/Constants.ts +++ b/src/Constants.ts @@ -1,7 +1,7 @@ import {IMap} from "./types"; export let CONSTANTS: IMap = { - Version: "0.44.0", + Version: "0.44.1", //Max level for any skill, assuming no multipliers. Determined by max numerical value in javascript for experience //and the skill level formula in Player.js. Note that all this means it that when experience hits MAX_INT, then From bd13234f069f1cd1b52129b39d1a54d92ae25d87 Mon Sep 17 00:00:00 2001 From: danielyxie Date: Mon, 4 Mar 2019 23:55:54 -0800 Subject: [PATCH 09/10] Corrected some incorrect imports --- src/CodingContractGenerator.js | 2 +- src/Player.js | 4 ++-- src/Prestige.js | 12 ++++++------ src/SaveObject.js | 4 ++-- src/Server/ServerPurchases.js | 4 ++-- 5 files changed, 13 insertions(+), 13 deletions(-) diff --git a/src/CodingContractGenerator.js b/src/CodingContractGenerator.js index 1538703bd..135d47863 100644 --- a/src/CodingContractGenerator.js +++ b/src/CodingContractGenerator.js @@ -3,7 +3,7 @@ import { CodingContract, CodingContractTypes } from "./CodingContracts"; import { Factions } from "./Faction/Factions"; import { Player } from "./Player"; -import { AllServers } from "./Server/Server"; +import { AllServers } from "./Server/AllServers"; import { GetServerByHostname } from "./Server/ServerHelpers"; import { getRandomInt } from "../utils/helpers/getRandomInt"; diff --git a/src/Player.js b/src/Player.js index a95234937..ca0c5c0ec 100644 --- a/src/Player.js +++ b/src/Player.js @@ -24,9 +24,9 @@ import {Gang, resetGangs} from "./Gang"; import {Locations} from "./Locations"; import {hasBn11SF, hasWallStreetSF,hasAISF} from "./NetscriptFunctions"; import { Sleeve } from "./PersonObjects/Sleeve/Sleeve"; -import { AllServers } from "./Server/AllServers"; +import { AllServers, + AddToAllServers } from "./Server/AllServers"; import { Server } from "./Server/Server"; -import { AddToAllServers } from "./Server/ServerHelpers"; import {Settings} from "./Settings/Settings"; import {SpecialServerIps, SpecialServerNames} from "./Server/SpecialServerIps"; import {SourceFiles, applySourceFile} from "./SourceFile"; diff --git a/src/Prestige.js b/src/Prestige.js index 1b3ba168d..0f12c1e97 100755 --- a/src/Prestige.js +++ b/src/Prestige.js @@ -16,19 +16,19 @@ import { Factions, import { joinFaction } from "./Faction/FactionHelpers"; import {deleteGangDisplayContent} from "./Gang"; import {Locations} from "./Location"; +import { Message } from "./Message/Message"; import { initMessages, - Messages, - Message } from "./Message/MessageHelpers"; + Messages } from "./Message/MessageHelpers"; import {initSingularitySFFlags, hasWallStreetSF}from "./NetscriptFunctions"; import {WorkerScript, workerScripts, prestigeWorkerScripts} from "./NetscriptWorker"; import {Player} from "./Player"; -import { AllServers } from "./Server/AllServers"; +import { AllServers, + AddToAllServers, + prestigeAllServers } from "./Server/AllServers"; import { Server } from "./Server/Server" -import { AddToAllServers, - initForeignServers, - prestigeAllServers, +import { initForeignServers, prestigeHomeComputer } from "./Server/ServerHelpers"; import { updateSourceFileFlags } from "./SourceFile/SourceFileFlags"; import { SpecialServerIps, diff --git a/src/SaveObject.js b/src/SaveObject.js index e59c0f777..27af3062a 100755 --- a/src/SaveObject.js +++ b/src/SaveObject.js @@ -14,8 +14,8 @@ import {processAllHacknetNodeEarnings} from "./HacknetNode"; import { loadMessages, initMessages, Messages } from "./Message/MessageHelpers"; import {Player, loadPlayer} from "./Player"; import { loadAllRunningScripts } from "./Script/ScriptHelpers"; -import { AllServers } from "./Server/AllServers"; -import { loadAllServers } from "./Server/ServerHelpers"; +import { AllServers, + loadAllServers } from "./Server/AllServers"; import { Settings } from "./Settings/Settings"; import { loadSpecialServerIps, SpecialServerIps } from "./Server/SpecialServerIps"; diff --git a/src/Server/ServerPurchases.js b/src/Server/ServerPurchases.js index 163a725d1..7805206f3 100644 --- a/src/Server/ServerPurchases.js +++ b/src/Server/ServerPurchases.js @@ -5,9 +5,9 @@ import { BitNodeMultipliers } from "../BitNode/BitNodeMultipliers"; import { CONSTANTS } from "../Constants"; import { Player } from "../Player"; -import { AllServers } from "../Server/AllServers"; +import { AllServers, + AddToAllServers } from "../Server/AllServers"; import { Server } from "../Server/Server"; -import { AddToAllServers } from "../Server/ServerHelpers"; import { dialogBoxCreate } from "../../utils/DialogBox"; import { createRandomIp } from "../../utils/IPAddress"; import { yesNoTxtInpBoxGetInput } from "../../utils/YesNoBox"; From d9ee715bc75e46b410f87fd8a62103be20f44b74 Mon Sep 17 00:00:00 2001 From: danielyxie Date: Tue, 5 Mar 2019 00:01:34 -0800 Subject: [PATCH 10/10] Fixed some more incorrect imports --- src/Location.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/Location.js b/src/Location.js index 42caaa0ba..33281738f 100644 --- a/src/Location.js +++ b/src/Location.js @@ -11,9 +11,9 @@ import {beginInfiltration} from "./Infiltration"; import {hasBladeburnerSF} from "./NetscriptFunctions"; import {Locations} from "./Locations"; import {Player} from "./Player"; -import { AllServers } from "./Server/AllServers"; +import { AllServers, + AddToAllServers } from "./Server/AllServers"; import { Server } from "./Server/Server"; -import { AddToAllServers } from "./Server/ServerHelpers"; import { getPurchaseServerCost, purchaseServer, purchaseRamForHomeComputer } from "./Server/ServerPurchases";