import SimpleRpcQuery from "./simple.js"; import { Buffer } from "buffer"; import { isPromise } from "../util.js"; import { clearTimeout, setTimeout } from "timers"; import { pack, unpack } from "msgpackr"; import type { RPCRequest, RPCResponse } from "@lumeweb/relay-types"; import RpcNetwork from "../network.js"; import { StreamingRpcQueryOptions } from "../types.js"; export default class StreamingRpcQuery extends SimpleRpcQuery { protected _options: StreamingRpcQueryOptions; protected _canceled = false; constructor( network: RpcNetwork, relay: string | Buffer, query: RPCRequest, options: StreamingRpcQueryOptions ) { super(network, relay, query, options); this._options = options; } public cancel() { this._canceled = true; } protected async queryRelay(relay: string | Buffer): Promise { let socket: any; let relayKey: Buffer = relay as Buffer; if (relay === "string") { relayKey = Buffer.from(relay, "hex"); } if (relay instanceof Buffer) { relayKey = relay; relay = relay.toString("hex"); } try { socket = this._network.dht.connect(relayKey); if (isPromise(socket)) { socket = await socket; } } catch (e) { return; } return new Promise((resolve, reject) => { const finish = () => { relay = relay as string; this._responses[relay] = {}; resolve(null); socket.end(); }; const listener = (res: Buffer) => { relay = relay as string; if (this._timeoutTimer) { clearTimeout(this._timeoutTimer as any); this._timeoutTimer = null; } if (this._canceled) { socket.write(pack({ cancel: true })); socket.off("data", listener); finish(); return; } const response = unpack(res as any) as RPCResponse; if (response && response.error) { this._errors[relay] = response.error; return reject(null); } if (response?.data.done) { finish(); return; } this._options.streamHandler(response?.data.data); }; socket.on("data", listener); socket.on("error", (error: any) => { relay = relay as string; this._errors[relay] = error; reject({ error }); }); socket.write("rpc"); socket.write(pack(this._query)); }); } }