mirror of
https://github.com/cerc-io/watcher-ts
synced 2025-01-24 03:59:06 +00:00
Invoke subgraph block handlers (#43)
* Add a test case to eden test to call the block handler * Add a block handler in example subgraph and call it in a watcher * Use an array map to call all the block handlers for a contract * Await on all the promises returned by block handlers map
This commit is contained in:
parent
06ba24e38f
commit
16bb955213
@ -8,7 +8,7 @@ import spies from 'chai-spies';
|
|||||||
|
|
||||||
import { getDummyEventData, getTestDatabase } from '../test/utils';
|
import { getDummyEventData, getTestDatabase } from '../test/utils';
|
||||||
import { instantiate } from './loader';
|
import { instantiate } from './loader';
|
||||||
import { createEvent, Block } from './utils';
|
import { createEvent, createBlock, Block } from './utils';
|
||||||
import { Database } from './database';
|
import { Database } from './database';
|
||||||
|
|
||||||
chai.use(spies);
|
chai.use(spies);
|
||||||
@ -19,7 +19,8 @@ describe('call handler in mapping code', () => {
|
|||||||
let exports: any;
|
let exports: any;
|
||||||
let db: Database;
|
let db: Database;
|
||||||
|
|
||||||
const eventData = getDummyEventData();
|
// Create dummy event data.
|
||||||
|
const dummyEventData = getDummyEventData();
|
||||||
|
|
||||||
before(async () => {
|
before(async () => {
|
||||||
db = getTestDatabase();
|
db = getTestDatabase();
|
||||||
@ -50,11 +51,11 @@ describe('call handler in mapping code', () => {
|
|||||||
|
|
||||||
it('should load the subgraph example wasm', async () => {
|
it('should load the subgraph example wasm', async () => {
|
||||||
const filePath = path.resolve(__dirname, '../test/subgraph/example1/build/Example1/Example1.wasm');
|
const filePath = path.resolve(__dirname, '../test/subgraph/example1/build/Example1/Example1.wasm');
|
||||||
const instance = await instantiate(db, { event: { block: eventData.block } }, filePath);
|
const instance = await instantiate(db, { event: { block: dummyEventData.block } }, filePath);
|
||||||
exports = instance.exports;
|
exports = instance.exports;
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should execute the handler function', async () => {
|
it('should execute the event handler function', async () => {
|
||||||
const {
|
const {
|
||||||
_start,
|
_start,
|
||||||
handleTest
|
handleTest
|
||||||
@ -65,7 +66,7 @@ describe('call handler in mapping code', () => {
|
|||||||
_start();
|
_start();
|
||||||
|
|
||||||
// Create event params data.
|
// Create event params data.
|
||||||
eventData.eventParams = [
|
dummyEventData.eventParams = [
|
||||||
{
|
{
|
||||||
name: 'param1',
|
name: 'param1',
|
||||||
value: 'abc',
|
value: 'abc',
|
||||||
@ -81,8 +82,8 @@ describe('call handler in mapping code', () => {
|
|||||||
// Dummy contract address string.
|
// Dummy contract address string.
|
||||||
const contractAddress = '0xCA6D29232D1435D8198E3E5302495417dD073d61';
|
const contractAddress = '0xCA6D29232D1435D8198E3E5302495417dD073d61';
|
||||||
|
|
||||||
// Create Test event to be passed to handler.
|
// Create an ethereum event Test to be passed to handler.
|
||||||
const test = await createEvent(exports, contractAddress, eventData);
|
const test = await createEvent(exports, contractAddress, dummyEventData);
|
||||||
|
|
||||||
await handleTest(test);
|
await handleTest(test);
|
||||||
|
|
||||||
@ -91,6 +92,20 @@ describe('call handler in mapping code', () => {
|
|||||||
expect(db.saveEntity).to.have.been.called();
|
expect(db.saveEntity).to.have.been.called();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should execute the block handler function', async () => {
|
||||||
|
const { _start, handleBlock } = exports;
|
||||||
|
const blockData = dummyEventData.block;
|
||||||
|
|
||||||
|
// Important to call _start for built subgraphs on instantiation!
|
||||||
|
// TODO: Check api version https://github.com/graphprotocol/graph-node/blob/6098daa8955bdfac597cec87080af5449807e874/runtime/wasm/src/module/mod.rs#L533
|
||||||
|
_start();
|
||||||
|
|
||||||
|
// Create an ethereum block to be passed to the handler.
|
||||||
|
const block = await createBlock(exports, blockData);
|
||||||
|
|
||||||
|
await handleBlock(block);
|
||||||
|
});
|
||||||
|
|
||||||
after(() => {
|
after(() => {
|
||||||
sandbox.restore();
|
sandbox.restore();
|
||||||
});
|
});
|
||||||
|
@ -9,7 +9,7 @@ import chai from 'chai';
|
|||||||
import spies from 'chai-spies';
|
import spies from 'chai-spies';
|
||||||
|
|
||||||
import { instantiate } from './loader';
|
import { instantiate } from './loader';
|
||||||
import { createEvent, Block } from './utils';
|
import { createEvent, Block, createBlock } from './utils';
|
||||||
import edenNetworkAbi from '../test/subgraph/eden/EdenNetwork/abis/EdenNetwork.json';
|
import edenNetworkAbi from '../test/subgraph/eden/EdenNetwork/abis/EdenNetwork.json';
|
||||||
import merkleDistributorAbi from '../test/subgraph/eden/EdenNetworkDistribution/abis/MerkleDistributor.json';
|
import merkleDistributorAbi from '../test/subgraph/eden/EdenNetworkDistribution/abis/MerkleDistributor.json';
|
||||||
import distributorGovernanceAbi from '../test/subgraph/eden/EdenNetworkGovernance/abis/DistributorGovernance.json';
|
import distributorGovernanceAbi from '../test/subgraph/eden/EdenNetworkGovernance/abis/DistributorGovernance.json';
|
||||||
@ -24,7 +24,9 @@ const sandbox = chai.spy.sandbox();
|
|||||||
|
|
||||||
describe('eden wasm loader tests', async () => {
|
describe('eden wasm loader tests', async () => {
|
||||||
let db: Database;
|
let db: Database;
|
||||||
const eventData = getDummyEventData();
|
|
||||||
|
// Create dummy event data.
|
||||||
|
const dummyEventData = getDummyEventData();
|
||||||
|
|
||||||
before(async () => {
|
before(async () => {
|
||||||
db = getTestDatabase();
|
db = getTestDatabase();
|
||||||
@ -65,7 +67,7 @@ describe('eden wasm loader tests', async () => {
|
|||||||
|
|
||||||
it('should load the subgraph network wasm', async () => {
|
it('should load the subgraph network wasm', async () => {
|
||||||
const filePath = path.resolve(__dirname, '../test/subgraph/eden/EdenNetwork/EdenNetwork.wasm');
|
const filePath = path.resolve(__dirname, '../test/subgraph/eden/EdenNetwork/EdenNetwork.wasm');
|
||||||
({ exports } = await instantiate(db, { event: { block: eventData.block } }, filePath, data));
|
({ exports } = await instantiate(db, { event: { block: dummyEventData.block } }, filePath, data));
|
||||||
const { _start } = exports;
|
const { _start } = exports;
|
||||||
_start();
|
_start();
|
||||||
});
|
});
|
||||||
@ -76,7 +78,7 @@ describe('eden wasm loader tests', async () => {
|
|||||||
} = exports;
|
} = exports;
|
||||||
|
|
||||||
// Create dummy SlotClaimedEvent params.
|
// Create dummy SlotClaimedEvent params.
|
||||||
eventData.eventParams = [
|
dummyEventData.eventParams = [
|
||||||
{
|
{
|
||||||
name: 'slot',
|
name: 'slot',
|
||||||
kind: 'uint8',
|
kind: 'uint8',
|
||||||
@ -114,8 +116,8 @@ describe('eden wasm loader tests', async () => {
|
|||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
// Create dummy SlotClaimedEvent to be passed to handler.
|
// Create an ethereum event SlotClaimedEvent to be passed to handler.
|
||||||
const slotClaimedEvent = await createEvent(exports, contractAddress, eventData);
|
const slotClaimedEvent = await createEvent(exports, contractAddress, dummyEventData);
|
||||||
|
|
||||||
await slotClaimed(slotClaimedEvent);
|
await slotClaimed(slotClaimedEvent);
|
||||||
});
|
});
|
||||||
@ -126,7 +128,7 @@ describe('eden wasm loader tests', async () => {
|
|||||||
} = exports;
|
} = exports;
|
||||||
|
|
||||||
// Create dummy SlotDelegateUpdatedEvent params.
|
// Create dummy SlotDelegateUpdatedEvent params.
|
||||||
eventData.eventParams = [
|
dummyEventData.eventParams = [
|
||||||
{
|
{
|
||||||
name: 'slot',
|
name: 'slot',
|
||||||
kind: 'uint8',
|
kind: 'uint8',
|
||||||
@ -149,8 +151,8 @@ describe('eden wasm loader tests', async () => {
|
|||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
// Create dummy SlotDelegateUpdatedEvent to be passed to handler.
|
// Create an ethereum event SlotDelegateUpdatedEvent to be passed to handler.
|
||||||
const slotClaimedEvent = await createEvent(exports, contractAddress, eventData);
|
const slotClaimedEvent = await createEvent(exports, contractAddress, dummyEventData);
|
||||||
|
|
||||||
await slotDelegateUpdated(slotClaimedEvent);
|
await slotDelegateUpdated(slotClaimedEvent);
|
||||||
});
|
});
|
||||||
@ -161,7 +163,7 @@ describe('eden wasm loader tests', async () => {
|
|||||||
} = exports;
|
} = exports;
|
||||||
|
|
||||||
// Create dummy StakeEvent params.
|
// Create dummy StakeEvent params.
|
||||||
eventData.eventParams = [
|
dummyEventData.eventParams = [
|
||||||
{
|
{
|
||||||
name: 'staker',
|
name: 'staker',
|
||||||
kind: 'address',
|
kind: 'address',
|
||||||
@ -174,8 +176,8 @@ describe('eden wasm loader tests', async () => {
|
|||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
// Create dummy StakeEvent to be passed to handler.
|
// Create an ethereum event StakeEvent to be passed to handler.
|
||||||
const stakeEvent = await createEvent(exports, contractAddress, eventData);
|
const stakeEvent = await createEvent(exports, contractAddress, dummyEventData);
|
||||||
|
|
||||||
await stake(stakeEvent);
|
await stake(stakeEvent);
|
||||||
});
|
});
|
||||||
@ -186,7 +188,7 @@ describe('eden wasm loader tests', async () => {
|
|||||||
} = exports;
|
} = exports;
|
||||||
|
|
||||||
// Create dummy UnstakeEvent params.
|
// Create dummy UnstakeEvent params.
|
||||||
eventData.eventParams = [
|
dummyEventData.eventParams = [
|
||||||
{
|
{
|
||||||
name: 'staker',
|
name: 'staker',
|
||||||
kind: 'address',
|
kind: 'address',
|
||||||
@ -199,8 +201,8 @@ describe('eden wasm loader tests', async () => {
|
|||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
// Create dummy UnstakeEvent to be passed to handler.
|
// Create an ethereum event UnstakeEvent to be passed to handler.
|
||||||
const unstakeEvent = await createEvent(exports, contractAddress, eventData);
|
const unstakeEvent = await createEvent(exports, contractAddress, dummyEventData);
|
||||||
|
|
||||||
await unstake(unstakeEvent);
|
await unstake(unstakeEvent);
|
||||||
});
|
});
|
||||||
@ -224,7 +226,7 @@ describe('eden wasm loader tests', async () => {
|
|||||||
|
|
||||||
it('should load the subgraph network distribution wasm', async () => {
|
it('should load the subgraph network distribution wasm', async () => {
|
||||||
const filePath = path.resolve(__dirname, '../test/subgraph/eden/EdenNetworkDistribution/EdenNetworkDistribution.wasm');
|
const filePath = path.resolve(__dirname, '../test/subgraph/eden/EdenNetworkDistribution/EdenNetworkDistribution.wasm');
|
||||||
({ exports } = await instantiate(db, { event: { block: eventData.block } }, filePath, data));
|
({ exports } = await instantiate(db, { event: { block: dummyEventData.block } }, filePath, data));
|
||||||
const { _start } = exports;
|
const { _start } = exports;
|
||||||
_start();
|
_start();
|
||||||
});
|
});
|
||||||
@ -235,7 +237,7 @@ describe('eden wasm loader tests', async () => {
|
|||||||
} = exports;
|
} = exports;
|
||||||
|
|
||||||
// Create dummy ClaimedEvent params.
|
// Create dummy ClaimedEvent params.
|
||||||
eventData.eventParams = [
|
dummyEventData.eventParams = [
|
||||||
{
|
{
|
||||||
name: 'index',
|
name: 'index',
|
||||||
kind: 'uint256',
|
kind: 'uint256',
|
||||||
@ -258,8 +260,8 @@ describe('eden wasm loader tests', async () => {
|
|||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
// Create dummy ClaimedEvent to be passed to handler.
|
// Create an ethereum event ClaimedEvent to be passed to handler.
|
||||||
const claimedEvent = await createEvent(exports, contractAddress, eventData);
|
const claimedEvent = await createEvent(exports, contractAddress, dummyEventData);
|
||||||
|
|
||||||
await claimed(claimedEvent);
|
await claimed(claimedEvent);
|
||||||
});
|
});
|
||||||
@ -270,7 +272,7 @@ describe('eden wasm loader tests', async () => {
|
|||||||
} = exports;
|
} = exports;
|
||||||
|
|
||||||
// Create dummy SlashedEvent params.
|
// Create dummy SlashedEvent params.
|
||||||
eventData.eventParams = [
|
dummyEventData.eventParams = [
|
||||||
{
|
{
|
||||||
name: 'account',
|
name: 'account',
|
||||||
kind: 'address',
|
kind: 'address',
|
||||||
@ -283,8 +285,8 @@ describe('eden wasm loader tests', async () => {
|
|||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
// Create dummy SlashedEvent to be passed to handler.
|
// Create an ethereum event SlashedEvent to be passed to handler.
|
||||||
const slashedEvent = await createEvent(exports, contractAddress, eventData);
|
const slashedEvent = await createEvent(exports, contractAddress, dummyEventData);
|
||||||
|
|
||||||
await slashed(slashedEvent);
|
await slashed(slashedEvent);
|
||||||
});
|
});
|
||||||
@ -295,7 +297,7 @@ describe('eden wasm loader tests', async () => {
|
|||||||
} = exports;
|
} = exports;
|
||||||
|
|
||||||
// Create dummy MerkleRootUpdatedEvent params.
|
// Create dummy MerkleRootUpdatedEvent params.
|
||||||
eventData.eventParams = [
|
dummyEventData.eventParams = [
|
||||||
{
|
{
|
||||||
name: 'merkleRoot',
|
name: 'merkleRoot',
|
||||||
kind: 'bytes32',
|
kind: 'bytes32',
|
||||||
@ -313,8 +315,8 @@ describe('eden wasm loader tests', async () => {
|
|||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
// Create dummy MerkleRootUpdatedEvent to be passed to handler.
|
// Create an ethereum event MerkleRootUpdatedEvent to be passed to handler.
|
||||||
const merkleRootUpdatedEvent = await createEvent(exports, contractAddress, eventData);
|
const merkleRootUpdatedEvent = await createEvent(exports, contractAddress, dummyEventData);
|
||||||
|
|
||||||
await merkleRootUpdated(merkleRootUpdatedEvent);
|
await merkleRootUpdated(merkleRootUpdatedEvent);
|
||||||
});
|
});
|
||||||
@ -325,7 +327,7 @@ describe('eden wasm loader tests', async () => {
|
|||||||
} = exports;
|
} = exports;
|
||||||
|
|
||||||
// Create dummy AccountUpdatedEvent params.
|
// Create dummy AccountUpdatedEvent params.
|
||||||
eventData.eventParams = [
|
dummyEventData.eventParams = [
|
||||||
{
|
{
|
||||||
name: 'account',
|
name: 'account',
|
||||||
kind: 'address',
|
kind: 'address',
|
||||||
@ -343,8 +345,8 @@ describe('eden wasm loader tests', async () => {
|
|||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
// Create dummy AccountUpdatedEvent to be passed to handler.
|
// Create an ethereum event AccountUpdatedEvent to be passed to handler.
|
||||||
const accountUpdatedEvent = await createEvent(exports, contractAddress, eventData);
|
const accountUpdatedEvent = await createEvent(exports, contractAddress, dummyEventData);
|
||||||
|
|
||||||
await accountUpdated(accountUpdatedEvent);
|
await accountUpdated(accountUpdatedEvent);
|
||||||
});
|
});
|
||||||
@ -368,7 +370,7 @@ describe('eden wasm loader tests', async () => {
|
|||||||
|
|
||||||
it('should load the subgraph network governance wasm', async () => {
|
it('should load the subgraph network governance wasm', async () => {
|
||||||
const filePath = path.resolve(__dirname, '../test/subgraph/eden/EdenNetworkGovernance/EdenNetworkGovernance.wasm');
|
const filePath = path.resolve(__dirname, '../test/subgraph/eden/EdenNetworkGovernance/EdenNetworkGovernance.wasm');
|
||||||
({ exports } = await instantiate(db, { event: { block: eventData.block } }, filePath, data));
|
({ exports } = await instantiate(db, { event: { block: dummyEventData.block } }, filePath, data));
|
||||||
const { _start } = exports;
|
const { _start } = exports;
|
||||||
_start();
|
_start();
|
||||||
});
|
});
|
||||||
@ -379,7 +381,7 @@ describe('eden wasm loader tests', async () => {
|
|||||||
} = exports;
|
} = exports;
|
||||||
|
|
||||||
// Create dummy BlockProducerAddedEvent params.
|
// Create dummy BlockProducerAddedEvent params.
|
||||||
eventData.eventParams = [
|
dummyEventData.eventParams = [
|
||||||
{
|
{
|
||||||
name: 'produces',
|
name: 'produces',
|
||||||
kind: 'address',
|
kind: 'address',
|
||||||
@ -387,8 +389,8 @@ describe('eden wasm loader tests', async () => {
|
|||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
// Create dummy BlockProducerAddedEvent to be passed to handler.
|
// Create an ethereum event BlockProducerAddedEvent to be passed to handler.
|
||||||
const blockProducerAddedEvent = await createEvent(exports, contractAddress, eventData);
|
const blockProducerAddedEvent = await createEvent(exports, contractAddress, dummyEventData);
|
||||||
|
|
||||||
await blockProducerAdded(blockProducerAddedEvent);
|
await blockProducerAdded(blockProducerAddedEvent);
|
||||||
});
|
});
|
||||||
@ -399,7 +401,7 @@ describe('eden wasm loader tests', async () => {
|
|||||||
} = exports;
|
} = exports;
|
||||||
|
|
||||||
// Create dummy BlockProducerRemovedEvent params.
|
// Create dummy BlockProducerRemovedEvent params.
|
||||||
eventData.eventParams = [
|
dummyEventData.eventParams = [
|
||||||
{
|
{
|
||||||
name: 'producer',
|
name: 'producer',
|
||||||
kind: 'address',
|
kind: 'address',
|
||||||
@ -407,8 +409,8 @@ describe('eden wasm loader tests', async () => {
|
|||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
// Create dummy BlockProducerRemovedEvent to be passed to handler.
|
// Create an ethereum event BlockProducerRemovedEvent to be passed to handler.
|
||||||
const blockProducerRemovedEvent = await createEvent(exports, contractAddress, eventData);
|
const blockProducerRemovedEvent = await createEvent(exports, contractAddress, dummyEventData);
|
||||||
|
|
||||||
await blockProducerRemoved(blockProducerRemovedEvent);
|
await blockProducerRemoved(blockProducerRemovedEvent);
|
||||||
});
|
});
|
||||||
@ -419,7 +421,7 @@ describe('eden wasm loader tests', async () => {
|
|||||||
} = exports;
|
} = exports;
|
||||||
|
|
||||||
// Create dummy BlockProducerRewardCollectorChangedEvent params.
|
// Create dummy BlockProducerRewardCollectorChangedEvent params.
|
||||||
eventData.eventParams = [
|
dummyEventData.eventParams = [
|
||||||
{
|
{
|
||||||
name: 'producer',
|
name: 'producer',
|
||||||
kind: 'address',
|
kind: 'address',
|
||||||
@ -437,8 +439,8 @@ describe('eden wasm loader tests', async () => {
|
|||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
// Create dummy BlockProducerRewardCollectorChangedEvent to be passed to handler.
|
// Create an ethereum event BlockProducerRewardCollectorChangedEvent to be passed to handler.
|
||||||
const blockProducerRewardCollectorChangedEvent = await createEvent(exports, contractAddress, eventData);
|
const blockProducerRewardCollectorChangedEvent = await createEvent(exports, contractAddress, dummyEventData);
|
||||||
|
|
||||||
await blockProducerRewardCollectorChanged(blockProducerRewardCollectorChangedEvent);
|
await blockProducerRewardCollectorChanged(blockProducerRewardCollectorChangedEvent);
|
||||||
});
|
});
|
||||||
@ -448,13 +450,23 @@ describe('eden wasm loader tests', async () => {
|
|||||||
rewardScheduleChanged
|
rewardScheduleChanged
|
||||||
} = exports;
|
} = exports;
|
||||||
|
|
||||||
eventData.eventParams = [];
|
dummyEventData.eventParams = [];
|
||||||
|
|
||||||
// Create dummy RewardScheduleChangedEvent to be passed to handler.
|
// Create an ethereum event RewardScheduleChangedEvent to be passed to handler.
|
||||||
const rewardScheduleChangedEvent = await createEvent(exports, contractAddress, eventData);
|
const rewardScheduleChangedEvent = await createEvent(exports, contractAddress, dummyEventData);
|
||||||
|
|
||||||
await rewardScheduleChanged(rewardScheduleChangedEvent);
|
await rewardScheduleChanged(rewardScheduleChangedEvent);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should call the block handler', async () => {
|
||||||
|
const { handleBlock } = exports;
|
||||||
|
const blockData = dummyEventData.block;
|
||||||
|
|
||||||
|
// Create an ethereum block to be passed to the handler.
|
||||||
|
const block = await createBlock(exports, blockData);
|
||||||
|
|
||||||
|
await handleBlock(block);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
after(() => {
|
after(() => {
|
||||||
|
@ -164,51 +164,7 @@ export const createEvent = async (instanceExports: any, contractAddress: string,
|
|||||||
id_of_type: idOfType
|
id_of_type: idOfType
|
||||||
} = instanceExports;
|
} = instanceExports;
|
||||||
|
|
||||||
// Fill block data.
|
const block = await createBlock(instanceExports, blockData);
|
||||||
const blockHashByteArray = await ByteArray.fromHexString(await __newString(blockData.blockHash));
|
|
||||||
const blockHash = await Bytes.fromByteArray(blockHashByteArray);
|
|
||||||
|
|
||||||
const parentHashByteArray = await ByteArray.fromHexString(await __newString(blockData.parentHash));
|
|
||||||
const parentHash = await Bytes.fromByteArray(parentHashByteArray);
|
|
||||||
|
|
||||||
const blockNumber = await BigInt.fromString(await __newString(blockData.blockNumber));
|
|
||||||
|
|
||||||
const blockTimestamp = await BigInt.fromString(await __newString(blockData.timestamp));
|
|
||||||
|
|
||||||
const stateRootByteArray = await ByteArray.fromHexString(await __newString(blockData.stateRoot));
|
|
||||||
const stateRoot = await Bytes.fromByteArray(stateRootByteArray);
|
|
||||||
|
|
||||||
const transactionsRootByteArray = await ByteArray.fromHexString(await __newString(blockData.txRoot));
|
|
||||||
const transactionsRoot = await Bytes.fromByteArray(transactionsRootByteArray);
|
|
||||||
|
|
||||||
const receiptsRootByteArray = await ByteArray.fromHexString(await __newString(blockData.receiptRoot));
|
|
||||||
const receiptsRoot = await Bytes.fromByteArray(receiptsRootByteArray);
|
|
||||||
|
|
||||||
const totalDifficulty = await BigInt.fromString(await __newString(blockData.td));
|
|
||||||
|
|
||||||
// Missing fields from watcher in block data:
|
|
||||||
// unclesHash
|
|
||||||
// author
|
|
||||||
// gasUsed
|
|
||||||
// gasLimit
|
|
||||||
// difficulty
|
|
||||||
// size
|
|
||||||
const block = await ethereum.Block.__new(
|
|
||||||
blockHash,
|
|
||||||
parentHash,
|
|
||||||
await Bytes.empty(),
|
|
||||||
await Address.zero(),
|
|
||||||
stateRoot,
|
|
||||||
transactionsRoot,
|
|
||||||
receiptsRoot,
|
|
||||||
blockNumber,
|
|
||||||
await BigInt.fromI32(0),
|
|
||||||
await BigInt.fromI32(0),
|
|
||||||
blockTimestamp,
|
|
||||||
await BigInt.fromI32(0),
|
|
||||||
totalDifficulty,
|
|
||||||
null
|
|
||||||
);
|
|
||||||
|
|
||||||
// Fill transaction data.
|
// Fill transaction data.
|
||||||
const txHashByteArray = await ByteArray.fromHexString(await __newString(tx.hash));
|
const txHashByteArray = await ByteArray.fromHexString(await __newString(tx.hash));
|
||||||
@ -264,6 +220,63 @@ export const createEvent = async (instanceExports: any, contractAddress: string,
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const createBlock = async (instanceExports: any, blockData: Block): Promise<any> => {
|
||||||
|
const {
|
||||||
|
__newString,
|
||||||
|
Address,
|
||||||
|
BigInt,
|
||||||
|
ethereum,
|
||||||
|
Bytes,
|
||||||
|
ByteArray
|
||||||
|
} = instanceExports;
|
||||||
|
|
||||||
|
// Fill block data.
|
||||||
|
const blockHashByteArray = await ByteArray.fromHexString(await __newString(blockData.blockHash));
|
||||||
|
const blockHash = await Bytes.fromByteArray(blockHashByteArray);
|
||||||
|
|
||||||
|
const parentHashByteArray = await ByteArray.fromHexString(await __newString(blockData.parentHash));
|
||||||
|
const parentHash = await Bytes.fromByteArray(parentHashByteArray);
|
||||||
|
|
||||||
|
const blockNumber = await BigInt.fromString(await __newString(blockData.blockNumber));
|
||||||
|
|
||||||
|
const blockTimestamp = await BigInt.fromString(await __newString(blockData.timestamp));
|
||||||
|
|
||||||
|
const stateRootByteArray = await ByteArray.fromHexString(await __newString(blockData.stateRoot));
|
||||||
|
const stateRoot = await Bytes.fromByteArray(stateRootByteArray);
|
||||||
|
|
||||||
|
const transactionsRootByteArray = await ByteArray.fromHexString(await __newString(blockData.txRoot));
|
||||||
|
const transactionsRoot = await Bytes.fromByteArray(transactionsRootByteArray);
|
||||||
|
|
||||||
|
const receiptsRootByteArray = await ByteArray.fromHexString(await __newString(blockData.receiptRoot));
|
||||||
|
const receiptsRoot = await Bytes.fromByteArray(receiptsRootByteArray);
|
||||||
|
|
||||||
|
const totalDifficulty = await BigInt.fromString(await __newString(blockData.td));
|
||||||
|
|
||||||
|
// Missing fields from watcher in block data:
|
||||||
|
// unclesHash
|
||||||
|
// author
|
||||||
|
// gasUsed
|
||||||
|
// gasLimit
|
||||||
|
// difficulty
|
||||||
|
// size
|
||||||
|
return await ethereum.Block.__new(
|
||||||
|
blockHash,
|
||||||
|
parentHash,
|
||||||
|
await Bytes.empty(),
|
||||||
|
await Address.zero(),
|
||||||
|
stateRoot,
|
||||||
|
transactionsRoot,
|
||||||
|
receiptsRoot,
|
||||||
|
blockNumber,
|
||||||
|
await BigInt.fromI32(0),
|
||||||
|
await BigInt.fromI32(0),
|
||||||
|
blockTimestamp,
|
||||||
|
await BigInt.fromI32(0),
|
||||||
|
totalDifficulty,
|
||||||
|
null
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
export const getSubgraphConfig = async (subgraphPath: string): Promise<any> => {
|
export const getSubgraphConfig = async (subgraphPath: string): Promise<any> => {
|
||||||
const configFilePath = path.resolve(path.join(subgraphPath, 'subgraph.yaml'));
|
const configFilePath = path.resolve(path.join(subgraphPath, 'subgraph.yaml'));
|
||||||
const fileExists = await fs.pathExists(configFilePath);
|
const fileExists = await fs.pathExists(configFilePath);
|
||||||
|
@ -13,7 +13,7 @@ import { ResultObject } from '@vulcanize/assemblyscript/lib/loader';
|
|||||||
import { EthClient } from '@vulcanize/ipld-eth-client';
|
import { EthClient } from '@vulcanize/ipld-eth-client';
|
||||||
import { IndexerInterface } from '@vulcanize/util';
|
import { IndexerInterface } from '@vulcanize/util';
|
||||||
|
|
||||||
import { createEvent, getSubgraphConfig } from './utils';
|
import { createBlock, createEvent, getSubgraphConfig } from './utils';
|
||||||
import { Context, instantiate } from './loader';
|
import { Context, instantiate } from './loader';
|
||||||
import { Database } from './database';
|
import { Database } from './database';
|
||||||
|
|
||||||
@ -149,6 +149,36 @@ export class GraphWatcher {
|
|||||||
await exports[eventHandler.handler](ethereumEvent);
|
await exports[eventHandler.handler](ethereumEvent);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async handleBlock (blockHash: string) {
|
||||||
|
const {
|
||||||
|
allEthHeaderCids: {
|
||||||
|
nodes: [
|
||||||
|
blockData
|
||||||
|
]
|
||||||
|
}
|
||||||
|
} = await this._postgraphileClient.getBlocks({ blockHash });
|
||||||
|
|
||||||
|
// Call block handler(s) for each contract.
|
||||||
|
for (const dataSource of this._dataSources) {
|
||||||
|
// Check if block handler(s) are configured.
|
||||||
|
if (!dataSource.mapping.blockHandlers) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const { instance: { exports } } = this._dataSourceMap[dataSource.source.address];
|
||||||
|
|
||||||
|
// Create ethereum block to be passed to a wasm block handler.
|
||||||
|
const ethereumBlock = await createBlock(exports, blockData);
|
||||||
|
|
||||||
|
// Call all the block handlers one after the another for a contract.
|
||||||
|
const blockHandlerPromises = dataSource.mapping.blockHandlers.map(async (blockHandler: any): Promise<void> => {
|
||||||
|
await exports[blockHandler.handler](ethereumBlock);
|
||||||
|
});
|
||||||
|
|
||||||
|
await Promise.all(blockHandlerPromises);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
setIndexer (indexer: IndexerInterface): void {
|
setIndexer (indexer: IndexerInterface): void {
|
||||||
this._indexer = indexer;
|
this._indexer = indexer;
|
||||||
}
|
}
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import { Address, log, BigInt, BigDecimal, ByteArray, dataSource } from '@graphprotocol/graph-ts';
|
import { Address, log, BigInt, BigDecimal, ByteArray, dataSource, ethereum } from '@graphprotocol/graph-ts';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
Example1,
|
Example1,
|
||||||
@ -54,6 +54,30 @@ export function handleTest (event: Test): void {
|
|||||||
// - contract.getMethod(...)
|
// - contract.getMethod(...)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function handleBlock (block: ethereum.Block): void {
|
||||||
|
log.debug('block.hash: {}', [block.hash.toHexString()]);
|
||||||
|
log.debug('block.parentHash: {}', [block.parentHash.toHexString()]);
|
||||||
|
log.debug('block.unclesHash: {}', [block.unclesHash.toHexString()]);
|
||||||
|
log.debug('block.author: {}', [block.author.toHexString()]);
|
||||||
|
log.debug('block.stateRoot: {}', [block.stateRoot.toHexString()]);
|
||||||
|
log.debug('block.transactionsRoot: {}', [block.transactionsRoot.toHexString()]);
|
||||||
|
log.debug('block.receiptsRoot: {}', [block.receiptsRoot.toHexString()]);
|
||||||
|
log.debug('block.number: {}', [block.number.toString()]);
|
||||||
|
log.debug('block.gasUsed: {}', [block.gasUsed.toString()]);
|
||||||
|
log.debug('block.gasLimit: {}', [block.gasLimit.toString()]);
|
||||||
|
log.debug('block.timestamp: {}', [block.timestamp.toString()]);
|
||||||
|
log.debug('block.difficulty: {}', [block.difficulty.toString()]);
|
||||||
|
log.debug('block.totalDifficulty: {}', [block.totalDifficulty.toString()]);
|
||||||
|
|
||||||
|
const blockSize = block.size;
|
||||||
|
|
||||||
|
if (blockSize) {
|
||||||
|
log.debug('block.size: {}', [blockSize.toString()]);
|
||||||
|
} else {
|
||||||
|
log.debug('block.size: {}', ['null']);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export function testEthCall (): void {
|
export function testEthCall (): void {
|
||||||
log.debug('In test eth call', []);
|
log.debug('In test eth call', []);
|
||||||
|
|
||||||
|
@ -20,4 +20,6 @@ dataSources:
|
|||||||
eventHandlers:
|
eventHandlers:
|
||||||
- event: Test(string,uint8)
|
- event: Test(string,uint8)
|
||||||
handler: handleTest
|
handler: handleTest
|
||||||
|
blockHandlers:
|
||||||
|
- handler: handleBlock
|
||||||
file: ./src/mapping.ts
|
file: ./src/mapping.ts
|
||||||
|
@ -181,6 +181,11 @@ export class Indexer {
|
|||||||
await handleEvent(this, resultEvent);
|
await handleEvent(this, resultEvent);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async processBlock (blockHash: string): Promise<void> {
|
||||||
|
// Call subgraph handler for block.
|
||||||
|
await this._graphWatcher.handleBlock(blockHash);
|
||||||
|
}
|
||||||
|
|
||||||
async processEvent (event: Event): Promise<void> {
|
async processEvent (event: Event): Promise<void> {
|
||||||
// Trigger indexing of data based on the event.
|
// Trigger indexing of data based on the event.
|
||||||
await this.triggerIndexingOnEvent(event);
|
await this.triggerIndexingOnEvent(event);
|
||||||
|
@ -19,7 +19,8 @@ import {
|
|||||||
QUEUE_EVENT_PROCESSING,
|
QUEUE_EVENT_PROCESSING,
|
||||||
JobQueueConfig,
|
JobQueueConfig,
|
||||||
DEFAULT_CONFIG_PATH,
|
DEFAULT_CONFIG_PATH,
|
||||||
getCustomProvider
|
getCustomProvider,
|
||||||
|
JOB_KIND_INDEX
|
||||||
} from '@vulcanize/util';
|
} from '@vulcanize/util';
|
||||||
import { GraphWatcher, Database as GraphDatabase } from '@vulcanize/graph-node';
|
import { GraphWatcher, Database as GraphDatabase } from '@vulcanize/graph-node';
|
||||||
|
|
||||||
@ -50,6 +51,12 @@ export class JobRunner {
|
|||||||
await this._jobQueue.subscribe(QUEUE_BLOCK_PROCESSING, async (job) => {
|
await this._jobQueue.subscribe(QUEUE_BLOCK_PROCESSING, async (job) => {
|
||||||
await this._baseJobRunner.processBlock(job);
|
await this._baseJobRunner.processBlock(job);
|
||||||
|
|
||||||
|
const { data: { kind, blockHash } } = job;
|
||||||
|
|
||||||
|
if (kind === JOB_KIND_INDEX) {
|
||||||
|
await this._indexer.processBlock(blockHash);
|
||||||
|
}
|
||||||
|
|
||||||
await this._jobQueue.markComplete(job);
|
await this._jobQueue.markComplete(job);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user