123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157 |
- import { assertArgument } from "../utils/index.js";
- import { JsonRpcApiPollingProvider } from "./provider-jsonrpc.js";
- import type {
- JsonRpcApiProviderOptions,
- JsonRpcError, JsonRpcPayload, JsonRpcResult,
- JsonRpcSigner
- } from "./provider-jsonrpc.js";
- import type { Network, Networkish } from "./network.js";
- /**
- * The interface to an [[link-eip-1193]] provider, which is a standard
- * used by most injected providers, which the [[BrowserProvider]] accepts
- * and exposes the API of.
- */
- export interface Eip1193Provider {
- /**
- * See [[link-eip-1193]] for details on this method.
- */
- request(request: { method: string, params?: Array<any> | Record<string, any> }): Promise<any>;
- };
- /**
- * The possible additional events dispatched when using the ``"debug"``
- * event on a [[BrowserProvider]].
- */
- export type DebugEventBrowserProvider = {
- action: "sendEip1193Payload",
- payload: { method: string, params: Array<any> }
- } | {
- action: "receiveEip1193Result",
- result: any
- } | {
- action: "receiveEip1193Error",
- error: Error
- };
- export type BrowserProviderOptions = {
- polling?: boolean;
- staticNetwork?: null | boolean | Network;
- cacheTimeout?: number;
- pollingInterval?: number;
- };
- /**
- * A **BrowserProvider** is intended to wrap an injected provider which
- * adheres to the [[link-eip-1193]] standard, which most (if not all)
- * currently do.
- */
- export class BrowserProvider extends JsonRpcApiPollingProvider {
- #request: (method: string, params: Array<any> | Record<string, any>) => Promise<any>;
- /**
- * Connect to the %%ethereum%% provider, optionally forcing the
- * %%network%%.
- */
- constructor(ethereum: Eip1193Provider, network?: Networkish, _options?: BrowserProviderOptions) {
- // Copy the options
- const options: JsonRpcApiProviderOptions = Object.assign({ },
- ((_options != null) ? _options: { }),
- { batchMaxCount: 1 });
- assertArgument(ethereum && ethereum.request, "invalid EIP-1193 provider", "ethereum", ethereum);
- super(network, options);
- this.#request = async (method: string, params: Array<any> | Record<string, any>) => {
- const payload = { method, params };
- this.emit("debug", { action: "sendEip1193Request", payload });
- try {
- const result = await ethereum.request(payload);
- this.emit("debug", { action: "receiveEip1193Result", result });
- return result;
- } catch (e: any) {
- const error = new Error(e.message);
- (<any>error).code = e.code;
- (<any>error).data = e.data;
- (<any>error).payload = payload;
- this.emit("debug", { action: "receiveEip1193Error", error });
- throw error;
- }
- };
- }
- async send(method: string, params: Array<any> | Record<string, any>): Promise<any> {
- await this._start();
- return await super.send(method, params);
- }
- async _send(payload: JsonRpcPayload | Array<JsonRpcPayload>): Promise<Array<JsonRpcResult | JsonRpcError>> {
- assertArgument(!Array.isArray(payload), "EIP-1193 does not support batch request", "payload", payload);
- try {
- const result = await this.#request(payload.method, payload.params || [ ]);
- return [ { id: payload.id, result } ];
- } catch (e: any) {
- return [ {
- id: payload.id,
- error: { code: e.code, data: e.data, message: e.message }
- } ];
- }
- }
- getRpcError(payload: JsonRpcPayload, error: JsonRpcError): Error {
- error = JSON.parse(JSON.stringify(error));
- // EIP-1193 gives us some machine-readable error codes, so rewrite
- // them into
- switch (error.error.code || -1) {
- case 4001:
- error.error.message = `ethers-user-denied: ${ error.error.message }`;
- break;
- case 4200:
- error.error.message = `ethers-unsupported: ${ error.error.message }`;
- break;
- }
- return super.getRpcError(payload, error);
- }
- /**
- * Resolves to ``true`` if the provider manages the %%address%%.
- */
- async hasSigner(address: number | string): Promise<boolean> {
- if (address == null) { address = 0; }
- const accounts = await this.send("eth_accounts", [ ]);
- if (typeof(address) === "number") {
- return (accounts.length > address);
- }
- address = address.toLowerCase();
- return accounts.filter((a: string) => (a.toLowerCase() === address)).length !== 0;
- }
- async getSigner(address?: number | string): Promise<JsonRpcSigner> {
- if (address == null) { address = 0; }
- if (!(await this.hasSigner(address))) {
- try {
- //const resp =
- await this.#request("eth_requestAccounts", [ ]);
- //console.log("RESP", resp);
- } catch (error: any) {
- const payload = error.payload;
- throw this.getRpcError(payload, { id: payload.id, error });
- }
- }
- return await super.getSigner(address);
- }
- }
|