Skip to content

Commit 4681b83

Browse files
committedNov 27, 2023
Add auto-detected static network support to providers and allow customizing socket provider options (#4199, #4418, #4441).
1 parent 9e74d14 commit 4681b83

File tree

4 files changed

+76
-28
lines changed

4 files changed

+76
-28
lines changed
 

‎src.ts/providers/provider-ipcsocket.ts

+4-2
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,8 @@ import { connect } from "net";
33
import { SocketProvider } from "./provider-socket.js";
44

55
import type { Socket } from "net";
6+
7+
import type { JsonRpcApiProviderOptions } from "./provider-jsonrpc.js";
68
import type { Networkish } from "./network.js";
79

810

@@ -35,8 +37,8 @@ export class IpcSocketProvider extends SocketProvider {
3537
*/
3638
get socket(): Socket { return this.#socket; }
3739

38-
constructor(path: string, network?: Networkish) {
39-
super(network);
40+
constructor(path: string, network?: Networkish, options?: JsonRpcApiProviderOptions) {
41+
super(network, options);
4042
this.#socket = connect(path);
4143

4244
this.socket.on("ready", async () => {

‎src.ts/providers/provider-jsonrpc.ts

+49-21
Original file line numberDiff line numberDiff line change
@@ -190,7 +190,7 @@ export type DebugEventJsonRpcApiProvider = {
190190
*/
191191
export type JsonRpcApiProviderOptions = {
192192
polling?: boolean;
193-
staticNetwork?: null | Network;
193+
staticNetwork?: null | boolean | Network;
194194
batchStallTime?: number;
195195
batchMaxSize?: number;
196196
batchMaxCount?: number;
@@ -463,6 +463,7 @@ export abstract class JsonRpcApiProvider extends AbstractProvider {
463463
};
464464

465465
#network: null | Network;
466+
#pendingDetectNetwork: null | Promise<Network>;
466467

467468
#scheduleDrain(): void {
468469
if (this.#drainTimer) { return; }
@@ -554,6 +555,7 @@ export abstract class JsonRpcApiProvider extends AbstractProvider {
554555
this.#drainTimer = null;
555556

556557
this.#network = null;
558+
this.#pendingDetectNetwork = null;
557559

558560
{
559561
let resolve: null | ((value: void) => void) = null;
@@ -563,9 +565,15 @@ export abstract class JsonRpcApiProvider extends AbstractProvider {
563565
this.#notReady = { promise, resolve };
564566
}
565567

566-
// Make sure any static network is compatbile with the provided netwrok
567568
const staticNetwork = this._getOption("staticNetwork");
568-
if (staticNetwork) {
569+
if (typeof(staticNetwork) === "boolean") {
570+
assertArgument(!staticNetwork || network !== "any", "staticNetwork cannot be used on special network 'any'", "options", options);
571+
if (staticNetwork && network != null) {
572+
this.#network = Network.from(network);
573+
}
574+
575+
} else if (staticNetwork) {
576+
// Make sure any static network is compatbile with the provided netwrok
569577
assertArgument(network == null || staticNetwork.matches(network),
570578
"staticNetwork MUST match network object", "options", options);
571579
this.#network = staticNetwork;
@@ -641,36 +649,56 @@ export abstract class JsonRpcApiProvider extends AbstractProvider {
641649
*/
642650
async _detectNetwork(): Promise<Network> {
643651
const network = this._getOption("staticNetwork");
644-
if (network) { return network; }
652+
if (network) {
653+
if (network === true) {
654+
if (this.#network) { return this.#network; }
655+
} else {
656+
return network;
657+
}
658+
}
659+
660+
if (this.#pendingDetectNetwork) {
661+
return await this.#pendingDetectNetwork;
662+
}
645663

646664
// If we are ready, use ``send``, which enabled requests to be batched
647665
if (this.ready) {
648-
return Network.from(getBigInt(await this.send("eth_chainId", [ ])));
666+
this.#pendingDetectNetwork = (async () => {
667+
const result = Network.from(getBigInt(await this.send("eth_chainId", [ ])));
668+
this.#pendingDetectNetwork = null;
669+
return result;
670+
})();
671+
return await this.#pendingDetectNetwork;
649672
}
650673

651674
// We are not ready yet; use the primitive _send
675+
this.#pendingDetectNetwork = (async () => {
676+
const payload: JsonRpcPayload = {
677+
id: this.#nextId++, method: "eth_chainId", params: [ ], jsonrpc: "2.0"
678+
};
652679

653-
const payload: JsonRpcPayload = {
654-
id: this.#nextId++, method: "eth_chainId", params: [ ], jsonrpc: "2.0"
655-
};
680+
this.emit("debug", { action: "sendRpcPayload", payload });
656681

657-
this.emit("debug", { action: "sendRpcPayload", payload });
682+
let result: JsonRpcResult | JsonRpcError;
683+
try {
684+
result = (await this._send(payload))[0];
685+
this.#pendingDetectNetwork = null;
686+
} catch (error) {
687+
this.#pendingDetectNetwork = null;
688+
this.emit("debug", { action: "receiveRpcError", error });
689+
throw error;
690+
}
658691

659-
let result: JsonRpcResult | JsonRpcError;
660-
try {
661-
result = (await this._send(payload))[0];
662-
} catch (error) {
663-
this.emit("debug", { action: "receiveRpcError", error });
664-
throw error;
665-
}
692+
this.emit("debug", { action: "receiveRpcResult", result });
666693

667-
this.emit("debug", { action: "receiveRpcResult", result });
694+
if ("result" in result) {
695+
return Network.from(getBigInt(result.result));
696+
}
668697

669-
if ("result" in result) {
670-
return Network.from(getBigInt(result.result));
671-
}
698+
throw this.getRpcError(payload, result);
699+
})();
672700

673-
throw this.getRpcError(payload, result);
701+
return await this.#pendingDetectNetwork;
674702
}
675703

676704
/**

‎src.ts/providers/provider-socket.ts

+20-3
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,9 @@ import { JsonRpcApiProvider } from "./provider-jsonrpc.js";
1515

1616
import type { Subscriber, Subscription } from "./abstract-provider.js";
1717
import type { EventFilter } from "./provider.js";
18-
import type { JsonRpcError, JsonRpcPayload, JsonRpcResult } from "./provider-jsonrpc.js";
18+
import type {
19+
JsonRpcApiProviderOptions, JsonRpcError, JsonRpcPayload, JsonRpcResult
20+
} from "./provider-jsonrpc.js";
1921
import type { Networkish } from "./network.js";
2022

2123

@@ -194,8 +196,23 @@ export class SocketProvider extends JsonRpcApiProvider {
194196
*
195197
* If unspecified, the network will be discovered.
196198
*/
197-
constructor(network?: Networkish) {
198-
super(network, { batchMaxCount: 1 });
199+
constructor(network?: Networkish, _options?: JsonRpcApiProviderOptions) {
200+
// Copy the options
201+
const options = Object.assign({ }, (_options != null) ? _options: { });
202+
203+
// Support for batches is generally not supported for
204+
// connection-base providers; if this changes in the future
205+
// the _send should be updated to reflect this
206+
assertArgument(options.batchMaxCount == null || options.batchMaxCount === 1,
207+
"sockets-based providers do not support batches", "options.batchMaxCount", _options);
208+
options.batchMaxCount = 1;
209+
210+
// Socket-based Providers (generally) cannot change their network,
211+
// since they have a long-lived connection; but let people override
212+
// this if they have just cause.
213+
if (options.staticNetwork == null) { options.staticNetwork = true; }
214+
215+
super(network, options);
199216
this.#callbacks = new Map();
200217
this.#subs = new Map();
201218
this.#pending = new Map();

‎src.ts/providers/provider-websocket.ts

+3-2
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ import { WebSocket as _WebSocket } from "./ws.js"; /*-browser*/
44

55
import { SocketProvider } from "./provider-socket.js";
66

7+
import type { JsonRpcApiProviderOptions} from "./provider-jsonrpc.js";
78
import type { Networkish } from "./network.js";
89

910
/**
@@ -45,8 +46,8 @@ export class WebSocketProvider extends SocketProvider {
4546
return this.#websocket;
4647
}
4748

48-
constructor(url: string | WebSocketLike | WebSocketCreator, network?: Networkish) {
49-
super(network);
49+
constructor(url: string | WebSocketLike | WebSocketCreator, network?: Networkish, options?: JsonRpcApiProviderOptions) {
50+
super(network, options);
5051
if (typeof(url) === "string") {
5152
this.#connect = () => { return new _WebSocket(url); };
5253
this.#websocket = this.#connect();

0 commit comments

Comments
 (0)
Please sign in to comment.