merge #4066 from jr-reverse-caching
cache registry reverses, completion in address selector
This commit is contained in:
commit
8958603f64
@ -14,7 +14,7 @@
|
|||||||
// You should have received a copy of the GNU General Public License
|
// You should have received a copy of the GNU General Public License
|
||||||
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
import subscribeToEvent from '../util/subscribe-to-event';
|
import subscribeToEvents from '../util/subscribe-to-events';
|
||||||
|
|
||||||
export const checkIfVerified = (contract, account) => {
|
export const checkIfVerified = (contract, account) => {
|
||||||
return contract.instance.certified.call({}, [account]);
|
return contract.instance.certified.call({}, [account]);
|
||||||
@ -72,7 +72,7 @@ export const awaitPuzzle = (api, contract, account) => {
|
|||||||
return blockNumber(api)
|
return blockNumber(api)
|
||||||
.then((block) => {
|
.then((block) => {
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
const subscription = subscribeToEvent(contract, 'Puzzled', {
|
const subscription = subscribeToEvents(contract, ['Puzzled'], {
|
||||||
from: block.toNumber(),
|
from: block.toNumber(),
|
||||||
filter: (log) => log.params.who.value === account
|
filter: (log) => log.params.who.value === account
|
||||||
});
|
});
|
||||||
|
@ -59,6 +59,9 @@ const STORE = {
|
|||||||
},
|
},
|
||||||
settings: {
|
settings: {
|
||||||
backgroundSeed: ''
|
backgroundSeed: ''
|
||||||
|
},
|
||||||
|
registry: {
|
||||||
|
reverse: {}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
@ -23,6 +23,7 @@ import SignerMiddleware from './providers/signerMiddleware';
|
|||||||
import statusMiddleware from '~/views/Status/middleware';
|
import statusMiddleware from '~/views/Status/middleware';
|
||||||
import CertificationsMiddleware from './providers/certifications/middleware';
|
import CertificationsMiddleware from './providers/certifications/middleware';
|
||||||
import ChainMiddleware from './providers/chainMiddleware';
|
import ChainMiddleware from './providers/chainMiddleware';
|
||||||
|
import RegistryMiddleware from './providers/registry/middleware';
|
||||||
|
|
||||||
export default function (api, browserHistory) {
|
export default function (api, browserHistory) {
|
||||||
const errors = new ErrorsMiddleware();
|
const errors = new ErrorsMiddleware();
|
||||||
@ -32,13 +33,15 @@ export default function (api, browserHistory) {
|
|||||||
const certifications = new CertificationsMiddleware();
|
const certifications = new CertificationsMiddleware();
|
||||||
const routeMiddleware = routerMiddleware(browserHistory);
|
const routeMiddleware = routerMiddleware(browserHistory);
|
||||||
const chain = new ChainMiddleware();
|
const chain = new ChainMiddleware();
|
||||||
|
const registry = new RegistryMiddleware(api);
|
||||||
|
|
||||||
const middleware = [
|
const middleware = [
|
||||||
settings.toMiddleware(),
|
settings.toMiddleware(),
|
||||||
signer.toMiddleware(),
|
signer.toMiddleware(),
|
||||||
errors.toMiddleware(),
|
errors.toMiddleware(),
|
||||||
certifications.toMiddleware(),
|
certifications.toMiddleware(),
|
||||||
chain.toMiddleware()
|
chain.toMiddleware(),
|
||||||
|
registry
|
||||||
];
|
];
|
||||||
|
|
||||||
return middleware.concat(status, routeMiddleware, thunk);
|
return middleware.concat(status, routeMiddleware, thunk);
|
||||||
|
28
js/src/redux/providers/registry/actions.js
Normal file
28
js/src/redux/providers/registry/actions.js
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
// Copyright 2015, 2016 Parity Technologies (UK) Ltd.
|
||||||
|
// This file is part of Parity.
|
||||||
|
|
||||||
|
// Parity is free software: you can redistribute it and/or modify
|
||||||
|
// it under the terms of the GNU General Public License as published by
|
||||||
|
// the Free Software Foundation, either version 3 of the License, or
|
||||||
|
// (at your option) any later version.
|
||||||
|
|
||||||
|
// Parity is distributed in the hope that it will be useful,
|
||||||
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
// GNU General Public License for more details.
|
||||||
|
|
||||||
|
// You should have received a copy of the GNU General Public License
|
||||||
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
export const setReverse = (address, reverse) => ({
|
||||||
|
type: 'setReverse',
|
||||||
|
address, reverse
|
||||||
|
});
|
||||||
|
|
||||||
|
export const startCachingReverses = () => ({
|
||||||
|
type: 'startCachingReverses'
|
||||||
|
});
|
||||||
|
|
||||||
|
export const stopCachingReverses = () => ({
|
||||||
|
type: 'stopCachingReverses'
|
||||||
|
});
|
99
js/src/redux/providers/registry/middleware.js
Normal file
99
js/src/redux/providers/registry/middleware.js
Normal file
@ -0,0 +1,99 @@
|
|||||||
|
// Copyright 2015, 2016 Parity Technologies (UK) Ltd.
|
||||||
|
// This file is part of Parity.
|
||||||
|
|
||||||
|
// Parity is free software: you can redistribute it and/or modify
|
||||||
|
// it under the terms of the GNU General Public License as published by
|
||||||
|
// the Free Software Foundation, either version 3 of the License, or
|
||||||
|
// (at your option) any later version.
|
||||||
|
|
||||||
|
// Parity is distributed in the hope that it will be useful,
|
||||||
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
// GNU General Public License for more details.
|
||||||
|
|
||||||
|
// You should have received a copy of the GNU General Public License
|
||||||
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
import Contracts from '~/contracts';
|
||||||
|
import subscribeToEvents from '~/util/subscribe-to-events';
|
||||||
|
|
||||||
|
import registryABI from '~/contracts/abi/registry.json';
|
||||||
|
|
||||||
|
import { setReverse, startCachingReverses } from './actions';
|
||||||
|
|
||||||
|
export default (api) => (store) => {
|
||||||
|
let contract, subscription, timeout, interval;
|
||||||
|
|
||||||
|
let addressesToCheck = {};
|
||||||
|
|
||||||
|
const onLog = (log) => {
|
||||||
|
switch (log.event) {
|
||||||
|
case 'ReverseConfirmed':
|
||||||
|
addressesToCheck[log.params.reverse.value] = true;
|
||||||
|
|
||||||
|
break;
|
||||||
|
case 'ReverseRemoved':
|
||||||
|
delete addressesToCheck[log.params.reverse.value];
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const checkReverses = () => {
|
||||||
|
Object
|
||||||
|
.keys(addressesToCheck)
|
||||||
|
.forEach((address) => {
|
||||||
|
contract
|
||||||
|
.instance
|
||||||
|
.reverse
|
||||||
|
.call({}, [ address ])
|
||||||
|
.then((reverse) => store.dispatch(setReverse(address, reverse)));
|
||||||
|
});
|
||||||
|
|
||||||
|
addressesToCheck = {};
|
||||||
|
};
|
||||||
|
|
||||||
|
return (next) => (action) => {
|
||||||
|
switch (action.type) {
|
||||||
|
case 'initAll':
|
||||||
|
next(action);
|
||||||
|
store.dispatch(startCachingReverses());
|
||||||
|
|
||||||
|
break;
|
||||||
|
case 'startCachingReverses':
|
||||||
|
const { registry } = Contracts.get();
|
||||||
|
|
||||||
|
registry.getInstance()
|
||||||
|
.then((instance) => api.newContract(registryABI, instance.address))
|
||||||
|
.then((_contract) => {
|
||||||
|
contract = _contract;
|
||||||
|
|
||||||
|
subscription = subscribeToEvents(_contract, ['ReverseConfirmed', 'ReverseRemoved']);
|
||||||
|
subscription.on('log', onLog);
|
||||||
|
|
||||||
|
timeout = setTimeout(checkReverses, 5000);
|
||||||
|
interval = setInterval(checkReverses, 20000);
|
||||||
|
})
|
||||||
|
.catch((err) => {
|
||||||
|
console.error('Failed to start caching reverses:', err);
|
||||||
|
throw err;
|
||||||
|
});
|
||||||
|
|
||||||
|
break;
|
||||||
|
case 'stopCachingReverses':
|
||||||
|
if (subscription) {
|
||||||
|
subscription.unsubscribe();
|
||||||
|
}
|
||||||
|
if (interval) {
|
||||||
|
clearInterval(interval);
|
||||||
|
}
|
||||||
|
if (timeout) {
|
||||||
|
clearTimeout(timeout);
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
next(action);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
};
|
33
js/src/redux/providers/registry/reducer.js
Normal file
33
js/src/redux/providers/registry/reducer.js
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
// Copyright 2015, 2016 Parity Technologies (UK) Ltd.
|
||||||
|
// This file is part of Parity.
|
||||||
|
|
||||||
|
// Parity is free software: you can redistribute it and/or modify
|
||||||
|
// it under the terms of the GNU General Public License as published by
|
||||||
|
// the Free Software Foundation, either version 3 of the License, or
|
||||||
|
// (at your option) any later version.
|
||||||
|
|
||||||
|
// Parity is distributed in the hope that it will be useful,
|
||||||
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
// GNU General Public License for more details.
|
||||||
|
|
||||||
|
// You should have received a copy of the GNU General Public License
|
||||||
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
const initialState = {
|
||||||
|
reverse: {} // cache for reverse lookup
|
||||||
|
};
|
||||||
|
|
||||||
|
export default (state = initialState, action) => {
|
||||||
|
if (action.type === 'setReverse') {
|
||||||
|
if (state.reverse[action.address] === action.reverse) {
|
||||||
|
return state;
|
||||||
|
}
|
||||||
|
|
||||||
|
return { ...state, reverse: {
|
||||||
|
...state.reverse, [ action.address ]: action.reverse
|
||||||
|
} };
|
||||||
|
}
|
||||||
|
|
||||||
|
return state;
|
||||||
|
};
|
@ -24,6 +24,7 @@ import {
|
|||||||
snackbarReducer, walletReducer
|
snackbarReducer, walletReducer
|
||||||
} from './providers';
|
} from './providers';
|
||||||
import certificationsReducer from './providers/certifications/reducer';
|
import certificationsReducer from './providers/certifications/reducer';
|
||||||
|
import registryReducer from './providers/registry/reducer';
|
||||||
|
|
||||||
import errorReducer from '~/ui/Errors/reducers';
|
import errorReducer from '~/ui/Errors/reducers';
|
||||||
import settingsReducer from '~/views/Settings/reducers';
|
import settingsReducer from '~/views/Settings/reducers';
|
||||||
@ -43,6 +44,7 @@ export default function () {
|
|||||||
images: imagesReducer,
|
images: imagesReducer,
|
||||||
nodeStatus: nodeStatusReducer,
|
nodeStatus: nodeStatusReducer,
|
||||||
personal: personalReducer,
|
personal: personalReducer,
|
||||||
|
registry: registryReducer,
|
||||||
signer: signerReducer,
|
signer: signerReducer,
|
||||||
snackbar: snackbarReducer,
|
snackbar: snackbarReducer,
|
||||||
wallet: walletReducer,
|
wallet: walletReducer,
|
||||||
|
@ -56,6 +56,7 @@ class AddressSelect extends Component {
|
|||||||
contacts: PropTypes.object,
|
contacts: PropTypes.object,
|
||||||
contracts: PropTypes.object,
|
contracts: PropTypes.object,
|
||||||
tokens: PropTypes.object,
|
tokens: PropTypes.object,
|
||||||
|
reverse: PropTypes.object,
|
||||||
|
|
||||||
// Optional props
|
// Optional props
|
||||||
allowCopy: PropTypes.bool,
|
allowCopy: PropTypes.bool,
|
||||||
@ -584,10 +585,12 @@ class AddressSelect extends Component {
|
|||||||
function mapStateToProps (state) {
|
function mapStateToProps (state) {
|
||||||
const { accountsInfo } = state.personal;
|
const { accountsInfo } = state.personal;
|
||||||
const { balances } = state.balances;
|
const { balances } = state.balances;
|
||||||
|
const { reverse } = state.registry;
|
||||||
|
|
||||||
return {
|
return {
|
||||||
accountsInfo,
|
accountsInfo,
|
||||||
balances
|
balances,
|
||||||
|
reverse
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -16,7 +16,7 @@
|
|||||||
|
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { observable, action } from 'mobx';
|
import { observable, action } from 'mobx';
|
||||||
import { flatMap } from 'lodash';
|
import { flatMap, uniqBy } from 'lodash';
|
||||||
import { FormattedMessage } from 'react-intl';
|
import { FormattedMessage } from 'react-intl';
|
||||||
|
|
||||||
import Contracts from '~/contracts';
|
import Contracts from '~/contracts';
|
||||||
@ -30,7 +30,48 @@ export default class AddressSelectStore {
|
|||||||
@observable registryValues = [];
|
@observable registryValues = [];
|
||||||
|
|
||||||
initValues = [];
|
initValues = [];
|
||||||
regLookups = [];
|
regLookups = [
|
||||||
|
(query) => {
|
||||||
|
query = query.toLowerCase().trim();
|
||||||
|
if (query.length === 0 || query === '0x') {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
const startsWithQuery = (s) => new RegExp('^' + query, 'i').test(s);
|
||||||
|
|
||||||
|
let address;
|
||||||
|
let name = this.reverse[query];
|
||||||
|
|
||||||
|
if (!name) {
|
||||||
|
const addr = Object
|
||||||
|
.keys(this.reverse)
|
||||||
|
.find((addr) => {
|
||||||
|
const name = this.reverse[addr];
|
||||||
|
return startsWithQuery(addr) || (name && startsWithQuery(name));
|
||||||
|
});
|
||||||
|
|
||||||
|
if (addr) {
|
||||||
|
address = addr;
|
||||||
|
name = this.reverse[addr];
|
||||||
|
} else {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
address,
|
||||||
|
name,
|
||||||
|
description: (
|
||||||
|
<FormattedMessage
|
||||||
|
id='addressSelect.fromRegistry'
|
||||||
|
defaultMessage='{name} (from registry)'
|
||||||
|
values={ {
|
||||||
|
name
|
||||||
|
} }
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
};
|
||||||
|
}
|
||||||
|
];
|
||||||
|
|
||||||
constructor (api) {
|
constructor (api) {
|
||||||
this.api = api;
|
this.api = api;
|
||||||
@ -114,7 +155,8 @@ export default class AddressSelectStore {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@action setValues (props) {
|
@action setValues (props) {
|
||||||
const { accounts = {}, contracts = {}, contacts = {} } = props;
|
const { accounts = {}, contracts = {}, contacts = {}, reverse = {} } = props;
|
||||||
|
this.reverse = reverse;
|
||||||
|
|
||||||
const accountsN = Object.keys(accounts).length;
|
const accountsN = Object.keys(accounts).length;
|
||||||
const contractsN = Object.keys(contracts).length;
|
const contractsN = Object.keys(contracts).length;
|
||||||
@ -194,6 +236,8 @@ export default class AddressSelectStore {
|
|||||||
.filter((result) => result && !ZERO.test(result.address));
|
.filter((result) => result && !ZERO.test(result.address));
|
||||||
})
|
})
|
||||||
.then((results) => {
|
.then((results) => {
|
||||||
|
results = uniqBy(results, (result) => result.address);
|
||||||
|
|
||||||
this.registryValues = results
|
this.registryValues = results
|
||||||
.map((result) => {
|
.map((result) => {
|
||||||
const lowercaseAddress = result.address.toLowerCase();
|
const lowercaseAddress = result.address.toLowerCase();
|
||||||
|
@ -1,77 +0,0 @@
|
|||||||
// Copyright 2015, 2016 Parity Technologies (UK) Ltd.
|
|
||||||
// This file is part of Parity.
|
|
||||||
|
|
||||||
// Parity is free software: you can redistribute it and/or modify
|
|
||||||
// it under the terms of the GNU General Public License as published by
|
|
||||||
// the Free Software Foundation, either version 3 of the License, or
|
|
||||||
// (at your option) any later version.
|
|
||||||
|
|
||||||
// Parity is distributed in the hope that it will be useful,
|
|
||||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
// GNU General Public License for more details.
|
|
||||||
|
|
||||||
// You should have received a copy of the GNU General Public License
|
|
||||||
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
|
|
||||||
import EventEmitter from 'eventemitter3';
|
|
||||||
|
|
||||||
const defaults = {
|
|
||||||
from: 0, // TODO
|
|
||||||
to: 'latest',
|
|
||||||
timeout: null,
|
|
||||||
filter: () => true
|
|
||||||
};
|
|
||||||
|
|
||||||
const subscribeToEvent = (contract, name, opt = {}) => {
|
|
||||||
opt = Object.assign({}, defaults, opt);
|
|
||||||
|
|
||||||
let subscription = null;
|
|
||||||
let timeout = null;
|
|
||||||
|
|
||||||
const unsubscribe = () => {
|
|
||||||
if (subscription) {
|
|
||||||
contract.unsubscribe(subscription);
|
|
||||||
subscription = null;
|
|
||||||
}
|
|
||||||
if (timeout) {
|
|
||||||
clearTimeout(timeout);
|
|
||||||
timeout = null;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const emitter = new EventEmitter();
|
|
||||||
emitter.unsubscribe = unsubscribe;
|
|
||||||
|
|
||||||
if (typeof opt.timeout === 'number') {
|
|
||||||
timeout = setTimeout(() => {
|
|
||||||
unsubscribe();
|
|
||||||
emitter.emit('timeout');
|
|
||||||
}, opt.timeout);
|
|
||||||
}
|
|
||||||
|
|
||||||
const callback = (err, logs) => {
|
|
||||||
if (err) {
|
|
||||||
return emitter.emit('error', err);
|
|
||||||
}
|
|
||||||
for (let log of logs) {
|
|
||||||
if (opt.filter(log)) {
|
|
||||||
emitter.emit('log', log);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
contract.subscribe(name, {
|
|
||||||
fromBlock: opt.from, toBlock: opt.to
|
|
||||||
}, callback)
|
|
||||||
.then((_subscription) => {
|
|
||||||
subscription = _subscription;
|
|
||||||
})
|
|
||||||
.catch((err) => {
|
|
||||||
emitter.emit('error', err);
|
|
||||||
});
|
|
||||||
|
|
||||||
return emitter;
|
|
||||||
};
|
|
||||||
|
|
||||||
export default subscribeToEvent;
|
|
97
js/src/util/subscribe-to-events.js
Normal file
97
js/src/util/subscribe-to-events.js
Normal file
@ -0,0 +1,97 @@
|
|||||||
|
// Copyright 2015, 2016 Parity Technologies (UK) Ltd.
|
||||||
|
// This file is part of Parity.
|
||||||
|
|
||||||
|
// Parity is free software: you can redistribute it and/or modify
|
||||||
|
// it under the terms of the GNU General Public License as published by
|
||||||
|
// the Free Software Foundation, either version 3 of the License, or
|
||||||
|
// (at your option) any later version.
|
||||||
|
|
||||||
|
// Parity is distributed in the hope that it will be useful,
|
||||||
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
// GNU General Public License for more details.
|
||||||
|
|
||||||
|
// You should have received a copy of the GNU General Public License
|
||||||
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
import EventEmitter from 'eventemitter3';
|
||||||
|
|
||||||
|
const defaults = {
|
||||||
|
from: 0,
|
||||||
|
to: 'latest',
|
||||||
|
interval: 5000,
|
||||||
|
filter: () => true
|
||||||
|
};
|
||||||
|
|
||||||
|
const subscribeToEvents = (contract, events, opt = {}) => {
|
||||||
|
const { api } = contract;
|
||||||
|
opt = Object.assign({}, defaults, opt);
|
||||||
|
|
||||||
|
let filter = null;
|
||||||
|
let interval = null;
|
||||||
|
|
||||||
|
const unsubscribe = () => {
|
||||||
|
if (filter) {
|
||||||
|
filter
|
||||||
|
.then((filterId) => {
|
||||||
|
return api.eth.uninstallFilter(filterId);
|
||||||
|
})
|
||||||
|
.catch((err) => {
|
||||||
|
emitter.emit('error', err);
|
||||||
|
});
|
||||||
|
filter = null;
|
||||||
|
}
|
||||||
|
if (interval) {
|
||||||
|
clearInterval(interval);
|
||||||
|
interval = null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const emitter = new EventEmitter();
|
||||||
|
emitter.unsubscribe = unsubscribe;
|
||||||
|
|
||||||
|
const fetcher = (method, filterId) => () => {
|
||||||
|
api
|
||||||
|
.eth[method](filterId)
|
||||||
|
.then((logs) => {
|
||||||
|
logs = contract.parseEventLogs(logs);
|
||||||
|
|
||||||
|
for (let log of logs) {
|
||||||
|
if (opt.filter(log)) {
|
||||||
|
emitter.emit('log', log);
|
||||||
|
emitter.emit(log.event, log);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.catch((err) => {
|
||||||
|
emitter.emit('error', err);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
const signatures = events
|
||||||
|
.filter((event) => contract.instance[event])
|
||||||
|
.map((event) => contract.instance[event].signature);
|
||||||
|
|
||||||
|
filter = api.eth
|
||||||
|
.newFilter({
|
||||||
|
fromBlock: opt.from, toBlock: opt.to,
|
||||||
|
address: contract.address,
|
||||||
|
topics: [signatures]
|
||||||
|
})
|
||||||
|
.then((filterId) => {
|
||||||
|
fetcher('getFilterLogs', filterId)(); // fetch immediately
|
||||||
|
|
||||||
|
const fetchChanges = fetcher('getFilterChanges', filterId);
|
||||||
|
interval = setInterval(fetchChanges, opt.interval);
|
||||||
|
|
||||||
|
return filterId;
|
||||||
|
})
|
||||||
|
.catch((err) => {
|
||||||
|
emitter.emit('error', err);
|
||||||
|
throw err; // reject Promise
|
||||||
|
});
|
||||||
|
|
||||||
|
return emitter;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default subscribeToEvents;
|
Loading…
Reference in New Issue
Block a user