@honeybbq/teamspeak-client 0.2.1 → 0.2.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/client.d.ts.map +1 -1
- package/dist/crypto/identity.d.ts.map +1 -1
- package/dist/discovery/index.cjs +1 -1
- package/dist/discovery/index.mjs +1 -1
- package/dist/discovery/resolver.d.ts +1 -1
- package/dist/discovery/resolver.d.ts.map +1 -1
- package/dist/handler-C_JhqGTd.js.map +1 -1
- package/dist/handler-CqCD93f0.cjs.map +1 -1
- package/dist/index.cjs +3 -1
- package/dist/index.cjs.map +1 -1
- package/dist/index.mjs +2 -2
- package/dist/index.mjs.map +1 -1
- package/dist/primitives-BIAYfJ2Y.js.map +1 -1
- package/dist/primitives-bj-ml31v.cjs.map +1 -1
- package/dist/{resolver-DDZWomrF.js → resolver-Br0QefWx.js} +18 -21
- package/dist/resolver-Br0QefWx.js.map +1 -0
- package/dist/resolver-DDD2FFhD.cjs +4 -0
- package/dist/resolver-DDD2FFhD.cjs.map +1 -0
- package/dist/transfer.d.ts +5 -9
- package/dist/transfer.d.ts.map +1 -1
- package/dist/transport/handler.d.ts.map +1 -1
- package/package.json +5 -5
- package/dist/resolver-DDZWomrF.js.map +0 -1
- package/dist/resolver-Dey6omBe.cjs +0 -4
- package/dist/resolver-Dey6omBe.cjs.map +0 -1
package/dist/client.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"client.d.ts","sourceRoot":"","sources":["../src/client.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,KAAK,QAAQ,EACb,KAAK,aAAa,EAClB,KAAK,iBAAiB,EACtB,KAAK,eAAe,EACpB,KAAK,MAAM,EAGX,YAAY,EAEb,MAAM,YAAY,CAAC;AAEpB,OAAO,EAAE,QAAQ,EAAE,KAAK,EAAE,MAAM,mBAAmB,CAAC;AAEpD,OAAO,EAAE,aAAa,EAAE,MAAM,wBAAwB,CAAC;
|
|
1
|
+
{"version":3,"file":"client.d.ts","sourceRoot":"","sources":["../src/client.ts"],"names":[],"mappings":"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"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"identity.d.ts","sourceRoot":"","sources":["../../src/crypto/identity.ts"],"names":[],"mappings":"AAAA,OAAO,
|
|
1
|
+
{"version":3,"file":"identity.d.ts","sourceRoot":"","sources":["../../src/crypto/identity.ts"],"names":[],"mappings":"AAAA,OAAO,EAML,KAAK,SAAS,EACf,MAAM,aAAa,CAAC;AAOrB,qBAAa,QAAQ;IACnB,QAAQ,CAAC,UAAU,EAAE,SAAS,CAAC;IAC/B,MAAM,EAAE,MAAM,CAAC;gBAEH,UAAU,EAAE,SAAS,EAAE,MAAM,EAAE,MAAM;IAKjD,eAAe,IAAI,MAAM;IAoBzB,QAAQ,IAAI,MAAM;IAUlB,aAAa,IAAI,MAAM;IAOjB,cAAc,CAAC,WAAW,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,WAAW,GAAG,OAAO,CAAC,IAAI,CAAC;CAe/E;AAED,wBAAgB,kBAAkB,CAAC,CAAC,EAAE,MAAM,GAAG,QAAQ,CA+BtD;AAED,wBAAgB,gBAAgB,CAAC,WAAW,EAAE,MAAM,GAAG,QAAQ,CAc9D;AAED,wBAAgB,mBAAmB,CAAC,SAAS,EAAE,MAAM,GAAG,MAAM,CAG7D;AAED,wBAAgB,OAAO,CAAC,IAAI,EAAE,UAAU,GAAG,UAAU,CAEpD;AA+BD;mDACmD;AACnD,wBAAgB,eAAe,CAAC,IAAI,EAAE,UAAU,GAAG,SAAS,CAS3D"}
|
package/dist/discovery/index.cjs
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
Object.defineProperty(exports,Symbol.toStringTag,{value:`Module`});const e=require(`../resolver-
|
|
1
|
+
Object.defineProperty(exports,Symbol.toStringTag,{value:`Module`});const e=require(`../resolver-DDD2FFhD.cjs`);exports.Resolver=e.t;
|
package/dist/discovery/index.mjs
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import { t as e } from "../resolver-
|
|
1
|
+
import { t as e } from "../resolver-Br0QefWx.js";
|
|
2
2
|
export { e as Resolver };
|
|
@@ -2,7 +2,7 @@ import type { Logger } from "../types.js";
|
|
|
2
2
|
import type { ResolvedAddr } from "../types.js";
|
|
3
3
|
export declare class Resolver {
|
|
4
4
|
#private;
|
|
5
|
-
constructor(
|
|
5
|
+
constructor(_log?: Logger);
|
|
6
6
|
resolve(inputAddr: string, signal?: AbortSignal): Promise<ResolvedAddr[]>;
|
|
7
7
|
}
|
|
8
8
|
//# sourceMappingURL=resolver.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"resolver.d.ts","sourceRoot":"","sources":["../../src/discovery/resolver.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AAC1C,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AAOhD,qBAAa,QAAQ;;
|
|
1
|
+
{"version":3,"file":"resolver.d.ts","sourceRoot":"","sources":["../../src/discovery/resolver.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AAC1C,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AAOhD,qBAAa,QAAQ;;gBAGP,IAAI,GAAE,MAAmB;IAE/B,OAAO,CAAC,SAAS,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,WAAW,GAAG,OAAO,CAAC,YAAY,EAAE,CAAC;CA+GhF"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"handler-C_JhqGTd.js","names":["#mod","#receiveWindow","#generation","#mappedBaseOffset","#isNextGen","#hashtable","#ensureControl","#processReference","#processLiteral","#updateHashtable","#crypt","#logger","#clientID","#conn","#closed","#lastMessageReceived","#handleRawPacket","#triggerClose","#packetCounter","#sendPacketRaw","#stopPing","#checkResends","#stopResend","#initPacketCheck","#sendSplitPacket","#generationCounter","#write","#nextPacketIdentity","#trackResend","#resolveGeneration","#decryptPacketData","#sendPong","#handleCommandWindowAndAck","#handlePacketQueue","#updatePostReceiveState","#recvWindowCommand","#recvWindowCommandLow","#sendWindowCommand","#sendWindowCommandLow","#sendAck","#commandQueue","#commandLowQueue","#fastForwardMissingPackets","#_nextCommandID","#_nextCommandLowID","#tryReassemble","#tryDecompress","#ackManager","#doResend"],"sources":["../src/transport/packet.ts","../src/transport/generation-window.ts","../src/transport/quicklz.ts","../src/transport/handler.ts"],"sourcesContent":["export const enum PacketType {\n Voice = 0,\n VoiceWhisper = 1,\n Command = 2,\n CommandLow = 3,\n Ping = 4,\n Pong = 5,\n Ack = 6,\n AckLow = 7,\n Init1 = 8,\n}\n\nexport const enum PacketFlags {\n Fragmented = 0x10,\n NewProtocol = 0x20,\n Compressed = 0x40,\n Unencrypted = 0x80,\n}\n\nexport interface Packet {\n /** Type byte combined with flags (low nibble = type, high nibble = flags). */\n typeFlagged: number;\n id: number;\n clientID: number;\n generationID: number;\n data: Uint8Array;\n receivedAt: number; // Date.now()\n}\n\nexport function packetType(p: Packet): PacketType {\n return (p.typeFlagged & 0x0f) as PacketType;\n}\n\nexport function packetFlags(p: Packet): number {\n return p.typeFlagged & 0xf0;\n}\n\nexport function isUnencrypted(p: Packet): boolean {\n return (packetFlags(p) & PacketFlags.Unencrypted) !== 0;\n}\n\n/** Build the 5-byte client-to-server header: [packetID(2), clientID(2), typeFlagged(1)]. */\nexport function buildC2SHeader(p: Packet): Uint8Array {\n const header = new Uint8Array(5);\n const view = new DataView(header.buffer);\n view.setUint16(0, p.id, false);\n view.setUint16(2, p.clientID, false);\n header[4] = p.typeFlagged;\n return header;\n}\n\n/** Parse a 3-byte server-to-client header. */\nexport function parseS2CHeader(raw: Uint8Array): Pick<Packet, \"id\" | \"typeFlagged\"> {\n const view = new DataView(raw.buffer, raw.byteOffset, raw.byteLength);\n return {\n id: view.getUint16(0, false),\n typeFlagged: raw[2]!,\n };\n}\n\n/** Parse a 5-byte client-to-server header. */\nexport function parseC2SHeader(raw: Uint8Array): Pick<Packet, \"id\" | \"clientID\" | \"typeFlagged\"> {\n const view = new DataView(raw.buffer, raw.byteOffset, raw.byteLength);\n return {\n id: view.getUint16(0, false),\n clientID: view.getUint16(2, false),\n typeFlagged: raw[4]!,\n };\n}\n","export class GenerationWindow {\n #mappedBaseOffset = 0;\n #generation = 0;\n readonly #mod: number;\n readonly #receiveWindow: number;\n\n constructor(mod: number, windowSize: number) {\n this.#mod = mod;\n this.#receiveWindow = windowSize;\n }\n\n get generation(): number {\n return this.#generation;\n }\n\n advance(amount: number): void {\n if (amount <= 0) return;\n const newBaseOffset = this.#mappedBaseOffset + amount;\n const genStep = Math.floor(newBaseOffset / this.#mod);\n if (genStep > 0) {\n this.#generation = Math.min(this.#generation + genStep, 0xffff_ffff);\n }\n this.#mappedBaseOffset = newBaseOffset % this.#mod;\n }\n\n advanceToExcluded(mappedValue: number): void {\n let moveDist = mappedValue - this.#mappedBaseOffset;\n if (moveDist < 0) moveDist += this.#mod;\n this.advance(moveDist + 1);\n }\n\n syncTo(mappedValue: number): void {\n let moveDist = mappedValue - this.#mappedBaseOffset;\n if (moveDist < 0) moveDist += this.#mod;\n this.advance(moveDist);\n }\n\n isInWindow(mappedValue: number): boolean {\n const maxOffset = this.#mappedBaseOffset + this.#receiveWindow;\n if (maxOffset < this.#mod) {\n return mappedValue >= this.#mappedBaseOffset && mappedValue < maxOffset;\n }\n return mappedValue >= this.#mappedBaseOffset || mappedValue < maxOffset - this.#mod;\n }\n\n mappedToIndex(mappedValue: number): number {\n if (this.#isNextGen(mappedValue)) {\n return mappedValue + this.#mod - this.#mappedBaseOffset;\n }\n return mappedValue - this.#mappedBaseOffset;\n }\n\n isOldPacket(mappedValue: number): boolean {\n return this.mappedToIndex(mappedValue) < 0;\n }\n\n isFuturePacket(mappedValue: number): boolean {\n return this.mappedToIndex(mappedValue) >= this.#receiveWindow;\n }\n\n #isNextGen(mappedValue: number): boolean {\n return (\n this.#mappedBaseOffset > this.#mod - this.#receiveWindow &&\n mappedValue < this.#mappedBaseOffset + this.#receiveWindow - this.#mod\n );\n }\n\n getGeneration(mappedValue: number): number {\n if (this.#isNextGen(mappedValue)) return this.#generation + 1;\n return this.#generation;\n }\n\n reset(): void {\n this.#mappedBaseOffset = 0;\n this.#generation = 0;\n }\n}\n","const TABLE_SIZE = 4096;\n\ninterface QlzState {\n control: number;\n sourcePos: number;\n destPos: number;\n nextHashed: number;\n}\n\nexport class Qlz {\n #hashtable = new Int32Array(TABLE_SIZE);\n\n decompress(data: Uint8Array): Uint8Array {\n const { headerLen, decompressedSize, flags } = parseQlzHeader(data);\n\n const dest = new Uint8Array(decompressedSize);\n\n if ((flags & 0x01) === 0) {\n dest.set(data.slice(headerLen, headerLen + decompressedSize));\n return dest;\n }\n\n this.#hashtable.fill(0);\n\n const state: QlzState = {\n control: 1,\n sourcePos: headerLen,\n destPos: 0,\n nextHashed: 0,\n };\n\n while (this.#ensureControl(data, state)) {\n if ((state.control & 1) !== 0) {\n if (!this.#processReference(data, dest, state)) break;\n } else {\n if (this.#processLiteral(data, dest, decompressedSize, state)) break;\n }\n }\n\n return dest;\n }\n\n #ensureControl(data: Uint8Array, st: QlzState): boolean {\n if (st.control !== 1) return true;\n if (st.sourcePos + 4 > data.length) return false;\n st.control =\n (data[st.sourcePos]! |\n (data[st.sourcePos + 1]! << 8) |\n (data[st.sourcePos + 2]! << 16) |\n (data[st.sourcePos + 3]! << 24)) >>>\n 0;\n st.sourcePos += 4;\n return true;\n }\n\n #processReference(data: Uint8Array, dest: Uint8Array, st: QlzState): boolean {\n st.control = (st.control >>> 1) >>> 0;\n if (st.sourcePos + 2 > data.length) return false;\n\n const b1 = data[st.sourcePos]!;\n const b2 = data[st.sourcePos + 1]!;\n st.sourcePos += 2;\n\n const hash = (b1 >> 4) | (b2 << 4);\n let matchlen = b1 & 0x0f;\n if (matchlen !== 0) {\n matchlen += 2;\n } else {\n if (st.sourcePos >= data.length) return false;\n matchlen = data[st.sourcePos]!;\n st.sourcePos++;\n }\n\n const offset = this.#hashtable[hash]!;\n for (let i = 0; i < matchlen; i++) {\n if (st.destPos < dest.length && offset + i < st.destPos) {\n dest[st.destPos] = dest[offset + i]!;\n st.destPos++;\n }\n }\n\n const end = st.destPos + 1 - matchlen;\n this.#updateHashtable(dest, st, end);\n st.nextHashed = st.destPos;\n\n return true;\n }\n\n #processLiteral(\n data: Uint8Array,\n dest: Uint8Array,\n decompressedSize: number,\n st: QlzState,\n ): boolean {\n const threshold = Math.max(decompressedSize, 10) - 10;\n if (st.destPos >= threshold) {\n while (st.destPos < decompressedSize) {\n if (st.control === 1) {\n st.sourcePos += 4;\n if (st.sourcePos > data.length) break;\n st.control =\n (data[st.sourcePos - 4]! |\n (data[st.sourcePos - 3]! << 8) |\n (data[st.sourcePos - 2]! << 16) |\n (data[st.sourcePos - 1]! << 24)) >>>\n 0;\n }\n if (st.sourcePos >= data.length) break;\n dest[st.destPos++] = data[st.sourcePos++]!;\n st.control = (st.control >>> 1) >>> 0;\n }\n return true;\n }\n\n if (st.sourcePos >= data.length || st.destPos >= dest.length) return true;\n\n dest[st.destPos++] = data[st.sourcePos++]!;\n st.control = (st.control >>> 1) >>> 0;\n\n const end = Math.max(st.destPos - 2, 0);\n this.#updateHashtable(dest, st, end);\n if (st.nextHashed < end) st.nextHashed = end;\n\n return false;\n }\n\n #updateHashtable(dest: Uint8Array, st: QlzState, end: number): void {\n while (st.nextHashed < end) {\n if (st.nextHashed + 3 > dest.length) break;\n const v =\n (dest[st.nextHashed]! |\n (dest[st.nextHashed + 1]! << 8) |\n (dest[st.nextHashed + 2]! << 16)) >>>\n 0;\n const hash = ((v >> 12) ^ v) & 0xfff;\n this.#hashtable[hash] = st.nextHashed;\n st.nextHashed++;\n }\n }\n}\n\nfunction parseQlzHeader(data: Uint8Array): {\n headerLen: number;\n decompressedSize: number;\n flags: number;\n} {\n if (data.length < 3) throw new Error(\"QuickLZ: data too short\");\n\n const flags = data[0]!;\n const level = (flags >> 2) & 0x03;\n if (level !== 1) throw new Error(\"QuickLZ: only level 1 is supported\");\n\n const headerLen = (flags & 0x02) !== 0 ? 9 : 3;\n if (data.length < headerLen) throw new Error(\"QuickLZ: data too short for header\");\n\n let decompressedSize: number;\n if ((flags & 0x02) !== 0) {\n decompressedSize = (data[5]! | (data[6]! << 8) | (data[7]! << 16) | (data[8]! << 24)) >>> 0;\n } else {\n decompressedSize = data[2]!;\n }\n\n return { headerLen, decompressedSize, flags };\n}\n","import { createSocket, type Socket as UdpSocket } from \"node:dgram\";\nimport type { Crypt } from \"../crypto/crypt.js\";\nimport type { Logger } from \"../types.js\";\nimport { noopLogger } from \"../types.js\";\nimport {\n type Packet,\n PacketType,\n PacketFlags,\n packetType,\n packetFlags,\n buildC2SHeader,\n parseS2CHeader,\n} from \"./packet.js\";\nimport { GenerationWindow } from \"./generation-window.js\";\nimport { Qlz } from \"./quicklz.js\";\nimport { processInit1 } from \"../handshake/crypt-handshake.js\";\n\nconst MAX_OUT_PACKET_SIZE = 500;\nconst RECEIVE_PACKET_WINDOW_SIZE = 1024;\nconst PING_INTERVAL_MS = 5_000;\nconst PACKET_TIMEOUT_MS = 60_000;\nconst MAX_RETRY_INTERVAL_MS = 1_000;\nconst UDP_READ_BUFFER_SIZE = 4096;\nconst HEADER_SIZE = 5;\nconst TAG_SIZE = 8;\nconst VOICE_HEADER_SIZE = 3;\nconst RESEND_BASE_INTERVAL_MS = 500;\nconst RESEND_LOOP_INTERVAL_MS = 100;\n\ninterface ResendPacket {\n packet: Packet;\n firstSend: number;\n lastSend: number;\n retryCount: number;\n nextInterval: number;\n}\n\nexport class PacketHandler {\n onPacket: ((p: Packet) => void) | null = null;\n onClosed: ((err: Error | null) => void) | null = null;\n\n readonly #crypt: Crypt;\n readonly #logger: Logger;\n\n #conn: UdpSocket | null = null;\n #clientID = 0;\n #closed = false;\n #stopPing: (() => void) | null = null;\n #stopResend: (() => void) | null = null;\n #lastMessageReceived = Date.now();\n\n #packetCounter = new Uint16Array(9);\n #generationCounter = new Uint32Array(9);\n\n #recvWindowCommand = new GenerationWindow(1 << 16, RECEIVE_PACKET_WINDOW_SIZE);\n #recvWindowCommandLow = new GenerationWindow(1 << 16, RECEIVE_PACKET_WINDOW_SIZE);\n #sendWindowCommand = new GenerationWindow(1 << 16, RECEIVE_PACKET_WINDOW_SIZE);\n #sendWindowCommandLow = new GenerationWindow(1 << 16, RECEIVE_PACKET_WINDOW_SIZE);\n\n #commandQueue = new Map<number, Packet>();\n #commandLowQueue = new Map<number, Packet>();\n #ackManager = new Map<number, ResendPacket>();\n #initPacketCheck: ResendPacket | null = null;\n\n constructor(crypt: Crypt, logger: Logger = noopLogger) {\n this.#crypt = crypt;\n this.#logger = logger;\n }\n\n setClientID(id: number): void {\n this.#clientID = id;\n }\n\n connect(addr: string): Promise<void> {\n return new Promise((resolve, reject) => {\n const [host, portStr] =\n addr.lastIndexOf(\":\") > 0\n ? [addr.slice(0, addr.lastIndexOf(\":\")), addr.slice(addr.lastIndexOf(\":\") + 1)]\n : [addr, \"9987\"];\n const port = parseInt(portStr, 10);\n\n const socket = createSocket(\"udp4\");\n socket.once(\"error\", reject);\n socket.connect(port, host, () => {\n socket.off(\"error\", reject);\n this.start(socket);\n resolve();\n });\n });\n }\n\n start(conn: UdpSocket): void {\n this.#conn = conn;\n this.#closed = false;\n this.#lastMessageReceived = Date.now();\n\n conn.on(\"message\", (msg) => {\n this.#lastMessageReceived = Date.now();\n this.#handleRawPacket(new Uint8Array(msg.buffer, msg.byteOffset, msg.byteLength));\n });\n conn.on(\"error\", (err) => {\n this.#logger.error(\"udp error\", err);\n this.#triggerClose(err);\n });\n conn.on(\"close\", () => this.#triggerClose(null));\n\n this.#packetCounter[PacketType.Command] = 1;\n this.#packetCounter[PacketType.Init1] = 101;\n\n const init1Data = processInit1(this.#crypt, null);\n if (init1Data) this.#sendPacketRaw(PacketType.Init1, init1Data, 0);\n\n // Ping loop\n const pingTimer = setInterval(() => {\n if (this.#crypt.cryptoInitComplete) {\n this.sendPacket(PacketType.Ping, new Uint8Array(0), PacketFlags.Unencrypted);\n }\n }, PING_INTERVAL_MS);\n this.#stopPing = () => clearInterval(pingTimer);\n\n // Resend loop\n const resendTimer = setInterval(() => this.#checkResends(), RESEND_LOOP_INTERVAL_MS);\n this.#stopResend = () => clearInterval(resendTimer);\n }\n\n receivedFinalInitAck(): void {\n this.#initPacketCheck = null;\n }\n\n sendPacket(pType: PacketType, data: Uint8Array, flags: number): void {\n const dummy = !this.#crypt.cryptoInitComplete;\n if (data.length > 487 && pType !== PacketType.Voice && pType !== PacketType.VoiceWhisper) {\n this.#sendSplitPacket(pType, data, flags, dummy);\n return;\n }\n this.#sendPacketRaw(pType, data, flags, dummy);\n }\n\n sendVoicePacket(data: Uint8Array, codec: number): void {\n const pID = this.#packetCounter[PacketType.Voice]!;\n const pGen = this.#generationCounter[PacketType.Voice]!;\n this.#packetCounter[PacketType.Voice] = (pID + 1) & 0xffff;\n if (this.#packetCounter[PacketType.Voice] === 0) {\n this.#generationCounter[PacketType.Voice]!++;\n }\n\n const payloadLen = VOICE_HEADER_SIZE + data.length;\n const voicePayload = new Uint8Array(payloadLen);\n new DataView(voicePayload.buffer).setUint16(0, pID, false);\n voicePayload[2] = codec;\n voicePayload.set(data, VOICE_HEADER_SIZE);\n\n const p: Packet = {\n typeFlagged: PacketType.Voice | PacketFlags.Unencrypted,\n id: pID,\n clientID: this.#clientID,\n generationID: pGen,\n data: voicePayload,\n receivedAt: 0,\n };\n\n const header = buildC2SHeader(p);\n const final = new Uint8Array(TAG_SIZE + HEADER_SIZE + payloadLen);\n final.set(this.#crypt.fakeSignature, 0);\n final.set(header, TAG_SIZE);\n final.set(voicePayload, TAG_SIZE + HEADER_SIZE);\n this.#write(final);\n }\n\n close(): void {\n if (this.#closed) return;\n this.#closed = true;\n this.#stopPing?.();\n this.#stopResend?.();\n this.#conn?.close();\n }\n\n // ---- Private ---------------------------------------------------------------\n\n #sendSplitPacket(pType: PacketType, data: Uint8Array, flags: number, dummy: boolean): void {\n const maxSize = MAX_OUT_PACKET_SIZE - HEADER_SIZE - TAG_SIZE; // 487\n let pos = 0;\n let first = true;\n\n while (pos < data.length) {\n const blockSize = Math.min(data.length - pos, maxSize);\n const last = pos + blockSize === data.length;\n\n let pFlags = flags;\n if (first !== last) pFlags |= PacketFlags.Fragmented;\n\n this.#sendPacketRaw(pType, data.slice(pos, pos + blockSize), pFlags, dummy);\n pos += blockSize;\n first = false;\n }\n }\n\n #sendPacketRaw(\n pType: PacketType,\n data: Uint8Array,\n flags: number,\n dummy = !this.#crypt.cryptoInitComplete,\n ): void {\n flags = applyProtocolFlags(pType, flags);\n const [pID, pGen] = this.#nextPacketIdentity(pType);\n const p: Packet = {\n typeFlagged: pType | flags,\n id: pID,\n clientID: this.#clientID,\n generationID: pGen,\n data,\n receivedAt: 0,\n };\n\n const unencrypted = (flags & PacketFlags.Unencrypted) !== 0;\n const header = buildC2SHeader(p);\n const [ciphertext, tag] = this.#crypt.encrypt(\n pType,\n pID,\n pGen,\n header,\n data,\n dummy,\n unencrypted,\n );\n\n const final = new Uint8Array(TAG_SIZE + HEADER_SIZE + ciphertext.length);\n final.set(tag.slice(0, TAG_SIZE), 0);\n final.set(header, TAG_SIZE);\n final.set(ciphertext, TAG_SIZE + HEADER_SIZE);\n this.#write(final);\n\n const rp: ResendPacket = {\n packet: p,\n firstSend: Date.now(),\n lastSend: Date.now(),\n retryCount: 0,\n nextInterval: RESEND_BASE_INTERVAL_MS,\n };\n this.#trackResend(pType, p, rp);\n }\n\n #write(data: Uint8Array): void {\n this.#conn?.send(Buffer.from(data), (err) => {\n if (err) this.#logger.warn(\"udp send error\", err);\n });\n }\n\n #handleRawPacket(raw: Uint8Array): void {\n if (raw.length < 11) return;\n\n const tag = raw.slice(0, TAG_SIZE);\n const header = raw.slice(TAG_SIZE, TAG_SIZE + 3);\n const ciphertext = raw.slice(TAG_SIZE + 3);\n\n const parsed = parseS2CHeader(header);\n const p: Packet = {\n ...parsed,\n clientID: 0,\n generationID: this.#resolveGeneration(parsed.id, parsed.typeFlagged & 0x0f),\n data: new Uint8Array(0),\n receivedAt: Date.now(),\n };\n\n const decrypted = this.#decryptPacketData(p, header, ciphertext, tag);\n if (decrypted === null) return;\n p.data = decrypted.plaintext;\n\n const pType = packetType(p);\n\n if (pType === PacketType.Ping) {\n this.#sendPong(p.id, decrypted.dummyUsed);\n return;\n }\n\n if (!this.#handleCommandWindowAndAck(p, decrypted.dummyUsed)) return;\n\n this.#handlePacketQueue(p);\n this.#updatePostReceiveState(p);\n }\n\n #resolveGeneration(id: number, pType: number): number {\n switch (pType as PacketType) {\n case PacketType.Command:\n return this.#recvWindowCommand.getGeneration(id);\n case PacketType.CommandLow:\n return this.#recvWindowCommandLow.getGeneration(id);\n case PacketType.Ack:\n return this.#sendWindowCommand.getGeneration(id);\n case PacketType.AckLow:\n return this.#sendWindowCommandLow.getGeneration(id);\n default:\n return 0;\n }\n }\n\n #decryptPacketData(\n p: Packet,\n header: Uint8Array,\n ciphertext: Uint8Array,\n tag: Uint8Array,\n ): { plaintext: Uint8Array; dummyUsed: boolean } | null {\n const unencrypted = (packetFlags(p) & PacketFlags.Unencrypted) !== 0;\n const dummy = !this.#crypt.cryptoInitComplete;\n let dummyUsed = dummy;\n const pType = packetType(p);\n const gen = p.generationID;\n\n try {\n const plaintext = this.#crypt.decrypt(\n pType,\n p.id,\n gen,\n header,\n ciphertext,\n tag,\n dummy,\n unencrypted,\n );\n return { plaintext, dummyUsed };\n } catch {\n // Try adjacent generations\n for (const offset of [-1, 1]) {\n const guessGen = gen + offset;\n if (guessGen < 0) continue;\n try {\n const plaintext = this.#crypt.decrypt(\n pType,\n p.id,\n guessGen,\n header,\n ciphertext,\n tag,\n false,\n false,\n );\n return { plaintext, dummyUsed: false };\n } catch {\n // continue\n }\n }\n\n // Try dummy fallback for command/ack types\n if (\n pType === PacketType.Command ||\n pType === PacketType.CommandLow ||\n pType === PacketType.Ack\n ) {\n try {\n const plaintext = this.#crypt.decrypt(\n pType,\n p.id,\n gen,\n header,\n ciphertext,\n tag,\n true,\n unencrypted,\n );\n return { plaintext, dummyUsed: true };\n } catch {\n // fall through\n }\n }\n\n return null;\n }\n }\n\n #handleCommandWindowAndAck(p: Packet, dummyUsed: boolean): boolean {\n const pType = packetType(p);\n if (pType !== PacketType.Command && pType !== PacketType.CommandLow) return true;\n\n const win = pType === PacketType.Command ? this.#recvWindowCommand : this.#recvWindowCommandLow;\n const ackType = pType === PacketType.Command ? PacketType.Ack : PacketType.AckLow;\n\n if (!win.isInWindow(p.id)) {\n if (win.isOldPacket(p.id)) {\n this.#sendAck(p.id, ackType, dummyUsed);\n }\n return false;\n }\n this.#sendAck(p.id, ackType, dummyUsed);\n return true;\n }\n\n #sendAck(packetID: number, ackType: PacketType, dummyUsed: boolean): void {\n const ackData = new Uint8Array(2);\n new DataView(ackData.buffer).setUint16(0, packetID, false);\n this.#sendPacketRaw(ackType, ackData, 0, dummyUsed);\n }\n\n #sendPong(pID: number, dummyUsed: boolean): void {\n const pongData = new Uint8Array(2);\n new DataView(pongData.buffer).setUint16(0, pID, false);\n this.#sendPacketRaw(PacketType.Pong, pongData, PacketFlags.Unencrypted, dummyUsed);\n }\n\n #handlePacketQueue(p: Packet): void {\n const pType = packetType(p);\n if (pType !== PacketType.Command && pType !== PacketType.CommandLow) {\n this.onPacket?.(p);\n return;\n }\n\n const isCommand = pType === PacketType.Command;\n const queue = isCommand ? this.#commandQueue : this.#commandLowQueue;\n const win = isCommand ? this.#recvWindowCommand : this.#recvWindowCommandLow;\n\n queue.set(p.id, p);\n\n if (isCommand) {\n this.#fastForwardMissingPackets(queue, win, true);\n } else {\n this.#fastForwardMissingPackets(queue, win, false);\n }\n\n while (true) {\n const current = isCommand ? this.#_nextCommandID : this.#_nextCommandLowID;\n const packet = queue.get(current);\n if (packet === undefined) break;\n\n const result = this.#tryReassemble(packet, queue, current, win);\n if (!result) break;\n\n const [reassembled, newNext] = result;\n if (isCommand) this.#_nextCommandID = newNext;\n else this.#_nextCommandLowID = newNext;\n\n this.#tryDecompress(reassembled);\n this.onPacket?.(reassembled);\n }\n }\n\n #_nextCommandID = 0;\n #_nextCommandLowID = 0;\n\n #fastForwardMissingPackets(\n queue: Map<number, Packet>,\n win: GenerationWindow,\n isCommand: boolean,\n ): void {\n let nextID = isCommand ? this.#_nextCommandID : this.#_nextCommandLowID;\n while (!queue.has(nextID) && hasOldNewerPacket(queue, nextID)) {\n nextID = (nextID + 1) & 0xffff;\n win.advance(1);\n }\n if (isCommand) this.#_nextCommandID = nextID;\n else this.#_nextCommandLowID = nextID;\n }\n\n #tryReassemble(\n startPacket: Packet,\n queue: Map<number, Packet>,\n nextID: number,\n win: GenerationWindow,\n ): [reassembled: Packet, newNextID: number] | null {\n if ((packetFlags(startPacket) & PacketFlags.Fragmented) === 0) {\n queue.delete(nextID);\n win.advance(1);\n return [startPacket, (nextID + 1) & 0xffff];\n }\n\n const fragments: Packet[] = [];\n let totalSize = 0;\n let currID = nextID;\n let startSeen = false;\n let complete = false;\n\n while (true) {\n const frag = queue.get(currID);\n if (!frag) return null;\n fragments.push(frag);\n totalSize += frag.data.length;\n\n const fragmented = (packetFlags(frag) & PacketFlags.Fragmented) !== 0;\n if (!startSeen) {\n startSeen = true;\n if (!fragmented) {\n complete = true;\n break;\n }\n } else if (fragmented) {\n complete = true;\n break;\n }\n currID = (currID + 1) & 0xffff;\n }\n\n if (!complete) return null;\n\n const combined = new Uint8Array(totalSize);\n let pos = 0;\n for (const frag of fragments) {\n combined.set(frag.data, pos);\n pos += frag.data.length;\n queue.delete(nextID);\n win.advance(1);\n nextID = (nextID + 1) & 0xffff;\n }\n\n startPacket.data = combined;\n startPacket.typeFlagged &= ~PacketFlags.Fragmented;\n return [startPacket, nextID];\n }\n\n #tryDecompress(packet: Packet): void {\n if ((packetFlags(packet) & PacketFlags.Compressed) === 0) return;\n try {\n const qlz = new Qlz();\n packet.data = qlz.decompress(packet.data);\n packet.typeFlagged &= ~PacketFlags.Compressed;\n } catch (err) {\n this.#logger.debug(\"decompression failed\", { id: packet.id, err });\n }\n }\n\n #updatePostReceiveState(p: Packet): void {\n const pType = packetType(p);\n\n if (pType === PacketType.Init1) {\n this.#initPacketCheck = null;\n return;\n }\n\n if ((pType === PacketType.Ack || pType === PacketType.AckLow) && p.data.length >= 2) {\n const ackID = new DataView(p.data.buffer, p.data.byteOffset).getUint16(0, false);\n const targetType = pType === PacketType.Ack ? PacketType.Command : PacketType.CommandLow;\n const key = (targetType << 16) | ackID;\n this.#ackManager.delete(key);\n }\n }\n\n #nextPacketIdentity(pType: PacketType): [id: number, gen: number] {\n const pID = this.#packetCounter[pType]!;\n const pGen = this.#generationCounter[pType]!;\n\n if (pType === PacketType.Init1) return [pID, pGen];\n\n this.#packetCounter[pType] = (pID + 1) & 0xffff;\n if (this.#packetCounter[pType] === 0) {\n this.#generationCounter[pType] = (pGen + 1) >>> 0;\n }\n\n if (pType === PacketType.Command) {\n this.#sendWindowCommand.advanceToExcluded(pID);\n } else if (pType === PacketType.CommandLow) {\n this.#sendWindowCommandLow.advanceToExcluded(pID);\n }\n\n return [pID, pGen];\n }\n\n #trackResend(pType: PacketType, p: Packet, rp: ResendPacket): void {\n if (pType === PacketType.Init1) {\n this.#initPacketCheck = rp;\n return;\n }\n if (pType === PacketType.Command || pType === PacketType.CommandLow) {\n const key = (pType << 16) | p.id;\n this.#ackManager.set(key, rp);\n }\n }\n\n #checkResends(): void {\n const now = Date.now();\n\n if (now - this.#lastMessageReceived > PACKET_TIMEOUT_MS) {\n this.#logger.warn(\"idle timeout\");\n this.#triggerClose(new Error(\"idle timeout\"));\n return;\n }\n\n if (this.#initPacketCheck) {\n this.#doResend(this.#initPacketCheck, now);\n }\n\n for (const [key, rp] of this.#ackManager) {\n if (now - rp.firstSend > PACKET_TIMEOUT_MS) {\n this.#ackManager.delete(key);\n this.#triggerClose(new Error(\"packet ack timeout\"));\n return;\n }\n this.#doResend(rp, now);\n }\n }\n\n #doResend(rp: ResendPacket, now: number): void {\n if (now - rp.lastSend < rp.nextInterval) return;\n\n rp.lastSend = now;\n rp.retryCount++;\n rp.nextInterval = Math.min(rp.nextInterval * 2, MAX_RETRY_INTERVAL_MS);\n\n const dummy = !this.#crypt.cryptoInitComplete;\n const unencrypted = (packetFlags(rp.packet) & PacketFlags.Unencrypted) !== 0;\n const header = buildC2SHeader(rp.packet);\n const pType = packetType(rp.packet);\n\n const [ciphertext, tag] = this.#crypt.encrypt(\n pType,\n rp.packet.id,\n rp.packet.generationID,\n header,\n rp.packet.data,\n dummy,\n unencrypted,\n );\n\n const final = new Uint8Array(TAG_SIZE + HEADER_SIZE + ciphertext.length);\n final.set(tag.slice(0, TAG_SIZE), 0);\n final.set(header, TAG_SIZE);\n final.set(ciphertext, TAG_SIZE + HEADER_SIZE);\n this.#write(final);\n }\n\n #triggerClose(err: Error | null): void {\n if (this.#closed) return;\n this.close();\n this.onClosed?.(err);\n }\n}\n\n// ---- Helpers ----------------------------------------------------------------\n\nfunction applyProtocolFlags(pType: PacketType, flags: number): number {\n if (pType === PacketType.Command || pType === PacketType.CommandLow) {\n return flags | PacketFlags.NewProtocol;\n }\n return flags;\n}\n\nfunction hasOldNewerPacket(queue: Map<number, Packet>, nextID: number): boolean {\n const now = Date.now();\n for (const [id, pkg] of queue) {\n const diff = (id - nextID + 0x10000) & 0xffff;\n if (diff < 0x8000 && now - pkg.receivedAt > 5_000) {\n return true;\n }\n }\n return false;\n}\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"handler-C_JhqGTd.js","names":["#mod","#receiveWindow","#generation","#mappedBaseOffset","#isNextGen","#hashtable","#ensureControl","#processReference","#processLiteral","#updateHashtable","#crypt","#logger","#clientID","#conn","#closed","#lastMessageReceived","#handleRawPacket","#triggerClose","#packetCounter","#sendPacketRaw","#stopPing","#checkResends","#stopResend","#initPacketCheck","#sendSplitPacket","#generationCounter","#write","#nextPacketIdentity","#trackResend","#resolveGeneration","#decryptPacketData","#sendPong","#handleCommandWindowAndAck","#handlePacketQueue","#updatePostReceiveState","#recvWindowCommand","#recvWindowCommandLow","#sendWindowCommand","#sendWindowCommandLow","#sendAck","#commandQueue","#commandLowQueue","#fastForwardMissingPackets","#_nextCommandID","#_nextCommandLowID","#tryReassemble","#tryDecompress","#ackManager","#doResend"],"sources":["../src/transport/packet.ts","../src/transport/generation-window.ts","../src/transport/quicklz.ts","../src/transport/handler.ts"],"sourcesContent":["export const enum PacketType {\n Voice = 0,\n VoiceWhisper = 1,\n Command = 2,\n CommandLow = 3,\n Ping = 4,\n Pong = 5,\n Ack = 6,\n AckLow = 7,\n Init1 = 8,\n}\n\nexport const enum PacketFlags {\n Fragmented = 0x10,\n NewProtocol = 0x20,\n Compressed = 0x40,\n Unencrypted = 0x80,\n}\n\nexport interface Packet {\n /** Type byte combined with flags (low nibble = type, high nibble = flags). */\n typeFlagged: number;\n id: number;\n clientID: number;\n generationID: number;\n data: Uint8Array;\n receivedAt: number; // Date.now()\n}\n\nexport function packetType(p: Packet): PacketType {\n return (p.typeFlagged & 0x0f) as PacketType;\n}\n\nexport function packetFlags(p: Packet): number {\n return p.typeFlagged & 0xf0;\n}\n\nexport function isUnencrypted(p: Packet): boolean {\n return (packetFlags(p) & PacketFlags.Unencrypted) !== 0;\n}\n\n/** Build the 5-byte client-to-server header: [packetID(2), clientID(2), typeFlagged(1)]. */\nexport function buildC2SHeader(p: Packet): Uint8Array {\n const header = new Uint8Array(5);\n const view = new DataView(header.buffer);\n view.setUint16(0, p.id, false);\n view.setUint16(2, p.clientID, false);\n header[4] = p.typeFlagged;\n return header;\n}\n\n/** Parse a 3-byte server-to-client header. */\nexport function parseS2CHeader(raw: Uint8Array): Pick<Packet, \"id\" | \"typeFlagged\"> {\n const view = new DataView(raw.buffer, raw.byteOffset, raw.byteLength);\n return {\n id: view.getUint16(0, false),\n typeFlagged: raw[2]!,\n };\n}\n\n/** Parse a 5-byte client-to-server header. */\nexport function parseC2SHeader(raw: Uint8Array): Pick<Packet, \"id\" | \"clientID\" | \"typeFlagged\"> {\n const view = new DataView(raw.buffer, raw.byteOffset, raw.byteLength);\n return {\n id: view.getUint16(0, false),\n clientID: view.getUint16(2, false),\n typeFlagged: raw[4]!,\n };\n}\n","export class GenerationWindow {\n #mappedBaseOffset = 0;\n #generation = 0;\n readonly #mod: number;\n readonly #receiveWindow: number;\n\n constructor(mod: number, windowSize: number) {\n this.#mod = mod;\n this.#receiveWindow = windowSize;\n }\n\n get generation(): number {\n return this.#generation;\n }\n\n advance(amount: number): void {\n if (amount <= 0) return;\n const newBaseOffset = this.#mappedBaseOffset + amount;\n const genStep = Math.floor(newBaseOffset / this.#mod);\n if (genStep > 0) {\n this.#generation = Math.min(this.#generation + genStep, 0xffff_ffff);\n }\n this.#mappedBaseOffset = newBaseOffset % this.#mod;\n }\n\n advanceToExcluded(mappedValue: number): void {\n let moveDist = mappedValue - this.#mappedBaseOffset;\n if (moveDist < 0) moveDist += this.#mod;\n this.advance(moveDist + 1);\n }\n\n syncTo(mappedValue: number): void {\n let moveDist = mappedValue - this.#mappedBaseOffset;\n if (moveDist < 0) moveDist += this.#mod;\n this.advance(moveDist);\n }\n\n isInWindow(mappedValue: number): boolean {\n const maxOffset = this.#mappedBaseOffset + this.#receiveWindow;\n if (maxOffset < this.#mod) {\n return mappedValue >= this.#mappedBaseOffset && mappedValue < maxOffset;\n }\n return mappedValue >= this.#mappedBaseOffset || mappedValue < maxOffset - this.#mod;\n }\n\n mappedToIndex(mappedValue: number): number {\n if (this.#isNextGen(mappedValue)) {\n return mappedValue + this.#mod - this.#mappedBaseOffset;\n }\n return mappedValue - this.#mappedBaseOffset;\n }\n\n isOldPacket(mappedValue: number): boolean {\n return this.mappedToIndex(mappedValue) < 0;\n }\n\n isFuturePacket(mappedValue: number): boolean {\n return this.mappedToIndex(mappedValue) >= this.#receiveWindow;\n }\n\n #isNextGen(mappedValue: number): boolean {\n return (\n this.#mappedBaseOffset > this.#mod - this.#receiveWindow &&\n mappedValue < this.#mappedBaseOffset + this.#receiveWindow - this.#mod\n );\n }\n\n getGeneration(mappedValue: number): number {\n if (this.#isNextGen(mappedValue)) return this.#generation + 1;\n return this.#generation;\n }\n\n reset(): void {\n this.#mappedBaseOffset = 0;\n this.#generation = 0;\n }\n}\n","const TABLE_SIZE = 4096;\n\ninterface QlzState {\n control: number;\n sourcePos: number;\n destPos: number;\n nextHashed: number;\n}\n\nexport class Qlz {\n #hashtable = new Int32Array(TABLE_SIZE);\n\n decompress(data: Uint8Array): Uint8Array {\n const { headerLen, decompressedSize, flags } = parseQlzHeader(data);\n\n const dest = new Uint8Array(decompressedSize);\n\n if ((flags & 0x01) === 0) {\n dest.set(data.slice(headerLen, headerLen + decompressedSize));\n return dest;\n }\n\n this.#hashtable.fill(0);\n\n const state: QlzState = {\n control: 1,\n sourcePos: headerLen,\n destPos: 0,\n nextHashed: 0,\n };\n\n while (this.#ensureControl(data, state)) {\n if ((state.control & 1) !== 0) {\n if (!this.#processReference(data, dest, state)) break;\n } else {\n if (this.#processLiteral(data, dest, decompressedSize, state)) break;\n }\n }\n\n return dest;\n }\n\n #ensureControl(data: Uint8Array, st: QlzState): boolean {\n if (st.control !== 1) return true;\n if (st.sourcePos + 4 > data.length) return false;\n st.control =\n (data[st.sourcePos]! |\n (data[st.sourcePos + 1]! << 8) |\n (data[st.sourcePos + 2]! << 16) |\n (data[st.sourcePos + 3]! << 24)) >>>\n 0;\n st.sourcePos += 4;\n return true;\n }\n\n #processReference(data: Uint8Array, dest: Uint8Array, st: QlzState): boolean {\n st.control = (st.control >>> 1) >>> 0;\n if (st.sourcePos + 2 > data.length) return false;\n\n const b1 = data[st.sourcePos]!;\n const b2 = data[st.sourcePos + 1]!;\n st.sourcePos += 2;\n\n const hash = (b1 >> 4) | (b2 << 4);\n let matchlen = b1 & 0x0f;\n if (matchlen !== 0) {\n matchlen += 2;\n } else {\n if (st.sourcePos >= data.length) return false;\n matchlen = data[st.sourcePos]!;\n st.sourcePos++;\n }\n\n const offset = this.#hashtable[hash]!;\n for (let i = 0; i < matchlen; i++) {\n if (st.destPos < dest.length && offset + i < st.destPos) {\n dest[st.destPos] = dest[offset + i]!;\n st.destPos++;\n }\n }\n\n const end = st.destPos + 1 - matchlen;\n this.#updateHashtable(dest, st, end);\n st.nextHashed = st.destPos;\n\n return true;\n }\n\n #processLiteral(\n data: Uint8Array,\n dest: Uint8Array,\n decompressedSize: number,\n st: QlzState,\n ): boolean {\n const threshold = Math.max(decompressedSize, 10) - 10;\n if (st.destPos >= threshold) {\n while (st.destPos < decompressedSize) {\n if (st.control === 1) {\n st.sourcePos += 4;\n if (st.sourcePos > data.length) break;\n st.control =\n (data[st.sourcePos - 4]! |\n (data[st.sourcePos - 3]! << 8) |\n (data[st.sourcePos - 2]! << 16) |\n (data[st.sourcePos - 1]! << 24)) >>>\n 0;\n }\n if (st.sourcePos >= data.length) break;\n dest[st.destPos++] = data[st.sourcePos++]!;\n st.control = (st.control >>> 1) >>> 0;\n }\n return true;\n }\n\n if (st.sourcePos >= data.length || st.destPos >= dest.length) return true;\n\n dest[st.destPos++] = data[st.sourcePos++]!;\n st.control = (st.control >>> 1) >>> 0;\n\n const end = Math.max(st.destPos - 2, 0);\n this.#updateHashtable(dest, st, end);\n if (st.nextHashed < end) st.nextHashed = end;\n\n return false;\n }\n\n #updateHashtable(dest: Uint8Array, st: QlzState, end: number): void {\n while (st.nextHashed < end) {\n if (st.nextHashed + 3 > dest.length) break;\n const v =\n (dest[st.nextHashed]! |\n (dest[st.nextHashed + 1]! << 8) |\n (dest[st.nextHashed + 2]! << 16)) >>>\n 0;\n const hash = ((v >> 12) ^ v) & 0xfff;\n this.#hashtable[hash] = st.nextHashed;\n st.nextHashed++;\n }\n }\n}\n\nfunction parseQlzHeader(data: Uint8Array): {\n headerLen: number;\n decompressedSize: number;\n flags: number;\n} {\n if (data.length < 3) throw new Error(\"QuickLZ: data too short\");\n\n const flags = data[0]!;\n const level = (flags >> 2) & 0x03;\n if (level !== 1) throw new Error(\"QuickLZ: only level 1 is supported\");\n\n const headerLen = (flags & 0x02) !== 0 ? 9 : 3;\n if (data.length < headerLen) throw new Error(\"QuickLZ: data too short for header\");\n\n let decompressedSize: number;\n if ((flags & 0x02) !== 0) {\n decompressedSize = (data[5]! | (data[6]! << 8) | (data[7]! << 16) | (data[8]! << 24)) >>> 0;\n } else {\n decompressedSize = data[2]!;\n }\n\n return { headerLen, decompressedSize, flags };\n}\n","import { createSocket, type Socket as UdpSocket } from \"node:dgram\";\nimport type { Crypt } from \"../crypto/crypt.js\";\nimport type { Logger } from \"../types.js\";\nimport { noopLogger } from \"../types.js\";\nimport {\n type Packet,\n PacketType,\n PacketFlags,\n packetType,\n packetFlags,\n buildC2SHeader,\n parseS2CHeader,\n} from \"./packet.js\";\nimport { GenerationWindow } from \"./generation-window.js\";\nimport { Qlz } from \"./quicklz.js\";\nimport { processInit1 } from \"../handshake/crypt-handshake.js\";\n\nconst MAX_OUT_PACKET_SIZE = 500;\nconst RECEIVE_PACKET_WINDOW_SIZE = 1024;\nconst PING_INTERVAL_MS = 5_000;\nconst PACKET_TIMEOUT_MS = 60_000;\nconst MAX_RETRY_INTERVAL_MS = 1_000;\nconst HEADER_SIZE = 5;\nconst TAG_SIZE = 8;\nconst VOICE_HEADER_SIZE = 3;\nconst RESEND_BASE_INTERVAL_MS = 500;\nconst RESEND_LOOP_INTERVAL_MS = 100;\n\ninterface ResendPacket {\n packet: Packet;\n firstSend: number;\n lastSend: number;\n retryCount: number;\n nextInterval: number;\n}\n\nexport class PacketHandler {\n onPacket: ((p: Packet) => void) | null = null;\n onClosed: ((err: Error | null) => void) | null = null;\n\n readonly #crypt: Crypt;\n readonly #logger: Logger;\n\n #conn: UdpSocket | null = null;\n #clientID = 0;\n #closed = false;\n #stopPing: (() => void) | null = null;\n #stopResend: (() => void) | null = null;\n #lastMessageReceived = Date.now();\n\n #packetCounter = new Uint16Array(9);\n #generationCounter = new Uint32Array(9);\n\n #recvWindowCommand = new GenerationWindow(1 << 16, RECEIVE_PACKET_WINDOW_SIZE);\n #recvWindowCommandLow = new GenerationWindow(1 << 16, RECEIVE_PACKET_WINDOW_SIZE);\n #sendWindowCommand = new GenerationWindow(1 << 16, RECEIVE_PACKET_WINDOW_SIZE);\n #sendWindowCommandLow = new GenerationWindow(1 << 16, RECEIVE_PACKET_WINDOW_SIZE);\n\n #commandQueue = new Map<number, Packet>();\n #commandLowQueue = new Map<number, Packet>();\n #ackManager = new Map<number, ResendPacket>();\n #initPacketCheck: ResendPacket | null = null;\n\n constructor(crypt: Crypt, logger: Logger = noopLogger) {\n this.#crypt = crypt;\n this.#logger = logger;\n }\n\n setClientID(id: number): void {\n this.#clientID = id;\n }\n\n connect(addr: string): Promise<void> {\n return new Promise((resolve, reject) => {\n const [host, portStr] =\n addr.lastIndexOf(\":\") > 0\n ? [addr.slice(0, addr.lastIndexOf(\":\")), addr.slice(addr.lastIndexOf(\":\") + 1)]\n : [addr, \"9987\"];\n const port = parseInt(portStr, 10);\n\n const socket = createSocket(\"udp4\");\n socket.once(\"error\", reject);\n socket.connect(port, host, () => {\n socket.off(\"error\", reject);\n this.start(socket);\n resolve();\n });\n });\n }\n\n start(conn: UdpSocket): void {\n this.#conn = conn;\n this.#closed = false;\n this.#lastMessageReceived = Date.now();\n\n conn.on(\"message\", (msg) => {\n this.#lastMessageReceived = Date.now();\n this.#handleRawPacket(new Uint8Array(msg.buffer, msg.byteOffset, msg.byteLength));\n });\n conn.on(\"error\", (err) => {\n this.#logger.error(\"udp error\", err);\n this.#triggerClose(err);\n });\n conn.on(\"close\", () => this.#triggerClose(null));\n\n this.#packetCounter[PacketType.Command] = 1;\n this.#packetCounter[PacketType.Init1] = 101;\n\n const init1Data = processInit1(this.#crypt, null);\n if (init1Data) this.#sendPacketRaw(PacketType.Init1, init1Data, 0);\n\n // Ping loop\n const pingTimer = setInterval(() => {\n if (this.#crypt.cryptoInitComplete) {\n this.sendPacket(PacketType.Ping, new Uint8Array(0), PacketFlags.Unencrypted);\n }\n }, PING_INTERVAL_MS);\n this.#stopPing = () => clearInterval(pingTimer);\n\n // Resend loop\n const resendTimer = setInterval(() => this.#checkResends(), RESEND_LOOP_INTERVAL_MS);\n this.#stopResend = () => clearInterval(resendTimer);\n }\n\n receivedFinalInitAck(): void {\n this.#initPacketCheck = null;\n }\n\n sendPacket(pType: PacketType, data: Uint8Array, flags: number): void {\n const dummy = !this.#crypt.cryptoInitComplete;\n if (data.length > 487 && pType !== PacketType.Voice && pType !== PacketType.VoiceWhisper) {\n this.#sendSplitPacket(pType, data, flags, dummy);\n return;\n }\n this.#sendPacketRaw(pType, data, flags, dummy);\n }\n\n sendVoicePacket(data: Uint8Array, codec: number): void {\n const pID = this.#packetCounter[PacketType.Voice]!;\n const pGen = this.#generationCounter[PacketType.Voice]!;\n this.#packetCounter[PacketType.Voice] = (pID + 1) & 0xffff;\n if (this.#packetCounter[PacketType.Voice] === 0) {\n this.#generationCounter[PacketType.Voice]!++;\n }\n\n const payloadLen = VOICE_HEADER_SIZE + data.length;\n const voicePayload = new Uint8Array(payloadLen);\n new DataView(voicePayload.buffer).setUint16(0, pID, false);\n voicePayload[2] = codec;\n voicePayload.set(data, VOICE_HEADER_SIZE);\n\n const p: Packet = {\n typeFlagged: PacketType.Voice | PacketFlags.Unencrypted,\n id: pID,\n clientID: this.#clientID,\n generationID: pGen,\n data: voicePayload,\n receivedAt: 0,\n };\n\n const header = buildC2SHeader(p);\n const final = new Uint8Array(TAG_SIZE + HEADER_SIZE + payloadLen);\n final.set(this.#crypt.fakeSignature, 0);\n final.set(header, TAG_SIZE);\n final.set(voicePayload, TAG_SIZE + HEADER_SIZE);\n this.#write(final);\n }\n\n close(): void {\n if (this.#closed) return;\n this.#closed = true;\n this.#stopPing?.();\n this.#stopResend?.();\n this.#conn?.close();\n }\n\n // ---- Private ---------------------------------------------------------------\n\n #sendSplitPacket(pType: PacketType, data: Uint8Array, flags: number, dummy: boolean): void {\n const maxSize = MAX_OUT_PACKET_SIZE - HEADER_SIZE - TAG_SIZE; // 487\n let pos = 0;\n let first = true;\n\n while (pos < data.length) {\n const blockSize = Math.min(data.length - pos, maxSize);\n const last = pos + blockSize === data.length;\n\n let pFlags = flags;\n if (first !== last) pFlags |= PacketFlags.Fragmented;\n\n this.#sendPacketRaw(pType, data.slice(pos, pos + blockSize), pFlags, dummy);\n pos += blockSize;\n first = false;\n }\n }\n\n #sendPacketRaw(\n pType: PacketType,\n data: Uint8Array,\n flags: number,\n dummy = !this.#crypt.cryptoInitComplete,\n ): void {\n flags = applyProtocolFlags(pType, flags);\n const [pID, pGen] = this.#nextPacketIdentity(pType);\n const p: Packet = {\n typeFlagged: pType | flags,\n id: pID,\n clientID: this.#clientID,\n generationID: pGen,\n data,\n receivedAt: 0,\n };\n\n const unencrypted = (flags & PacketFlags.Unencrypted) !== 0;\n const header = buildC2SHeader(p);\n const [ciphertext, tag] = this.#crypt.encrypt(\n pType,\n pID,\n pGen,\n header,\n data,\n dummy,\n unencrypted,\n );\n\n const final = new Uint8Array(TAG_SIZE + HEADER_SIZE + ciphertext.length);\n final.set(tag.slice(0, TAG_SIZE), 0);\n final.set(header, TAG_SIZE);\n final.set(ciphertext, TAG_SIZE + HEADER_SIZE);\n this.#write(final);\n\n const rp: ResendPacket = {\n packet: p,\n firstSend: Date.now(),\n lastSend: Date.now(),\n retryCount: 0,\n nextInterval: RESEND_BASE_INTERVAL_MS,\n };\n this.#trackResend(pType, p, rp);\n }\n\n #write(data: Uint8Array): void {\n this.#conn?.send(Buffer.from(data), (err) => {\n if (err) this.#logger.warn(\"udp send error\", err);\n });\n }\n\n #handleRawPacket(raw: Uint8Array): void {\n if (raw.length < 11) return;\n\n const tag = raw.slice(0, TAG_SIZE);\n const header = raw.slice(TAG_SIZE, TAG_SIZE + 3);\n const ciphertext = raw.slice(TAG_SIZE + 3);\n\n const parsed = parseS2CHeader(header);\n const p: Packet = {\n ...parsed,\n clientID: 0,\n generationID: this.#resolveGeneration(parsed.id, parsed.typeFlagged & 0x0f),\n data: new Uint8Array(0),\n receivedAt: Date.now(),\n };\n\n const decrypted = this.#decryptPacketData(p, header, ciphertext, tag);\n if (decrypted === null) return;\n p.data = decrypted.plaintext;\n\n const pType = packetType(p);\n\n if (pType === PacketType.Ping) {\n this.#sendPong(p.id, decrypted.dummyUsed);\n return;\n }\n\n if (!this.#handleCommandWindowAndAck(p, decrypted.dummyUsed)) return;\n\n this.#handlePacketQueue(p);\n this.#updatePostReceiveState(p);\n }\n\n #resolveGeneration(id: number, pType: number): number {\n switch (pType as PacketType) {\n case PacketType.Command:\n return this.#recvWindowCommand.getGeneration(id);\n case PacketType.CommandLow:\n return this.#recvWindowCommandLow.getGeneration(id);\n case PacketType.Ack:\n return this.#sendWindowCommand.getGeneration(id);\n case PacketType.AckLow:\n return this.#sendWindowCommandLow.getGeneration(id);\n default:\n return 0;\n }\n }\n\n #decryptPacketData(\n p: Packet,\n header: Uint8Array,\n ciphertext: Uint8Array,\n tag: Uint8Array,\n ): { plaintext: Uint8Array; dummyUsed: boolean } | null {\n const unencrypted = (packetFlags(p) & PacketFlags.Unencrypted) !== 0;\n const dummy = !this.#crypt.cryptoInitComplete;\n let dummyUsed = dummy;\n const pType = packetType(p);\n const gen = p.generationID;\n\n try {\n const plaintext = this.#crypt.decrypt(\n pType,\n p.id,\n gen,\n header,\n ciphertext,\n tag,\n dummy,\n unencrypted,\n );\n return { plaintext, dummyUsed };\n } catch {\n // Try adjacent generations\n for (const offset of [-1, 1]) {\n const guessGen = gen + offset;\n if (guessGen < 0) continue;\n try {\n const plaintext = this.#crypt.decrypt(\n pType,\n p.id,\n guessGen,\n header,\n ciphertext,\n tag,\n false,\n false,\n );\n return { plaintext, dummyUsed: false };\n } catch {\n // continue\n }\n }\n\n // Try dummy fallback for command/ack types\n if (\n pType === PacketType.Command ||\n pType === PacketType.CommandLow ||\n pType === PacketType.Ack\n ) {\n try {\n const plaintext = this.#crypt.decrypt(\n pType,\n p.id,\n gen,\n header,\n ciphertext,\n tag,\n true,\n unencrypted,\n );\n return { plaintext, dummyUsed: true };\n } catch {\n // fall through\n }\n }\n\n return null;\n }\n }\n\n #handleCommandWindowAndAck(p: Packet, dummyUsed: boolean): boolean {\n const pType = packetType(p);\n if (pType !== PacketType.Command && pType !== PacketType.CommandLow) return true;\n\n const win = pType === PacketType.Command ? this.#recvWindowCommand : this.#recvWindowCommandLow;\n const ackType = pType === PacketType.Command ? PacketType.Ack : PacketType.AckLow;\n\n if (!win.isInWindow(p.id)) {\n if (win.isOldPacket(p.id)) {\n this.#sendAck(p.id, ackType, dummyUsed);\n }\n return false;\n }\n this.#sendAck(p.id, ackType, dummyUsed);\n return true;\n }\n\n #sendAck(packetID: number, ackType: PacketType, dummyUsed: boolean): void {\n const ackData = new Uint8Array(2);\n new DataView(ackData.buffer).setUint16(0, packetID, false);\n this.#sendPacketRaw(ackType, ackData, 0, dummyUsed);\n }\n\n #sendPong(pID: number, dummyUsed: boolean): void {\n const pongData = new Uint8Array(2);\n new DataView(pongData.buffer).setUint16(0, pID, false);\n this.#sendPacketRaw(PacketType.Pong, pongData, PacketFlags.Unencrypted, dummyUsed);\n }\n\n #handlePacketQueue(p: Packet): void {\n const pType = packetType(p);\n if (pType !== PacketType.Command && pType !== PacketType.CommandLow) {\n this.onPacket?.(p);\n return;\n }\n\n const isCommand = pType === PacketType.Command;\n const queue = isCommand ? this.#commandQueue : this.#commandLowQueue;\n const win = isCommand ? this.#recvWindowCommand : this.#recvWindowCommandLow;\n\n queue.set(p.id, p);\n\n if (isCommand) {\n this.#fastForwardMissingPackets(queue, win, true);\n } else {\n this.#fastForwardMissingPackets(queue, win, false);\n }\n\n while (true) {\n const current = isCommand ? this.#_nextCommandID : this.#_nextCommandLowID;\n const packet = queue.get(current);\n if (packet === undefined) break;\n\n const result = this.#tryReassemble(packet, queue, current, win);\n if (!result) break;\n\n const [reassembled, newNext] = result;\n if (isCommand) this.#_nextCommandID = newNext;\n else this.#_nextCommandLowID = newNext;\n\n this.#tryDecompress(reassembled);\n this.onPacket?.(reassembled);\n }\n }\n\n #_nextCommandID = 0;\n #_nextCommandLowID = 0;\n\n #fastForwardMissingPackets(\n queue: Map<number, Packet>,\n win: GenerationWindow,\n isCommand: boolean,\n ): void {\n let nextID = isCommand ? this.#_nextCommandID : this.#_nextCommandLowID;\n while (!queue.has(nextID) && hasOldNewerPacket(queue, nextID)) {\n nextID = (nextID + 1) & 0xffff;\n win.advance(1);\n }\n if (isCommand) this.#_nextCommandID = nextID;\n else this.#_nextCommandLowID = nextID;\n }\n\n #tryReassemble(\n startPacket: Packet,\n queue: Map<number, Packet>,\n nextID: number,\n win: GenerationWindow,\n ): [reassembled: Packet, newNextID: number] | null {\n if ((packetFlags(startPacket) & PacketFlags.Fragmented) === 0) {\n queue.delete(nextID);\n win.advance(1);\n return [startPacket, (nextID + 1) & 0xffff];\n }\n\n const fragments: Packet[] = [];\n let totalSize = 0;\n let currID = nextID;\n let startSeen = false;\n let complete = false;\n\n while (true) {\n const frag = queue.get(currID);\n if (!frag) return null;\n fragments.push(frag);\n totalSize += frag.data.length;\n\n const fragmented = (packetFlags(frag) & PacketFlags.Fragmented) !== 0;\n if (!startSeen) {\n startSeen = true;\n if (!fragmented) {\n complete = true;\n break;\n }\n } else if (fragmented) {\n complete = true;\n break;\n }\n currID = (currID + 1) & 0xffff;\n }\n\n if (!complete) return null;\n\n const combined = new Uint8Array(totalSize);\n let pos = 0;\n for (const frag of fragments) {\n combined.set(frag.data, pos);\n pos += frag.data.length;\n queue.delete(nextID);\n win.advance(1);\n nextID = (nextID + 1) & 0xffff;\n }\n\n startPacket.data = combined;\n startPacket.typeFlagged &= ~PacketFlags.Fragmented;\n return [startPacket, nextID];\n }\n\n #tryDecompress(packet: Packet): void {\n if ((packetFlags(packet) & PacketFlags.Compressed) === 0) return;\n try {\n const qlz = new Qlz();\n packet.data = qlz.decompress(packet.data);\n packet.typeFlagged &= ~PacketFlags.Compressed;\n } catch (err) {\n this.#logger.debug(\"decompression failed\", { id: packet.id, err });\n }\n }\n\n #updatePostReceiveState(p: Packet): void {\n const pType = packetType(p);\n\n if (pType === PacketType.Init1) {\n this.#initPacketCheck = null;\n return;\n }\n\n if ((pType === PacketType.Ack || pType === PacketType.AckLow) && p.data.length >= 2) {\n const ackID = new DataView(p.data.buffer, p.data.byteOffset).getUint16(0, false);\n const targetType = pType === PacketType.Ack ? PacketType.Command : PacketType.CommandLow;\n const key = (targetType << 16) | ackID;\n this.#ackManager.delete(key);\n }\n }\n\n #nextPacketIdentity(pType: PacketType): [id: number, gen: number] {\n const pID = this.#packetCounter[pType]!;\n const pGen = this.#generationCounter[pType]!;\n\n if (pType === PacketType.Init1) return [pID, pGen];\n\n this.#packetCounter[pType] = (pID + 1) & 0xffff;\n if (this.#packetCounter[pType] === 0) {\n this.#generationCounter[pType] = (pGen + 1) >>> 0;\n }\n\n if (pType === PacketType.Command) {\n this.#sendWindowCommand.advanceToExcluded(pID);\n } else if (pType === PacketType.CommandLow) {\n this.#sendWindowCommandLow.advanceToExcluded(pID);\n }\n\n return [pID, pGen];\n }\n\n #trackResend(pType: PacketType, p: Packet, rp: ResendPacket): void {\n if (pType === PacketType.Init1) {\n this.#initPacketCheck = rp;\n return;\n }\n if (pType === PacketType.Command || pType === PacketType.CommandLow) {\n const key = (pType << 16) | p.id;\n this.#ackManager.set(key, rp);\n }\n }\n\n #checkResends(): void {\n const now = Date.now();\n\n if (now - this.#lastMessageReceived > PACKET_TIMEOUT_MS) {\n this.#logger.warn(\"idle timeout\");\n this.#triggerClose(new Error(\"idle timeout\"));\n return;\n }\n\n if (this.#initPacketCheck) {\n this.#doResend(this.#initPacketCheck, now);\n }\n\n for (const [key, rp] of this.#ackManager) {\n if (now - rp.firstSend > PACKET_TIMEOUT_MS) {\n this.#ackManager.delete(key);\n this.#triggerClose(new Error(\"packet ack timeout\"));\n return;\n }\n this.#doResend(rp, now);\n }\n }\n\n #doResend(rp: ResendPacket, now: number): void {\n if (now - rp.lastSend < rp.nextInterval) return;\n\n rp.lastSend = now;\n rp.retryCount++;\n rp.nextInterval = Math.min(rp.nextInterval * 2, MAX_RETRY_INTERVAL_MS);\n\n const dummy = !this.#crypt.cryptoInitComplete;\n const unencrypted = (packetFlags(rp.packet) & PacketFlags.Unencrypted) !== 0;\n const header = buildC2SHeader(rp.packet);\n const pType = packetType(rp.packet);\n\n const [ciphertext, tag] = this.#crypt.encrypt(\n pType,\n rp.packet.id,\n rp.packet.generationID,\n header,\n rp.packet.data,\n dummy,\n unencrypted,\n );\n\n const final = new Uint8Array(TAG_SIZE + HEADER_SIZE + ciphertext.length);\n final.set(tag.slice(0, TAG_SIZE), 0);\n final.set(header, TAG_SIZE);\n final.set(ciphertext, TAG_SIZE + HEADER_SIZE);\n this.#write(final);\n }\n\n #triggerClose(err: Error | null): void {\n if (this.#closed) return;\n this.close();\n this.onClosed?.(err);\n }\n}\n\n// ---- Helpers ----------------------------------------------------------------\n\nfunction applyProtocolFlags(pType: PacketType, flags: number): number {\n if (pType === PacketType.Command || pType === PacketType.CommandLow) {\n return flags | PacketFlags.NewProtocol;\n }\n return flags;\n}\n\nfunction hasOldNewerPacket(queue: Map<number, Packet>, nextID: number): boolean {\n const now = Date.now();\n for (const [id, pkg] of queue) {\n const diff = (id - nextID + 0x10000) & 0xffff;\n if (diff < 0x8000 && now - pkg.receivedAt > 5_000) {\n return true;\n }\n }\n return false;\n}\n"],"mappings":"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"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"handler-CqCD93f0.cjs","names":["#mod","#receiveWindow","#generation","#mappedBaseOffset","#isNextGen","#hashtable","#ensureControl","#processReference","#processLiteral","#updateHashtable","#crypt","#logger","#clientID","#conn","#closed","#lastMessageReceived","#handleRawPacket","#triggerClose","#packetCounter","#sendPacketRaw","#stopPing","#checkResends","#stopResend","#initPacketCheck","#sendSplitPacket","#generationCounter","#write","#nextPacketIdentity","#trackResend","#resolveGeneration","#decryptPacketData","#sendPong","#handleCommandWindowAndAck","#handlePacketQueue","#updatePostReceiveState","#recvWindowCommand","#recvWindowCommandLow","#sendWindowCommand","#sendWindowCommandLow","#sendAck","#commandQueue","#commandLowQueue","#fastForwardMissingPackets","#_nextCommandID","#_nextCommandLowID","#tryReassemble","#tryDecompress","#ackManager","#doResend"],"sources":["../src/transport/packet.ts","../src/transport/generation-window.ts","../src/transport/quicklz.ts","../src/transport/handler.ts"],"sourcesContent":["export const enum PacketType {\n Voice = 0,\n VoiceWhisper = 1,\n Command = 2,\n CommandLow = 3,\n Ping = 4,\n Pong = 5,\n Ack = 6,\n AckLow = 7,\n Init1 = 8,\n}\n\nexport const enum PacketFlags {\n Fragmented = 0x10,\n NewProtocol = 0x20,\n Compressed = 0x40,\n Unencrypted = 0x80,\n}\n\nexport interface Packet {\n /** Type byte combined with flags (low nibble = type, high nibble = flags). */\n typeFlagged: number;\n id: number;\n clientID: number;\n generationID: number;\n data: Uint8Array;\n receivedAt: number; // Date.now()\n}\n\nexport function packetType(p: Packet): PacketType {\n return (p.typeFlagged & 0x0f) as PacketType;\n}\n\nexport function packetFlags(p: Packet): number {\n return p.typeFlagged & 0xf0;\n}\n\nexport function isUnencrypted(p: Packet): boolean {\n return (packetFlags(p) & PacketFlags.Unencrypted) !== 0;\n}\n\n/** Build the 5-byte client-to-server header: [packetID(2), clientID(2), typeFlagged(1)]. */\nexport function buildC2SHeader(p: Packet): Uint8Array {\n const header = new Uint8Array(5);\n const view = new DataView(header.buffer);\n view.setUint16(0, p.id, false);\n view.setUint16(2, p.clientID, false);\n header[4] = p.typeFlagged;\n return header;\n}\n\n/** Parse a 3-byte server-to-client header. */\nexport function parseS2CHeader(raw: Uint8Array): Pick<Packet, \"id\" | \"typeFlagged\"> {\n const view = new DataView(raw.buffer, raw.byteOffset, raw.byteLength);\n return {\n id: view.getUint16(0, false),\n typeFlagged: raw[2]!,\n };\n}\n\n/** Parse a 5-byte client-to-server header. */\nexport function parseC2SHeader(raw: Uint8Array): Pick<Packet, \"id\" | \"clientID\" | \"typeFlagged\"> {\n const view = new DataView(raw.buffer, raw.byteOffset, raw.byteLength);\n return {\n id: view.getUint16(0, false),\n clientID: view.getUint16(2, false),\n typeFlagged: raw[4]!,\n };\n}\n","export class GenerationWindow {\n #mappedBaseOffset = 0;\n #generation = 0;\n readonly #mod: number;\n readonly #receiveWindow: number;\n\n constructor(mod: number, windowSize: number) {\n this.#mod = mod;\n this.#receiveWindow = windowSize;\n }\n\n get generation(): number {\n return this.#generation;\n }\n\n advance(amount: number): void {\n if (amount <= 0) return;\n const newBaseOffset = this.#mappedBaseOffset + amount;\n const genStep = Math.floor(newBaseOffset / this.#mod);\n if (genStep > 0) {\n this.#generation = Math.min(this.#generation + genStep, 0xffff_ffff);\n }\n this.#mappedBaseOffset = newBaseOffset % this.#mod;\n }\n\n advanceToExcluded(mappedValue: number): void {\n let moveDist = mappedValue - this.#mappedBaseOffset;\n if (moveDist < 0) moveDist += this.#mod;\n this.advance(moveDist + 1);\n }\n\n syncTo(mappedValue: number): void {\n let moveDist = mappedValue - this.#mappedBaseOffset;\n if (moveDist < 0) moveDist += this.#mod;\n this.advance(moveDist);\n }\n\n isInWindow(mappedValue: number): boolean {\n const maxOffset = this.#mappedBaseOffset + this.#receiveWindow;\n if (maxOffset < this.#mod) {\n return mappedValue >= this.#mappedBaseOffset && mappedValue < maxOffset;\n }\n return mappedValue >= this.#mappedBaseOffset || mappedValue < maxOffset - this.#mod;\n }\n\n mappedToIndex(mappedValue: number): number {\n if (this.#isNextGen(mappedValue)) {\n return mappedValue + this.#mod - this.#mappedBaseOffset;\n }\n return mappedValue - this.#mappedBaseOffset;\n }\n\n isOldPacket(mappedValue: number): boolean {\n return this.mappedToIndex(mappedValue) < 0;\n }\n\n isFuturePacket(mappedValue: number): boolean {\n return this.mappedToIndex(mappedValue) >= this.#receiveWindow;\n }\n\n #isNextGen(mappedValue: number): boolean {\n return (\n this.#mappedBaseOffset > this.#mod - this.#receiveWindow &&\n mappedValue < this.#mappedBaseOffset + this.#receiveWindow - this.#mod\n );\n }\n\n getGeneration(mappedValue: number): number {\n if (this.#isNextGen(mappedValue)) return this.#generation + 1;\n return this.#generation;\n }\n\n reset(): void {\n this.#mappedBaseOffset = 0;\n this.#generation = 0;\n }\n}\n","const TABLE_SIZE = 4096;\n\ninterface QlzState {\n control: number;\n sourcePos: number;\n destPos: number;\n nextHashed: number;\n}\n\nexport class Qlz {\n #hashtable = new Int32Array(TABLE_SIZE);\n\n decompress(data: Uint8Array): Uint8Array {\n const { headerLen, decompressedSize, flags } = parseQlzHeader(data);\n\n const dest = new Uint8Array(decompressedSize);\n\n if ((flags & 0x01) === 0) {\n dest.set(data.slice(headerLen, headerLen + decompressedSize));\n return dest;\n }\n\n this.#hashtable.fill(0);\n\n const state: QlzState = {\n control: 1,\n sourcePos: headerLen,\n destPos: 0,\n nextHashed: 0,\n };\n\n while (this.#ensureControl(data, state)) {\n if ((state.control & 1) !== 0) {\n if (!this.#processReference(data, dest, state)) break;\n } else {\n if (this.#processLiteral(data, dest, decompressedSize, state)) break;\n }\n }\n\n return dest;\n }\n\n #ensureControl(data: Uint8Array, st: QlzState): boolean {\n if (st.control !== 1) return true;\n if (st.sourcePos + 4 > data.length) return false;\n st.control =\n (data[st.sourcePos]! |\n (data[st.sourcePos + 1]! << 8) |\n (data[st.sourcePos + 2]! << 16) |\n (data[st.sourcePos + 3]! << 24)) >>>\n 0;\n st.sourcePos += 4;\n return true;\n }\n\n #processReference(data: Uint8Array, dest: Uint8Array, st: QlzState): boolean {\n st.control = (st.control >>> 1) >>> 0;\n if (st.sourcePos + 2 > data.length) return false;\n\n const b1 = data[st.sourcePos]!;\n const b2 = data[st.sourcePos + 1]!;\n st.sourcePos += 2;\n\n const hash = (b1 >> 4) | (b2 << 4);\n let matchlen = b1 & 0x0f;\n if (matchlen !== 0) {\n matchlen += 2;\n } else {\n if (st.sourcePos >= data.length) return false;\n matchlen = data[st.sourcePos]!;\n st.sourcePos++;\n }\n\n const offset = this.#hashtable[hash]!;\n for (let i = 0; i < matchlen; i++) {\n if (st.destPos < dest.length && offset + i < st.destPos) {\n dest[st.destPos] = dest[offset + i]!;\n st.destPos++;\n }\n }\n\n const end = st.destPos + 1 - matchlen;\n this.#updateHashtable(dest, st, end);\n st.nextHashed = st.destPos;\n\n return true;\n }\n\n #processLiteral(\n data: Uint8Array,\n dest: Uint8Array,\n decompressedSize: number,\n st: QlzState,\n ): boolean {\n const threshold = Math.max(decompressedSize, 10) - 10;\n if (st.destPos >= threshold) {\n while (st.destPos < decompressedSize) {\n if (st.control === 1) {\n st.sourcePos += 4;\n if (st.sourcePos > data.length) break;\n st.control =\n (data[st.sourcePos - 4]! |\n (data[st.sourcePos - 3]! << 8) |\n (data[st.sourcePos - 2]! << 16) |\n (data[st.sourcePos - 1]! << 24)) >>>\n 0;\n }\n if (st.sourcePos >= data.length) break;\n dest[st.destPos++] = data[st.sourcePos++]!;\n st.control = (st.control >>> 1) >>> 0;\n }\n return true;\n }\n\n if (st.sourcePos >= data.length || st.destPos >= dest.length) return true;\n\n dest[st.destPos++] = data[st.sourcePos++]!;\n st.control = (st.control >>> 1) >>> 0;\n\n const end = Math.max(st.destPos - 2, 0);\n this.#updateHashtable(dest, st, end);\n if (st.nextHashed < end) st.nextHashed = end;\n\n return false;\n }\n\n #updateHashtable(dest: Uint8Array, st: QlzState, end: number): void {\n while (st.nextHashed < end) {\n if (st.nextHashed + 3 > dest.length) break;\n const v =\n (dest[st.nextHashed]! |\n (dest[st.nextHashed + 1]! << 8) |\n (dest[st.nextHashed + 2]! << 16)) >>>\n 0;\n const hash = ((v >> 12) ^ v) & 0xfff;\n this.#hashtable[hash] = st.nextHashed;\n st.nextHashed++;\n }\n }\n}\n\nfunction parseQlzHeader(data: Uint8Array): {\n headerLen: number;\n decompressedSize: number;\n flags: number;\n} {\n if (data.length < 3) throw new Error(\"QuickLZ: data too short\");\n\n const flags = data[0]!;\n const level = (flags >> 2) & 0x03;\n if (level !== 1) throw new Error(\"QuickLZ: only level 1 is supported\");\n\n const headerLen = (flags & 0x02) !== 0 ? 9 : 3;\n if (data.length < headerLen) throw new Error(\"QuickLZ: data too short for header\");\n\n let decompressedSize: number;\n if ((flags & 0x02) !== 0) {\n decompressedSize = (data[5]! | (data[6]! << 8) | (data[7]! << 16) | (data[8]! << 24)) >>> 0;\n } else {\n decompressedSize = data[2]!;\n }\n\n return { headerLen, decompressedSize, flags };\n}\n","import { createSocket, type Socket as UdpSocket } from \"node:dgram\";\nimport type { Crypt } from \"../crypto/crypt.js\";\nimport type { Logger } from \"../types.js\";\nimport { noopLogger } from \"../types.js\";\nimport {\n type Packet,\n PacketType,\n PacketFlags,\n packetType,\n packetFlags,\n buildC2SHeader,\n parseS2CHeader,\n} from \"./packet.js\";\nimport { GenerationWindow } from \"./generation-window.js\";\nimport { Qlz } from \"./quicklz.js\";\nimport { processInit1 } from \"../handshake/crypt-handshake.js\";\n\nconst MAX_OUT_PACKET_SIZE = 500;\nconst RECEIVE_PACKET_WINDOW_SIZE = 1024;\nconst PING_INTERVAL_MS = 5_000;\nconst PACKET_TIMEOUT_MS = 60_000;\nconst MAX_RETRY_INTERVAL_MS = 1_000;\nconst UDP_READ_BUFFER_SIZE = 4096;\nconst HEADER_SIZE = 5;\nconst TAG_SIZE = 8;\nconst VOICE_HEADER_SIZE = 3;\nconst RESEND_BASE_INTERVAL_MS = 500;\nconst RESEND_LOOP_INTERVAL_MS = 100;\n\ninterface ResendPacket {\n packet: Packet;\n firstSend: number;\n lastSend: number;\n retryCount: number;\n nextInterval: number;\n}\n\nexport class PacketHandler {\n onPacket: ((p: Packet) => void) | null = null;\n onClosed: ((err: Error | null) => void) | null = null;\n\n readonly #crypt: Crypt;\n readonly #logger: Logger;\n\n #conn: UdpSocket | null = null;\n #clientID = 0;\n #closed = false;\n #stopPing: (() => void) | null = null;\n #stopResend: (() => void) | null = null;\n #lastMessageReceived = Date.now();\n\n #packetCounter = new Uint16Array(9);\n #generationCounter = new Uint32Array(9);\n\n #recvWindowCommand = new GenerationWindow(1 << 16, RECEIVE_PACKET_WINDOW_SIZE);\n #recvWindowCommandLow = new GenerationWindow(1 << 16, RECEIVE_PACKET_WINDOW_SIZE);\n #sendWindowCommand = new GenerationWindow(1 << 16, RECEIVE_PACKET_WINDOW_SIZE);\n #sendWindowCommandLow = new GenerationWindow(1 << 16, RECEIVE_PACKET_WINDOW_SIZE);\n\n #commandQueue = new Map<number, Packet>();\n #commandLowQueue = new Map<number, Packet>();\n #ackManager = new Map<number, ResendPacket>();\n #initPacketCheck: ResendPacket | null = null;\n\n constructor(crypt: Crypt, logger: Logger = noopLogger) {\n this.#crypt = crypt;\n this.#logger = logger;\n }\n\n setClientID(id: number): void {\n this.#clientID = id;\n }\n\n connect(addr: string): Promise<void> {\n return new Promise((resolve, reject) => {\n const [host, portStr] =\n addr.lastIndexOf(\":\") > 0\n ? [addr.slice(0, addr.lastIndexOf(\":\")), addr.slice(addr.lastIndexOf(\":\") + 1)]\n : [addr, \"9987\"];\n const port = parseInt(portStr, 10);\n\n const socket = createSocket(\"udp4\");\n socket.once(\"error\", reject);\n socket.connect(port, host, () => {\n socket.off(\"error\", reject);\n this.start(socket);\n resolve();\n });\n });\n }\n\n start(conn: UdpSocket): void {\n this.#conn = conn;\n this.#closed = false;\n this.#lastMessageReceived = Date.now();\n\n conn.on(\"message\", (msg) => {\n this.#lastMessageReceived = Date.now();\n this.#handleRawPacket(new Uint8Array(msg.buffer, msg.byteOffset, msg.byteLength));\n });\n conn.on(\"error\", (err) => {\n this.#logger.error(\"udp error\", err);\n this.#triggerClose(err);\n });\n conn.on(\"close\", () => this.#triggerClose(null));\n\n this.#packetCounter[PacketType.Command] = 1;\n this.#packetCounter[PacketType.Init1] = 101;\n\n const init1Data = processInit1(this.#crypt, null);\n if (init1Data) this.#sendPacketRaw(PacketType.Init1, init1Data, 0);\n\n // Ping loop\n const pingTimer = setInterval(() => {\n if (this.#crypt.cryptoInitComplete) {\n this.sendPacket(PacketType.Ping, new Uint8Array(0), PacketFlags.Unencrypted);\n }\n }, PING_INTERVAL_MS);\n this.#stopPing = () => clearInterval(pingTimer);\n\n // Resend loop\n const resendTimer = setInterval(() => this.#checkResends(), RESEND_LOOP_INTERVAL_MS);\n this.#stopResend = () => clearInterval(resendTimer);\n }\n\n receivedFinalInitAck(): void {\n this.#initPacketCheck = null;\n }\n\n sendPacket(pType: PacketType, data: Uint8Array, flags: number): void {\n const dummy = !this.#crypt.cryptoInitComplete;\n if (data.length > 487 && pType !== PacketType.Voice && pType !== PacketType.VoiceWhisper) {\n this.#sendSplitPacket(pType, data, flags, dummy);\n return;\n }\n this.#sendPacketRaw(pType, data, flags, dummy);\n }\n\n sendVoicePacket(data: Uint8Array, codec: number): void {\n const pID = this.#packetCounter[PacketType.Voice]!;\n const pGen = this.#generationCounter[PacketType.Voice]!;\n this.#packetCounter[PacketType.Voice] = (pID + 1) & 0xffff;\n if (this.#packetCounter[PacketType.Voice] === 0) {\n this.#generationCounter[PacketType.Voice]!++;\n }\n\n const payloadLen = VOICE_HEADER_SIZE + data.length;\n const voicePayload = new Uint8Array(payloadLen);\n new DataView(voicePayload.buffer).setUint16(0, pID, false);\n voicePayload[2] = codec;\n voicePayload.set(data, VOICE_HEADER_SIZE);\n\n const p: Packet = {\n typeFlagged: PacketType.Voice | PacketFlags.Unencrypted,\n id: pID,\n clientID: this.#clientID,\n generationID: pGen,\n data: voicePayload,\n receivedAt: 0,\n };\n\n const header = buildC2SHeader(p);\n const final = new Uint8Array(TAG_SIZE + HEADER_SIZE + payloadLen);\n final.set(this.#crypt.fakeSignature, 0);\n final.set(header, TAG_SIZE);\n final.set(voicePayload, TAG_SIZE + HEADER_SIZE);\n this.#write(final);\n }\n\n close(): void {\n if (this.#closed) return;\n this.#closed = true;\n this.#stopPing?.();\n this.#stopResend?.();\n this.#conn?.close();\n }\n\n // ---- Private ---------------------------------------------------------------\n\n #sendSplitPacket(pType: PacketType, data: Uint8Array, flags: number, dummy: boolean): void {\n const maxSize = MAX_OUT_PACKET_SIZE - HEADER_SIZE - TAG_SIZE; // 487\n let pos = 0;\n let first = true;\n\n while (pos < data.length) {\n const blockSize = Math.min(data.length - pos, maxSize);\n const last = pos + blockSize === data.length;\n\n let pFlags = flags;\n if (first !== last) pFlags |= PacketFlags.Fragmented;\n\n this.#sendPacketRaw(pType, data.slice(pos, pos + blockSize), pFlags, dummy);\n pos += blockSize;\n first = false;\n }\n }\n\n #sendPacketRaw(\n pType: PacketType,\n data: Uint8Array,\n flags: number,\n dummy = !this.#crypt.cryptoInitComplete,\n ): void {\n flags = applyProtocolFlags(pType, flags);\n const [pID, pGen] = this.#nextPacketIdentity(pType);\n const p: Packet = {\n typeFlagged: pType | flags,\n id: pID,\n clientID: this.#clientID,\n generationID: pGen,\n data,\n receivedAt: 0,\n };\n\n const unencrypted = (flags & PacketFlags.Unencrypted) !== 0;\n const header = buildC2SHeader(p);\n const [ciphertext, tag] = this.#crypt.encrypt(\n pType,\n pID,\n pGen,\n header,\n data,\n dummy,\n unencrypted,\n );\n\n const final = new Uint8Array(TAG_SIZE + HEADER_SIZE + ciphertext.length);\n final.set(tag.slice(0, TAG_SIZE), 0);\n final.set(header, TAG_SIZE);\n final.set(ciphertext, TAG_SIZE + HEADER_SIZE);\n this.#write(final);\n\n const rp: ResendPacket = {\n packet: p,\n firstSend: Date.now(),\n lastSend: Date.now(),\n retryCount: 0,\n nextInterval: RESEND_BASE_INTERVAL_MS,\n };\n this.#trackResend(pType, p, rp);\n }\n\n #write(data: Uint8Array): void {\n this.#conn?.send(Buffer.from(data), (err) => {\n if (err) this.#logger.warn(\"udp send error\", err);\n });\n }\n\n #handleRawPacket(raw: Uint8Array): void {\n if (raw.length < 11) return;\n\n const tag = raw.slice(0, TAG_SIZE);\n const header = raw.slice(TAG_SIZE, TAG_SIZE + 3);\n const ciphertext = raw.slice(TAG_SIZE + 3);\n\n const parsed = parseS2CHeader(header);\n const p: Packet = {\n ...parsed,\n clientID: 0,\n generationID: this.#resolveGeneration(parsed.id, parsed.typeFlagged & 0x0f),\n data: new Uint8Array(0),\n receivedAt: Date.now(),\n };\n\n const decrypted = this.#decryptPacketData(p, header, ciphertext, tag);\n if (decrypted === null) return;\n p.data = decrypted.plaintext;\n\n const pType = packetType(p);\n\n if (pType === PacketType.Ping) {\n this.#sendPong(p.id, decrypted.dummyUsed);\n return;\n }\n\n if (!this.#handleCommandWindowAndAck(p, decrypted.dummyUsed)) return;\n\n this.#handlePacketQueue(p);\n this.#updatePostReceiveState(p);\n }\n\n #resolveGeneration(id: number, pType: number): number {\n switch (pType as PacketType) {\n case PacketType.Command:\n return this.#recvWindowCommand.getGeneration(id);\n case PacketType.CommandLow:\n return this.#recvWindowCommandLow.getGeneration(id);\n case PacketType.Ack:\n return this.#sendWindowCommand.getGeneration(id);\n case PacketType.AckLow:\n return this.#sendWindowCommandLow.getGeneration(id);\n default:\n return 0;\n }\n }\n\n #decryptPacketData(\n p: Packet,\n header: Uint8Array,\n ciphertext: Uint8Array,\n tag: Uint8Array,\n ): { plaintext: Uint8Array; dummyUsed: boolean } | null {\n const unencrypted = (packetFlags(p) & PacketFlags.Unencrypted) !== 0;\n const dummy = !this.#crypt.cryptoInitComplete;\n let dummyUsed = dummy;\n const pType = packetType(p);\n const gen = p.generationID;\n\n try {\n const plaintext = this.#crypt.decrypt(\n pType,\n p.id,\n gen,\n header,\n ciphertext,\n tag,\n dummy,\n unencrypted,\n );\n return { plaintext, dummyUsed };\n } catch {\n // Try adjacent generations\n for (const offset of [-1, 1]) {\n const guessGen = gen + offset;\n if (guessGen < 0) continue;\n try {\n const plaintext = this.#crypt.decrypt(\n pType,\n p.id,\n guessGen,\n header,\n ciphertext,\n tag,\n false,\n false,\n );\n return { plaintext, dummyUsed: false };\n } catch {\n // continue\n }\n }\n\n // Try dummy fallback for command/ack types\n if (\n pType === PacketType.Command ||\n pType === PacketType.CommandLow ||\n pType === PacketType.Ack\n ) {\n try {\n const plaintext = this.#crypt.decrypt(\n pType,\n p.id,\n gen,\n header,\n ciphertext,\n tag,\n true,\n unencrypted,\n );\n return { plaintext, dummyUsed: true };\n } catch {\n // fall through\n }\n }\n\n return null;\n }\n }\n\n #handleCommandWindowAndAck(p: Packet, dummyUsed: boolean): boolean {\n const pType = packetType(p);\n if (pType !== PacketType.Command && pType !== PacketType.CommandLow) return true;\n\n const win = pType === PacketType.Command ? this.#recvWindowCommand : this.#recvWindowCommandLow;\n const ackType = pType === PacketType.Command ? PacketType.Ack : PacketType.AckLow;\n\n if (!win.isInWindow(p.id)) {\n if (win.isOldPacket(p.id)) {\n this.#sendAck(p.id, ackType, dummyUsed);\n }\n return false;\n }\n this.#sendAck(p.id, ackType, dummyUsed);\n return true;\n }\n\n #sendAck(packetID: number, ackType: PacketType, dummyUsed: boolean): void {\n const ackData = new Uint8Array(2);\n new DataView(ackData.buffer).setUint16(0, packetID, false);\n this.#sendPacketRaw(ackType, ackData, 0, dummyUsed);\n }\n\n #sendPong(pID: number, dummyUsed: boolean): void {\n const pongData = new Uint8Array(2);\n new DataView(pongData.buffer).setUint16(0, pID, false);\n this.#sendPacketRaw(PacketType.Pong, pongData, PacketFlags.Unencrypted, dummyUsed);\n }\n\n #handlePacketQueue(p: Packet): void {\n const pType = packetType(p);\n if (pType !== PacketType.Command && pType !== PacketType.CommandLow) {\n this.onPacket?.(p);\n return;\n }\n\n const isCommand = pType === PacketType.Command;\n const queue = isCommand ? this.#commandQueue : this.#commandLowQueue;\n const win = isCommand ? this.#recvWindowCommand : this.#recvWindowCommandLow;\n\n queue.set(p.id, p);\n\n if (isCommand) {\n this.#fastForwardMissingPackets(queue, win, true);\n } else {\n this.#fastForwardMissingPackets(queue, win, false);\n }\n\n while (true) {\n const current = isCommand ? this.#_nextCommandID : this.#_nextCommandLowID;\n const packet = queue.get(current);\n if (packet === undefined) break;\n\n const result = this.#tryReassemble(packet, queue, current, win);\n if (!result) break;\n\n const [reassembled, newNext] = result;\n if (isCommand) this.#_nextCommandID = newNext;\n else this.#_nextCommandLowID = newNext;\n\n this.#tryDecompress(reassembled);\n this.onPacket?.(reassembled);\n }\n }\n\n #_nextCommandID = 0;\n #_nextCommandLowID = 0;\n\n #fastForwardMissingPackets(\n queue: Map<number, Packet>,\n win: GenerationWindow,\n isCommand: boolean,\n ): void {\n let nextID = isCommand ? this.#_nextCommandID : this.#_nextCommandLowID;\n while (!queue.has(nextID) && hasOldNewerPacket(queue, nextID)) {\n nextID = (nextID + 1) & 0xffff;\n win.advance(1);\n }\n if (isCommand) this.#_nextCommandID = nextID;\n else this.#_nextCommandLowID = nextID;\n }\n\n #tryReassemble(\n startPacket: Packet,\n queue: Map<number, Packet>,\n nextID: number,\n win: GenerationWindow,\n ): [reassembled: Packet, newNextID: number] | null {\n if ((packetFlags(startPacket) & PacketFlags.Fragmented) === 0) {\n queue.delete(nextID);\n win.advance(1);\n return [startPacket, (nextID + 1) & 0xffff];\n }\n\n const fragments: Packet[] = [];\n let totalSize = 0;\n let currID = nextID;\n let startSeen = false;\n let complete = false;\n\n while (true) {\n const frag = queue.get(currID);\n if (!frag) return null;\n fragments.push(frag);\n totalSize += frag.data.length;\n\n const fragmented = (packetFlags(frag) & PacketFlags.Fragmented) !== 0;\n if (!startSeen) {\n startSeen = true;\n if (!fragmented) {\n complete = true;\n break;\n }\n } else if (fragmented) {\n complete = true;\n break;\n }\n currID = (currID + 1) & 0xffff;\n }\n\n if (!complete) return null;\n\n const combined = new Uint8Array(totalSize);\n let pos = 0;\n for (const frag of fragments) {\n combined.set(frag.data, pos);\n pos += frag.data.length;\n queue.delete(nextID);\n win.advance(1);\n nextID = (nextID + 1) & 0xffff;\n }\n\n startPacket.data = combined;\n startPacket.typeFlagged &= ~PacketFlags.Fragmented;\n return [startPacket, nextID];\n }\n\n #tryDecompress(packet: Packet): void {\n if ((packetFlags(packet) & PacketFlags.Compressed) === 0) return;\n try {\n const qlz = new Qlz();\n packet.data = qlz.decompress(packet.data);\n packet.typeFlagged &= ~PacketFlags.Compressed;\n } catch (err) {\n this.#logger.debug(\"decompression failed\", { id: packet.id, err });\n }\n }\n\n #updatePostReceiveState(p: Packet): void {\n const pType = packetType(p);\n\n if (pType === PacketType.Init1) {\n this.#initPacketCheck = null;\n return;\n }\n\n if ((pType === PacketType.Ack || pType === PacketType.AckLow) && p.data.length >= 2) {\n const ackID = new DataView(p.data.buffer, p.data.byteOffset).getUint16(0, false);\n const targetType = pType === PacketType.Ack ? PacketType.Command : PacketType.CommandLow;\n const key = (targetType << 16) | ackID;\n this.#ackManager.delete(key);\n }\n }\n\n #nextPacketIdentity(pType: PacketType): [id: number, gen: number] {\n const pID = this.#packetCounter[pType]!;\n const pGen = this.#generationCounter[pType]!;\n\n if (pType === PacketType.Init1) return [pID, pGen];\n\n this.#packetCounter[pType] = (pID + 1) & 0xffff;\n if (this.#packetCounter[pType] === 0) {\n this.#generationCounter[pType] = (pGen + 1) >>> 0;\n }\n\n if (pType === PacketType.Command) {\n this.#sendWindowCommand.advanceToExcluded(pID);\n } else if (pType === PacketType.CommandLow) {\n this.#sendWindowCommandLow.advanceToExcluded(pID);\n }\n\n return [pID, pGen];\n }\n\n #trackResend(pType: PacketType, p: Packet, rp: ResendPacket): void {\n if (pType === PacketType.Init1) {\n this.#initPacketCheck = rp;\n return;\n }\n if (pType === PacketType.Command || pType === PacketType.CommandLow) {\n const key = (pType << 16) | p.id;\n this.#ackManager.set(key, rp);\n }\n }\n\n #checkResends(): void {\n const now = Date.now();\n\n if (now - this.#lastMessageReceived > PACKET_TIMEOUT_MS) {\n this.#logger.warn(\"idle timeout\");\n this.#triggerClose(new Error(\"idle timeout\"));\n return;\n }\n\n if (this.#initPacketCheck) {\n this.#doResend(this.#initPacketCheck, now);\n }\n\n for (const [key, rp] of this.#ackManager) {\n if (now - rp.firstSend > PACKET_TIMEOUT_MS) {\n this.#ackManager.delete(key);\n this.#triggerClose(new Error(\"packet ack timeout\"));\n return;\n }\n this.#doResend(rp, now);\n }\n }\n\n #doResend(rp: ResendPacket, now: number): void {\n if (now - rp.lastSend < rp.nextInterval) return;\n\n rp.lastSend = now;\n rp.retryCount++;\n rp.nextInterval = Math.min(rp.nextInterval * 2, MAX_RETRY_INTERVAL_MS);\n\n const dummy = !this.#crypt.cryptoInitComplete;\n const unencrypted = (packetFlags(rp.packet) & PacketFlags.Unencrypted) !== 0;\n const header = buildC2SHeader(rp.packet);\n const pType = packetType(rp.packet);\n\n const [ciphertext, tag] = this.#crypt.encrypt(\n pType,\n rp.packet.id,\n rp.packet.generationID,\n header,\n rp.packet.data,\n dummy,\n unencrypted,\n );\n\n const final = new Uint8Array(TAG_SIZE + HEADER_SIZE + ciphertext.length);\n final.set(tag.slice(0, TAG_SIZE), 0);\n final.set(header, TAG_SIZE);\n final.set(ciphertext, TAG_SIZE + HEADER_SIZE);\n this.#write(final);\n }\n\n #triggerClose(err: Error | null): void {\n if (this.#closed) return;\n this.close();\n this.onClosed?.(err);\n }\n}\n\n// ---- Helpers ----------------------------------------------------------------\n\nfunction applyProtocolFlags(pType: PacketType, flags: number): number {\n if (pType === PacketType.Command || pType === PacketType.CommandLow) {\n return flags | PacketFlags.NewProtocol;\n }\n return flags;\n}\n\nfunction hasOldNewerPacket(queue: Map<number, Packet>, nextID: number): boolean {\n const now = Date.now();\n for (const [id, pkg] of queue) {\n const diff = (id - nextID + 0x10000) & 0xffff;\n if (diff < 0x8000 && now - pkg.receivedAt > 5_000) {\n return true;\n }\n }\n return false;\n}\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"handler-CqCD93f0.cjs","names":["#mod","#receiveWindow","#generation","#mappedBaseOffset","#isNextGen","#hashtable","#ensureControl","#processReference","#processLiteral","#updateHashtable","#crypt","#logger","#clientID","#conn","#closed","#lastMessageReceived","#handleRawPacket","#triggerClose","#packetCounter","#sendPacketRaw","#stopPing","#checkResends","#stopResend","#initPacketCheck","#sendSplitPacket","#generationCounter","#write","#nextPacketIdentity","#trackResend","#resolveGeneration","#decryptPacketData","#sendPong","#handleCommandWindowAndAck","#handlePacketQueue","#updatePostReceiveState","#recvWindowCommand","#recvWindowCommandLow","#sendWindowCommand","#sendWindowCommandLow","#sendAck","#commandQueue","#commandLowQueue","#fastForwardMissingPackets","#_nextCommandID","#_nextCommandLowID","#tryReassemble","#tryDecompress","#ackManager","#doResend"],"sources":["../src/transport/packet.ts","../src/transport/generation-window.ts","../src/transport/quicklz.ts","../src/transport/handler.ts"],"sourcesContent":["export const enum PacketType {\n Voice = 0,\n VoiceWhisper = 1,\n Command = 2,\n CommandLow = 3,\n Ping = 4,\n Pong = 5,\n Ack = 6,\n AckLow = 7,\n Init1 = 8,\n}\n\nexport const enum PacketFlags {\n Fragmented = 0x10,\n NewProtocol = 0x20,\n Compressed = 0x40,\n Unencrypted = 0x80,\n}\n\nexport interface Packet {\n /** Type byte combined with flags (low nibble = type, high nibble = flags). */\n typeFlagged: number;\n id: number;\n clientID: number;\n generationID: number;\n data: Uint8Array;\n receivedAt: number; // Date.now()\n}\n\nexport function packetType(p: Packet): PacketType {\n return (p.typeFlagged & 0x0f) as PacketType;\n}\n\nexport function packetFlags(p: Packet): number {\n return p.typeFlagged & 0xf0;\n}\n\nexport function isUnencrypted(p: Packet): boolean {\n return (packetFlags(p) & PacketFlags.Unencrypted) !== 0;\n}\n\n/** Build the 5-byte client-to-server header: [packetID(2), clientID(2), typeFlagged(1)]. */\nexport function buildC2SHeader(p: Packet): Uint8Array {\n const header = new Uint8Array(5);\n const view = new DataView(header.buffer);\n view.setUint16(0, p.id, false);\n view.setUint16(2, p.clientID, false);\n header[4] = p.typeFlagged;\n return header;\n}\n\n/** Parse a 3-byte server-to-client header. */\nexport function parseS2CHeader(raw: Uint8Array): Pick<Packet, \"id\" | \"typeFlagged\"> {\n const view = new DataView(raw.buffer, raw.byteOffset, raw.byteLength);\n return {\n id: view.getUint16(0, false),\n typeFlagged: raw[2]!,\n };\n}\n\n/** Parse a 5-byte client-to-server header. */\nexport function parseC2SHeader(raw: Uint8Array): Pick<Packet, \"id\" | \"clientID\" | \"typeFlagged\"> {\n const view = new DataView(raw.buffer, raw.byteOffset, raw.byteLength);\n return {\n id: view.getUint16(0, false),\n clientID: view.getUint16(2, false),\n typeFlagged: raw[4]!,\n };\n}\n","export class GenerationWindow {\n #mappedBaseOffset = 0;\n #generation = 0;\n readonly #mod: number;\n readonly #receiveWindow: number;\n\n constructor(mod: number, windowSize: number) {\n this.#mod = mod;\n this.#receiveWindow = windowSize;\n }\n\n get generation(): number {\n return this.#generation;\n }\n\n advance(amount: number): void {\n if (amount <= 0) return;\n const newBaseOffset = this.#mappedBaseOffset + amount;\n const genStep = Math.floor(newBaseOffset / this.#mod);\n if (genStep > 0) {\n this.#generation = Math.min(this.#generation + genStep, 0xffff_ffff);\n }\n this.#mappedBaseOffset = newBaseOffset % this.#mod;\n }\n\n advanceToExcluded(mappedValue: number): void {\n let moveDist = mappedValue - this.#mappedBaseOffset;\n if (moveDist < 0) moveDist += this.#mod;\n this.advance(moveDist + 1);\n }\n\n syncTo(mappedValue: number): void {\n let moveDist = mappedValue - this.#mappedBaseOffset;\n if (moveDist < 0) moveDist += this.#mod;\n this.advance(moveDist);\n }\n\n isInWindow(mappedValue: number): boolean {\n const maxOffset = this.#mappedBaseOffset + this.#receiveWindow;\n if (maxOffset < this.#mod) {\n return mappedValue >= this.#mappedBaseOffset && mappedValue < maxOffset;\n }\n return mappedValue >= this.#mappedBaseOffset || mappedValue < maxOffset - this.#mod;\n }\n\n mappedToIndex(mappedValue: number): number {\n if (this.#isNextGen(mappedValue)) {\n return mappedValue + this.#mod - this.#mappedBaseOffset;\n }\n return mappedValue - this.#mappedBaseOffset;\n }\n\n isOldPacket(mappedValue: number): boolean {\n return this.mappedToIndex(mappedValue) < 0;\n }\n\n isFuturePacket(mappedValue: number): boolean {\n return this.mappedToIndex(mappedValue) >= this.#receiveWindow;\n }\n\n #isNextGen(mappedValue: number): boolean {\n return (\n this.#mappedBaseOffset > this.#mod - this.#receiveWindow &&\n mappedValue < this.#mappedBaseOffset + this.#receiveWindow - this.#mod\n );\n }\n\n getGeneration(mappedValue: number): number {\n if (this.#isNextGen(mappedValue)) return this.#generation + 1;\n return this.#generation;\n }\n\n reset(): void {\n this.#mappedBaseOffset = 0;\n this.#generation = 0;\n }\n}\n","const TABLE_SIZE = 4096;\n\ninterface QlzState {\n control: number;\n sourcePos: number;\n destPos: number;\n nextHashed: number;\n}\n\nexport class Qlz {\n #hashtable = new Int32Array(TABLE_SIZE);\n\n decompress(data: Uint8Array): Uint8Array {\n const { headerLen, decompressedSize, flags } = parseQlzHeader(data);\n\n const dest = new Uint8Array(decompressedSize);\n\n if ((flags & 0x01) === 0) {\n dest.set(data.slice(headerLen, headerLen + decompressedSize));\n return dest;\n }\n\n this.#hashtable.fill(0);\n\n const state: QlzState = {\n control: 1,\n sourcePos: headerLen,\n destPos: 0,\n nextHashed: 0,\n };\n\n while (this.#ensureControl(data, state)) {\n if ((state.control & 1) !== 0) {\n if (!this.#processReference(data, dest, state)) break;\n } else {\n if (this.#processLiteral(data, dest, decompressedSize, state)) break;\n }\n }\n\n return dest;\n }\n\n #ensureControl(data: Uint8Array, st: QlzState): boolean {\n if (st.control !== 1) return true;\n if (st.sourcePos + 4 > data.length) return false;\n st.control =\n (data[st.sourcePos]! |\n (data[st.sourcePos + 1]! << 8) |\n (data[st.sourcePos + 2]! << 16) |\n (data[st.sourcePos + 3]! << 24)) >>>\n 0;\n st.sourcePos += 4;\n return true;\n }\n\n #processReference(data: Uint8Array, dest: Uint8Array, st: QlzState): boolean {\n st.control = (st.control >>> 1) >>> 0;\n if (st.sourcePos + 2 > data.length) return false;\n\n const b1 = data[st.sourcePos]!;\n const b2 = data[st.sourcePos + 1]!;\n st.sourcePos += 2;\n\n const hash = (b1 >> 4) | (b2 << 4);\n let matchlen = b1 & 0x0f;\n if (matchlen !== 0) {\n matchlen += 2;\n } else {\n if (st.sourcePos >= data.length) return false;\n matchlen = data[st.sourcePos]!;\n st.sourcePos++;\n }\n\n const offset = this.#hashtable[hash]!;\n for (let i = 0; i < matchlen; i++) {\n if (st.destPos < dest.length && offset + i < st.destPos) {\n dest[st.destPos] = dest[offset + i]!;\n st.destPos++;\n }\n }\n\n const end = st.destPos + 1 - matchlen;\n this.#updateHashtable(dest, st, end);\n st.nextHashed = st.destPos;\n\n return true;\n }\n\n #processLiteral(\n data: Uint8Array,\n dest: Uint8Array,\n decompressedSize: number,\n st: QlzState,\n ): boolean {\n const threshold = Math.max(decompressedSize, 10) - 10;\n if (st.destPos >= threshold) {\n while (st.destPos < decompressedSize) {\n if (st.control === 1) {\n st.sourcePos += 4;\n if (st.sourcePos > data.length) break;\n st.control =\n (data[st.sourcePos - 4]! |\n (data[st.sourcePos - 3]! << 8) |\n (data[st.sourcePos - 2]! << 16) |\n (data[st.sourcePos - 1]! << 24)) >>>\n 0;\n }\n if (st.sourcePos >= data.length) break;\n dest[st.destPos++] = data[st.sourcePos++]!;\n st.control = (st.control >>> 1) >>> 0;\n }\n return true;\n }\n\n if (st.sourcePos >= data.length || st.destPos >= dest.length) return true;\n\n dest[st.destPos++] = data[st.sourcePos++]!;\n st.control = (st.control >>> 1) >>> 0;\n\n const end = Math.max(st.destPos - 2, 0);\n this.#updateHashtable(dest, st, end);\n if (st.nextHashed < end) st.nextHashed = end;\n\n return false;\n }\n\n #updateHashtable(dest: Uint8Array, st: QlzState, end: number): void {\n while (st.nextHashed < end) {\n if (st.nextHashed + 3 > dest.length) break;\n const v =\n (dest[st.nextHashed]! |\n (dest[st.nextHashed + 1]! << 8) |\n (dest[st.nextHashed + 2]! << 16)) >>>\n 0;\n const hash = ((v >> 12) ^ v) & 0xfff;\n this.#hashtable[hash] = st.nextHashed;\n st.nextHashed++;\n }\n }\n}\n\nfunction parseQlzHeader(data: Uint8Array): {\n headerLen: number;\n decompressedSize: number;\n flags: number;\n} {\n if (data.length < 3) throw new Error(\"QuickLZ: data too short\");\n\n const flags = data[0]!;\n const level = (flags >> 2) & 0x03;\n if (level !== 1) throw new Error(\"QuickLZ: only level 1 is supported\");\n\n const headerLen = (flags & 0x02) !== 0 ? 9 : 3;\n if (data.length < headerLen) throw new Error(\"QuickLZ: data too short for header\");\n\n let decompressedSize: number;\n if ((flags & 0x02) !== 0) {\n decompressedSize = (data[5]! | (data[6]! << 8) | (data[7]! << 16) | (data[8]! << 24)) >>> 0;\n } else {\n decompressedSize = data[2]!;\n }\n\n return { headerLen, decompressedSize, flags };\n}\n","import { createSocket, type Socket as UdpSocket } from \"node:dgram\";\nimport type { Crypt } from \"../crypto/crypt.js\";\nimport type { Logger } from \"../types.js\";\nimport { noopLogger } from \"../types.js\";\nimport {\n type Packet,\n PacketType,\n PacketFlags,\n packetType,\n packetFlags,\n buildC2SHeader,\n parseS2CHeader,\n} from \"./packet.js\";\nimport { GenerationWindow } from \"./generation-window.js\";\nimport { Qlz } from \"./quicklz.js\";\nimport { processInit1 } from \"../handshake/crypt-handshake.js\";\n\nconst MAX_OUT_PACKET_SIZE = 500;\nconst RECEIVE_PACKET_WINDOW_SIZE = 1024;\nconst PING_INTERVAL_MS = 5_000;\nconst PACKET_TIMEOUT_MS = 60_000;\nconst MAX_RETRY_INTERVAL_MS = 1_000;\nconst HEADER_SIZE = 5;\nconst TAG_SIZE = 8;\nconst VOICE_HEADER_SIZE = 3;\nconst RESEND_BASE_INTERVAL_MS = 500;\nconst RESEND_LOOP_INTERVAL_MS = 100;\n\ninterface ResendPacket {\n packet: Packet;\n firstSend: number;\n lastSend: number;\n retryCount: number;\n nextInterval: number;\n}\n\nexport class PacketHandler {\n onPacket: ((p: Packet) => void) | null = null;\n onClosed: ((err: Error | null) => void) | null = null;\n\n readonly #crypt: Crypt;\n readonly #logger: Logger;\n\n #conn: UdpSocket | null = null;\n #clientID = 0;\n #closed = false;\n #stopPing: (() => void) | null = null;\n #stopResend: (() => void) | null = null;\n #lastMessageReceived = Date.now();\n\n #packetCounter = new Uint16Array(9);\n #generationCounter = new Uint32Array(9);\n\n #recvWindowCommand = new GenerationWindow(1 << 16, RECEIVE_PACKET_WINDOW_SIZE);\n #recvWindowCommandLow = new GenerationWindow(1 << 16, RECEIVE_PACKET_WINDOW_SIZE);\n #sendWindowCommand = new GenerationWindow(1 << 16, RECEIVE_PACKET_WINDOW_SIZE);\n #sendWindowCommandLow = new GenerationWindow(1 << 16, RECEIVE_PACKET_WINDOW_SIZE);\n\n #commandQueue = new Map<number, Packet>();\n #commandLowQueue = new Map<number, Packet>();\n #ackManager = new Map<number, ResendPacket>();\n #initPacketCheck: ResendPacket | null = null;\n\n constructor(crypt: Crypt, logger: Logger = noopLogger) {\n this.#crypt = crypt;\n this.#logger = logger;\n }\n\n setClientID(id: number): void {\n this.#clientID = id;\n }\n\n connect(addr: string): Promise<void> {\n return new Promise((resolve, reject) => {\n const [host, portStr] =\n addr.lastIndexOf(\":\") > 0\n ? [addr.slice(0, addr.lastIndexOf(\":\")), addr.slice(addr.lastIndexOf(\":\") + 1)]\n : [addr, \"9987\"];\n const port = parseInt(portStr, 10);\n\n const socket = createSocket(\"udp4\");\n socket.once(\"error\", reject);\n socket.connect(port, host, () => {\n socket.off(\"error\", reject);\n this.start(socket);\n resolve();\n });\n });\n }\n\n start(conn: UdpSocket): void {\n this.#conn = conn;\n this.#closed = false;\n this.#lastMessageReceived = Date.now();\n\n conn.on(\"message\", (msg) => {\n this.#lastMessageReceived = Date.now();\n this.#handleRawPacket(new Uint8Array(msg.buffer, msg.byteOffset, msg.byteLength));\n });\n conn.on(\"error\", (err) => {\n this.#logger.error(\"udp error\", err);\n this.#triggerClose(err);\n });\n conn.on(\"close\", () => this.#triggerClose(null));\n\n this.#packetCounter[PacketType.Command] = 1;\n this.#packetCounter[PacketType.Init1] = 101;\n\n const init1Data = processInit1(this.#crypt, null);\n if (init1Data) this.#sendPacketRaw(PacketType.Init1, init1Data, 0);\n\n // Ping loop\n const pingTimer = setInterval(() => {\n if (this.#crypt.cryptoInitComplete) {\n this.sendPacket(PacketType.Ping, new Uint8Array(0), PacketFlags.Unencrypted);\n }\n }, PING_INTERVAL_MS);\n this.#stopPing = () => clearInterval(pingTimer);\n\n // Resend loop\n const resendTimer = setInterval(() => this.#checkResends(), RESEND_LOOP_INTERVAL_MS);\n this.#stopResend = () => clearInterval(resendTimer);\n }\n\n receivedFinalInitAck(): void {\n this.#initPacketCheck = null;\n }\n\n sendPacket(pType: PacketType, data: Uint8Array, flags: number): void {\n const dummy = !this.#crypt.cryptoInitComplete;\n if (data.length > 487 && pType !== PacketType.Voice && pType !== PacketType.VoiceWhisper) {\n this.#sendSplitPacket(pType, data, flags, dummy);\n return;\n }\n this.#sendPacketRaw(pType, data, flags, dummy);\n }\n\n sendVoicePacket(data: Uint8Array, codec: number): void {\n const pID = this.#packetCounter[PacketType.Voice]!;\n const pGen = this.#generationCounter[PacketType.Voice]!;\n this.#packetCounter[PacketType.Voice] = (pID + 1) & 0xffff;\n if (this.#packetCounter[PacketType.Voice] === 0) {\n this.#generationCounter[PacketType.Voice]!++;\n }\n\n const payloadLen = VOICE_HEADER_SIZE + data.length;\n const voicePayload = new Uint8Array(payloadLen);\n new DataView(voicePayload.buffer).setUint16(0, pID, false);\n voicePayload[2] = codec;\n voicePayload.set(data, VOICE_HEADER_SIZE);\n\n const p: Packet = {\n typeFlagged: PacketType.Voice | PacketFlags.Unencrypted,\n id: pID,\n clientID: this.#clientID,\n generationID: pGen,\n data: voicePayload,\n receivedAt: 0,\n };\n\n const header = buildC2SHeader(p);\n const final = new Uint8Array(TAG_SIZE + HEADER_SIZE + payloadLen);\n final.set(this.#crypt.fakeSignature, 0);\n final.set(header, TAG_SIZE);\n final.set(voicePayload, TAG_SIZE + HEADER_SIZE);\n this.#write(final);\n }\n\n close(): void {\n if (this.#closed) return;\n this.#closed = true;\n this.#stopPing?.();\n this.#stopResend?.();\n this.#conn?.close();\n }\n\n // ---- Private ---------------------------------------------------------------\n\n #sendSplitPacket(pType: PacketType, data: Uint8Array, flags: number, dummy: boolean): void {\n const maxSize = MAX_OUT_PACKET_SIZE - HEADER_SIZE - TAG_SIZE; // 487\n let pos = 0;\n let first = true;\n\n while (pos < data.length) {\n const blockSize = Math.min(data.length - pos, maxSize);\n const last = pos + blockSize === data.length;\n\n let pFlags = flags;\n if (first !== last) pFlags |= PacketFlags.Fragmented;\n\n this.#sendPacketRaw(pType, data.slice(pos, pos + blockSize), pFlags, dummy);\n pos += blockSize;\n first = false;\n }\n }\n\n #sendPacketRaw(\n pType: PacketType,\n data: Uint8Array,\n flags: number,\n dummy = !this.#crypt.cryptoInitComplete,\n ): void {\n flags = applyProtocolFlags(pType, flags);\n const [pID, pGen] = this.#nextPacketIdentity(pType);\n const p: Packet = {\n typeFlagged: pType | flags,\n id: pID,\n clientID: this.#clientID,\n generationID: pGen,\n data,\n receivedAt: 0,\n };\n\n const unencrypted = (flags & PacketFlags.Unencrypted) !== 0;\n const header = buildC2SHeader(p);\n const [ciphertext, tag] = this.#crypt.encrypt(\n pType,\n pID,\n pGen,\n header,\n data,\n dummy,\n unencrypted,\n );\n\n const final = new Uint8Array(TAG_SIZE + HEADER_SIZE + ciphertext.length);\n final.set(tag.slice(0, TAG_SIZE), 0);\n final.set(header, TAG_SIZE);\n final.set(ciphertext, TAG_SIZE + HEADER_SIZE);\n this.#write(final);\n\n const rp: ResendPacket = {\n packet: p,\n firstSend: Date.now(),\n lastSend: Date.now(),\n retryCount: 0,\n nextInterval: RESEND_BASE_INTERVAL_MS,\n };\n this.#trackResend(pType, p, rp);\n }\n\n #write(data: Uint8Array): void {\n this.#conn?.send(Buffer.from(data), (err) => {\n if (err) this.#logger.warn(\"udp send error\", err);\n });\n }\n\n #handleRawPacket(raw: Uint8Array): void {\n if (raw.length < 11) return;\n\n const tag = raw.slice(0, TAG_SIZE);\n const header = raw.slice(TAG_SIZE, TAG_SIZE + 3);\n const ciphertext = raw.slice(TAG_SIZE + 3);\n\n const parsed = parseS2CHeader(header);\n const p: Packet = {\n ...parsed,\n clientID: 0,\n generationID: this.#resolveGeneration(parsed.id, parsed.typeFlagged & 0x0f),\n data: new Uint8Array(0),\n receivedAt: Date.now(),\n };\n\n const decrypted = this.#decryptPacketData(p, header, ciphertext, tag);\n if (decrypted === null) return;\n p.data = decrypted.plaintext;\n\n const pType = packetType(p);\n\n if (pType === PacketType.Ping) {\n this.#sendPong(p.id, decrypted.dummyUsed);\n return;\n }\n\n if (!this.#handleCommandWindowAndAck(p, decrypted.dummyUsed)) return;\n\n this.#handlePacketQueue(p);\n this.#updatePostReceiveState(p);\n }\n\n #resolveGeneration(id: number, pType: number): number {\n switch (pType as PacketType) {\n case PacketType.Command:\n return this.#recvWindowCommand.getGeneration(id);\n case PacketType.CommandLow:\n return this.#recvWindowCommandLow.getGeneration(id);\n case PacketType.Ack:\n return this.#sendWindowCommand.getGeneration(id);\n case PacketType.AckLow:\n return this.#sendWindowCommandLow.getGeneration(id);\n default:\n return 0;\n }\n }\n\n #decryptPacketData(\n p: Packet,\n header: Uint8Array,\n ciphertext: Uint8Array,\n tag: Uint8Array,\n ): { plaintext: Uint8Array; dummyUsed: boolean } | null {\n const unencrypted = (packetFlags(p) & PacketFlags.Unencrypted) !== 0;\n const dummy = !this.#crypt.cryptoInitComplete;\n let dummyUsed = dummy;\n const pType = packetType(p);\n const gen = p.generationID;\n\n try {\n const plaintext = this.#crypt.decrypt(\n pType,\n p.id,\n gen,\n header,\n ciphertext,\n tag,\n dummy,\n unencrypted,\n );\n return { plaintext, dummyUsed };\n } catch {\n // Try adjacent generations\n for (const offset of [-1, 1]) {\n const guessGen = gen + offset;\n if (guessGen < 0) continue;\n try {\n const plaintext = this.#crypt.decrypt(\n pType,\n p.id,\n guessGen,\n header,\n ciphertext,\n tag,\n false,\n false,\n );\n return { plaintext, dummyUsed: false };\n } catch {\n // continue\n }\n }\n\n // Try dummy fallback for command/ack types\n if (\n pType === PacketType.Command ||\n pType === PacketType.CommandLow ||\n pType === PacketType.Ack\n ) {\n try {\n const plaintext = this.#crypt.decrypt(\n pType,\n p.id,\n gen,\n header,\n ciphertext,\n tag,\n true,\n unencrypted,\n );\n return { plaintext, dummyUsed: true };\n } catch {\n // fall through\n }\n }\n\n return null;\n }\n }\n\n #handleCommandWindowAndAck(p: Packet, dummyUsed: boolean): boolean {\n const pType = packetType(p);\n if (pType !== PacketType.Command && pType !== PacketType.CommandLow) return true;\n\n const win = pType === PacketType.Command ? this.#recvWindowCommand : this.#recvWindowCommandLow;\n const ackType = pType === PacketType.Command ? PacketType.Ack : PacketType.AckLow;\n\n if (!win.isInWindow(p.id)) {\n if (win.isOldPacket(p.id)) {\n this.#sendAck(p.id, ackType, dummyUsed);\n }\n return false;\n }\n this.#sendAck(p.id, ackType, dummyUsed);\n return true;\n }\n\n #sendAck(packetID: number, ackType: PacketType, dummyUsed: boolean): void {\n const ackData = new Uint8Array(2);\n new DataView(ackData.buffer).setUint16(0, packetID, false);\n this.#sendPacketRaw(ackType, ackData, 0, dummyUsed);\n }\n\n #sendPong(pID: number, dummyUsed: boolean): void {\n const pongData = new Uint8Array(2);\n new DataView(pongData.buffer).setUint16(0, pID, false);\n this.#sendPacketRaw(PacketType.Pong, pongData, PacketFlags.Unencrypted, dummyUsed);\n }\n\n #handlePacketQueue(p: Packet): void {\n const pType = packetType(p);\n if (pType !== PacketType.Command && pType !== PacketType.CommandLow) {\n this.onPacket?.(p);\n return;\n }\n\n const isCommand = pType === PacketType.Command;\n const queue = isCommand ? this.#commandQueue : this.#commandLowQueue;\n const win = isCommand ? this.#recvWindowCommand : this.#recvWindowCommandLow;\n\n queue.set(p.id, p);\n\n if (isCommand) {\n this.#fastForwardMissingPackets(queue, win, true);\n } else {\n this.#fastForwardMissingPackets(queue, win, false);\n }\n\n while (true) {\n const current = isCommand ? this.#_nextCommandID : this.#_nextCommandLowID;\n const packet = queue.get(current);\n if (packet === undefined) break;\n\n const result = this.#tryReassemble(packet, queue, current, win);\n if (!result) break;\n\n const [reassembled, newNext] = result;\n if (isCommand) this.#_nextCommandID = newNext;\n else this.#_nextCommandLowID = newNext;\n\n this.#tryDecompress(reassembled);\n this.onPacket?.(reassembled);\n }\n }\n\n #_nextCommandID = 0;\n #_nextCommandLowID = 0;\n\n #fastForwardMissingPackets(\n queue: Map<number, Packet>,\n win: GenerationWindow,\n isCommand: boolean,\n ): void {\n let nextID = isCommand ? this.#_nextCommandID : this.#_nextCommandLowID;\n while (!queue.has(nextID) && hasOldNewerPacket(queue, nextID)) {\n nextID = (nextID + 1) & 0xffff;\n win.advance(1);\n }\n if (isCommand) this.#_nextCommandID = nextID;\n else this.#_nextCommandLowID = nextID;\n }\n\n #tryReassemble(\n startPacket: Packet,\n queue: Map<number, Packet>,\n nextID: number,\n win: GenerationWindow,\n ): [reassembled: Packet, newNextID: number] | null {\n if ((packetFlags(startPacket) & PacketFlags.Fragmented) === 0) {\n queue.delete(nextID);\n win.advance(1);\n return [startPacket, (nextID + 1) & 0xffff];\n }\n\n const fragments: Packet[] = [];\n let totalSize = 0;\n let currID = nextID;\n let startSeen = false;\n let complete = false;\n\n while (true) {\n const frag = queue.get(currID);\n if (!frag) return null;\n fragments.push(frag);\n totalSize += frag.data.length;\n\n const fragmented = (packetFlags(frag) & PacketFlags.Fragmented) !== 0;\n if (!startSeen) {\n startSeen = true;\n if (!fragmented) {\n complete = true;\n break;\n }\n } else if (fragmented) {\n complete = true;\n break;\n }\n currID = (currID + 1) & 0xffff;\n }\n\n if (!complete) return null;\n\n const combined = new Uint8Array(totalSize);\n let pos = 0;\n for (const frag of fragments) {\n combined.set(frag.data, pos);\n pos += frag.data.length;\n queue.delete(nextID);\n win.advance(1);\n nextID = (nextID + 1) & 0xffff;\n }\n\n startPacket.data = combined;\n startPacket.typeFlagged &= ~PacketFlags.Fragmented;\n return [startPacket, nextID];\n }\n\n #tryDecompress(packet: Packet): void {\n if ((packetFlags(packet) & PacketFlags.Compressed) === 0) return;\n try {\n const qlz = new Qlz();\n packet.data = qlz.decompress(packet.data);\n packet.typeFlagged &= ~PacketFlags.Compressed;\n } catch (err) {\n this.#logger.debug(\"decompression failed\", { id: packet.id, err });\n }\n }\n\n #updatePostReceiveState(p: Packet): void {\n const pType = packetType(p);\n\n if (pType === PacketType.Init1) {\n this.#initPacketCheck = null;\n return;\n }\n\n if ((pType === PacketType.Ack || pType === PacketType.AckLow) && p.data.length >= 2) {\n const ackID = new DataView(p.data.buffer, p.data.byteOffset).getUint16(0, false);\n const targetType = pType === PacketType.Ack ? PacketType.Command : PacketType.CommandLow;\n const key = (targetType << 16) | ackID;\n this.#ackManager.delete(key);\n }\n }\n\n #nextPacketIdentity(pType: PacketType): [id: number, gen: number] {\n const pID = this.#packetCounter[pType]!;\n const pGen = this.#generationCounter[pType]!;\n\n if (pType === PacketType.Init1) return [pID, pGen];\n\n this.#packetCounter[pType] = (pID + 1) & 0xffff;\n if (this.#packetCounter[pType] === 0) {\n this.#generationCounter[pType] = (pGen + 1) >>> 0;\n }\n\n if (pType === PacketType.Command) {\n this.#sendWindowCommand.advanceToExcluded(pID);\n } else if (pType === PacketType.CommandLow) {\n this.#sendWindowCommandLow.advanceToExcluded(pID);\n }\n\n return [pID, pGen];\n }\n\n #trackResend(pType: PacketType, p: Packet, rp: ResendPacket): void {\n if (pType === PacketType.Init1) {\n this.#initPacketCheck = rp;\n return;\n }\n if (pType === PacketType.Command || pType === PacketType.CommandLow) {\n const key = (pType << 16) | p.id;\n this.#ackManager.set(key, rp);\n }\n }\n\n #checkResends(): void {\n const now = Date.now();\n\n if (now - this.#lastMessageReceived > PACKET_TIMEOUT_MS) {\n this.#logger.warn(\"idle timeout\");\n this.#triggerClose(new Error(\"idle timeout\"));\n return;\n }\n\n if (this.#initPacketCheck) {\n this.#doResend(this.#initPacketCheck, now);\n }\n\n for (const [key, rp] of this.#ackManager) {\n if (now - rp.firstSend > PACKET_TIMEOUT_MS) {\n this.#ackManager.delete(key);\n this.#triggerClose(new Error(\"packet ack timeout\"));\n return;\n }\n this.#doResend(rp, now);\n }\n }\n\n #doResend(rp: ResendPacket, now: number): void {\n if (now - rp.lastSend < rp.nextInterval) return;\n\n rp.lastSend = now;\n rp.retryCount++;\n rp.nextInterval = Math.min(rp.nextInterval * 2, MAX_RETRY_INTERVAL_MS);\n\n const dummy = !this.#crypt.cryptoInitComplete;\n const unencrypted = (packetFlags(rp.packet) & PacketFlags.Unencrypted) !== 0;\n const header = buildC2SHeader(rp.packet);\n const pType = packetType(rp.packet);\n\n const [ciphertext, tag] = this.#crypt.encrypt(\n pType,\n rp.packet.id,\n rp.packet.generationID,\n header,\n rp.packet.data,\n dummy,\n unencrypted,\n );\n\n const final = new Uint8Array(TAG_SIZE + HEADER_SIZE + ciphertext.length);\n final.set(tag.slice(0, TAG_SIZE), 0);\n final.set(header, TAG_SIZE);\n final.set(ciphertext, TAG_SIZE + HEADER_SIZE);\n this.#write(final);\n }\n\n #triggerClose(err: Error | null): void {\n if (this.#closed) return;\n this.close();\n this.onClosed?.(err);\n }\n}\n\n// ---- Helpers ----------------------------------------------------------------\n\nfunction applyProtocolFlags(pType: PacketType, flags: number): number {\n if (pType === PacketType.Command || pType === PacketType.CommandLow) {\n return flags | PacketFlags.NewProtocol;\n }\n return flags;\n}\n\nfunction hasOldNewerPacket(queue: Map<number, Packet>, nextID: number): boolean {\n const now = Date.now();\n for (const [id, pkg] of queue) {\n const diff = (id - nextID + 0x10000) & 0xffff;\n if (diff < 0x8000 && now - pkg.receivedAt > 5_000) {\n return true;\n }\n }\n return false;\n}\n"],"mappings":"gHAAA,IAAkB,EAAX,SAAA,EAAA,OACL,GAAA,EAAA,MAAA,GAAA,QACA,EAAA,EAAA,aAAA,GAAA,eACA,EAAA,EAAA,QAAA,GAAA,UACA,EAAA,EAAA,WAAA,GAAA,aACA,EAAA,EAAA,KAAA,GAAA,OACA,EAAA,EAAA,KAAA,GAAA,OACA,EAAA,EAAA,IAAA,GAAA,MACA,EAAA,EAAA,OAAA,GAAA,SACA,EAAA,EAAA,MAAA,GAAA,cACD,CAEiB,EAAX,SAAA,EAAA,OACL,GAAA,EAAA,WAAA,IAAA,aACA,EAAA,EAAA,YAAA,IAAA,cACA,EAAA,EAAA,WAAA,IAAA,aACA,EAAA,EAAA,YAAA,KAAA,oBACD,CAYD,SAAgB,EAAW,EAAuB,CAChD,OAAQ,EAAE,YAAc,GAG1B,SAAgB,EAAY,EAAmB,CAC7C,OAAO,EAAE,YAAc,IAGzB,SAAgB,EAAc,EAAoB,CAChD,OAAQ,EAAY,EAAE,CAAG,EAAY,eAAiB,EAIxD,SAAgB,EAAe,EAAuB,CACpD,IAAM,EAAS,IAAI,WAAW,EAAE,CAC1B,EAAO,IAAI,SAAS,EAAO,OAAO,CAIxC,OAHA,EAAK,UAAU,EAAG,EAAE,GAAI,GAAM,CAC9B,EAAK,UAAU,EAAG,EAAE,SAAU,GAAM,CACpC,EAAO,GAAK,EAAE,YACP,EAIT,SAAgB,EAAe,EAAqD,CAElF,MAAO,CACL,GAFW,IAAI,SAAS,EAAI,OAAQ,EAAI,WAAY,EAAI,WAAW,CAE1D,UAAU,EAAG,GAAM,CAC5B,YAAa,EAAI,GAClB,CAIH,SAAgB,EAAe,EAAkE,CAC/F,IAAM,EAAO,IAAI,SAAS,EAAI,OAAQ,EAAI,WAAY,EAAI,WAAW,CACrE,MAAO,CACL,GAAI,EAAK,UAAU,EAAG,GAAM,CAC5B,SAAU,EAAK,UAAU,EAAG,GAAM,CAClC,YAAa,EAAI,GAClB,CCnEH,IAAa,EAAb,KAA8B,CAC5B,GAAoB,EACpB,GAAc,EACd,GACA,GAEA,YAAY,EAAa,EAAoB,CAC3C,MAAA,EAAY,EACZ,MAAA,EAAsB,EAGxB,IAAI,YAAqB,CACvB,OAAO,MAAA,EAGT,QAAQ,EAAsB,CAC5B,GAAI,GAAU,EAAG,OACjB,IAAM,EAAgB,MAAA,EAAyB,EACzC,EAAU,KAAK,MAAM,EAAgB,MAAA,EAAU,CACjD,EAAU,IACZ,MAAA,EAAmB,KAAK,IAAI,MAAA,EAAmB,EAAS,WAAY,EAEtE,MAAA,EAAyB,EAAgB,MAAA,EAG3C,kBAAkB,EAA2B,CAC3C,IAAI,EAAW,EAAc,MAAA,EACzB,EAAW,IAAG,GAAY,MAAA,GAC9B,KAAK,QAAQ,EAAW,EAAE,CAG5B,OAAO,EAA2B,CAChC,IAAI,EAAW,EAAc,MAAA,EACzB,EAAW,IAAG,GAAY,MAAA,GAC9B,KAAK,QAAQ,EAAS,CAGxB,WAAW,EAA8B,CACvC,IAAM,EAAY,MAAA,EAAyB,MAAA,EAI3C,OAHI,EAAY,MAAA,EACP,GAAe,MAAA,GAA0B,EAAc,EAEzD,GAAe,MAAA,GAA0B,EAAc,EAAY,MAAA,EAG5E,cAAc,EAA6B,CAIzC,OAHI,MAAA,EAAgB,EAAY,CACvB,EAAc,MAAA,EAAY,MAAA,EAE5B,EAAc,MAAA,EAGvB,YAAY,EAA8B,CACxC,OAAO,KAAK,cAAc,EAAY,CAAG,EAG3C,eAAe,EAA8B,CAC3C,OAAO,KAAK,cAAc,EAAY,EAAI,MAAA,EAG5C,GAAW,EAA8B,CACvC,OACE,MAAA,EAAyB,MAAA,EAAY,MAAA,GACrC,EAAc,MAAA,EAAyB,MAAA,EAAsB,MAAA,EAIjE,cAAc,EAA6B,CAEzC,OADI,MAAA,EAAgB,EAAY,CAAS,MAAA,EAAmB,EACrD,MAAA,EAGT,OAAc,CACZ,MAAA,EAAyB,EACzB,MAAA,EAAmB,IC1EjB,EAAa,KASN,EAAb,KAAiB,CACf,GAAa,IAAI,WAAW,EAAW,CAEvC,WAAW,EAA8B,CACvC,GAAM,CAAE,YAAW,mBAAkB,SAAU,EAAe,EAAK,CAE7D,EAAO,IAAI,WAAW,EAAiB,CAE7C,GAAA,EAAK,EAAQ,GAEX,OADA,EAAK,IAAI,EAAK,MAAM,EAAW,EAAY,EAAiB,CAAC,CACtD,EAGT,MAAA,EAAgB,KAAK,EAAE,CAEvB,IAAM,EAAkB,CACtB,QAAS,EACT,UAAW,EACX,QAAS,EACT,WAAY,EACb,CAED,KAAO,MAAA,EAAoB,EAAM,EAAM,EACrC,GAAK,EAAM,QAAU,MACf,CAAC,MAAA,EAAuB,EAAM,EAAM,EAAM,CAAE,cAE5C,MAAA,EAAqB,EAAM,EAAM,EAAkB,EAAM,CAAE,MAInE,OAAO,EAGT,GAAe,EAAkB,EAAuB,CAUtD,OATI,EAAG,UAAY,EACf,EAAG,UAAY,EAAI,EAAK,OAAe,IAC3C,EAAG,SACA,EAAK,EAAG,WACN,EAAK,EAAG,UAAY,IAAO,EAC3B,EAAK,EAAG,UAAY,IAAO,GAC3B,EAAK,EAAG,UAAY,IAAO,MAC9B,EACF,EAAG,WAAa,EACT,IATsB,GAY/B,GAAkB,EAAkB,EAAkB,EAAuB,CAE3E,GADA,EAAG,QAAW,EAAG,UAAY,IAAO,EAChC,EAAG,UAAY,EAAI,EAAK,OAAQ,MAAO,GAE3C,IAAM,EAAK,EAAK,EAAG,WACb,EAAK,EAAK,EAAG,UAAY,GAC/B,EAAG,WAAa,EAEhB,IAAM,EAAQ,GAAM,EAAM,GAAM,EAC5B,EAAW,EAAK,GACpB,GAAI,IAAa,EACf,GAAY,MACP,CACL,GAAI,EAAG,WAAa,EAAK,OAAQ,MAAO,GACxC,EAAW,EAAK,EAAG,WACnB,EAAG,YAGL,IAAM,EAAS,MAAA,EAAgB,GAC/B,IAAK,IAAI,EAAI,EAAG,EAAI,EAAU,IACxB,EAAG,QAAU,EAAK,QAAU,EAAS,EAAI,EAAG,UAC9C,EAAK,EAAG,SAAW,EAAK,EAAS,GACjC,EAAG,WAIP,IAAM,EAAM,EAAG,QAAU,EAAI,EAI7B,OAHA,MAAA,EAAsB,EAAM,EAAI,EAAI,CACpC,EAAG,WAAa,EAAG,QAEZ,GAGT,GACE,EACA,EACA,EACA,EACS,CACT,IAAM,EAAY,KAAK,IAAI,EAAkB,GAAG,CAAG,GACnD,GAAI,EAAG,SAAW,EAAW,CAC3B,KAAO,EAAG,QAAU,GAAkB,CACpC,GAAI,EAAG,UAAY,EAAG,CAEpB,GADA,EAAG,WAAa,EACZ,EAAG,UAAY,EAAK,OAAQ,MAChC,EAAG,SACA,EAAK,EAAG,UAAY,GAClB,EAAK,EAAG,UAAY,IAAO,EAC3B,EAAK,EAAG,UAAY,IAAO,GAC3B,EAAK,EAAG,UAAY,IAAO,MAC9B,EAEJ,GAAI,EAAG,WAAa,EAAK,OAAQ,MACjC,EAAK,EAAG,WAAa,EAAK,EAAG,aAC7B,EAAG,QAAW,EAAG,UAAY,IAAO,EAEtC,MAAO,GAGT,GAAI,EAAG,WAAa,EAAK,QAAU,EAAG,SAAW,EAAK,OAAQ,MAAO,GAErE,EAAK,EAAG,WAAa,EAAK,EAAG,aAC7B,EAAG,QAAW,EAAG,UAAY,IAAO,EAEpC,IAAM,EAAM,KAAK,IAAI,EAAG,QAAU,EAAG,EAAE,CAIvC,OAHA,MAAA,EAAsB,EAAM,EAAI,EAAI,CAChC,EAAG,WAAa,IAAK,EAAG,WAAa,GAElC,GAGT,GAAiB,EAAkB,EAAc,EAAmB,CAClE,KAAO,EAAG,WAAa,GACjB,IAAG,WAAa,EAAI,EAAK,SADH,CAE1B,IAAM,GACH,EAAK,EAAG,YACN,EAAK,EAAG,WAAa,IAAO,EAC5B,EAAK,EAAG,WAAa,IAAO,MAC/B,EACI,GAAS,GAAK,GAAM,GAAK,KAC/B,MAAA,EAAgB,GAAQ,EAAG,WAC3B,EAAG,gBAKT,SAAS,EAAe,EAItB,CACA,GAAI,EAAK,OAAS,EAAG,MAAU,MAAM,0BAA0B,CAE/D,IAAM,EAAQ,EAAK,GAEnB,IADe,GAAS,EAAK,IACf,EAAG,MAAU,MAAM,qCAAqC,CAEtE,IAAM,EAAa,EAAQ,EAAc,EAAI,EAC7C,GAAI,EAAK,OAAS,EAAW,MAAU,MAAM,qCAAqC,CAElF,IAAI,EAOJ,MANA,CAGE,EAHG,EAAQ,GACS,EAAK,GAAO,EAAK,IAAO,EAAM,EAAK,IAAO,GAAO,EAAK,IAAO,MAAS,EAEvE,EAAK,GAGnB,CAAE,YAAW,mBAAkB,QAAO,CCjJ/C,IAAM,EAAsB,IACtB,EAA6B,KAC7B,EAAmB,IACnB,EAAoB,IACpB,EAAwB,IACxB,EAAc,EACd,EAAW,EACX,EAAoB,EACpB,EAA0B,IAC1B,EAA0B,IAUnB,EAAb,KAA2B,CACzB,SAAyC,KACzC,SAAiD,KAEjD,GACA,GAEA,GAA0B,KAC1B,GAAY,EACZ,GAAU,GACV,GAAiC,KACjC,GAAmC,KACnC,GAAuB,KAAK,KAAK,CAEjC,GAAiB,IAAI,YAAY,EAAE,CACnC,GAAqB,IAAI,YAAY,EAAE,CAEvC,GAAqB,IAAI,EAAiB,MAAS,EAA2B,CAC9E,GAAwB,IAAI,EAAiB,MAAS,EAA2B,CACjF,GAAqB,IAAI,EAAiB,MAAS,EAA2B,CAC9E,GAAwB,IAAI,EAAiB,MAAS,EAA2B,CAEjF,GAAgB,IAAI,IACpB,GAAmB,IAAI,IACvB,GAAc,IAAI,IAClB,GAAwC,KAExC,YAAY,EAAc,EAAiB,EAAA,EAAY,CACrD,MAAA,EAAc,EACd,MAAA,EAAe,EAGjB,YAAY,EAAkB,CAC5B,MAAA,EAAiB,EAGnB,QAAQ,EAA6B,CACnC,OAAO,IAAI,SAAS,EAAS,IAAW,CACtC,GAAM,CAAC,EAAM,GACX,EAAK,YAAY,IAAI,CAAG,EACpB,CAAC,EAAK,MAAM,EAAG,EAAK,YAAY,IAAI,CAAC,CAAE,EAAK,MAAM,EAAK,YAAY,IAAI,CAAG,EAAE,CAAC,CAC7E,CAAC,EAAM,OAAO,CACd,EAAO,SAAS,EAAS,GAAG,CAE5B,GAAA,EAAA,EAAA,cAAsB,OAAO,CACnC,EAAO,KAAK,QAAS,EAAO,CAC5B,EAAO,QAAQ,EAAM,MAAY,CAC/B,EAAO,IAAI,QAAS,EAAO,CAC3B,KAAK,MAAM,EAAO,CAClB,GAAS,EACT,EACF,CAGJ,MAAM,EAAuB,CAC3B,MAAA,EAAa,EACb,MAAA,EAAe,GACf,MAAA,EAA4B,KAAK,KAAK,CAEtC,EAAK,GAAG,UAAY,GAAQ,CAC1B,MAAA,EAA4B,KAAK,KAAK,CACtC,MAAA,EAAsB,IAAI,WAAW,EAAI,OAAQ,EAAI,WAAY,EAAI,WAAW,CAAC,EACjF,CACF,EAAK,GAAG,QAAU,GAAQ,CACxB,MAAA,EAAa,MAAM,YAAa,EAAI,CACpC,MAAA,EAAmB,EAAI,EACvB,CACF,EAAK,GAAG,YAAe,MAAA,EAAmB,KAAK,CAAC,CAEhD,MAAA,EAAoB,EAAW,SAAW,EAC1C,MAAA,EAAoB,EAAW,OAAS,IAExC,IAAM,EAAY,EAAA,EAAa,MAAA,EAAa,KAAK,CAC7C,GAAW,MAAA,EAAoB,EAAW,MAAO,EAAW,EAAE,CAGlE,IAAM,EAAY,gBAAkB,CAC9B,MAAA,EAAY,oBACd,KAAK,WAAW,EAAW,KAAM,IAAI,WAAe,EAAY,YAAY,EAE7E,EAAiB,CACpB,MAAA,MAAuB,cAAc,EAAU,CAG/C,IAAM,EAAc,gBAAkB,MAAA,GAAoB,CAAE,EAAwB,CACpF,MAAA,MAAyB,cAAc,EAAY,CAGrD,sBAA6B,CAC3B,MAAA,EAAwB,KAG1B,WAAW,EAAmB,EAAkB,EAAqB,CACnE,IAAM,EAAQ,CAAC,MAAA,EAAY,mBAC3B,GAAI,EAAK,OAAS,KAAO,IAAU,EAAW,OAAS,IAAU,EAAW,aAAc,CACxF,MAAA,EAAsB,EAAO,EAAM,EAAO,EAAM,CAChD,OAEF,MAAA,EAAoB,EAAO,EAAM,EAAO,EAAM,CAGhD,gBAAgB,EAAkB,EAAqB,CACrD,IAAM,EAAM,MAAA,EAAoB,EAAW,OACrC,EAAO,MAAA,EAAwB,EAAW,OAChD,MAAA,EAAoB,EAAW,OAAU,EAAM,EAAK,MAChD,MAAA,EAAoB,EAAW,SAAW,GAC5C,MAAA,EAAwB,EAAW,SAGrC,IAAM,EAAa,EAAoB,EAAK,OACtC,EAAe,IAAI,WAAW,EAAW,CAC/C,IAAI,SAAS,EAAa,OAAO,CAAC,UAAU,EAAG,EAAK,GAAM,CAC1D,EAAa,GAAK,EAClB,EAAa,IAAI,EAAM,EAAkB,CAWzC,IAAM,EAAS,EATG,CAChB,YAAa,EAAW,MAAQ,EAAY,YAC5C,GAAI,EACJ,SAAU,MAAA,EACV,aAAc,EACd,KAAM,EACN,WAAY,EACb,CAE+B,CAC1B,EAAQ,IAAI,WAAW,EAAW,EAAc,EAAW,CACjE,EAAM,IAAI,MAAA,EAAY,cAAe,EAAE,CACvC,EAAM,IAAI,EAAQ,EAAS,CAC3B,EAAM,IAAI,EAAc,EAAW,EAAY,CAC/C,MAAA,EAAY,EAAM,CAGpB,OAAc,CACR,MAAA,IACJ,MAAA,EAAe,GACf,MAAA,KAAkB,CAClB,MAAA,KAAoB,CACpB,MAAA,GAAY,OAAO,EAKrB,GAAiB,EAAmB,EAAkB,EAAe,EAAsB,CACzF,IAAM,EAAU,EAAsB,EAAc,EAChD,EAAM,EACN,EAAQ,GAEZ,KAAO,EAAM,EAAK,QAAQ,CACxB,IAAM,EAAY,KAAK,IAAI,EAAK,OAAS,EAAK,EAAQ,CAChD,EAAO,EAAM,IAAc,EAAK,OAElC,EAAS,EACT,IAAU,IAAM,GAAU,EAAY,YAE1C,MAAA,EAAoB,EAAO,EAAK,MAAM,EAAK,EAAM,EAAU,CAAE,EAAQ,EAAM,CAC3E,GAAO,EACP,EAAQ,IAIZ,GACE,EACA,EACA,EACA,EAAQ,CAAC,MAAA,EAAY,mBACf,CACN,EAAQ,EAAmB,EAAO,EAAM,CACxC,GAAM,CAAC,EAAK,GAAQ,MAAA,EAAyB,EAAM,CAC7C,EAAY,CAChB,YAAa,EAAQ,EACrB,GAAI,EACJ,SAAU,MAAA,EACV,aAAc,EACd,OACA,WAAY,EACb,CAEK,GAAe,EAAQ,EAAY,eAAiB,EACpD,EAAS,EAAe,EAAE,CAC1B,CAAC,EAAY,GAAO,MAAA,EAAY,QACpC,EACA,EACA,EACA,EACA,EACA,EACA,EACD,CAEK,EAAQ,IAAI,WAAW,EAAW,EAAc,EAAW,OAAO,CACxE,EAAM,IAAI,EAAI,MAAM,EAAG,EAAS,CAAE,EAAE,CACpC,EAAM,IAAI,EAAQ,EAAS,CAC3B,EAAM,IAAI,EAAY,EAAW,EAAY,CAC7C,MAAA,EAAY,EAAM,CAElB,IAAM,EAAmB,CACvB,OAAQ,EACR,UAAW,KAAK,KAAK,CACrB,SAAU,KAAK,KAAK,CACpB,WAAY,EACZ,aAAc,EACf,CACD,MAAA,EAAkB,EAAO,EAAG,EAAG,CAGjC,GAAO,EAAwB,CAC7B,MAAA,GAAY,KAAK,OAAO,KAAK,EAAK,CAAG,GAAQ,CACvC,GAAK,MAAA,EAAa,KAAK,iBAAkB,EAAI,EACjD,CAGJ,GAAiB,EAAuB,CACtC,GAAI,EAAI,OAAS,GAAI,OAErB,IAAM,EAAM,EAAI,MAAM,EAAG,EAAS,CAC5B,EAAS,EAAI,MAAM,EAAU,EAAW,EAAE,CAC1C,EAAa,EAAI,MAAM,EAAW,EAAE,CAEpC,EAAS,EAAe,EAAO,CAC/B,EAAY,CAChB,GAAG,EACH,SAAU,EACV,aAAc,MAAA,EAAwB,EAAO,GAAI,EAAO,YAAc,GAAK,CAC3E,KAAM,IAAI,WACV,WAAY,KAAK,KAAK,CACvB,CAEK,EAAY,MAAA,EAAwB,EAAG,EAAQ,EAAY,EAAI,CACjE,OAAc,KAKlB,IAJA,EAAE,KAAO,EAAU,UAEL,EAAW,EAAE,GAEb,EAAW,KAAM,CAC7B,MAAA,EAAe,EAAE,GAAI,EAAU,UAAU,CACzC,OAGG,MAAA,EAAgC,EAAG,EAAU,UAAU,GAE5D,MAAA,EAAwB,EAAE,CAC1B,MAAA,EAA6B,EAAE,GAGjC,GAAmB,EAAY,EAAuB,CACpD,OAAQ,EAAR,CACE,KAAK,EAAW,QACd,OAAO,MAAA,EAAwB,cAAc,EAAG,CAClD,KAAK,EAAW,WACd,OAAO,MAAA,EAA2B,cAAc,EAAG,CACrD,KAAK,EAAW,IACd,OAAO,MAAA,EAAwB,cAAc,EAAG,CAClD,KAAK,EAAW,OACd,OAAO,MAAA,EAA2B,cAAc,EAAG,CACrD,QACE,MAAO,IAIb,GACE,EACA,EACA,EACA,EACsD,CACtD,IAAM,GAAe,EAAY,EAAE,CAAG,EAAY,eAAiB,EAC7D,EAAQ,CAAC,MAAA,EAAY,mBACvB,EAAY,EACV,EAAQ,EAAW,EAAE,CACrB,EAAM,EAAE,aAEd,GAAI,CAWF,MAAO,CAAE,UAVS,MAAA,EAAY,QAC5B,EACA,EAAE,GACF,EACA,EACA,EACA,EACA,EACA,EACD,CACmB,YAAW,MACzB,CAEN,IAAK,IAAM,IAAU,CAAC,GAAI,EAAE,CAAE,CAC5B,IAAM,EAAW,EAAM,EACnB,OAAW,GACf,GAAI,CAWF,MAAO,CAAE,UAVS,MAAA,EAAY,QAC5B,EACA,EAAE,GACF,EACA,EACA,EACA,EACA,GACA,GACD,CACmB,UAAW,GAAO,MAChC,GAMV,GACE,IAAU,EAAW,SACrB,IAAU,EAAW,YACrB,IAAU,EAAW,IAErB,GAAI,CAWF,MAAO,CAAE,UAVS,MAAA,EAAY,QAC5B,EACA,EAAE,GACF,EACA,EACA,EACA,EACA,GACA,EACD,CACmB,UAAW,GAAM,MAC/B,EAKV,OAAO,MAIX,GAA2B,EAAW,EAA6B,CACjE,IAAM,EAAQ,EAAW,EAAE,CAC3B,GAAI,IAAU,EAAW,SAAW,IAAU,EAAW,WAAY,MAAO,GAE5E,IAAM,EAAM,IAAU,EAAW,QAAU,MAAA,EAA0B,MAAA,EAC/D,EAAU,IAAU,EAAW,QAAU,EAAW,IAAM,EAAW,OAS3E,OAPK,EAAI,WAAW,EAAE,GAAG,EAMzB,MAAA,EAAc,EAAE,GAAI,EAAS,EAAU,CAChC,KAND,EAAI,YAAY,EAAE,GAAG,EACvB,MAAA,EAAc,EAAE,GAAI,EAAS,EAAU,CAElC,IAMX,GAAS,EAAkB,EAAqB,EAA0B,CACxE,IAAM,EAAU,IAAI,WAAW,EAAE,CACjC,IAAI,SAAS,EAAQ,OAAO,CAAC,UAAU,EAAG,EAAU,GAAM,CAC1D,MAAA,EAAoB,EAAS,EAAS,EAAG,EAAU,CAGrD,GAAU,EAAa,EAA0B,CAC/C,IAAM,EAAW,IAAI,WAAW,EAAE,CAClC,IAAI,SAAS,EAAS,OAAO,CAAC,UAAU,EAAG,EAAK,GAAM,CACtD,MAAA,EAAoB,EAAW,KAAM,EAAU,EAAY,YAAa,EAAU,CAGpF,GAAmB,EAAiB,CAClC,IAAM,EAAQ,EAAW,EAAE,CAC3B,GAAI,IAAU,EAAW,SAAW,IAAU,EAAW,WAAY,CACnE,KAAK,WAAW,EAAE,CAClB,OAGF,IAAM,EAAY,IAAU,EAAW,QACjC,EAAQ,EAAY,MAAA,EAAqB,MAAA,EACzC,EAAM,EAAY,MAAA,EAA0B,MAAA,EAUlD,IARA,EAAM,IAAI,EAAE,GAAI,EAAE,CAEd,EACF,MAAA,EAAgC,EAAO,EAAK,GAAK,CAEjD,MAAA,EAAgC,EAAO,EAAK,GAAM,GAGvC,CACX,IAAM,EAAU,EAAY,MAAA,EAAuB,MAAA,EAC7C,EAAS,EAAM,IAAI,EAAQ,CACjC,GAAI,IAAW,IAAA,GAAW,MAE1B,IAAM,EAAS,MAAA,EAAoB,EAAQ,EAAO,EAAS,EAAI,CAC/D,GAAI,CAAC,EAAQ,MAEb,GAAM,CAAC,EAAa,GAAW,EAC3B,EAAW,MAAA,EAAuB,EACjC,MAAA,EAA0B,EAE/B,MAAA,EAAoB,EAAY,CAChC,KAAK,WAAW,EAAY,EAIhC,GAAkB,EAClB,GAAqB,EAErB,GACE,EACA,EACA,EACM,CACN,IAAI,EAAS,EAAY,MAAA,EAAuB,MAAA,EAChD,KAAO,CAAC,EAAM,IAAI,EAAO,EAAI,EAAkB,EAAO,EAAO,EAC3D,EAAU,EAAS,EAAK,MACxB,EAAI,QAAQ,EAAE,CAEZ,EAAW,MAAA,EAAuB,EACjC,MAAA,EAA0B,EAGjC,GACE,EACA,EACA,EACA,EACiD,CACjD,IAAK,EAAY,EAAY,CAAG,EAAY,cAAgB,EAG1D,OAFA,EAAM,OAAO,EAAO,CACpB,EAAI,QAAQ,EAAE,CACP,CAAC,EAAc,EAAS,EAAK,MAAO,CAG7C,IAAM,EAAsB,EAAE,CAC1B,EAAY,EACZ,EAAS,EACT,EAAY,GACZ,EAAW,GAEf,OAAa,CACX,IAAM,EAAO,EAAM,IAAI,EAAO,CAC9B,GAAI,CAAC,EAAM,OAAO,KAClB,EAAU,KAAK,EAAK,CACpB,GAAa,EAAK,KAAK,OAEvB,IAAM,GAAc,EAAY,EAAK,CAAG,EAAY,cAAgB,EACpE,GAAI,CAAC,EAEH,IADA,EAAY,GACR,CAAC,EAAY,CACf,EAAW,GACX,eAEO,EAAY,CACrB,EAAW,GACX,MAEF,EAAU,EAAS,EAAK,MAG1B,GAAI,CAAC,EAAU,OAAO,KAEtB,IAAM,EAAW,IAAI,WAAW,EAAU,CACtC,EAAM,EACV,IAAK,IAAM,KAAQ,EACjB,EAAS,IAAI,EAAK,KAAM,EAAI,CAC5B,GAAO,EAAK,KAAK,OACjB,EAAM,OAAO,EAAO,CACpB,EAAI,QAAQ,EAAE,CACd,EAAU,EAAS,EAAK,MAK1B,MAFA,GAAY,KAAO,EACnB,EAAY,aAAe,CAAC,EAAY,WACjC,CAAC,EAAa,EAAO,CAG9B,GAAe,EAAsB,CACnC,IAAK,EAAY,EAAO,CAAG,EAAY,cAAgB,EACvD,GAAI,CAEF,EAAO,KADK,IAAI,GAAK,CACH,WAAW,EAAO,KAAK,CACzC,EAAO,aAAe,CAAC,EAAY,iBAC5B,EAAK,CACZ,MAAA,EAAa,MAAM,uBAAwB,CAAE,GAAI,EAAO,GAAI,MAAK,CAAC,EAItE,GAAwB,EAAiB,CACvC,IAAM,EAAQ,EAAW,EAAE,CAE3B,GAAI,IAAU,EAAW,MAAO,CAC9B,MAAA,EAAwB,KACxB,OAGF,IAAK,IAAU,EAAW,KAAO,IAAU,EAAW,SAAW,EAAE,KAAK,QAAU,EAAG,CACnF,IAAM,EAAQ,IAAI,SAAS,EAAE,KAAK,OAAQ,EAAE,KAAK,WAAW,CAAC,UAAU,EAAG,GAAM,CAE1E,GADa,IAAU,EAAW,IAAM,EAAW,QAAU,EAAW,aACnD,GAAM,EACjC,MAAA,EAAiB,OAAO,EAAI,EAIhC,GAAoB,EAA8C,CAChE,IAAM,EAAM,MAAA,EAAoB,GAC1B,EAAO,MAAA,EAAwB,GAerC,OAbI,IAAU,EAAW,MAAc,CAAC,EAAK,EAAK,EAElD,MAAA,EAAoB,GAAU,EAAM,EAAK,MACrC,MAAA,EAAoB,KAAW,IACjC,MAAA,EAAwB,GAAU,EAAO,IAAO,GAG9C,IAAU,EAAW,QACvB,MAAA,EAAwB,kBAAkB,EAAI,CACrC,IAAU,EAAW,YAC9B,MAAA,EAA2B,kBAAkB,EAAI,CAG5C,CAAC,EAAK,EAAK,EAGpB,GAAa,EAAmB,EAAW,EAAwB,CACjE,GAAI,IAAU,EAAW,MAAO,CAC9B,MAAA,EAAwB,EACxB,OAEF,GAAI,IAAU,EAAW,SAAW,IAAU,EAAW,WAAY,CACnE,IAAM,EAAO,GAAS,GAAM,EAAE,GAC9B,MAAA,EAAiB,IAAI,EAAK,EAAG,EAIjC,IAAsB,CACpB,IAAM,EAAM,KAAK,KAAK,CAEtB,GAAI,EAAM,MAAA,EAA4B,EAAmB,CACvD,MAAA,EAAa,KAAK,eAAe,CACjC,MAAA,EAAuB,MAAM,eAAe,CAAC,CAC7C,OAGE,MAAA,GACF,MAAA,EAAe,MAAA,EAAuB,EAAI,CAG5C,IAAK,GAAM,CAAC,EAAK,KAAO,MAAA,EAAkB,CACxC,GAAI,EAAM,EAAG,UAAY,EAAmB,CAC1C,MAAA,EAAiB,OAAO,EAAI,CAC5B,MAAA,EAAuB,MAAM,qBAAqB,CAAC,CACnD,OAEF,MAAA,EAAe,EAAI,EAAI,EAI3B,GAAU,EAAkB,EAAmB,CAC7C,GAAI,EAAM,EAAG,SAAW,EAAG,aAAc,OAEzC,EAAG,SAAW,EACd,EAAG,aACH,EAAG,aAAe,KAAK,IAAI,EAAG,aAAe,EAAG,EAAsB,CAEtE,IAAM,EAAQ,CAAC,MAAA,EAAY,mBACrB,GAAe,EAAY,EAAG,OAAO,CAAG,EAAY,eAAiB,EACrE,EAAS,EAAe,EAAG,OAAO,CAClC,EAAQ,EAAW,EAAG,OAAO,CAE7B,CAAC,EAAY,GAAO,MAAA,EAAY,QACpC,EACA,EAAG,OAAO,GACV,EAAG,OAAO,aACV,EACA,EAAG,OAAO,KACV,EACA,EACD,CAEK,EAAQ,IAAI,WAAW,EAAW,EAAc,EAAW,OAAO,CACxE,EAAM,IAAI,EAAI,MAAM,EAAG,EAAS,CAAE,EAAE,CACpC,EAAM,IAAI,EAAQ,EAAS,CAC3B,EAAM,IAAI,EAAY,EAAW,EAAY,CAC7C,MAAA,EAAY,EAAM,CAGpB,GAAc,EAAyB,CACjC,MAAA,IACJ,KAAK,OAAO,CACZ,KAAK,WAAW,EAAI,IAMxB,SAAS,EAAmB,EAAmB,EAAuB,CAIpE,OAHI,IAAU,EAAW,SAAW,IAAU,EAAW,WAChD,EAAQ,EAAY,YAEtB,EAGT,SAAS,EAAkB,EAA4B,EAAyB,CAC9E,IAAM,EAAM,KAAK,KAAK,CACtB,IAAK,GAAM,CAAC,EAAI,KAAQ,EAEtB,IADc,EAAK,EAAS,MAAW,OAC5B,OAAU,EAAM,EAAI,WAAa,IAC1C,MAAO,GAGX,MAAO"}
|
package/dist/index.cjs
CHANGED
|
@@ -1,2 +1,4 @@
|
|
|
1
|
-
Object.defineProperty(exports,Symbol.toStringTag,{value:`Module`});const e=require(`./types-DrnoCdSW.cjs`),t=require(`./primitives-bj-ml31v.cjs`),n=require(`./crypto-2veVY1fC.cjs`),r=require(`./resolver-Dey6omBe.cjs`),i=require(`./handler-CqCD93f0.cjs`),a=require(`./command-Cu2v-5-K.cjs`),o=require(`./crypt-handshake-Dbj2cSBZ.cjs`),s=require(`./crypt-init2-C2oihZ7e.cjs`),c=require(`./parser-DhAWj-TI.cjs`);let l=require(`node:crypto`),u=require(`node:net`);var d=class{#e=new Map;#t=0;#n=[];#r=!1;register(){this.#t++;let e=this.#t;return[e,new Promise(t=>{this.#e.set(e,t)})]}unregister(e){this.#e.delete(e)}signalWelcomeComplete(){this.#r=!0,this.#n=[]}buffer(e){this.#r&&this.#e.size!==0&&this.#n.push(e)}resolve(e,t){let n=this.#e.get(e);if(!n){this.#n=[];return}let r=this.#n;this.#n=[],this.#e.delete(e),n({err:t,data:r})}discardBuffer(){this.#n=[]}reset(){this.#e.clear(),this.#n=[],this.#r=!1,this.#t=0}};function f(e){let n=e.id??`0`,r=e.msg??``,i=e.return_code,a=null;n!==`0`&&(a=new t.C(n,r));let o=null;if(i!==void 0&&i!==``){let e=parseInt(i,10);isNaN(e)||(o=e)}return{err:a,rc:o}}function p(e,t){return e.includes(`return_code=`)?e:`${e} return_code=${t}`}var m=class{#e=new Map;#t=0;register(){this.#t++,this.#t>65535&&(this.#t=1);let e=this.#t;return[e,new Promise(t=>{this.#e.set(e,t)})]}unregister(e){this.#e.delete(e)}notify(e,t){let n=this.#e.get(e);n&&n(t)}reset(){this.#e.clear(),this.#t=0}};function h(e,n,r){return new Promise((i,a)=>{let o=(0,u.createConnection)({host:e,port:n},()=>{o.write(r,e=>{e?(o.destroy(),a(new t.b(`failed to send transfer key: ${e.message}`))):i(o)})});o.setTimeout(1e4),o.once(`error`,a),o.once(`timeout`,()=>{o.destroy(),a(new t.b(`connection timeout`))})})}async function g(e,t,n){let r=await h(e,t.port,t.fileTransferKey);await new Promise((e,t)=>{n.pipe(r),r.on(`finish`,e),r.on(`error`,t)})}async function _(e,t,n){let r=await h(e,t.port,t.fileTransferKey);await new Promise((e,t)=>{r.pipe(n),n.on(`finish`,e),r.on(`error`,t),n.on(`error`,t)})}function v(e,t,n,r,i,o){let s=t.startsWith(`/`)?t:`/${t}`;return a.t(`ftinitupload`,{cid:String(e),name:s,cpw:n,size:String(r),clientftfid:String(i),overwrite:o?`1`:`0`,resume:`0`})}function y(e,t,n,r){let i=t.startsWith(`/`)?t:`/${t}`;return a.t(`ftinitdownload`,{cid:String(e),name:i,cpw:n,clientftfid:String(r),seekpos:`0`})}function b(e,t){let n=t;for(let t=e.length-1;t>=0;t--)n=e[t](n);return n}function x(e,t){let n=t;for(let t=e.length-1;t>=0;t--)n=e[t](n);return n}function S(e,t){let n=t.alpha??``,r=t.beta??``,i=t.omega??``;e.crypt.initCrypto(n,r,i),e.logger.info(`crypto initialized (P-256 path), sending clientinit`),O(e)}function C(e,t){e.logger.info(`received initivexpand2`),e.handler.receivedFinalInitAck();let n=t.l??``,r=t.omega??``,i=t.proof??``,a=t.beta??``,o=T(e,a);s.t(e.crypt,n,r,i,a,o),O(e)}function w(e,t){let n=t.aclid??t.clid??``,r=n?parseInt(n,10):0;r>0&&(e.clid=r,e.handler.setClientID(r)),e.logger.info(`connected to server`,{selfId:e.clid}),e._markConnected(),setImmediate(()=>{let t=a.t(`clientupdate`,{client_input_muted:`0`,client_output_muted:`0`});e.sendCommandNoWait(t).catch(()=>{})})}function T(e,n){let[r,o]=t.r(),s=Buffer.from(r).toString(`base64`),c=E(e,r,n),l=a.n(`clientek`,[[`ek`,s],[`proof`,c]]);return e.handler.sendPacket(i.a.Command,Buffer.from(l),0),o}function E(e,n,r){let i=Buffer.from(r,`base64`),a=new Uint8Array(86);a.set(n.slice(0,32)),a.set(i.slice(0,Math.min(54,i.length)),32);let o=t.o(e.crypt.identity.privateKey,a);return Buffer.from(o).toString(`base64`)}function D(e){return e===``?``:(0,l.createHash)(`sha1`).update(e).digest().toString(`base64`)}function O(e){let t=k(e);e.handler.sendPacket(i.a.Command,Buffer.from(t),0)}function k(e){let t=e.crypt.identity.publicKeyBase64(),n=e.getClientInitOptions(),r=D(n.defaultChannelPassword),i=D(n.serverPassword),o=(0,l.createHash)(`sha1`).update(t).digest().toString(`base64`);return a.n(`clientinit`,[[`client_nickname`,e.nickname],[`client_version`,`3.?.? [Build: 5680278000]`],[`client_platform`,`Windows`],[`client_input_hardware`,`1`],[`client_output_hardware`,`1`],[`client_default_channel`,n.defaultChannel],[`client_default_channel_password`,r],[`client_server_password`,i],[`client_meta_data`,``],[`client_version_sign`,`DX5NIYLvfJEUjuIbCidnoeozxIDRRkpq3I9vVMBmE9L2qnekOoBzSenkzsg2lC9CMv8K5hkEzhr2TYUYSwUXCg==`],[`client_key_offset`,String(e.crypt.identity.offset)],[`client_nickname_phonetic`,``],[`client_default_token`,``],[`hwid`,o]])}function A(e){if(e===``||e===void 0)return 0n;try{return BigInt(e)}catch{return 0n}}function j(e){let t=parseInt(e,10);return isNaN(t)||t<0||t>65535?0:t}function M(e){let t=parseInt(e,10);return isNaN(t)?0:t}function N(e,t){if(t===e)return!0;if(!t.startsWith(e))return!1;let n=t.slice(e.length);return/^\d+$/.test(n)}function P(e){let t=e.indexOf(` `);if(t<0)return[e];let n=e.slice(0,t),r=e.slice(t+1);if(!r.includes(`|`))return[e];let i=r.split(`|`),a=[];for(let e of i)e!==``&&a.push(`${n} ${e}`);return a.length===0?[e]:a}function F(e,t,n,r){switch(e.name){case`notifycliententerview`:return I(e,n,r);case`notifyclientleftview`:return L(e,t,n);case`notifyclientmoved`:return R(e,n);case`notifytextmessage`:return z(e,n);case`notifyclientpoke`:return U(e);case`notifystartupload`:return{kind:`startUpload`,info:B(e)};case`notifystartdownload`:return{kind:`startDownload`,info:V(e)};case`notifystatusfiletransfer`:return{kind:`fileTransferStatus`,info:H(e)};default:return{kind:`unknown`}}}function I(e,t,n){let r=j(e.params.clid??``),i=A(e.params.cid??``),a=M(e.params.client_type??``),o=e.params.client_servergroups??``,s={id:r,nickname:e.params.client_nickname??``,uid:e.params.client_unique_identifier??``,channelID:i,type:a,serverGroups:o?o.split(`,`):[]};return r!==0&&t.set(r,s),{kind:`clientEnter`,info:s}}function L(e,t,n){let r=j(e.params.clid??``),i=M(e.params.reasonid??``),a=r===t;return r!==0&&n.delete(r),{kind:`clientLeave`,event:{id:r,reasonID:i,reasonMsg:e.params.reasonmsg??``,targetID:j(e.params.targetid??``)},isSelf:a}}function R(e,t){let n=j(e.params.clid??``),r=A(e.params.ctid??``);if(n!==0){let e=t.get(n);e&&t.set(n,{...e,channelID:r})}return{kind:`clientMoved`,event:{id:n,targetChannelID:r,reasonID:M(e.params.reasonid??``),invokerID:j(e.params.invokerid??``),invokerName:e.params.invokername??``,invokerUID:e.params.invokeruid??``}}}function z(e,t){let n=j(e.params.invokerid??``),r=t.get(n);return{kind:`textMessage`,message:{targetMode:M(e.params.targetmode??``),targetID:A(e.params.target??``),invokerID:n,invokerName:e.params.invokername??``,invokerUID:e.params.invokeruid??r?.uid??``,message:a.i(e.params.msg??``),invokerGroups:r?.serverGroups??[]}}}function B(e){return{clientFileTransferID:j(e.params.clientftfid??``),serverFileTransferID:j(e.params.serverftfid??``),fileTransferKey:e.params.ftkey??``,port:j(e.params.port??``),seekPosition:A(e.params.seekpos??``)}}function V(e){return{clientFileTransferID:j(e.params.clientftfid??``),serverFileTransferID:j(e.params.serverftfid??``),fileTransferKey:e.params.ftkey??``,port:j(e.params.port??``),size:A(e.params.size??``)}}function H(e){return{clientFileTransferID:j(e.params.clientftfid??``),status:M(e.params.status??``),message:e.params.msg??``}}function U(e){return{kind:`poked`,event:{invokerID:j(e.params.invokerid??``),invokerName:a.i(e.params.invokername??``),invokerUID:e.params.invokeruid??``,message:a.i(e.params.msg??``)}}}var W=class e{static TOKEN_RATE=4;static TOKEN_MAX=8;#e=5;#t=Date.now();async wait(t){for(;;){if(t?.aborted)throw t.reason;let n=Date.now(),r=(n-this.#t)/1e3;if(this.#e=Math.min(this.#e+r*e.TOKEN_RATE,e.TOKEN_MAX),this.#t=n,this.#e>=1){--this.#e;return}let i=Math.ceil((1-this.#e)/e.TOKEN_RATE*1e3)+10;await new Promise((e,n)=>{let r=setTimeout(e,i);t&&t.addEventListener(`abort`,()=>{clearTimeout(r),n(t.reason)},{once:!0})})}}};function G(e){return{serverPassword:e.serverPassword??``,defaultChannel:e.defaultChannel??``,defaultChannelPassword:e.defaultChannelPassword??``}}var K=class{crypt;handler;logger;nickname;clid=0;#e;#t;#n;#r;#i=e.t.Disconnected;#a=new W;#o=new d;#s=new m;#c=new Map;#l=[];#u=[];#d=[];#f=[];#p=[];#m=[];#h=[];#g=[];#_=[];#v=[];#y=[];#b=[];#x;#S;constructor(t,a,o,s={}){this.#e=t,this.#t=a,this.nickname=o,this.logger=s.logger??e.n,this.#n=s.resolver??new r.t(this.logger),this.#r=G(s),this.crypt=new n.t(t),this.handler=new i.t(this.crypt,this.logger),this.handler.onPacket=e=>this.#T(e),this.handler.onClosed=e=>this.#P(e),s.commandMiddleware&&this.#y.push(...s.commandMiddleware),s.eventMiddleware&&this.#b.push(...s.eventMiddleware),this.#x=this.#F(),this.#S=this.#I()}get status(){return this.#i}getClientInitOptions(){return this.#r}async connect(){if(this.#i!==e.t.Disconnected)throw new t.h;this.#C(),this.#i=e.t.Connecting;let n=await this.#w();this.logger.info(`connecting to server`,{address:n}),await this.handler.connect(n)}async disconnect(){if(this.#i===e.t.Disconnected)return;let t=this.#i===e.t.Connected;if(this.#i=e.t.Disconnected,this.logger.info(`disconnecting from server`),t)try{await this.execCommand(`clientdisconnect reasonmsg=Shutdown`,1e3)}catch{}this.handler.close();let n=this.#_.slice();for(let e of n)setImmediate(()=>e(void 0))}waitConnected(t){return this.#i===e.t.Connected?Promise.resolve():new Promise((e,n)=>{this.#l.push(e),t&&t.addEventListener(`abort`,()=>n(t.reason),{once:!0})})}async sendCommandNoWait(e){await this.#a.wait(),await this.#x(e)}async execCommand(e,t=1e4){await this.execCommandWithResponse(e,t)}async execCommandWithResponse(e,n=1e4){let[r,i]=this.#o.register(),a=p(e,r);try{await this.#a.wait(),await this.#x(a)}catch(e){throw this.#o.unregister(r),e}let o=await Promise.race([i,new Promise((r,i)=>setTimeout(()=>i(new t.g(e)),n))]);if(this.#o.unregister(r),o.err)throw o.err;return o.data}on(e,t){switch(e){case`textMessage`:this.#u.push(t);break;case`clientEnter`:this.#d.push(t);break;case`clientLeave`:this.#f.push(t);break;case`clientMoved`:this.#p.push(t);break;case`poked`:this.#m.push(t);break;case`voiceData`:this.#h.push(t);break;case`connected`:this.#g.push(t);break;case`disconnected`:this.#_.push(t);break;case`kicked`:this.#v.push(t);break}return this}useCommandMiddleware(...e){return this.#y.push(...e),this.#x=this.#F(),this}useEventMiddleware(...e){return this.#b.push(...e),this.#S=this.#I(),this}clientID(){return this.clid}channelID(){return this.#c.get(this.clid)?.channelID??0n}sendVoice(e,t){this.handler.sendVoicePacket(e,t)}async fileTransferInitUpload(e,n,r,i,a=!1){let[o,s]=this.#s.register(),c=v(e,n,r,i,o,a);try{await this.execCommand(c,1e4)}catch(e){throw this.#s.unregister(o),e}let l=await Promise.race([s,new Promise((e,n)=>setTimeout(()=>n(new t.x),1e4))]);if(this.#s.unregister(o),`size`in l)throw new t.b(`unexpected download response`);if(`status`in l){let e=l;throw new t.b(`${e.message} (status=${e.status})`)}return l}async fileTransferInitDownload(e,n,r){let[i,a]=this.#s.register(),o=y(e,n,r,i);try{await this.execCommand(o,1e4)}catch(e){throw this.#s.unregister(i),e}let s=await Promise.race([a,new Promise((e,n)=>setTimeout(()=>n(new t.x),1e4))]);if(this.#s.unregister(i),`seekPosition`in s)throw new t.b(`unexpected upload response`);if(`status`in s){let e=s;throw new t.b(`${e.message} (status=${e.status})`)}return s}uploadFileData(e,t,n){return g(e,t,n)}downloadFileData(e,t,n){return _(e,t,n)}_markConnected(){this.#i=e.t.Connected;for(let e of this.#l)e();this.#l=[];let t=this.#g.slice();for(let e of t)setImmediate(()=>e())}#C(){this.handler.close(),this.crypt=new n.t(this.#e),this.handler=new i.t(this.crypt,this.logger),this.handler.onPacket=e=>this.#T(e),this.handler.onClosed=e=>this.#P(e),this.#o.reset(),this.#s.reset(),this.#c.clear(),this.clid=0,this.#x=this.#F()}async#w(){let e=this.#t.includes(`:`)?this.#t:`${this.#t}:9987`;try{return(await this.#n.resolve(this.#t))[0]?.addr??e}catch{return e}}#T(e){this.#E(e)}#E(e){let t=e.typeFlagged&15;if(t===8){let t=o.n(this.crypt,e.data);t&&this.handler.sendPacket(i.a.Init1,t,0);return}if((t===0||t===1)&&e.data.length>5){this.#D(e.data);return}(t===2||t===3)&&e.data.length>0&&this.#O(Buffer.from(e.data).toString(`utf8`))}#D(e){if(this.#h.length===0)return;let t=new DataView(e.buffer,e.byteOffset,e.byteLength).getUint16(2,!1);if(t===this.clid)return;let n={clientId:t,codec:e[4],data:e.subarray(5)};for(let e of this.#h)setImmediate(()=>e(n))}#O(e){if(!e)return;let t=e.split(/[\n\0]/);for(let e of t){let t=e.replace(/\r$/,``);if(t)for(let e of P(t))this.#k(e)}}#k(e){let t=c.t(e);if(!(!t||!t.name)){if(t.name.startsWith(`notify`)){let e=F(t,this.clid,this.#c,this.nickname);this.#j(e,t.params);return}switch(t.name){case`clientinitiv`:S(this,t.params);break;case`initivexpand2`:C(this,t.params);break;case`initserver`:w(this,t.params);break;case`error`:this.#A(t.params);break;default:{let e=t.params;if(t.name.includes(`=`)){let n=t.name.indexOf(`=`),r=t.name.slice(0,n),i=t.name.slice(n+1);e={[r]:i,...t.params}}this.#o.buffer(e);break}}}}#A(e){let{err:t,rc:n}=f(e);n===null?this.#o.discardBuffer():this.#o.resolve(n,t),(e.id??`0`)===`3329`&&setImmediate(()=>this.disconnect().catch(()=>{}))}#j(e,t){switch(e.kind){case`clientEnter`:{let t=e.info;t.id!==0&&N(this.nickname,t.nickname)&&(this.clid=t.id,this.handler.setClientID(t.id),this.#o.signalWelcomeComplete()),this.#M(`clientEnter`,t);break}case`clientLeave`:if(this.#M(`clientLeave`,e.event),e.isSelf&&(e.event.reasonID===4||e.event.reasonID===5)){let t=e.event.reasonMsg;for(let e of this.#v)setImmediate(()=>e(t))}break;case`clientMoved`:this.#M(`clientMoved`,e.event);break;case`textMessage`:this.#M(`textMessage`,e.message);break;case`poked`:this.#M(`poked`,e.event);break;case`startUpload`:this.#s.notify(e.info.clientFileTransferID,e.info);break;case`startDownload`:this.#s.notify(e.info.clientFileTransferID,e.info);break;case`fileTransferStatus`:this.#s.notify(e.info.clientFileTransferID,e.info);break}}#M(e,t){this.#S(t)}#N(e,t){switch(e){case`textMessage`:for(let e of this.#u)setImmediate(()=>e(t));break;case`clientEnter`:for(let e of this.#d)setImmediate(()=>e(t));break;case`clientLeave`:for(let e of this.#f)setImmediate(()=>e(t));break;case`clientMoved`:for(let e of this.#p)setImmediate(()=>e(t));break;case`poked`:for(let e of this.#m)setImmediate(()=>e(t));break}}#P(t){if(this.#i===e.t.Disconnected)return;this.#i=e.t.Disconnected;let n=this.#_.slice();for(let e of n)setImmediate(()=>e(t??void 0))}#F(){return b(this.#y,async e=>{this.handler.sendPacket(i.a.Command,Buffer.from(e),0)})}#I(){return x(this.#b,e=>{typeof e==`object`&&e&&`invokerName`in e&&`message`in e&&`targetMode`in e?this.#N(`textMessage`,e):typeof e==`object`&&e&&`invokerName`in e&&`message`in e&&!(`targetMode`in e)?this.#N(`poked`,e):typeof e==`object`&&e&&`id`in e&&`uid`in e?this.#N(`clientEnter`,e):typeof e==`object`&&e&&`id`in e&&`reasonID`in e&&`targetChannelID`in e?this.#N(`clientMoved`,e):typeof e==`object`&&e&&`id`in e&&`reasonID`in e&&this.#N(`clientLeave`,e)})}};async function q(e,t,n,r){let i=a.n(`sendtextmessage`,[[`targetmode`,String(t)],[`target`,String(n)],[`msg`,r]]);await e.sendCommandNoWait(i)}async function J(e,t,n,r=``){let i=[[`clid`,String(t)],[`cid`,String(n)]];r&&i.push([`cpw`,r]);let o=a.n(`clientmove`,i);await e.execCommand(o,1e4)}async function Y(e,t,n){let r=a.n(`clientpoke`,[[`clid`,String(t)],[`msg`,n]]);await e.execCommand(r,1e4)}async function X(e,t){let n=(await e.execCommandWithResponse(`clientinfo clid=${t}`,5e3))[0];if(!n)throw Error(`no data returned for client ${t}`);return n}async function Z(e){return(await e.execCommandWithResponse(`channellist`,5e3)).map(e=>({id:BigInt(e.cid??`0`),parentID:BigInt(e.pid??`0`),name:a.i(e.channel_name??``),description:``}))}async function Q(e){return(await e.execCommandWithResponse(`clientlist -uid -away -voice -groups`,5e3)).map(e=>{let t=e.client_servergroups??``;return{id:parseInt(e.clid??`0`,10),nickname:a.i(e.client_nickname??``),uid:e.client_unique_identifier??``,channelID:BigInt(e.cid??`0`),type:parseInt(e.client_type??`0`,10),serverGroups:t?t.split(`,`):[]}})}async function $(e,t,n){if(n.length===0)return;let r=n.join(`|`),i=a.t(`ftdeletefile`,{cid:String(t),cpw:``,name:r});await e.execCommand(i,1e4)}exports.AlreadyConnectedError=t.h,exports.Client=K,exports.ClientStatus=e.t,exports.CommandTimeoutError=t.g,exports.CryptoInitError=t._,exports.EAXTagMismatchError=t.v,exports.FakeSignatureMismatchError=t.y,exports.FileTransferError=t.b,exports.FileTransferTimeoutError=t.x,exports.Identity=t.l,exports.InvalidIdentityError=t.S,exports.ServerError=t.C,exports.TeamspeakError=t.w,exports.clientMove=J,exports.consoleLogger=e.n,exports.dialFileTransfer=h,exports.downloadFileData=_,exports.fileTransferDeleteFile=$,exports.generateIdentity=t.u,exports.getClientInfo=X,exports.getUidFromPublicKey=t.d,exports.identityFromString=t.p,exports.listChannels=Z,exports.listClients=Q,exports.noopLogger=e.r,exports.poke=Y,exports.sendTextMessage=q,exports.uploadFileData=g;
|
|
1
|
+
Object.defineProperty(exports,Symbol.toStringTag,{value:`Module`});const e=require(`./types-DrnoCdSW.cjs`),t=require(`./primitives-bj-ml31v.cjs`),n=require(`./crypto-2veVY1fC.cjs`),r=require(`./resolver-DDD2FFhD.cjs`),i=require(`./handler-CqCD93f0.cjs`),a=require(`./command-Cu2v-5-K.cjs`),o=require(`./crypt-handshake-Dbj2cSBZ.cjs`),s=require(`./crypt-init2-C2oihZ7e.cjs`),c=require(`./parser-DhAWj-TI.cjs`);let l=require(`node:crypto`),u=require(`node:net`);var d=class{#e=new Map;#t=0;#n=[];#r=!1;register(){this.#t++;let e=this.#t;return[e,new Promise(t=>{this.#e.set(e,t)})]}unregister(e){this.#e.delete(e)}signalWelcomeComplete(){this.#r=!0,this.#n=[]}buffer(e){this.#r&&this.#e.size!==0&&this.#n.push(e)}resolve(e,t){let n=this.#e.get(e);if(!n){this.#n=[];return}let r=this.#n;this.#n=[],this.#e.delete(e),n({err:t,data:r})}discardBuffer(){this.#n=[]}reset(){this.#e.clear(),this.#n=[],this.#r=!1,this.#t=0}};function f(e){let n=e.id??`0`,r=e.msg??``,i=e.return_code,a=null;n!==`0`&&(a=new t.C(n,r));let o=null;if(i!==void 0&&i!==``){let e=parseInt(i,10);isNaN(e)||(o=e)}return{err:a,rc:o}}function p(e,t){return e.includes(`return_code=`)?e:`${e} return_code=${t}`}var m=class{#e=new Map;#t=0;register(){this.#t++,this.#t>65535&&(this.#t=1);let e=this.#t;return[e,new Promise(t=>{this.#e.set(e,t)})]}unregister(e){this.#e.delete(e)}notify(e,t){let n=this.#e.get(e);n&&n(t)}reset(){this.#e.clear(),this.#t=0}};function h(e,n,r){return new Promise((i,a)=>{let o=(0,u.createConnection)({host:e,port:n},()=>{o.write(r,e=>{e?(o.destroy(),a(new t.b(`failed to send transfer key: ${e.message}`))):i(o)})});o.setTimeout(1e4),o.once(`error`,a),o.once(`timeout`,()=>{o.destroy(),a(new t.b(`connection timeout`))})})}async function g(e,t,n){let r=await h(e,t.port,t.fileTransferKey);await new Promise((e,t)=>{n.pipe(r),r.on(`finish`,e),r.on(`error`,t)})}async function _(e,t,n){let r=await h(e,t.port,t.fileTransferKey);await new Promise((e,t)=>{r.pipe(n),n.on(`finish`,e),r.on(`error`,t),n.on(`error`,t)})}function v(e,t,n,r,i,o){let s=t.startsWith(`/`)?t:`/${t}`;return a.t(`ftinitupload`,{cid:String(e),name:s,cpw:n,size:String(r),clientftfid:String(i),overwrite:o?`1`:`0`,resume:`0`})}function y(e,t,n,r){let i=t.startsWith(`/`)?t:`/${t}`;return a.t(`ftinitdownload`,{cid:String(e),name:i,cpw:n,clientftfid:String(r),seekpos:`0`})}function b(e,t){let n=t;for(let t=e.length-1;t>=0;t--)n=e[t](n);return n}function x(e,t){let n=t;for(let t=e.length-1;t>=0;t--)n=e[t](n);return n}function S(e,t){let n=t.alpha??``,r=t.beta??``,i=t.omega??``;e.crypt.initCrypto(n,r,i),e.logger.info(`crypto initialized (P-256 path), sending clientinit`),O(e)}function C(e,t){e.logger.info(`received initivexpand2`),e.handler.receivedFinalInitAck();let n=t.l??``,r=t.omega??``,i=t.proof??``,a=t.beta??``,o=T(e,a);s.t(e.crypt,n,r,i,a,o),O(e)}function w(e,t){let n=t.aclid??t.clid??``,r=n?parseInt(n,10):0;r>0&&(e.clid=r,e.handler.setClientID(r)),e.logger.info(`connected to server`,{selfId:e.clid}),e._markConnected(),setImmediate(()=>{let t=a.t(`clientupdate`,{client_input_muted:`0`,client_output_muted:`0`});e.sendCommandNoWait(t).catch(()=>{})})}function T(e,n){let[r,o]=t.r(),s=Buffer.from(r).toString(`base64`),c=E(e,r,n),l=a.n(`clientek`,[[`ek`,s],[`proof`,c]]);return e.handler.sendPacket(i.a.Command,Buffer.from(l),0),o}function E(e,n,r){let i=Buffer.from(r,`base64`),a=new Uint8Array(86);a.set(n.slice(0,32)),a.set(i.slice(0,Math.min(54,i.length)),32);let o=t.o(e.crypt.identity.privateKey,a);return Buffer.from(o).toString(`base64`)}function D(e){return e===``?``:(0,l.createHash)(`sha1`).update(e).digest().toString(`base64`)}function O(e){let t=k(e);e.handler.sendPacket(i.a.Command,Buffer.from(t),0)}function k(e){let t=e.crypt.identity.publicKeyBase64(),n=e.getClientInitOptions(),r=D(n.defaultChannelPassword),i=D(n.serverPassword),o=(0,l.createHash)(`sha1`).update(t).digest().toString(`base64`);return a.n(`clientinit`,[[`client_nickname`,e.nickname],[`client_version`,`3.?.? [Build: 5680278000]`],[`client_platform`,`Windows`],[`client_input_hardware`,`1`],[`client_output_hardware`,`1`],[`client_default_channel`,n.defaultChannel],[`client_default_channel_password`,r],[`client_server_password`,i],[`client_meta_data`,``],[`client_version_sign`,`DX5NIYLvfJEUjuIbCidnoeozxIDRRkpq3I9vVMBmE9L2qnekOoBzSenkzsg2lC9CMv8K5hkEzhr2TYUYSwUXCg==`],[`client_key_offset`,String(e.crypt.identity.offset)],[`client_nickname_phonetic`,``],[`client_default_token`,``],[`hwid`,o]])}function A(e){if(e===``||e===void 0)return 0n;try{return BigInt(e)}catch{return 0n}}function j(e){let t=parseInt(e,10);return isNaN(t)||t<0||t>65535?0:t}function M(e){let t=parseInt(e,10);return isNaN(t)?0:t}function N(e,t){if(t===e)return!0;if(!t.startsWith(e))return!1;let n=t.slice(e.length);return/^\d+$/.test(n)}function P(e){let t=e.indexOf(` `);if(t<0)return[e];let n=e.slice(0,t),r=e.slice(t+1);if(!r.includes(`|`))return[e];let i=r.split(`|`),a=[];for(let e of i)e!==``&&a.push(`${n} ${e}`);return a.length===0?[e]:a}function F(e,t,n,r){switch(e.name){case`notifycliententerview`:return I(e,n,r);case`notifyclientleftview`:return L(e,t,n);case`notifyclientmoved`:return R(e,n);case`notifytextmessage`:return z(e,n);case`notifyclientpoke`:return U(e);case`notifystartupload`:return{kind:`startUpload`,info:B(e)};case`notifystartdownload`:return{kind:`startDownload`,info:V(e)};case`notifystatusfiletransfer`:return{kind:`fileTransferStatus`,info:H(e)};default:return{kind:`unknown`}}}function I(e,t,n){let r=j(e.params.clid??``),i=A(e.params.cid??``),a=M(e.params.client_type??``),o=e.params.client_servergroups??``,s={id:r,nickname:e.params.client_nickname??``,uid:e.params.client_unique_identifier??``,channelID:i,type:a,serverGroups:o?o.split(`,`):[]};return r!==0&&t.set(r,s),{kind:`clientEnter`,info:s}}function L(e,t,n){let r=j(e.params.clid??``),i=M(e.params.reasonid??``),a=r===t;return r!==0&&n.delete(r),{kind:`clientLeave`,event:{id:r,reasonID:i,reasonMsg:e.params.reasonmsg??``,targetID:j(e.params.targetid??``)},isSelf:a}}function R(e,t){let n=j(e.params.clid??``),r=A(e.params.ctid??``);if(n!==0){let e=t.get(n);e&&t.set(n,{...e,channelID:r})}return{kind:`clientMoved`,event:{id:n,targetChannelID:r,reasonID:M(e.params.reasonid??``),invokerID:j(e.params.invokerid??``),invokerName:e.params.invokername??``,invokerUID:e.params.invokeruid??``}}}function z(e,t){let n=j(e.params.invokerid??``),r=t.get(n);return{kind:`textMessage`,message:{targetMode:M(e.params.targetmode??``),targetID:A(e.params.target??``),invokerID:n,invokerName:e.params.invokername??``,invokerUID:e.params.invokeruid??r?.uid??``,message:a.i(e.params.msg??``),invokerGroups:r?.serverGroups??[]}}}function B(e){return{clientFileTransferID:j(e.params.clientftfid??``),serverFileTransferID:j(e.params.serverftfid??``),fileTransferKey:e.params.ftkey??``,port:j(e.params.port??``),seekPosition:A(e.params.seekpos??``)}}function V(e){return{clientFileTransferID:j(e.params.clientftfid??``),serverFileTransferID:j(e.params.serverftfid??``),fileTransferKey:e.params.ftkey??``,port:j(e.params.port??``),size:A(e.params.size??``)}}function H(e){return{clientFileTransferID:j(e.params.clientftfid??``),status:M(e.params.status??``),message:e.params.msg??``}}function U(e){return{kind:`poked`,event:{invokerID:j(e.params.invokerid??``),invokerName:a.i(e.params.invokername??``),invokerUID:e.params.invokeruid??``,message:a.i(e.params.msg??``)}}}var W=class e{static TOKEN_RATE=4;static TOKEN_MAX=8;#e=5;#t=Date.now();async wait(t){for(;;){if(t?.aborted)throw t.reason;let n=Date.now(),r=(n-this.#t)/1e3;if(this.#e=Math.min(this.#e+r*e.TOKEN_RATE,e.TOKEN_MAX),this.#t=n,this.#e>=1){--this.#e;return}let i=Math.ceil((1-this.#e)/e.TOKEN_RATE*1e3)+10;await new Promise((e,n)=>{let r=setTimeout(e,i);t&&t.addEventListener(`abort`,()=>{clearTimeout(r),n(t.reason)},{once:!0})})}}};function G(e){return{serverPassword:e.serverPassword??``,defaultChannel:e.defaultChannel??``,defaultChannelPassword:e.defaultChannelPassword??``}}var K=class{crypt;handler;logger;nickname;clid=0;#e;#t;#n;#r;#i=e.t.Disconnected;#a=new W;#o=new d;#s=new m;#c=new Map;#l=[];#u=[];#d=[];#f=[];#p=[];#m=[];#h=[];#g=[];#_=[];#v=[];#y=[];#b=[];#x;#S;constructor(t,a,o,s={}){this.#e=t,this.#t=a,this.nickname=o,this.logger=s.logger??e.n,this.#n=s.resolver??new r.t(this.logger),this.#r=G(s),this.crypt=new n.t(t),this.handler=new i.t(this.crypt,this.logger),this.handler.onPacket=e=>this.#T(e),this.handler.onClosed=e=>this.#P(e),s.commandMiddleware&&this.#y.push(...s.commandMiddleware),s.eventMiddleware&&this.#b.push(...s.eventMiddleware),this.#x=this.#F(),this.#S=this.#I()}get status(){return this.#i}getClientInitOptions(){return this.#r}async connect(){if(this.#i!==e.t.Disconnected)throw new t.h;this.#C(),this.#i=e.t.Connecting;let n=await this.#w();this.logger.info(`connecting to server`,{address:n}),await this.handler.connect(n)}async disconnect(){if(this.#i===e.t.Disconnected)return;let t=this.#i===e.t.Connected;if(this.#i=e.t.Disconnected,this.logger.info(`disconnecting from server`),t)try{await this.execCommand(`clientdisconnect reasonmsg=Shutdown`,1e3)}catch{}this.handler.close();let n=this.#_.slice();for(let e of n)setImmediate(()=>e(void 0))}waitConnected(t){return this.#i===e.t.Connected?Promise.resolve():new Promise((e,n)=>{this.#l.push(e),t&&t.addEventListener(`abort`,()=>n(t.reason),{once:!0})})}async sendCommandNoWait(e){await this.#a.wait(),await this.#x(e)}async execCommand(e,t=1e4){await this.execCommandWithResponse(e,t)}async execCommandWithResponse(e,n=1e4){let[r,i]=this.#o.register(),a=p(e,r);try{await this.#a.wait(),await this.#x(a)}catch(e){throw this.#o.unregister(r),e}let o=await Promise.race([i,new Promise((r,i)=>setTimeout(()=>i(new t.g(e)),n))]);if(this.#o.unregister(r),o.err)throw o.err;return o.data}on(e,t){switch(e){case`textMessage`:this.#u.push(t);break;case`clientEnter`:this.#d.push(t);break;case`clientLeave`:this.#f.push(t);break;case`clientMoved`:this.#p.push(t);break;case`poked`:this.#m.push(t);break;case`voiceData`:this.#h.push(t);break;case`connected`:this.#g.push(t);break;case`disconnected`:this.#_.push(t);break;case`kicked`:this.#v.push(t);break}return this}useCommandMiddleware(...e){return this.#y.push(...e),this.#x=this.#F(),this}useEventMiddleware(...e){return this.#b.push(...e),this.#S=this.#I(),this}clientID(){return this.clid}channelID(){return this.#c.get(this.clid)?.channelID??0n}sendVoice(e,t){this.handler.sendVoicePacket(e,t)}async fileTransferInitUpload(e,n,r,i,a=!1){let[o,s]=this.#s.register(),c=v(e,n,r,i,o,a);try{await this.execCommand(c,1e4)}catch(e){throw this.#s.unregister(o),e}let l=await Promise.race([s,new Promise((e,n)=>setTimeout(()=>n(new t.x),1e4))]);if(this.#s.unregister(o),`size`in l)throw new t.b(`unexpected download response`);if(`status`in l){let e=l;throw new t.b(`${e.message} (status=${e.status})`)}return l}async fileTransferInitDownload(e,n,r){let[i,a]=this.#s.register(),o=y(e,n,r,i);try{await this.execCommand(o,1e4)}catch(e){throw this.#s.unregister(i),e}let s=await Promise.race([a,new Promise((e,n)=>setTimeout(()=>n(new t.x),1e4))]);if(this.#s.unregister(i),`seekPosition`in s)throw new t.b(`unexpected upload response`);if(`status`in s){let e=s;throw new t.b(`${e.message} (status=${e.status})`)}return s}uploadFileData(e,t,n){return g(e,t,n)}downloadFileData(e,t,n){return _(e,t,n)}_markConnected(){this.#i=e.t.Connected;for(let e of this.#l)e();this.#l=[];let t=this.#g.slice();for(let e of t)setImmediate(()=>e())}#C(){this.handler.close(),this.crypt=new n.t(this.#e),this.handler=new i.t(this.crypt,this.logger),this.handler.onPacket=e=>this.#T(e),this.handler.onClosed=e=>this.#P(e),this.#o.reset(),this.#s.reset(),this.#c.clear(),this.clid=0,this.#x=this.#F()}async#w(){let e=this.#t.includes(`:`)?this.#t:`${this.#t}:9987`;try{return(await this.#n.resolve(this.#t))[0]?.addr??e}catch{return e}}#T(e){this.#E(e)}#E(e){let t=e.typeFlagged&15;if(t===8){let t=o.n(this.crypt,e.data);t&&this.handler.sendPacket(i.a.Init1,t,0);return}if((t===0||t===1)&&e.data.length>5){this.#D(e.data);return}(t===2||t===3)&&e.data.length>0&&this.#O(Buffer.from(e.data).toString(`utf8`))}#D(e){if(this.#h.length===0)return;let t=new DataView(e.buffer,e.byteOffset,e.byteLength).getUint16(2,!1);if(t===this.clid)return;let n={clientId:t,codec:e[4],data:e.subarray(5)};for(let e of this.#h)setImmediate(()=>e(n))}#O(e){if(!e)return;let t=e.split(`\0`).join(`
|
|
2
|
+
`).split(`
|
|
3
|
+
`);for(let e of t){let t=e.replace(/\r$/,``);if(t)for(let e of P(t))this.#k(e)}}#k(e){let t=c.t(e);if(!(!t||!t.name)){if(t.name.startsWith(`notify`)){let e=F(t,this.clid,this.#c,this.nickname);this.#j(e,t.params);return}switch(t.name){case`clientinitiv`:S(this,t.params);break;case`initivexpand2`:C(this,t.params);break;case`initserver`:w(this,t.params);break;case`error`:this.#A(t.params);break;default:{let e=t.params;if(t.name.includes(`=`)){let n=t.name.indexOf(`=`),r=t.name.slice(0,n),i=t.name.slice(n+1);e={[r]:i,...t.params}}this.#o.buffer(e);break}}}}#A(e){let{err:t,rc:n}=f(e);n===null?this.#o.discardBuffer():this.#o.resolve(n,t),(e.id??`0`)===`3329`&&setImmediate(()=>this.disconnect().catch(()=>{}))}#j(e,t){switch(e.kind){case`clientEnter`:{let t=e.info;t.id!==0&&N(this.nickname,t.nickname)&&(this.clid=t.id,this.handler.setClientID(t.id),this.#o.signalWelcomeComplete()),this.#M(`clientEnter`,t);break}case`clientLeave`:if(this.#M(`clientLeave`,e.event),e.isSelf&&(e.event.reasonID===4||e.event.reasonID===5)){let t=e.event.reasonMsg;for(let e of this.#v)setImmediate(()=>e(t))}break;case`clientMoved`:this.#M(`clientMoved`,e.event);break;case`textMessage`:this.#M(`textMessage`,e.message);break;case`poked`:this.#M(`poked`,e.event);break;case`startUpload`:this.#s.notify(e.info.clientFileTransferID,e.info);break;case`startDownload`:this.#s.notify(e.info.clientFileTransferID,e.info);break;case`fileTransferStatus`:this.#s.notify(e.info.clientFileTransferID,e.info);break}}#M(e,t){this.#S(t)}#N(e,t){switch(e){case`textMessage`:for(let e of this.#u)setImmediate(()=>e(t));break;case`clientEnter`:for(let e of this.#d)setImmediate(()=>e(t));break;case`clientLeave`:for(let e of this.#f)setImmediate(()=>e(t));break;case`clientMoved`:for(let e of this.#p)setImmediate(()=>e(t));break;case`poked`:for(let e of this.#m)setImmediate(()=>e(t));break}}#P(t){if(this.#i===e.t.Disconnected)return;this.#i=e.t.Disconnected;let n=this.#_.slice();for(let e of n)setImmediate(()=>e(t??void 0))}#F(){return b(this.#y,async e=>{this.handler.sendPacket(i.a.Command,Buffer.from(e),0)})}#I(){return x(this.#b,e=>{typeof e==`object`&&e&&`invokerName`in e&&`message`in e&&`targetMode`in e?this.#N(`textMessage`,e):typeof e==`object`&&e&&`invokerName`in e&&`message`in e&&!(`targetMode`in e)?this.#N(`poked`,e):typeof e==`object`&&e&&`id`in e&&`uid`in e?this.#N(`clientEnter`,e):typeof e==`object`&&e&&`id`in e&&`reasonID`in e&&`targetChannelID`in e?this.#N(`clientMoved`,e):typeof e==`object`&&e&&`id`in e&&`reasonID`in e&&this.#N(`clientLeave`,e)})}};async function q(e,t,n,r){let i=a.n(`sendtextmessage`,[[`targetmode`,String(t)],[`target`,String(n)],[`msg`,r]]);await e.sendCommandNoWait(i)}async function J(e,t,n,r=``){let i=[[`clid`,String(t)],[`cid`,String(n)]];r&&i.push([`cpw`,r]);let o=a.n(`clientmove`,i);await e.execCommand(o,1e4)}async function Y(e,t,n){let r=a.n(`clientpoke`,[[`clid`,String(t)],[`msg`,n]]);await e.execCommand(r,1e4)}async function X(e,t){let n=(await e.execCommandWithResponse(`clientinfo clid=${t}`,5e3))[0];if(!n)throw Error(`no data returned for client ${t}`);return n}async function Z(e){return(await e.execCommandWithResponse(`channellist`,5e3)).map(e=>({id:BigInt(e.cid??`0`),parentID:BigInt(e.pid??`0`),name:a.i(e.channel_name??``),description:``}))}async function Q(e){return(await e.execCommandWithResponse(`clientlist -uid -away -voice -groups`,5e3)).map(e=>{let t=e.client_servergroups??``;return{id:parseInt(e.clid??`0`,10),nickname:a.i(e.client_nickname??``),uid:e.client_unique_identifier??``,channelID:BigInt(e.cid??`0`),type:parseInt(e.client_type??`0`,10),serverGroups:t?t.split(`,`):[]}})}async function $(e,t,n){if(n.length===0)return;let r=n.join(`|`),i=a.t(`ftdeletefile`,{cid:String(t),cpw:``,name:r});await e.execCommand(i,1e4)}exports.AlreadyConnectedError=t.h,exports.Client=K,exports.ClientStatus=e.t,exports.CommandTimeoutError=t.g,exports.CryptoInitError=t._,exports.EAXTagMismatchError=t.v,exports.FakeSignatureMismatchError=t.y,exports.FileTransferError=t.b,exports.FileTransferTimeoutError=t.x,exports.Identity=t.l,exports.InvalidIdentityError=t.S,exports.ServerError=t.C,exports.TeamspeakError=t.w,exports.clientMove=J,exports.consoleLogger=e.n,exports.dialFileTransfer=h,exports.downloadFileData=_,exports.fileTransferDeleteFile=$,exports.generateIdentity=t.u,exports.getClientInfo=X,exports.getUidFromPublicKey=t.d,exports.identityFromString=t.p,exports.listChannels=Z,exports.listClients=Q,exports.noopLogger=e.r,exports.poke=Y,exports.sendTextMessage=q,exports.uploadFileData=g;
|
|
2
4
|
//# sourceMappingURL=index.cjs.map
|