import { Settings } from "./Settings/Settings"; import { NetscriptPort } from "@nsdefs"; import { NetscriptPorts } from "./NetscriptWorker"; import { PositiveInteger } from "./types"; type Resolver = () => void; const emptyPortData = "NULL PORT DATA"; /** The object property is for typechecking and is not present at runtime */ export type PortNumber = PositiveInteger & { __PortNumber: true }; function isObjectLike(value: unknown): value is object { return (typeof value === "object" && value !== null) || typeof value === "function"; } /** Gets the numbered port, initializing it if it doesn't already exist. * Only using for functions that write data/resolvers. Use NetscriptPorts.get(n) for */ export function getPort(n: PortNumber) { let port = NetscriptPorts.get(n); if (port) return port; port = new Port(); NetscriptPorts.set(n, port); return port; } export class Port { data: any[] = []; resolver: Resolver | null = null; promise: Promise | null = null; add(data: any) { this.data.push(data); if (!this.resolver) return; this.resolver(); this.resolver = null; this.promise = null; } } export function portHandle(n: PortNumber): NetscriptPort { return { write: (value: unknown) => writePort(n, value), tryWrite: (value: unknown) => tryWritePort(n, value), read: () => readPort(n), peek: () => peekPort(n), nextWrite: () => nextPortWrite(n), full: () => isFullPort(n), empty: () => isEmptyPort(n), clear: () => clearPort(n), }; } export function writePort(n: PortNumber, value: unknown): unknown { const port = getPort(n); // Primitives don't need to be cloned. port.add(isObjectLike(value) ? structuredClone(value) : value); if (port.data.length > Settings.MaxPortCapacity) return port.data.shift(); return null; } export function tryWritePort(n: PortNumber, value: unknown): boolean { const port = getPort(n); if (port.data.length >= Settings.MaxPortCapacity) return false; // Primitives don't need to be cloned. port.add(isObjectLike(value) ? structuredClone(value) : value); return true; } export function readPort(n: PortNumber): unknown { const port = NetscriptPorts.get(n); if (!port || !port.data.length) return emptyPortData; const returnVal: unknown = port.data.shift(); if (!port.data.length && !port.resolver) NetscriptPorts.delete(n); return returnVal; } export function peekPort(n: PortNumber): unknown { const port = NetscriptPorts.get(n); if (!port || !port.data.length) return emptyPortData; // Needed to avoid exposing internal objects. return isObjectLike(port.data[0]) ? structuredClone(port.data[0]) : port.data[0]; } export function nextPortWrite(n: PortNumber) { const port = getPort(n); if (!port.promise) port.promise = new Promise((res) => (port.resolver = res)); return port.promise; } function isFullPort(n: PortNumber) { const port = NetscriptPorts.get(n); if (!port) return false; return port.data.length >= Settings.MaxPortCapacity; } function isEmptyPort(n: PortNumber) { const port = NetscriptPorts.get(n); if (!port) return true; return port.data.length === 0; } export function clearPort(n: PortNumber) { const port = NetscriptPorts.get(n); if (!port) return; if (!port.resolver) NetscriptPorts.delete(n); port.data.length = 0; }