2020-03-12 19:14:50 +01:00
|
|
|
"use strict";
|
|
|
|
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
|
|
|
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
|
|
|
return new (P || (P = Promise))(function (resolve, reject) {
|
|
|
|
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
|
|
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
|
|
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
|
|
|
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
|
|
});
|
|
|
|
};
|
|
|
|
import { BigNumber } from "@ethersproject/bignumber";
|
|
|
|
import { defineReadOnly } from "@ethersproject/properties";
|
|
|
|
import { JsonRpcProvider } from "./json-rpc-provider";
|
2020-11-16 23:07:24 -05:00
|
|
|
import { WebSocket } from "./ws";
|
2020-03-12 19:14:50 +01:00
|
|
|
import { Logger } from "@ethersproject/logger";
|
|
|
|
import { version } from "./_version";
|
|
|
|
const logger = new Logger(version);
|
|
|
|
/**
|
|
|
|
* Notes:
|
|
|
|
*
|
|
|
|
* This provider differs a bit from the polling providers. One main
|
|
|
|
* difference is how it handles consistency. The polling providers
|
|
|
|
* will stall responses to ensure a consistent state, while this
|
|
|
|
* WebSocket provider assumes the connected backend will manage this.
|
|
|
|
*
|
|
|
|
* For example, if a polling provider emits an event which indicats
|
|
|
|
* the event occurred in blockhash XXX, a call to fetch that block by
|
|
|
|
* its hash XXX, if not present will retry until it is present. This
|
|
|
|
* can occur when querying a pool of nodes that are mildly out of sync
|
|
|
|
* with each other.
|
|
|
|
*/
|
|
|
|
let NextId = 1;
|
2020-04-15 18:28:04 -04:00
|
|
|
// For more info about the Real-time Event API see:
|
|
|
|
// https://geth.ethereum.org/docs/rpc/pubsub
|
2020-03-12 19:14:50 +01:00
|
|
|
export class WebSocketProvider extends JsonRpcProvider {
|
|
|
|
constructor(url, network) {
|
2020-06-03 03:47:17 -04:00
|
|
|
// This will be added in the future; please open an issue to expedite
|
|
|
|
if (network === "any") {
|
|
|
|
logger.throwError("WebSocketProvider does not support 'any' network yet", Logger.errors.UNSUPPORTED_OPERATION, {
|
|
|
|
operation: "network:any"
|
|
|
|
});
|
|
|
|
}
|
2020-03-12 19:14:50 +01:00
|
|
|
super(url, network);
|
|
|
|
this._pollingInterval = -1;
|
2020-10-05 15:46:12 -04:00
|
|
|
this._wsReady = false;
|
2020-03-12 19:14:50 +01:00
|
|
|
defineReadOnly(this, "_websocket", new WebSocket(this.connection.url));
|
|
|
|
defineReadOnly(this, "_requests", {});
|
|
|
|
defineReadOnly(this, "_subs", {});
|
|
|
|
defineReadOnly(this, "_subIds", {});
|
2020-10-05 15:46:12 -04:00
|
|
|
defineReadOnly(this, "_detectNetwork", super.detectNetwork());
|
2020-03-12 19:14:50 +01:00
|
|
|
// Stall sending requests until the socket is open...
|
|
|
|
this._websocket.onopen = () => {
|
|
|
|
this._wsReady = true;
|
|
|
|
Object.keys(this._requests).forEach((id) => {
|
|
|
|
this._websocket.send(this._requests[id].payload);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
this._websocket.onmessage = (messageEvent) => {
|
|
|
|
const data = messageEvent.data;
|
|
|
|
const result = JSON.parse(data);
|
2020-05-03 17:53:58 -04:00
|
|
|
if (result.id != null) {
|
2020-03-12 19:14:50 +01:00
|
|
|
const id = String(result.id);
|
|
|
|
const request = this._requests[id];
|
|
|
|
delete this._requests[id];
|
2020-05-03 17:53:58 -04:00
|
|
|
if (result.result !== undefined) {
|
2020-03-12 19:14:50 +01:00
|
|
|
request.callback(null, result.result);
|
2020-10-07 00:25:28 -04:00
|
|
|
this.emit("debug", {
|
|
|
|
action: "response",
|
|
|
|
request: JSON.parse(request.payload),
|
|
|
|
response: result.result,
|
|
|
|
provider: this
|
|
|
|
});
|
2020-03-12 19:14:50 +01:00
|
|
|
}
|
|
|
|
else {
|
2020-10-07 00:25:28 -04:00
|
|
|
let error = null;
|
2020-03-12 19:14:50 +01:00
|
|
|
if (result.error) {
|
2020-10-07 00:25:28 -04:00
|
|
|
error = new Error(result.error.message || "unknown error");
|
2020-03-12 19:14:50 +01:00
|
|
|
defineReadOnly(error, "code", result.error.code || null);
|
|
|
|
defineReadOnly(error, "response", data);
|
|
|
|
}
|
|
|
|
else {
|
2020-10-07 00:25:28 -04:00
|
|
|
error = new Error("unknown error");
|
2020-03-12 19:14:50 +01:00
|
|
|
}
|
2020-10-07 00:25:28 -04:00
|
|
|
request.callback(error, undefined);
|
|
|
|
this.emit("debug", {
|
|
|
|
action: "response",
|
|
|
|
error: error,
|
|
|
|
request: JSON.parse(request.payload),
|
|
|
|
provider: this
|
|
|
|
});
|
2020-03-12 19:14:50 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (result.method === "eth_subscription") {
|
|
|
|
// Subscription...
|
|
|
|
const sub = this._subs[result.params.subscription];
|
|
|
|
if (sub) {
|
|
|
|
//this.emit.apply(this, );
|
|
|
|
sub.processFunc(result.params.result);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
console.warn("this should not happen");
|
|
|
|
}
|
|
|
|
};
|
2020-05-29 21:27:59 -04:00
|
|
|
// This Provider does not actually poll, but we want to trigger
|
|
|
|
// poll events for things that depend on them (like stalling for
|
|
|
|
// block and transaction lookups)
|
|
|
|
const fauxPoll = setInterval(() => {
|
|
|
|
this.emit("poll");
|
|
|
|
}, 1000);
|
|
|
|
if (fauxPoll.unref) {
|
|
|
|
fauxPoll.unref();
|
|
|
|
}
|
2020-03-12 19:14:50 +01:00
|
|
|
}
|
2020-10-05 15:46:12 -04:00
|
|
|
detectNetwork() {
|
|
|
|
return this._detectNetwork;
|
|
|
|
}
|
2020-03-12 19:14:50 +01:00
|
|
|
get pollingInterval() {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
resetEventsBlock(blockNumber) {
|
|
|
|
logger.throwError("cannot reset events block on WebSocketProvider", Logger.errors.UNSUPPORTED_OPERATION, {
|
|
|
|
operation: "resetEventBlock"
|
|
|
|
});
|
|
|
|
}
|
|
|
|
set pollingInterval(value) {
|
|
|
|
logger.throwError("cannot set polling interval on WebSocketProvider", Logger.errors.UNSUPPORTED_OPERATION, {
|
|
|
|
operation: "setPollingInterval"
|
|
|
|
});
|
|
|
|
}
|
|
|
|
poll() {
|
|
|
|
return __awaiter(this, void 0, void 0, function* () {
|
|
|
|
return null;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
set polling(value) {
|
|
|
|
if (!value) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
logger.throwError("cannot set polling on WebSocketProvider", Logger.errors.UNSUPPORTED_OPERATION, {
|
|
|
|
operation: "setPolling"
|
|
|
|
});
|
|
|
|
}
|
|
|
|
send(method, params) {
|
|
|
|
const rid = NextId++;
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
function callback(error, result) {
|
|
|
|
if (error) {
|
|
|
|
return reject(error);
|
|
|
|
}
|
|
|
|
return resolve(result);
|
|
|
|
}
|
|
|
|
const payload = JSON.stringify({
|
|
|
|
method: method,
|
|
|
|
params: params,
|
|
|
|
id: rid,
|
|
|
|
jsonrpc: "2.0"
|
|
|
|
});
|
2020-10-07 00:25:28 -04:00
|
|
|
this.emit("debug", {
|
|
|
|
action: "request",
|
|
|
|
request: JSON.parse(payload),
|
|
|
|
provider: this
|
|
|
|
});
|
2020-03-12 19:14:50 +01:00
|
|
|
this._requests[String(rid)] = { callback, payload };
|
|
|
|
if (this._wsReady) {
|
|
|
|
this._websocket.send(payload);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
static defaultUrl() {
|
2020-04-23 23:35:39 -04:00
|
|
|
return "ws:/\/localhost:8546";
|
2020-03-12 19:14:50 +01:00
|
|
|
}
|
|
|
|
_subscribe(tag, param, processFunc) {
|
|
|
|
return __awaiter(this, void 0, void 0, function* () {
|
|
|
|
let subIdPromise = this._subIds[tag];
|
|
|
|
if (subIdPromise == null) {
|
2020-04-23 23:35:39 -04:00
|
|
|
subIdPromise = Promise.all(param).then((param) => {
|
|
|
|
return this.send("eth_subscribe", param);
|
|
|
|
});
|
2020-03-12 19:14:50 +01:00
|
|
|
this._subIds[tag] = subIdPromise;
|
|
|
|
}
|
|
|
|
const subId = yield subIdPromise;
|
|
|
|
this._subs[subId] = { tag, processFunc };
|
|
|
|
});
|
|
|
|
}
|
|
|
|
_startEvent(event) {
|
|
|
|
switch (event.type) {
|
|
|
|
case "block":
|
2020-05-29 21:27:59 -04:00
|
|
|
this._subscribe("block", ["newHeads"], (result) => {
|
|
|
|
const blockNumber = BigNumber.from(result.number).toNumber();
|
|
|
|
this._emitted.block = blockNumber;
|
|
|
|
this.emit("block", blockNumber);
|
2020-03-12 19:14:50 +01:00
|
|
|
});
|
|
|
|
break;
|
|
|
|
case "pending":
|
|
|
|
this._subscribe("pending", ["newPendingTransactions"], (result) => {
|
|
|
|
this.emit("pending", result);
|
|
|
|
});
|
|
|
|
break;
|
|
|
|
case "filter":
|
2020-04-23 23:35:39 -04:00
|
|
|
this._subscribe(event.tag, ["logs", this._getFilter(event.filter)], (result) => {
|
|
|
|
if (result.removed == null) {
|
|
|
|
result.removed = false;
|
|
|
|
}
|
|
|
|
this.emit(event.filter, this.formatter.filterLog(result));
|
2020-03-12 19:14:50 +01:00
|
|
|
});
|
|
|
|
break;
|
|
|
|
case "tx": {
|
|
|
|
const emitReceipt = (event) => {
|
|
|
|
const hash = event.hash;
|
|
|
|
this.getTransactionReceipt(hash).then((receipt) => {
|
|
|
|
if (!receipt) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
this.emit(hash, receipt);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
// In case it is already mined
|
|
|
|
emitReceipt(event);
|
|
|
|
// To keep things simple, we start up a single newHeads subscription
|
|
|
|
// to keep an eye out for transactions we are watching for.
|
|
|
|
// Starting a subscription for an event (i.e. "tx") that is already
|
|
|
|
// running is (basically) a nop.
|
2020-05-29 21:27:59 -04:00
|
|
|
this._subscribe("tx", ["newHeads"], (result) => {
|
2020-03-12 19:14:50 +01:00
|
|
|
this._events.filter((e) => (e.type === "tx")).forEach(emitReceipt);
|
|
|
|
});
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// Nothing is needed
|
|
|
|
case "debug":
|
2020-05-29 21:27:59 -04:00
|
|
|
case "poll":
|
|
|
|
case "willPoll":
|
|
|
|
case "didPoll":
|
2020-03-12 19:14:50 +01:00
|
|
|
case "error":
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
console.log("unhandled:", event);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_stopEvent(event) {
|
|
|
|
let tag = event.tag;
|
|
|
|
if (event.type === "tx") {
|
|
|
|
// There are remaining transaction event listeners
|
|
|
|
if (this._events.filter((e) => (e.type === "tx")).length) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
tag = "tx";
|
|
|
|
}
|
2020-04-15 18:28:04 -04:00
|
|
|
else if (this.listenerCount(event.event)) {
|
2020-03-12 19:14:50 +01:00
|
|
|
// There are remaining event listeners
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const subId = this._subIds[tag];
|
|
|
|
if (!subId) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
delete this._subIds[tag];
|
|
|
|
subId.then((subId) => {
|
|
|
|
if (!this._subs[subId]) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
delete this._subs[subId];
|
|
|
|
this.send("eth_unsubscribe", [subId]);
|
|
|
|
});
|
|
|
|
}
|
2020-07-13 08:03:56 -04:00
|
|
|
destroy() {
|
|
|
|
return __awaiter(this, void 0, void 0, function* () {
|
|
|
|
// Wait until we have connected before trying to disconnect
|
|
|
|
if (this._websocket.readyState === WebSocket.CONNECTING) {
|
2020-07-13 08:50:41 -04:00
|
|
|
yield (new Promise((resolve) => {
|
|
|
|
this._websocket.onopen = function () {
|
2020-07-13 08:03:56 -04:00
|
|
|
resolve(true);
|
2020-07-13 08:50:41 -04:00
|
|
|
};
|
|
|
|
this._websocket.onerror = function () {
|
2020-07-13 08:03:56 -04:00
|
|
|
resolve(false);
|
2020-07-13 08:50:41 -04:00
|
|
|
};
|
|
|
|
}));
|
2020-07-13 08:03:56 -04:00
|
|
|
}
|
2020-07-13 09:34:20 -04:00
|
|
|
// Hangup
|
|
|
|
// See: https://developer.mozilla.org/en-US/docs/Web/API/CloseEvent#Status_codes
|
|
|
|
this._websocket.close(1000);
|
2020-07-13 08:03:56 -04:00
|
|
|
});
|
|
|
|
}
|
2020-03-12 19:14:50 +01:00
|
|
|
}
|
2020-07-13 08:03:56 -04:00
|
|
|
//# sourceMappingURL=websocket-provider.js.map
|