feat: observables for callable functions (#71)
* return eth subscription and observable array * adding checksum address function to utils * use observables triggered by a subject
This commit is contained in:
parent
2646cd58d1
commit
8dea143adc
|
@ -67,7 +67,7 @@ class EventSyncer {
|
||||||
|
|
||||||
if (this.isWebsocketProvider) {
|
if (this.isWebsocketProvider) {
|
||||||
const fnSubscribe = this.subscribeToEvent(eventKey, contractInstance, eventName);
|
const fnSubscribe = this.subscribeToEvent(eventKey, contractInstance, eventName);
|
||||||
const eth_subscribe = this.eventScanner.scan(
|
const ethSubscription = this.eventScanner.scan(
|
||||||
fnDBEvents,
|
fnDBEvents,
|
||||||
fnPastEvents,
|
fnPastEvents,
|
||||||
fnSubscribe,
|
fnSubscribe,
|
||||||
|
@ -75,25 +75,11 @@ class EventSyncer {
|
||||||
lastKnownBlock,
|
lastKnownBlock,
|
||||||
filterConditions
|
filterConditions
|
||||||
);
|
);
|
||||||
|
return [sub, ethSubscription];
|
||||||
const og_subscribe = sub.subscribe;
|
|
||||||
sub.subscribe = async (next, error, complete) => {
|
|
||||||
const s = og_subscribe.apply(sub, [next, error, complete]);
|
|
||||||
s.add(() => {
|
|
||||||
// Removing web3js subscription when rxJS unsubscribe is executed
|
|
||||||
eth_subscribe.then(susc => {
|
|
||||||
if (susc) {
|
|
||||||
susc.unsubscribe();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
return s;
|
|
||||||
};
|
|
||||||
} else {
|
} else {
|
||||||
this.eventScanner.scan(fnDBEvents, fnPastEvents, lastKnownBlock, filterConditions);
|
this.eventScanner.scan(fnDBEvents, fnPastEvents, lastKnownBlock, filterConditions);
|
||||||
|
return [sub, undefined];
|
||||||
}
|
}
|
||||||
|
|
||||||
return sub;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
getPastEvents = (eventKey, contractInstance, eventName, filters) => async (fromBlock, toBlock, hardLimit) => {
|
getPastEvents = (eventKey, contractInstance, eventName, filters) => async (fromBlock, toBlock, hardLimit) => {
|
||||||
|
|
|
@ -67,24 +67,14 @@ class LogSyncer {
|
||||||
this.events.emit("updateDB");
|
this.events.emit("updateDB");
|
||||||
});
|
});
|
||||||
|
|
||||||
const eth_subscribe = this._retrieveEvents(
|
const ethSubscription = this._retrieveEvents(
|
||||||
eventKey,
|
eventKey,
|
||||||
eventSummary.firstKnownBlock,
|
eventSummary.firstKnownBlock,
|
||||||
eventSummary.lastKnownBlock,
|
eventSummary.lastKnownBlock,
|
||||||
filterConditions
|
filterConditions
|
||||||
);
|
);
|
||||||
|
|
||||||
const og_subscribe = sub.subscribe;
|
return [sub, ethSubscription];
|
||||||
sub.subscribe = (next, error, complete) => {
|
|
||||||
const s = og_subscribe.apply(sub, [next, error, complete]);
|
|
||||||
s.add(() => {
|
|
||||||
// Removing web3js subscription when rxJS unsubscribe is executed
|
|
||||||
if (eth_subscribe) eth_subscribe.unsubscribe();
|
|
||||||
});
|
|
||||||
return s;
|
|
||||||
};
|
|
||||||
|
|
||||||
return sub;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
_retrieveEvents(eventKey, firstKnownBlock, lastKnownBlock, filterConditions) {
|
_retrieveEvents(eventKey, firstKnownBlock, lastKnownBlock, filterConditions) {
|
||||||
|
|
191
src/subspace.js
191
src/subspace.js
|
@ -1,5 +1,5 @@
|
||||||
import {ReplaySubject, BehaviorSubject} from "rxjs";
|
import {BehaviorSubject, from} from "rxjs";
|
||||||
import {distinctUntilChanged, map} from "rxjs/operators";
|
import {distinctUntilChanged, map, exhaustMap, shareReplay} from "rxjs/operators";
|
||||||
import equal from "fast-deep-equal";
|
import equal from "fast-deep-equal";
|
||||||
import Database from "./database/database.js";
|
import Database from "./database/database.js";
|
||||||
import NullDatabase from "./database/nullDatabase.js";
|
import NullDatabase from "./database/nullDatabase.js";
|
||||||
|
@ -13,8 +13,7 @@ import LogSyncer from "./logSyncer";
|
||||||
import hash from "object-hash";
|
import hash from "object-hash";
|
||||||
|
|
||||||
export default class Subspace {
|
export default class Subspace {
|
||||||
subjects = {};
|
observables = {};
|
||||||
callables = [];
|
|
||||||
|
|
||||||
newBlocksSubscription = null;
|
newBlocksSubscription = null;
|
||||||
intervalTracker = null;
|
intervalTracker = null;
|
||||||
|
@ -40,6 +39,7 @@ export default class Subspace {
|
||||||
|
|
||||||
this.networkId = undefined;
|
this.networkId = undefined;
|
||||||
this.isWebsocketProvider = options.disableSubscriptions ? false : !!provider.on;
|
this.isWebsocketProvider = options.disableSubscriptions ? false : !!provider.on;
|
||||||
|
this.triggerSubject = new BehaviorSubject();
|
||||||
}
|
}
|
||||||
|
|
||||||
init() {
|
init() {
|
||||||
|
@ -154,32 +154,36 @@ export default class Subspace {
|
||||||
console.error(err);
|
console.error(err);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
this.triggerSubject.next();
|
||||||
this.callables.forEach(fn => fn());
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
_initCallInterval() {
|
_initCallInterval() {
|
||||||
if (this.intervalTracker != null || this.options.callInterval === 0) return;
|
if (this.intervalTracker != null || this.options.callInterval === 0) return;
|
||||||
|
this.intervalTracker = setInterval(() => this.triggerSubject.next(), this.options.callInterval);
|
||||||
this.intervalTracker = setInterval(() => {
|
|
||||||
this.callables.forEach(fn => fn());
|
|
||||||
}, this.options.callInterval);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
_getSubject(subjectHash, subjectCB) {
|
_getObservable(subjectHash, observableBuilder) {
|
||||||
if (this.subjects[subjectHash]) return this.subjects[subjectHash];
|
if (this.observables[subjectHash]) return this.observables[subjectHash];
|
||||||
this.subjects[subjectHash] = subjectCB();
|
this.observables[subjectHash] = observableBuilder();
|
||||||
return this.subjects[subjectHash];
|
return this.observables[subjectHash];
|
||||||
}
|
}
|
||||||
|
|
||||||
_addDistinctCallable(trackAttribute, cbBuilder, SubjectType, subjectArg = undefined) {
|
_getDistinctObservableFromPromise(subjectName, promiseCB, cb) {
|
||||||
return this._getSubject(trackAttribute, () => {
|
return this._getObservable(subjectName, () => {
|
||||||
const sub = new SubjectType(subjectArg);
|
let observable = this.triggerSubject.pipe(
|
||||||
const cb = cbBuilder(sub);
|
exhaustMap(() => from(promiseCB())),
|
||||||
cb();
|
distinctUntilChanged((a, b) => equal(a, b))
|
||||||
this.callables.push(cb);
|
);
|
||||||
return sub.pipe(distinctUntilChanged((a, b) => equal(a, b)));
|
|
||||||
|
if(cb){
|
||||||
|
observable = observable.pipe(map(x => {
|
||||||
|
cb(x);
|
||||||
|
return x;
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
return observable.pipe(shareReplay({refCount: true, bufferSize: 1}));
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -190,11 +194,14 @@ export default class Subspace {
|
||||||
eventName,
|
eventName,
|
||||||
filterConditions
|
filterConditions
|
||||||
});
|
});
|
||||||
return this._getSubject(subjectHash, () => {
|
|
||||||
let deleteFrom = this.latestBlockNumber - this.options.refreshLastNBlocks;
|
|
||||||
let returnSub = this.eventSyncer.track(contractInstance, eventName, filterConditions, deleteFrom, this.networkId);
|
|
||||||
|
|
||||||
returnSub.map = prop => {
|
return this._getObservable(subjectHash, () => {
|
||||||
|
const deleteFrom = this.latestBlockNumber - this.options.refreshLastNBlocks;
|
||||||
|
const [subject, ethSubscription] = this.eventSyncer.track(contractInstance, eventName, filterConditions, deleteFrom, this.networkId);
|
||||||
|
|
||||||
|
// TODO: remove eth subscription
|
||||||
|
|
||||||
|
subject.map = prop => {
|
||||||
return returnSub.pipe(
|
return returnSub.pipe(
|
||||||
map(x => {
|
map(x => {
|
||||||
if (typeof prop === "string") {
|
if (typeof prop === "string") {
|
||||||
|
@ -211,7 +218,7 @@ export default class Subspace {
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
return returnSub;
|
return subject;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -219,13 +226,20 @@ export default class Subspace {
|
||||||
if (!this.isWebsocketProvider) console.warn("This method only works with websockets");
|
if (!this.isWebsocketProvider) console.warn("This method only works with websockets");
|
||||||
|
|
||||||
const subjectHash = hash({inputsABI, options});
|
const subjectHash = hash({inputsABI, options});
|
||||||
return this._getSubject(subjectHash, () =>
|
return this._getObservable(subjectHash, () => {
|
||||||
this.logSyncer.track(options, inputsABI, this.latestBlockNumber - this.options.refreshLastNBlocks, this.networkId)
|
const [subject, ethSubscription] = this.logSyncer.track(
|
||||||
|
options,
|
||||||
|
inputsABI,
|
||||||
|
this.latestBlockNumber - this.options.refreshLastNBlocks,
|
||||||
|
this.networkId
|
||||||
);
|
);
|
||||||
|
// TODO: remove eth subscription
|
||||||
|
return subject;
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
trackProperty(contractInstance, propName, methodArgs = [], callArgs = {}) {
|
trackProperty(contractInstance, propName, methodArgs = [], callArgs = {}) {
|
||||||
const subjectHash = hash({
|
const identifier = hash({
|
||||||
address: contractInstance.options.address,
|
address: contractInstance.options.address,
|
||||||
networkId: this.networkId,
|
networkId: this.networkId,
|
||||||
propName,
|
propName,
|
||||||
|
@ -233,36 +247,16 @@ export default class Subspace {
|
||||||
callArgs
|
callArgs
|
||||||
});
|
});
|
||||||
|
|
||||||
return this._getSubject(subjectHash, () => {
|
const observable = this._getDistinctObservableFromPromise(identifier, () => {
|
||||||
const subject = new ReplaySubject(1);
|
|
||||||
|
|
||||||
if (!Array.isArray(methodArgs)) {
|
if (!Array.isArray(methodArgs)) {
|
||||||
methodArgs = [methodArgs];
|
methodArgs = [methodArgs];
|
||||||
}
|
}
|
||||||
|
|
||||||
const method = contractInstance.methods[propName].apply(contractInstance.methods[propName], methodArgs);
|
const method = contractInstance.methods[propName].apply(contractInstance.methods[propName], methodArgs);
|
||||||
|
return method.call.apply(method.call, [callArgs]);
|
||||||
|
});
|
||||||
|
|
||||||
const callContractMethod = () => {
|
observable.map = prop => {
|
||||||
method.call.apply(method.call, [
|
return observable.pipe(
|
||||||
callArgs,
|
|
||||||
(err, result) => {
|
|
||||||
if (err) {
|
|
||||||
subject.error(err);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
subject.next(result);
|
|
||||||
}
|
|
||||||
]);
|
|
||||||
};
|
|
||||||
|
|
||||||
callContractMethod();
|
|
||||||
|
|
||||||
this.callables.push(callContractMethod);
|
|
||||||
|
|
||||||
const returnSub = subject.pipe(distinctUntilChanged((a, b) => equal(a, b)));
|
|
||||||
|
|
||||||
returnSub.map = prop => {
|
|
||||||
return returnSub.pipe(
|
|
||||||
map(x => {
|
map(x => {
|
||||||
if (typeof prop === "string") {
|
if (typeof prop === "string") {
|
||||||
return x[prop];
|
return x[prop];
|
||||||
|
@ -278,99 +272,47 @@ export default class Subspace {
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
return returnSub;
|
return observable;
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
trackBalance(address, erc20Address) {
|
trackBalance(address, erc20Address) {
|
||||||
if (!isAddress(address)) throw "invalid address";
|
if (!isAddress(address)) throw "invalid address";
|
||||||
if (erc20Address && !isAddress(erc20Address)) throw "invalid ERC20 contract address";
|
if (erc20Address && !isAddress(erc20Address)) throw "invalid ERC20 contract address";
|
||||||
|
|
||||||
const subjectHash = hash({address, erc20Address});
|
address = toChecksumAddress(address);
|
||||||
|
erc20Address = toChecksumAddress(address);
|
||||||
|
|
||||||
const getETHBalance = cb => {
|
return this._getDistinctObservableFromPromise(hash({address, erc20Address}), () => {
|
||||||
const fn = this.web3.getBalance;
|
if (!erc20Address) {
|
||||||
fn.apply(fn, [address, cb]);
|
return this.web3.getBalance(address);
|
||||||
};
|
} else {
|
||||||
|
|
||||||
const getTokenBalance = cb => {
|
|
||||||
const fn = this.web3.call;
|
|
||||||
// balanceOf
|
// balanceOf
|
||||||
const data = "0x70a08231" + "000000000000000000000000" + stripHexPrefix(address);
|
const data = "0x70a08231" + "000000000000000000000000" + stripHexPrefix(address);
|
||||||
fn.apply(fn, [{to: erc20Address, data}, cb]);
|
return new Promise((resolve, reject) => this.web3.call({to: erc20Address, data}).then(balance => resolve(hexToDec(balance))).catch(reject));
|
||||||
};
|
|
||||||
|
|
||||||
let callFn;
|
|
||||||
if (!erc20Address) {
|
|
||||||
callFn = subject => () =>
|
|
||||||
getETHBalance((err, balance) => {
|
|
||||||
if (err) {
|
|
||||||
subject.error(err);
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
subject.next(balance);
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
callFn = subject => () =>
|
|
||||||
getTokenBalance((err, balance) => {
|
|
||||||
if (err) {
|
|
||||||
subject.error(err);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
subject.next(hexToDec(balance));
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
return this._addDistinctCallable(subjectHash, callFn, ReplaySubject, 1);
|
trackBlockNumber() {
|
||||||
|
return this._getDistinctObservableFromPromise("blockNumber", () => this.web3.getBlockNumber());
|
||||||
}
|
}
|
||||||
|
|
||||||
trackBlock() {
|
trackBlock() {
|
||||||
const blockCB = subject => () => {
|
return this._getDistinctObservableFromPromise("gasPrice", () => this.web3.getBlock("latest"), block => {
|
||||||
this.web3
|
|
||||||
.getBlock("latest")
|
|
||||||
.then(block => {
|
|
||||||
if (this.latest10Blocks[this.latest10Blocks.length - 1].number === block.number) return;
|
if (this.latest10Blocks[this.latest10Blocks.length - 1].number === block.number) return;
|
||||||
|
|
||||||
this.latest10Blocks.push(block);
|
this.latest10Blocks.push(block);
|
||||||
if (this.latest10Blocks.length > 10) {
|
if (this.latest10Blocks.length > 10) {
|
||||||
this.latest10Blocks.shift();
|
this.latest10Blocks.shift();
|
||||||
}
|
}
|
||||||
subject.next(block);
|
});
|
||||||
})
|
|
||||||
.catch(error => subject.error(error));
|
|
||||||
};
|
|
||||||
|
|
||||||
return this._addDistinctCallable(
|
|
||||||
"blockObservable",
|
|
||||||
blockCB,
|
|
||||||
BehaviorSubject,
|
|
||||||
this.latest10Blocks[this.latest10Blocks.length - 1]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
trackBlockNumber() {
|
|
||||||
const blockNumberCB = subject => () => {
|
|
||||||
this.web3
|
|
||||||
.getBlockNumber()
|
|
||||||
.then(result => subject.next(result))
|
|
||||||
.catch(error => subject.error(error));
|
|
||||||
};
|
|
||||||
return this._addDistinctCallable("blockNumberObservable", blockNumberCB, ReplaySubject, 1);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
trackGasPrice() {
|
trackGasPrice() {
|
||||||
const gasPriceCB = subject => () => {
|
return this._getDistinctObservableFromPromise("gasPrice", () => this.web3.getGasPrice());
|
||||||
this.web3
|
|
||||||
.getGasPrice()
|
|
||||||
.then(result => subject.next(result))
|
|
||||||
.catch(error => subject.error(error));
|
|
||||||
};
|
|
||||||
return this._addDistinctCallable("gasPriceObservable", gasPriceCB, ReplaySubject, 1);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
trackAverageBlocktime() {
|
trackAverageBlocktime() {
|
||||||
this.trackBlock();
|
return this._getObservable("avgBlockTime", () => {
|
||||||
|
|
||||||
const calcAverage = () => {
|
const calcAverage = () => {
|
||||||
const times = [];
|
const times = [];
|
||||||
for (let i = 1; i < this.latest10Blocks.length; i++) {
|
for (let i = 1; i < this.latest10Blocks.length; i++) {
|
||||||
|
@ -380,9 +322,11 @@ export default class Subspace {
|
||||||
return times.length ? Math.round(times.reduce((a, b) => a + b) / times.length) * 1000 : 0;
|
return times.length ? Math.round(times.reduce((a, b) => a + b) / times.length) * 1000 : 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
const avgTimeCB = subject => () => subject.next(calcAverage());
|
return this.trackBlock().pipe(
|
||||||
|
map(() => calcAverage()),
|
||||||
return this._addDistinctCallable("blockTimeObservable", avgTimeCB, BehaviorSubject, calcAverage());
|
distinctUntilChanged((a, b) => equal(a, b))
|
||||||
|
);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
close() {
|
close() {
|
||||||
|
@ -390,6 +334,5 @@ export default class Subspace {
|
||||||
if (this.newBlocksSubscription) this.newBlocksSubscription.unsubscribe();
|
if (this.newBlocksSubscription) this.newBlocksSubscription.unsubscribe();
|
||||||
this.eventSyncer.close();
|
this.eventSyncer.close();
|
||||||
this.intervalTracker = null;
|
this.intervalTracker = null;
|
||||||
this.callables = [];
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
18
src/utils.js
18
src/utils.js
|
@ -2,6 +2,24 @@ export function isAddress(address) {
|
||||||
return /^(0x)?[0-9a-fA-F]{40}$/i.test(address);
|
return /^(0x)?[0-9a-fA-F]{40}$/i.test(address);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function toChecksumAddress(address) {
|
||||||
|
if (typeof address === "undefined") return "";
|
||||||
|
|
||||||
|
address = address.toLowerCase().replace(/^0x/i, "");
|
||||||
|
var addressHash = utils.sha3(address).replace(/^0x/i, "");
|
||||||
|
var checksumAddress = "0x";
|
||||||
|
|
||||||
|
for (var i = 0; i < address.length; i++) {
|
||||||
|
// If ith character is 9 to f then make it uppercase
|
||||||
|
if (parseInt(addressHash[i], 16) > 7) {
|
||||||
|
checksumAddress += address[i].toUpperCase();
|
||||||
|
} else {
|
||||||
|
checksumAddress += address[i];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return checksumAddress;
|
||||||
|
}
|
||||||
|
|
||||||
export function sleep(milliseconds) {
|
export function sleep(milliseconds) {
|
||||||
return new Promise(resolve => setTimeout(resolve, milliseconds));
|
return new Promise(resolve => setTimeout(resolve, milliseconds));
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue