feat: store event data and retrieve from db (#29)
* feat: store event data and retrieve from db * refactor: event tracking
This commit is contained in:
parent
bbcada804e
commit
2a213f053b
|
@ -22,8 +22,9 @@
|
||||||
"@babel/core": "^7.1.6",
|
"@babel/core": "^7.1.6",
|
||||||
"@babel/plugin-proposal-class-properties": "^7.5.5",
|
"@babel/plugin-proposal-class-properties": "^7.5.5",
|
||||||
"@babel/preset-env": "^7.1.6",
|
"@babel/preset-env": "^7.1.6",
|
||||||
"babel-loader": "^8.0.4",
|
|
||||||
"add-module-exports-webpack-plugin": "^1.0.0",
|
"add-module-exports-webpack-plugin": "^1.0.0",
|
||||||
|
"babel-loader": "^8.0.4",
|
||||||
|
"ganache-core": "^2.7.0",
|
||||||
"cross-env": "^5.2.0",
|
"cross-env": "^5.2.0",
|
||||||
"npm-run-all": "^4.1.5",
|
"npm-run-all": "^4.1.5",
|
||||||
"rimraf": "^2.6.2",
|
"rimraf": "^2.6.2",
|
||||||
|
@ -36,6 +37,7 @@
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"fast-deep-equal": "^2.0.1",
|
"fast-deep-equal": "^2.0.1",
|
||||||
"lokijs": "^1.5.6",
|
"lokijs": "^1.5.6",
|
||||||
|
"object-hash": "^1.3.1",
|
||||||
"rxjs": "^6.5.2",
|
"rxjs": "^6.5.2",
|
||||||
"web3-eth": "^1.2.1"
|
"web3-eth": "^1.2.1"
|
||||||
},
|
},
|
||||||
|
|
|
@ -1,5 +1,4 @@
|
||||||
import { fromEvent, interval, ReplaySubject } from 'rxjs';
|
import { fromEvent } from 'rxjs';
|
||||||
import { throttle, distinctUntilChanged } from 'rxjs/operators';
|
|
||||||
import loki from 'lokijs';
|
import loki from 'lokijs';
|
||||||
|
|
||||||
const getENV = function () {
|
const getENV = function () {
|
||||||
|
@ -49,47 +48,47 @@ class Database {
|
||||||
children = this.db.addCollection('children')
|
children = this.db.addCollection('children')
|
||||||
this.db.saveDatabase()
|
this.db.saveDatabase()
|
||||||
}
|
}
|
||||||
let tracked = this.db.getCollection('tracked')
|
|
||||||
if (!tracked) {
|
|
||||||
tracked = this.db.addCollection('tracked')
|
|
||||||
this.db.saveDatabase()
|
|
||||||
}
|
|
||||||
|
|
||||||
let dbChanges = fromEvent(this.events, "updateDB")
|
let dbChanges = fromEvent(this.events, "updateDB")
|
||||||
dbChanges.pipe(throttle(val => interval(400))).subscribe(() => {
|
dbChanges.subscribe(() => {
|
||||||
this.db.saveDatabase()
|
this.db.saveDatabase()
|
||||||
})
|
})
|
||||||
|
|
||||||
cb();
|
cb();
|
||||||
}
|
}
|
||||||
|
|
||||||
getLastKnownEvent(eventName) {
|
getLastKnownEvent(eventKey) {
|
||||||
let tracked = this.db.getCollection('tracked');
|
const collection = this.db.getCollection(eventKey);
|
||||||
let lastEvent = tracked.find({ "eventName": eventName })[0];
|
|
||||||
if (!lastEvent || lastEvent.length <= 0) {
|
let firstKnownBlock = 0;
|
||||||
tracked.insert({ "eventName": eventName, id: 0 });
|
let lastKnownBlock = 0;
|
||||||
lastEvent = tracked.find({ "eventName": eventName })[0];
|
|
||||||
}
|
if(collection && collection.count()){
|
||||||
return lastEvent;
|
firstKnownBlock = collection.min('blockNumber');
|
||||||
|
lastKnownBlock = collection.max('blockNumber');
|
||||||
|
} else {
|
||||||
|
this.db.addCollection(eventKey);
|
||||||
}
|
}
|
||||||
|
|
||||||
updateEventId(eventName, eventId) {
|
return {
|
||||||
let tracked = this.db.getCollection('tracked');
|
firstKnownBlock,
|
||||||
tracked.updateWhere(((x) => x.eventName === eventName), ((x) => x.id = eventId));
|
lastKnownBlock
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
getEventsFor(eventKey) {
|
getEventsFor(eventKey) {
|
||||||
let children = this.db.getCollection('children');
|
let children = this.db.getCollection(eventKey);
|
||||||
return children.find({ 'eventKey': eventKey });
|
return children.find();
|
||||||
}
|
}
|
||||||
|
|
||||||
eventExists(eventId) {
|
eventExists(eventKey, eventId) {
|
||||||
let children = this.db.getCollection('children');
|
let children = this.db.getCollection(eventKey);
|
||||||
return (children.find({ 'id': eventId }).length > 0);
|
return (children.find({ 'id': eventId }).length > 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
recordEvent(values) {
|
recordEvent(eventKey, values) {
|
||||||
let children = this.db.getCollection('children');
|
let children = this.db.getCollection(eventKey);
|
||||||
children.insert(values);
|
children.insert(values);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,8 +1,7 @@
|
||||||
import { fromEvent, interval, ReplaySubject } from 'rxjs';
|
import { fromEvent, ReplaySubject } from 'rxjs';
|
||||||
import { throttle, distinctUntilChanged } from 'rxjs/operators';
|
import { distinctUntilChanged } from 'rxjs/operators';
|
||||||
import { randomString } from './utils.js';
|
|
||||||
import equal from 'fast-deep-equal';
|
import equal from 'fast-deep-equal';
|
||||||
|
import hash from 'object-hash';
|
||||||
import Database from './database.js';
|
import Database from './database.js';
|
||||||
import Events from 'events';
|
import Events from 'events';
|
||||||
import Web3Eth from 'web3-eth';
|
import Web3Eth from 'web3-eth';
|
||||||
|
@ -33,60 +32,148 @@ export default class EventSyncer {
|
||||||
|
|
||||||
// TODO: get contract abi/address instead
|
// TODO: get contract abi/address instead
|
||||||
trackEvent(contractInstance, eventName, filterConditionsOrCb) {
|
trackEvent(contractInstance, eventName, filterConditionsOrCb) {
|
||||||
// let eventKey = eventName + "-from0x123";
|
let eventKey = eventName + '-' + hash(filterConditionsOrCb);
|
||||||
let eventKey = eventName;
|
|
||||||
|
|
||||||
let filterConditions, filterConditionsCb;
|
let filterConditions = {fromBlock: 0, toBlock: "latest"};
|
||||||
|
let filterConditionsCb;
|
||||||
if (typeof filterConditionsOrCb === 'function') {
|
if (typeof filterConditionsOrCb === 'function') {
|
||||||
filterConditionsCb = filterConditionsOrCb
|
filterConditionsCb = filterConditionsOrCb;
|
||||||
} else {
|
} else {
|
||||||
filterConditions = filterConditionsOrCb
|
filterConditions = Object.assign(filterConditions, filterConditionsOrCb || {});
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: should use this to resume events tracking
|
let eventSummary = this._db.getLastKnownEvent(eventKey);
|
||||||
// let lastEvent = this._db.getLastKnownEvent(eventName)
|
|
||||||
|
|
||||||
let sub = new ReplaySubject();
|
let sub = new ReplaySubject();
|
||||||
|
let contractObserver = fromEvent(this.events, eventKey)
|
||||||
|
|
||||||
this._db.getEventsFor(eventKey).forEach(sub.next);
|
contractObserver.subscribe((e) => {
|
||||||
|
if(!e) return;
|
||||||
|
|
||||||
let eventbusKey = "event-" + eventName + "-" + randomString();
|
// TODO: would be nice if trackEvent was smart enough to understand the type of returnValues and do the needed conversions
|
||||||
let contractObserver = fromEvent(this.events, eventbusKey)
|
|
||||||
|
|
||||||
|
const eventData = {
|
||||||
|
id: hash({eventName, blockNumber: e.blockNumber, transactionIndex: e.transactionIndex, logIndex: e.logIndex}),
|
||||||
|
returnValues: {...e.returnValues},
|
||||||
|
blockNumber: e.blockNumber,
|
||||||
|
transactionIndex: e.transactionIndex,
|
||||||
|
logIndex: e.logIndex
|
||||||
|
}
|
||||||
|
|
||||||
|
sub.next({blockNumber: e.blockNumber, ...e.returnValues});
|
||||||
|
|
||||||
|
if (this._db.eventExists(eventKey, eventData.id)) return;
|
||||||
|
|
||||||
|
this._db.recordEvent(eventKey, eventData);
|
||||||
|
|
||||||
|
this.events.emit("updateDB");
|
||||||
|
});
|
||||||
|
|
||||||
|
this._retrieveEvents(eventKey,
|
||||||
|
eventSummary.firstKnownBlock,
|
||||||
|
eventSummary.lastKnownBlock,
|
||||||
|
filterConditions,
|
||||||
|
filterConditionsCb,
|
||||||
|
contractInstance,
|
||||||
|
eventName);
|
||||||
|
|
||||||
|
return sub;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
_retrieveEvents(eventKey, firstKnownBlock, lastKnownBlock, filterConditions, filterConditionsCb, contractInstance, eventName) {
|
||||||
// TODO: this should be moved to a 'smart' module
|
// TODO: this should be moved to a 'smart' module
|
||||||
// for e.g, it should start fromBlock, from the latest known block (which means it should store block info)
|
|
||||||
// it should be able to do events X at the time to avoid slow downs as well as the 10k limit
|
// it should be able to do events X at the time to avoid slow downs as well as the 10k limit
|
||||||
contractInstance.events[eventName].apply(contractInstance.events[eventName], [(filterConditions || {fromBlock: 0}), (err, event) => {
|
// TODO: filter subscriptions with fromBlock and toBlock
|
||||||
|
|
||||||
|
if (firstKnownBlock == 0 || (firstKnownBlock > 0 && firstKnownBlock <= filterConditions.fromBlock)) {
|
||||||
|
if (filterConditions.toBlock === 'latest') {
|
||||||
|
// emit DB Events [fromBlock, lastKnownBlock]
|
||||||
|
this._serveDBEvents(eventKey, filterConditions.fromBlock, lastKnownBlock, filterConditions, filterConditionsCb);
|
||||||
|
// create a event subscription [lastKnownBlock + 1, ...]
|
||||||
|
let filters = Object.assign({}, filterConditions, { fromBlock: filterConditions.fromBlock > lastKnownBlock ? filterConditions.fromBlock : lastKnownBlock + 1 });
|
||||||
|
this._subscribeToEvent(contractInstance.events[eventName], filters, filterConditionsCb, eventKey);
|
||||||
|
}
|
||||||
|
else if (filterConditions.toBlock <= lastKnownBlock) {
|
||||||
|
// emit DB Events [fromBlock, toBlock]
|
||||||
|
this._serveDBEvents(eventKey, filterConditions.fromBlock, filterConditions.toBlock, filterConditions, filterConditionsCb);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
// emit DB Events [fromBlock, lastKnownBlock]
|
||||||
|
this._serveDBEvents(eventKey, filterConditions.fromBlock, lastKnownBlock, filterConditions, filterConditionsCb);
|
||||||
|
// create a past event subscription [lastKnownBlock + 1, toBlock]
|
||||||
|
let filters = Object.assign({}, filterConditions, { fromBlock: filterConditions.fromBlock > lastKnownBlock ? filterConditions.fromBlock : lastKnownBlock + 1 });
|
||||||
|
this._getPastEvents(contractInstance, eventName, filters, filterConditionsCb, eventKey);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (firstKnownBlock > 0) {
|
||||||
|
// create a past event subscription [ firstKnownBlock > fromBlock ? fromBlock : 0, firstKnownBlock - 1]
|
||||||
|
let fromBlock = firstKnownBlock > filterConditions.fromBlock ? filterConditions.fromBlock : 0;
|
||||||
|
let filters = Object.assign({}, filterConditions, { fromBlock, toBlock: firstKnownBlock - 1 });
|
||||||
|
this._getPastEvents(contractInstance, eventName, filters, filterConditionsCb, eventKey);
|
||||||
|
if (filterConditions.toBlock === 'latest') {
|
||||||
|
// emit DB Events [firstKnownBlock, lastKnownBlock]
|
||||||
|
this._serveDBEvents(eventKey, firstKnownBlock, lastKnownBlock, filterConditions, filterConditionsCb);
|
||||||
|
// create a subscription [lastKnownBlock + 1, ...]
|
||||||
|
const filters = Object.assign({}, filterConditions, { fromBlock: lastKnownBlock + 1 });
|
||||||
|
this._subscribeToEvent(contractInstance.events[eventName], filters, filterConditionsCb, eventKey);
|
||||||
|
}
|
||||||
|
else if (filterConditions.toBlock <= lastKnownBlock) {
|
||||||
|
// emit DB Events [fromBlock, toBlock]
|
||||||
|
this._serveDBEvents(eventKey, filterConditions.fromBlock, filterConditions.toBlock, filterConditions, filterConditionsCb);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
// emit DB Events [fromBlock, lastKnownBlock]
|
||||||
|
this._serveDBEvents(eventKey, filterConditions.fromBlock, lastKnownBlock, filterConditions, filterConditionsCb);
|
||||||
|
// create a past event subscription [lastKnownBlock + 1, toBlock]
|
||||||
|
let filters = Object.assign({}, filterConditions, { fromBlock: lastKnownBlock + 1, toBlock: filterConditions.toBlock });
|
||||||
|
this._getPastEvents(contractInstance, eventName, filters, filterConditionsCb, eventKey);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_serveDBEvents(eventKey, firstKnownBlock, lastKnownBlock, filterConditions, filterConditionsCb) {
|
||||||
|
const cb = this._parseEventCBFactory(filterConditions, filterConditionsCb, eventKey);
|
||||||
|
const storedEvents = this._db.getEventsFor(eventKey).filter(x => x.blockNumber >= firstKnownBlock && x.blockNumber <= lastKnownBlock);
|
||||||
|
storedEvents.forEach(ev => {
|
||||||
|
cb(null, ev);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
_getPastEvents(contract, eventName, filterConditions, filterConditionsCb, eventKey) {
|
||||||
|
const cb = this._parseEventCBFactory(filterConditions, filterConditionsCb, eventKey);
|
||||||
|
contract.getPastEvents.apply(contract, [eventName, filterConditions, (err, events) => {
|
||||||
|
events.forEach(ev => {
|
||||||
|
cb(err, ev);
|
||||||
|
});
|
||||||
|
} ]);
|
||||||
|
}
|
||||||
|
|
||||||
|
_subscribeToEvent(event, filterConditions, filterConditionsCb, eventKey) {
|
||||||
|
event.apply(event, [filterConditions, this._parseEventCBFactory(filterConditions, filterConditionsCb, eventKey) ]);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
_parseEventCBFactory = (filterConditions, filterConditionsCb, eventKey) => (err, ev) => {
|
||||||
|
if(err) return;
|
||||||
|
|
||||||
if (filterConditions) {
|
if (filterConditions) {
|
||||||
let propsToFilter = [];
|
let propsToFilter = [];
|
||||||
for (let prop in filterConditions.filter) {
|
for (let prop in filterConditions.filter) {
|
||||||
if (Object.keys(event.returnValues).indexOf(prop) >= 0) {
|
if (Object.keys(ev.returnValues).indexOf(prop) >= 0) {
|
||||||
propsToFilter.push(prop)
|
propsToFilter.push(prop);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for (let prop of propsToFilter) {
|
for (let prop of propsToFilter) {
|
||||||
if (filterConditions.filter[prop] !== event.returnValues[prop]) return;
|
if (filterConditions.filter[prop] !== ev.returnValues[prop])
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
} else if (filterConditionsCb && !filterConditionsCb(event.returnValues)) {
|
}
|
||||||
|
else if (filterConditionsCb && !filterConditionsCb(ev.returnValues)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.events.emit(eventbusKey, event);
|
this.events.emit(eventKey, ev);
|
||||||
}])
|
|
||||||
|
|
||||||
// TODO: would be nice if trackEvent was smart enough to understand the type of returnValues and do the needed conversions
|
|
||||||
contractObserver.pipe().subscribe((e) => {
|
|
||||||
e.eventKey = eventKey
|
|
||||||
if (this._db.eventExists(e.id)) return;
|
|
||||||
if (e.returnValues['$loki']) return sub.next(e.returnValues)
|
|
||||||
|
|
||||||
this._db.recordEvent(e.returnValues);
|
|
||||||
this._db.updateEventId(eventName, e.id)
|
|
||||||
this.events.emit("updateDB")
|
|
||||||
sub.next(e.returnValues)
|
|
||||||
})
|
|
||||||
|
|
||||||
return sub;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
_initNewBlocksSubscription() {
|
_initNewBlocksSubscription() {
|
||||||
|
@ -115,7 +202,7 @@ export default class EventSyncer {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: should save value in database
|
// TODO: should save value in database?
|
||||||
trackProperty(contractInstance, propName, methodArgs = [], callArgs = {}) {
|
trackProperty(contractInstance, propName, methodArgs = [], callArgs = {}) {
|
||||||
|
|
||||||
const sub = new ReplaySubject();
|
const sub = new ReplaySubject();
|
||||||
|
@ -141,7 +228,7 @@ export default class EventSyncer {
|
||||||
return sub.pipe(distinctUntilChanged((a, b) => equal(a, b)));
|
return sub.pipe(distinctUntilChanged((a, b) => equal(a, b)));
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: should save value in database (?)
|
// TODO: should save value in database?
|
||||||
trackBalance(address, erc20Address) {
|
trackBalance(address, erc20Address) {
|
||||||
const sub = new ReplaySubject();
|
const sub = new ReplaySubject();
|
||||||
|
|
||||||
|
@ -151,6 +238,7 @@ export default class EventSyncer {
|
||||||
if(!erc20Address){
|
if(!erc20Address){
|
||||||
callFn = () => {
|
callFn = () => {
|
||||||
const fn = this.web3.getBalance;
|
const fn = this.web3.getBalance;
|
||||||
|
|
||||||
fn.apply(fn, [address, (err, balance) => {
|
fn.apply(fn, [address, (err, balance) => {
|
||||||
if(err) {
|
if(err) {
|
||||||
sub.error(err);
|
sub.error(err);
|
||||||
|
@ -192,4 +280,3 @@ export default class EventSyncer {
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,23 +1,24 @@
|
||||||
|
const ganache = require("ganache-core");
|
||||||
const Web3Eth = require('web3-eth');
|
const Web3Eth = require('web3-eth');
|
||||||
|
|
||||||
let eth = new Web3Eth("ws://localhost:8545");
|
console.log("The following error is emitted by ganache - https://github.com/trufflesuite/ganache-core/issues/267")
|
||||||
|
let eth = new Web3Eth(ganache.provider());
|
||||||
|
|
||||||
async function run() {
|
async function run() {
|
||||||
let accounts = await eth.getAccounts();
|
let accounts = await eth.getAccounts();
|
||||||
|
|
||||||
|
|
||||||
setTimeout(async () => {
|
setTimeout(async () => {
|
||||||
await eth.sendTransaction({from: accounts[0], to: accounts[1], value: "100000000"});
|
await eth.sendTransaction({from: accounts[0], to: accounts[1], value: "100000000"});
|
||||||
await eth.sendTransaction({from: accounts[2], to: accounts[0], value: "999999999"});
|
await eth.sendTransaction({from: accounts[2], to: accounts[0], value: "999999999"});
|
||||||
await eth.sendTransaction({from: accounts[2], to: accounts[0], value: "232433434"});
|
await eth.sendTransaction({from: accounts[2], to: accounts[0], value: "232433434"});
|
||||||
}, 3000);
|
}, 2000);
|
||||||
|
|
||||||
const EventSyncer = require('../dist/node.js');
|
const EventSyncer = require('../dist/node.js');
|
||||||
const eventSyncer = new EventSyncer(eth.currentProvider);
|
const eventSyncer = new EventSyncer(eth.currentProvider);
|
||||||
|
|
||||||
await eventSyncer.init();
|
await eventSyncer.init();
|
||||||
|
|
||||||
eventSyncer.trackBalance(accounts[0]).pipe().subscribe((balance) => {
|
eventSyncer.trackBalance(accounts[0]).subscribe((balance) => {
|
||||||
console.log("accounts[0] balance is ", balance);
|
console.log("accounts[0] balance is ", balance);
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,75 @@
|
||||||
|
const Web3Eth = require('web3-eth');
|
||||||
|
const {deployRatingContract, mine} = require('./utils-web3');
|
||||||
|
const ganache = require("ganache-core");
|
||||||
|
|
||||||
|
console.log("The following error is emitted by ganache - https://github.com/trufflesuite/ganache-core/issues/267")
|
||||||
|
let eth = new Web3Eth(ganache.provider());
|
||||||
|
|
||||||
|
async function run() {
|
||||||
|
let accounts = await eth.getAccounts();
|
||||||
|
var RatingContract = await deployRatingContract(eth)
|
||||||
|
|
||||||
|
// Events are generated in these blocks:
|
||||||
|
// x x x x x x x x x
|
||||||
|
// 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20
|
||||||
|
await mine(eth);
|
||||||
|
await RatingContract.methods.doRating(1, 5).send({from: accounts[0]})
|
||||||
|
await mine(eth);
|
||||||
|
await mine(eth);
|
||||||
|
await RatingContract.methods.doRating(2, 3).send({from: accounts[0]})
|
||||||
|
await RatingContract.methods.doRating(3, 1).send({from: accounts[0]})
|
||||||
|
await RatingContract.methods.doRating(4, 5).send({from: accounts[0]})
|
||||||
|
await RatingContract.methods.doRating(5, 5).send({from: accounts[0]})
|
||||||
|
await mine(eth);
|
||||||
|
await RatingContract.methods.doRating(6, 5).send({from: accounts[0]})
|
||||||
|
await mine(eth);
|
||||||
|
await mine(eth);
|
||||||
|
await RatingContract.methods.doRating(7, 5).send({from: accounts[0]})
|
||||||
|
await RatingContract.methods.doRating(8, 5).send({from: accounts[0]})
|
||||||
|
await mine(eth);
|
||||||
|
await mine(eth);
|
||||||
|
await mine(eth);
|
||||||
|
await mine(eth);
|
||||||
|
await RatingContract.methods.doRating(8, 5).send({from: accounts[0]})
|
||||||
|
|
||||||
|
const EventSyncer = require('../dist/node.js');
|
||||||
|
const eventSyncer = new EventSyncer(eth.currentProvider);
|
||||||
|
|
||||||
|
await eventSyncer.init()
|
||||||
|
|
||||||
|
// Testing single block with a event
|
||||||
|
eventSyncer.trackEvent(RatingContract, 'Rating', {fromBlock: 3, toBlock: 3}).subscribe((v) => {
|
||||||
|
console.log("A", v)
|
||||||
|
});
|
||||||
|
|
||||||
|
// Testing blocks that have no events in between
|
||||||
|
eventSyncer.trackEvent(RatingContract, 'Rating', {fromBlock: 8, toBlock: 11}).subscribe((v) => {
|
||||||
|
console.log("B", v)
|
||||||
|
});
|
||||||
|
|
||||||
|
// Testing blocks that begin with no events
|
||||||
|
eventSyncer.trackEvent(RatingContract, 'Rating', {fromBlock: 12, toBlock: 15}).subscribe((v) => {
|
||||||
|
console.log("C", v)
|
||||||
|
});
|
||||||
|
|
||||||
|
// Testing all blocks
|
||||||
|
eventSyncer.trackEvent(RatingContract, 'Rating', {}).subscribe((v) => {
|
||||||
|
console.log("D", v)
|
||||||
|
});
|
||||||
|
|
||||||
|
// Testing blocks that end in no events
|
||||||
|
eventSyncer.trackEvent(RatingContract, 'Rating', {fromBlock: 14, toBlock: 18}).subscribe((v) => {
|
||||||
|
console.log("E", v)
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
setTimeout(() => {
|
||||||
|
// Testing if events come from the DB instead of a subscription
|
||||||
|
eventSyncer.trackEvent(RatingContract, 'Rating', {fromBlock: 7, toBlock: 11}).subscribe((v) => {
|
||||||
|
console.log("E", v)
|
||||||
|
});
|
||||||
|
}, 5000);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
run()
|
|
@ -118,7 +118,8 @@ async function deployRatingContract(eth) {
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
|
||||||
var contract = new eth.Contract(abi)
|
var contract = new eth.Contract(abi);
|
||||||
|
|
||||||
let instance = await contract.deploy({
|
let instance = await contract.deploy({
|
||||||
data: '0x608060405234801561001057600080fd5b5060e78061001f6000396000f3fe6080604052600436106039576000357c010000000000000000000000000000000000000000000000000000000090048063f60781a914603e575b600080fd5b348015604957600080fd5b50607d60048036036040811015605e57600080fd5b810190808035906020019092919080359060200190929190505050607f565b005b817ffdefdf8d82459f7b1eb157e5c44cbe6ee73d8ddd387511fe3622a3ee663b4697826040518082815260200191505060405180910390a2505056fea165627a7a7230582067833697a0e2bccb8bd624c0b06b2183641addb24f7931d8ec3979982bb663790029',
|
data: '0x608060405234801561001057600080fd5b5060e78061001f6000396000f3fe6080604052600436106039576000357c010000000000000000000000000000000000000000000000000000000090048063f60781a914603e575b600080fd5b348015604957600080fd5b50607d60048036036040811015605e57600080fd5b810190808035906020019092919080359060200190929190505050607f565b005b817ffdefdf8d82459f7b1eb157e5c44cbe6ee73d8ddd387511fe3622a3ee663b4697826040518082815260200191505060405180910390a2505056fea165627a7a7230582067833697a0e2bccb8bd624c0b06b2183641addb24f7931d8ec3979982bb663790029',
|
||||||
arguments: []
|
arguments: []
|
||||||
|
@ -129,8 +130,23 @@ async function deployRatingContract(eth) {
|
||||||
return instance
|
return instance
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
const mine = (web3) => {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
web3.currentProvider.send({
|
||||||
|
jsonrpc: '2.0',
|
||||||
|
method: 'evm_mine',
|
||||||
|
id: new Date().getTime()
|
||||||
|
}, (err, result) => {
|
||||||
|
if (err) { return reject(err) }
|
||||||
|
return resolve(result)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
deployEscrowContract,
|
deployEscrowContract,
|
||||||
deployRatingContract
|
deployRatingContract,
|
||||||
|
mine
|
||||||
};
|
};
|
||||||
|
|
Loading…
Reference in New Issue