2021-10-20 12:25:54 +00:00
|
|
|
import _ from 'lodash';
|
2022-08-26 06:32:39 +00:00
|
|
|
import debug from 'debug';
|
2023-05-15 06:40:27 +00:00
|
|
|
import { sha256 } from 'multiformats/hashes/sha2';
|
|
|
|
import { CID } from 'multiformats/cid';
|
|
|
|
|
2022-10-19 09:54:14 +00:00
|
|
|
import * as codec from '@ipld/dag-cbor';
|
2022-08-26 06:32:39 +00:00
|
|
|
|
2023-05-15 06:40:27 +00:00
|
|
|
import { BlockProgressInterface, GraphDatabaseInterface, StateInterface, StateKind } from './types';
|
2022-08-26 06:32:39 +00:00
|
|
|
import { jsonBigIntStringReplacer } from './misc';
|
2022-10-19 09:54:14 +00:00
|
|
|
import { ResultState } from './indexer';
|
2022-08-26 06:32:39 +00:00
|
|
|
|
2022-10-19 09:54:14 +00:00
|
|
|
const log = debug('vulcanize:state-helper');
|
2021-10-20 12:25:54 +00:00
|
|
|
|
2023-05-15 06:40:27 +00:00
|
|
|
export interface StateDataMeta {
|
|
|
|
id: string
|
|
|
|
kind: StateKind
|
|
|
|
parent: {
|
|
|
|
'/': string | null
|
|
|
|
},
|
|
|
|
ethBlock: {
|
|
|
|
cid: {
|
|
|
|
'/': string
|
|
|
|
},
|
|
|
|
num: number
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
interface StateData {
|
|
|
|
meta?: StateDataMeta;
|
|
|
|
state: any
|
|
|
|
}
|
|
|
|
|
2022-06-15 05:10:40 +00:00
|
|
|
export const updateStateForElementaryType = (initialObject: any, stateVariable: string, value: any): any => {
|
2021-10-20 12:25:54 +00:00
|
|
|
const object = _.cloneDeep(initialObject);
|
|
|
|
const path = ['state', stateVariable];
|
|
|
|
|
|
|
|
return _.set(object, path, value);
|
|
|
|
};
|
|
|
|
|
2022-06-15 05:10:40 +00:00
|
|
|
export const updateStateForMappingType = (initialObject: any, stateVariable: string, keys: string[], value: any): any => {
|
2021-10-20 12:25:54 +00:00
|
|
|
const object = _.cloneDeep(initialObject);
|
|
|
|
keys.unshift('state', stateVariable);
|
|
|
|
|
|
|
|
// Use _.setWith() with Object as customizer as _.set() treats numeric value in path as an index to an array.
|
|
|
|
return _.setWith(object, keys, value, Object);
|
|
|
|
};
|
2022-08-26 06:32:39 +00:00
|
|
|
|
2022-10-19 09:54:14 +00:00
|
|
|
export const verifyCheckpointData = async (database: GraphDatabaseInterface, block: BlockProgressInterface, data: any): Promise<void> => {
|
2022-08-26 06:32:39 +00:00
|
|
|
const { state } = data;
|
|
|
|
|
|
|
|
for (const [entityName, idEntityMap] of Object.entries(state)) {
|
2022-10-19 09:54:14 +00:00
|
|
|
for (const [id, stateEntity] of Object.entries(idEntityMap as {[key: string]: any})) {
|
2022-08-26 06:32:39 +00:00
|
|
|
const entityData = await database.getEntity(entityName, id, block.blockHash) as any;
|
|
|
|
|
|
|
|
// Compare entities.
|
2022-10-19 09:54:14 +00:00
|
|
|
const diffFound = Object.keys(stateEntity)
|
2022-08-26 06:32:39 +00:00
|
|
|
.some(key => {
|
2022-10-19 09:54:14 +00:00
|
|
|
let stateValue = stateEntity[key];
|
2022-08-26 06:32:39 +00:00
|
|
|
|
|
|
|
if (key === 'blockNumber') {
|
|
|
|
entityData.blockNumber = entityData._blockNumber;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (key === 'blockHash') {
|
|
|
|
entityData.blockHash = entityData._blockHash;
|
|
|
|
}
|
|
|
|
|
2022-10-19 09:54:14 +00:00
|
|
|
if (typeof stateEntity[key] === 'object' && stateEntity[key]?.id) {
|
|
|
|
stateValue = stateEntity[key].id;
|
2022-08-26 06:32:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (
|
2022-10-19 09:54:14 +00:00
|
|
|
Array.isArray(stateEntity[key]) &&
|
|
|
|
stateEntity[key].length &&
|
|
|
|
stateEntity[key][0].id
|
2022-08-26 06:32:39 +00:00
|
|
|
) {
|
2022-10-19 09:54:14 +00:00
|
|
|
// Map State entity 1 to N relation field array to match DB entity.
|
|
|
|
stateValue = stateEntity[key].map(({ id }: { id: string }) => id);
|
2022-08-26 06:32:39 +00:00
|
|
|
|
|
|
|
// Sort DB entity 1 to N relation field array.
|
|
|
|
entityData[key] = entityData[key].sort((a: string, b: string) => a.localeCompare(b));
|
|
|
|
}
|
|
|
|
|
2022-10-19 09:54:14 +00:00
|
|
|
return JSON.stringify(stateValue) !== JSON.stringify(entityData[key], jsonBigIntStringReplacer);
|
2022-08-26 06:32:39 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
if (diffFound) {
|
|
|
|
const message = `Diff found for checkpoint at block ${block.blockNumber} in entity ${entityName} id ${id}`;
|
|
|
|
log(message);
|
|
|
|
throw new Error(message);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
2022-10-19 09:54:14 +00:00
|
|
|
|
|
|
|
export const getResultState = (state: StateInterface): ResultState => {
|
|
|
|
const block = state.block;
|
|
|
|
|
|
|
|
const data = codec.decode(Buffer.from(state.data)) as any;
|
|
|
|
|
|
|
|
return {
|
|
|
|
block: {
|
|
|
|
cid: block.cid,
|
|
|
|
hash: block.blockHash,
|
|
|
|
number: block.blockNumber,
|
|
|
|
timestamp: block.blockTimestamp,
|
|
|
|
parentHash: block.parentHash
|
|
|
|
},
|
|
|
|
contractAddress: state.contractAddress,
|
|
|
|
cid: state.cid,
|
|
|
|
kind: state.kind,
|
|
|
|
data: JSON.stringify(data)
|
|
|
|
};
|
|
|
|
};
|
2023-05-15 06:40:27 +00:00
|
|
|
|
|
|
|
export const createOrUpdateStateData = async (
|
|
|
|
data: StateData,
|
|
|
|
meta?: StateDataMeta
|
|
|
|
): Promise<{ cid: CID, data: StateData, bytes: codec.ByteView<StateData> }> => {
|
|
|
|
if (meta) {
|
|
|
|
data.meta = meta;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Encoding the data using dag-cbor codec.
|
|
|
|
const bytes = codec.encode(data);
|
|
|
|
|
|
|
|
// Calculating sha256 (multi)hash of the encoded data.
|
|
|
|
const hash = await sha256.digest(bytes);
|
|
|
|
|
|
|
|
// Calculating the CID: v1, code: dag-cbor, hash.
|
|
|
|
const cid = CID.create(1, codec.code, hash);
|
|
|
|
|
|
|
|
return { cid, data, bytes };
|
|
|
|
};
|