-
Notifications
You must be signed in to change notification settings - Fork 2
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Integrate receipts extraction into ETHWorker #201
base: master
Are you sure you want to change the base?
Changes from all commits
91c3578
23cc720
f88493e
9af0ebb
6263bef
d34e0b3
09c33af
c63ada0
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -2,11 +2,12 @@ export KAFKA_URL=kafka-hz.stage.san:30911 | |
export ZOOKEEPER_URL=zookeeper-hz.stage.san:30921 | ||
export NODE_URL=https://ethereum.santiment.net | ||
export START_BLOCK="15676731" | ||
export BLOCK_INTERVAL="50" | ||
export BLOCK_INTERVAL="5" | ||
export EXPORT_TIMEOUT_MLS=300000 | ||
export CONTRACT_MODE="extract_exact_overwrite" | ||
export BLOCKCHAIN="eth" | ||
export KAFKA_TOPIC="erc20_exporter_test_topic" | ||
#export KAFKA_TOPIC="erc20_exporter_test_topic" | ||
export KAFKA_TOPIC='native_token_transfers:erc20_exporter_test_topic' | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Besides a single topic, we now support multiple ones. The format is comma separate There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. this is kind of ugly. I believe we can use several arguments instead of packing multiple values inside a single one.
|
||
export CARDANO_GRAPHQL_URL=https://cardano.santiment.net | ||
export ZOOKEEPER_SESSION_TIMEOUT=20000 | ||
export IS_ETH=false | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -6,7 +6,6 @@ import { ETHWorker } from './eth/eth_worker'; | |
import { ETHBlocksWorker } from './eth_blocks/eth_blocks_worker'; | ||
import { ETHContractsWorker } from './eth_contracts/eth_contracts_worker'; | ||
import { MaticWorker } from './matic/matic_worker'; | ||
import { ReceiptsWorker } from './receipts/receipts_worker'; | ||
import { UTXOWorker } from './utxo/utxo_worker'; | ||
import { XRPWorker } from './xrp/xrp_worker'; | ||
|
||
|
@@ -22,8 +21,6 @@ export function constructWorker(blockchain: string, settings: any): BaseWorker { | |
return new ETHBlocksWorker(settings); | ||
case 'matic': | ||
return new MaticWorker(settings); | ||
case 'receipts': | ||
return new ReceiptsWorker(settings); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Separate worker is no longer needed. We would use the
If we want to extract transfers and receipts at once. If for some reason we want to have only receipts deploy, we can do:
|
||
case 'utxo': | ||
return new UTXOWorker(settings); | ||
case 'xrp': | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,11 +1,11 @@ | ||
'use strict'; | ||
import { logger } from '../../lib/logger'; | ||
import { Exporter } from '../../lib/kafka_storage'; | ||
import { KafkaStorage } from '../../lib/kafka_storage'; | ||
import { constructRPCClient } from '../../lib/http_client'; | ||
import { extendEventsWithPrimaryKey } from './lib/extend_events_key'; | ||
import { ContractOverwrite, changeContractAddresses, extractChangedContractAddresses } from './lib/contract_overwrite'; | ||
import { stableSort, readJsonFile } from './lib/util'; | ||
import { BaseWorker } from '../../lib/worker_base'; | ||
import { BaseWorker, WorkResult, WorkResultMultiMode } from '../../lib/worker_base'; | ||
import { nextIntervalCalculator, setWorkerSleepTime, analyzeWorkerContext, NO_WORK_SLEEP } from '../eth/lib/next_interval_calculator'; | ||
import { Web3Interface, constructWeb3Wrapper } from '../eth/lib/web3_wrapper'; | ||
import { TimestampsCache } from './lib/timestamps_cache'; | ||
|
@@ -59,7 +59,7 @@ export class ERC20Worker extends BaseWorker { | |
this.allOldContracts = []; | ||
} | ||
|
||
async init(exporter?: Exporter) { | ||
async init(storage: KafkaStorage | Map<string, KafkaStorage>) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Workers receive information for which output Kafka topics need to be filled and produce the needed data. |
||
this.lastConfirmedBlock = await this.web3Wrapper.getBlockNumber() - this.settings.CONFIRMATIONS; | ||
|
||
if (this.settings.EXPORT_BLOCKS_LIST) { | ||
|
@@ -84,10 +84,10 @@ export class ERC20Worker extends BaseWorker { | |
} | ||
|
||
if (this.settings.EVENTS_IN_SAME_PARTITION) { | ||
if (exporter === undefined) { | ||
throw Error('Exporter reference need to be provided for events in same partition') | ||
if (!(storage instanceof KafkaStorage)) { | ||
throw Error('Single Kafka storage needs to be provided for events in same partition') | ||
} | ||
await exporter.initPartitioner((event: any) => simpleHash(event.contract)); | ||
await storage.initPartitioner((event: any) => simpleHash(event.contract)); | ||
} | ||
} | ||
|
||
|
@@ -112,7 +112,7 @@ export class ERC20Worker extends BaseWorker { | |
}; | ||
} | ||
|
||
async work() { | ||
async work(): Promise<WorkResult | WorkResultMultiMode> { | ||
const workerContext = await analyzeWorkerContext(this); | ||
setWorkerSleepTime(this, workerContext); | ||
if (workerContext === NO_WORK_SLEEP) return []; | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -3,23 +3,25 @@ import { constructRPCClient } from '../../lib/http_client'; | |
import { injectDAOHackTransfers, DAO_HACK_FORK_BLOCK } from './lib/dao_hack'; | ||
import { getGenesisTransfers } from './lib/genesis_transfers'; | ||
import { transactionOrder, stableSort } from './lib/util'; | ||
import { BaseWorker } from '../../lib/worker_base'; | ||
import { BaseWorker, WorkResult, WorkResultMultiMode } from '../../lib/worker_base'; | ||
import { Web3Interface, constructWeb3Wrapper, safeCastToNumber } from './lib/web3_wrapper'; | ||
import { decodeTransferTrace } from './lib/decode_transfers'; | ||
import { FeesDecoder } from './lib/fees_decoder'; | ||
import { nextIntervalCalculator, analyzeWorkerContext, setWorkerSleepTime, NO_WORK_SLEEP } from './lib/next_interval_calculator'; | ||
import { WithdrawalsDecoder } from './lib/withdrawals_decoder'; | ||
import { fetchEthInternalTrx, fetchBlocks, fetchReceipts } from './lib/fetch_data'; | ||
import { HTTPClientInterface } from '../../types'; | ||
import { Trace, ETHBlock, ETHTransfer, ETHReceiptsMap } from './eth_types'; | ||
import { Trace, ETHBlock, ETHTransfer, ETHReceipt } from './eth_types'; | ||
import { EOB, collectEndOfBlocks } from './lib/end_of_block'; | ||
|
||
import { assertIsDefined, parseKafkaTopicToObject } from '../../lib/utils'; | ||
import { decodeReceipt } from './lib/helper_receipts' | ||
|
||
export class ETHWorker extends BaseWorker { | ||
private web3Wrapper: Web3Interface; | ||
private ethClient: HTTPClientInterface; | ||
private feesDecoder: FeesDecoder; | ||
private withdrawalsDecoder: WithdrawalsDecoder; | ||
private modes: string[]; | ||
|
||
constructor(settings: any) { | ||
super(settings); | ||
|
@@ -31,19 +33,18 @@ export class ETHWorker extends BaseWorker { | |
|
||
this.feesDecoder = new FeesDecoder(this.web3Wrapper); | ||
this.withdrawalsDecoder = new WithdrawalsDecoder(this.web3Wrapper); | ||
this.modes = []; | ||
} | ||
|
||
async fetchData(fromBlock: number, toBlock: number): Promise<[Trace[], Map<number, ETHBlock>, ETHReceiptsMap]> { | ||
return await Promise.all([ | ||
fetchEthInternalTrx(this.ethClient, this.web3Wrapper, fromBlock, toBlock), | ||
fetchBlocks(this.ethClient, this.web3Wrapper, fromBlock, toBlock, true), | ||
fetchReceipts(this.ethClient, this.web3Wrapper, | ||
this.settings.RECEIPTS_API_METHOD, fromBlock, toBlock), | ||
]); | ||
async fetchData(fromBlock: number, toBlock: number): Promise<[Trace[], Map<number, ETHBlock>, ETHReceipt[]]> { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. We fetch raw data depending on what outputs are needed. Some basic raw data like block times is needed for any output. |
||
const traces: Promise<Trace[]> = this.isTracesNeeded() ? fetchEthInternalTrx(this.ethClient, this.web3Wrapper, fromBlock, toBlock) : Promise.resolve([]); | ||
const blocks: Promise<Map<number, ETHBlock>> = fetchBlocks(this.ethClient, this.web3Wrapper, fromBlock, toBlock, true); | ||
const receipts: Promise<ETHReceipt[]> = this.isReceiptsNeeded() ? fetchReceipts(this.ethClient, this.web3Wrapper, | ||
this.settings.RECEIPTS_API_METHOD, fromBlock, toBlock) : Promise.resolve([]); | ||
return await Promise.all([traces, blocks, receipts]); | ||
} | ||
|
||
transformPastEvents(fromBlock: number, toBlock: number, traces: Trace[], | ||
blocks: any, receipts: ETHReceiptsMap): ETHTransfer[] { | ||
transformPastEvents(fromBlock: number, toBlock: number, traces: Trace[], blocks: any, receipts: ETHReceipt[]): ETHTransfer[] { | ||
let events: ETHTransfer[] = []; | ||
if (fromBlock === 0) { | ||
logger.info('Adding the GENESIS transfers'); | ||
|
@@ -78,7 +79,7 @@ export class ETHWorker extends BaseWorker { | |
return result; | ||
} | ||
|
||
transformPastTransactionEvents(blocks: ETHBlock[], receipts: ETHReceiptsMap): ETHTransfer[] { | ||
transformPastTransactionEvents(blocks: ETHBlock[], receipts: ETHReceipt[]): ETHTransfer[] { | ||
const result: ETHTransfer[] = []; | ||
|
||
for (const block of blocks) { | ||
|
@@ -95,31 +96,84 @@ export class ETHWorker extends BaseWorker { | |
return result; | ||
} | ||
|
||
async work(): Promise<(ETHTransfer | EOB)[]> { | ||
const workerContext = await analyzeWorkerContext(this); | ||
setWorkerSleepTime(this, workerContext); | ||
if (workerContext === NO_WORK_SLEEP) return []; | ||
isReceiptsNeeded(): boolean { | ||
return this.modes.includes(this.settings.NATIVE_TOKEN_MODE) || this.modes.includes(this.settings.RECEIPTS_MODE) | ||
} | ||
|
||
const { fromBlock, toBlock } = nextIntervalCalculator(this); | ||
logger.info(`Fetching transfer events for interval ${fromBlock}:${toBlock}`); | ||
const [traces, blocks, receipts] = await this.fetchData(fromBlock, toBlock); | ||
let events: (ETHTransfer | EOB)[] = this.transformPastEvents(fromBlock, toBlock, traces, blocks, receipts); | ||
isTracesNeeded(): boolean { | ||
return this.modes.includes(this.settings.NATIVE_TOKEN_MODE) | ||
} | ||
|
||
getTransfersOutput(fromBlock: number, toBlock: number, traces: Trace[], | ||
blocks: Map<number, ETHBlock>, receipts: ETHReceipt[], endOfBlockEvents: EOB[]): WorkResult { | ||
assertIsDefined(traces, "Traces are needed for native token transfers"); | ||
assertIsDefined(receipts, "Receipts are needed for native token transfers"); | ||
|
||
events.push(...collectEndOfBlocks(fromBlock, toBlock, blocks, this.web3Wrapper)) | ||
const events: (ETHTransfer | EOB)[] = this.transformPastEvents(fromBlock, toBlock, traces, blocks, receipts); | ||
|
||
|
||
events.push(...endOfBlockEvents) | ||
if (events.length > 0) { | ||
stableSort(events, transactionOrder); | ||
extendEventsWithPrimaryKey(events, this.lastPrimaryKey); | ||
|
||
this.lastPrimaryKey += events.length; | ||
} | ||
|
||
return events; | ||
} | ||
|
||
getReceiptsOutput(blocks: Map<number, ETHBlock>, receipts: ETHReceipt[]): WorkResult { | ||
assertIsDefined(receipts, "Receipts are needed for receipts extraction"); | ||
assertIsDefined(blocks, "Blocks are needed for extraction"); | ||
const decodedReceipts = receipts.map((receipt: any) => decodeReceipt(receipt, this.web3Wrapper)); | ||
decodedReceipts.forEach(receipt => { | ||
const block = blocks.get(receipt.blockNumber) | ||
assertIsDefined(block, `Block ${receipt.blockNumber} is missing`) | ||
receipt.timestamp = block.timestamp; | ||
}); | ||
return decodedReceipts; | ||
} | ||
|
||
async work(): Promise<WorkResultMultiMode> { | ||
const result: WorkResultMultiMode = {}; | ||
const workerContext = await analyzeWorkerContext(this); | ||
setWorkerSleepTime(this, workerContext); | ||
if (workerContext === NO_WORK_SLEEP) return result; | ||
|
||
const { fromBlock, toBlock } = nextIntervalCalculator(this); | ||
|
||
logger.info(`Fetching events for interval ${fromBlock}:${toBlock}`); | ||
|
||
const [traces, blocks, receipts] = await this.fetchData(fromBlock, toBlock); | ||
|
||
|
||
this.lastExportedBlock = toBlock; | ||
|
||
return events; | ||
assertIsDefined(blocks, "Blocks are needed for extraction"); | ||
// TODO consider if EOB events should also be present in other output topics | ||
const endOfBlockEvents = collectEndOfBlocks(fromBlock, toBlock, blocks, this.web3Wrapper) | ||
if (this.modes.includes(this.settings.NATIVE_TOKEN_MODE)) { | ||
result[this.settings.NATIVE_TOKEN_MODE] = this.getTransfersOutput(fromBlock, toBlock, traces, blocks, | ||
receipts, endOfBlockEvents); | ||
} | ||
if (this.modes.includes(this.settings.RECEIPTS_MODE)) { | ||
result[this.settings.RECEIPTS_MODE] = this.getReceiptsOutput(blocks, receipts); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. this is not really modular design because each module has a footprint in the
|
||
} | ||
|
||
return result; | ||
} | ||
|
||
async init(): Promise<void> { | ||
this.lastConfirmedBlock = await this.web3Wrapper.getBlockNumber() - this.settings.CONFIRMATIONS; | ||
|
||
if (!this.settings.KAFKA_TOPIC.includes(":")) { | ||
throw new Error("ETH worker, expects KAFKA_TOPIC in mode:name format") | ||
} | ||
else { | ||
const mapping = parseKafkaTopicToObject(this.settings.KAFKA_TOPIC) | ||
this.modes = Object.keys(mapping); | ||
} | ||
} | ||
} | ||
|
||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
is it a change required for this PR?