2022-09-05 23:57:11 +03:00
|
|
|
"use strict";
|
|
|
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
|
|
exports.PollingEventSubscriber = exports.PollingTransactionSubscriber = exports.PollingOrphanSubscriber = exports.OnBlockSubscriber = exports.PollingBlockSubscriber = exports.getPollingSubscriber = void 0;
|
2022-09-16 05:58:45 +03:00
|
|
|
const index_js_1 = require("../utils/index.js");
|
2022-09-05 23:57:11 +03:00
|
|
|
function copy(obj) {
|
|
|
|
return JSON.parse(JSON.stringify(obj));
|
|
|
|
}
|
2022-11-30 23:44:23 +03:00
|
|
|
/**
|
|
|
|
* @TODO
|
|
|
|
*
|
|
|
|
* @_docloc: api/providers/abstract-provider
|
|
|
|
*/
|
2022-09-05 23:57:11 +03:00
|
|
|
function getPollingSubscriber(provider, event) {
|
|
|
|
if (event === "block") {
|
|
|
|
return new PollingBlockSubscriber(provider);
|
|
|
|
}
|
2022-09-16 05:58:45 +03:00
|
|
|
if ((0, index_js_1.isHexString)(event, 32)) {
|
2022-09-05 23:57:11 +03:00
|
|
|
return new PollingTransactionSubscriber(provider, event);
|
|
|
|
}
|
2022-11-09 10:57:02 +03:00
|
|
|
(0, index_js_1.assert)(false, "unsupported polling event", "UNSUPPORTED_OPERATION", {
|
2022-09-05 23:57:11 +03:00
|
|
|
operation: "getPollingSubscriber", info: { event }
|
|
|
|
});
|
|
|
|
}
|
|
|
|
exports.getPollingSubscriber = getPollingSubscriber;
|
|
|
|
// @TODO: refactor this
|
2022-11-30 23:44:23 +03:00
|
|
|
/**
|
|
|
|
* @TODO
|
|
|
|
*
|
|
|
|
* @_docloc: api/providers/abstract-provider
|
|
|
|
*/
|
2022-09-05 23:57:11 +03:00
|
|
|
class PollingBlockSubscriber {
|
|
|
|
#provider;
|
|
|
|
#poller;
|
|
|
|
#interval;
|
|
|
|
// The most recent block we have scanned for events. The value -2
|
|
|
|
// indicates we still need to fetch an initial block number
|
|
|
|
#blockNumber;
|
|
|
|
constructor(provider) {
|
|
|
|
this.#provider = provider;
|
|
|
|
this.#poller = null;
|
|
|
|
this.#interval = 4000;
|
|
|
|
this.#blockNumber = -2;
|
|
|
|
}
|
|
|
|
get pollingInterval() { return this.#interval; }
|
|
|
|
set pollingInterval(value) { this.#interval = value; }
|
|
|
|
async #poll() {
|
|
|
|
const blockNumber = await this.#provider.getBlockNumber();
|
|
|
|
if (this.#blockNumber === -2) {
|
|
|
|
this.#blockNumber = blockNumber;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// @TODO: Put a cap on the maximum number of events per loop?
|
|
|
|
if (blockNumber !== this.#blockNumber) {
|
|
|
|
for (let b = this.#blockNumber + 1; b <= blockNumber; b++) {
|
|
|
|
this.#provider.emit("block", b);
|
|
|
|
}
|
|
|
|
this.#blockNumber = blockNumber;
|
|
|
|
}
|
|
|
|
this.#poller = this.#provider._setTimeout(this.#poll.bind(this), this.#interval);
|
|
|
|
}
|
|
|
|
start() {
|
|
|
|
if (this.#poller) {
|
|
|
|
throw new Error("subscriber already running");
|
|
|
|
}
|
|
|
|
this.#poll();
|
|
|
|
this.#poller = this.#provider._setTimeout(this.#poll.bind(this), this.#interval);
|
|
|
|
}
|
|
|
|
stop() {
|
|
|
|
if (!this.#poller) {
|
|
|
|
throw new Error("subscriber not running");
|
|
|
|
}
|
|
|
|
this.#provider._clearTimeout(this.#poller);
|
|
|
|
this.#poller = null;
|
|
|
|
}
|
|
|
|
pause(dropWhilePaused) {
|
|
|
|
this.stop();
|
|
|
|
if (dropWhilePaused) {
|
|
|
|
this.#blockNumber = -2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
resume() {
|
|
|
|
this.start();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
exports.PollingBlockSubscriber = PollingBlockSubscriber;
|
2022-11-30 23:44:23 +03:00
|
|
|
/**
|
|
|
|
* @TODO
|
|
|
|
*
|
|
|
|
* @_docloc: api/providers/abstract-provider
|
|
|
|
*/
|
2022-09-05 23:57:11 +03:00
|
|
|
class OnBlockSubscriber {
|
|
|
|
#provider;
|
|
|
|
#poll;
|
|
|
|
constructor(provider) {
|
|
|
|
this.#provider = provider;
|
|
|
|
this.#poll = (blockNumber) => {
|
|
|
|
this._poll(blockNumber, this.#provider);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
async _poll(blockNumber, provider) {
|
|
|
|
throw new Error("sub-classes must override this");
|
|
|
|
}
|
|
|
|
start() {
|
|
|
|
this.#poll(-2);
|
|
|
|
this.#provider.on("block", this.#poll);
|
|
|
|
}
|
|
|
|
stop() {
|
|
|
|
this.#provider.off("block", this.#poll);
|
|
|
|
}
|
|
|
|
pause(dropWhilePaused) { this.stop(); }
|
|
|
|
resume() { this.start(); }
|
|
|
|
}
|
|
|
|
exports.OnBlockSubscriber = OnBlockSubscriber;
|
2022-11-30 23:44:23 +03:00
|
|
|
/**
|
|
|
|
* @TODO
|
|
|
|
*
|
|
|
|
* @_docloc: api/providers/abstract-provider
|
|
|
|
*/
|
2022-09-05 23:57:11 +03:00
|
|
|
class PollingOrphanSubscriber extends OnBlockSubscriber {
|
|
|
|
#filter;
|
|
|
|
constructor(provider, filter) {
|
|
|
|
super(provider);
|
|
|
|
this.#filter = copy(filter);
|
|
|
|
}
|
|
|
|
async _poll(blockNumber, provider) {
|
|
|
|
throw new Error("@TODO");
|
|
|
|
console.log(this.#filter);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
exports.PollingOrphanSubscriber = PollingOrphanSubscriber;
|
2022-11-30 23:44:23 +03:00
|
|
|
/**
|
|
|
|
* @TODO
|
|
|
|
*
|
|
|
|
* @_docloc: api/providers/abstract-provider
|
|
|
|
*/
|
2022-09-05 23:57:11 +03:00
|
|
|
class PollingTransactionSubscriber extends OnBlockSubscriber {
|
|
|
|
#hash;
|
|
|
|
constructor(provider, hash) {
|
|
|
|
super(provider);
|
|
|
|
this.#hash = hash;
|
|
|
|
}
|
|
|
|
async _poll(blockNumber, provider) {
|
|
|
|
const tx = await provider.getTransactionReceipt(this.#hash);
|
|
|
|
if (tx) {
|
|
|
|
provider.emit(this.#hash, tx);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
exports.PollingTransactionSubscriber = PollingTransactionSubscriber;
|
2022-11-30 23:44:23 +03:00
|
|
|
/**
|
|
|
|
* @TODO
|
|
|
|
*
|
|
|
|
* @_docloc: api/providers/abstract-provider
|
|
|
|
*/
|
2022-09-05 23:57:11 +03:00
|
|
|
class PollingEventSubscriber {
|
|
|
|
#provider;
|
|
|
|
#filter;
|
|
|
|
#poller;
|
|
|
|
// The most recent block we have scanned for events. The value -2
|
|
|
|
// indicates we still need to fetch an initial block number
|
|
|
|
#blockNumber;
|
|
|
|
constructor(provider, filter) {
|
|
|
|
this.#provider = provider;
|
|
|
|
this.#filter = copy(filter);
|
|
|
|
this.#poller = this.#poll.bind(this);
|
|
|
|
this.#blockNumber = -2;
|
|
|
|
}
|
|
|
|
async #poll(blockNumber) {
|
|
|
|
// The initial block hasn't been determined yet
|
|
|
|
if (this.#blockNumber === -2) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const filter = copy(this.#filter);
|
|
|
|
filter.fromBlock = this.#blockNumber + 1;
|
|
|
|
filter.toBlock = blockNumber;
|
|
|
|
const logs = await this.#provider.getLogs(filter);
|
|
|
|
// No logs could just mean the node has not indexed them yet,
|
|
|
|
// so we keep a sliding window of 60 blocks to keep scanning
|
|
|
|
if (logs.length === 0) {
|
|
|
|
if (this.#blockNumber < blockNumber - 60) {
|
|
|
|
this.#blockNumber = blockNumber - 60;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
this.#blockNumber = blockNumber;
|
|
|
|
for (const log of logs) {
|
|
|
|
this.#provider.emit(this.#filter, log);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
start() {
|
|
|
|
if (this.#blockNumber === -2) {
|
|
|
|
this.#provider.getBlockNumber().then((blockNumber) => {
|
|
|
|
this.#blockNumber = blockNumber;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
this.#provider.on("block", this.#poller);
|
|
|
|
}
|
|
|
|
stop() {
|
|
|
|
this.#provider.off("block", this.#poller);
|
|
|
|
}
|
|
|
|
pause(dropWhilePaused) {
|
|
|
|
this.stop();
|
|
|
|
if (dropWhilePaused) {
|
|
|
|
this.#blockNumber = -2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
resume() {
|
|
|
|
this.start();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
exports.PollingEventSubscriber = PollingEventSubscriber;
|
|
|
|
//# sourceMappingURL=subscriber-polling.js.map
|