otterscan/src/useErigonHooks.ts

443 lines
12 KiB
TypeScript
Raw Normal View History

import { useState, useEffect, useMemo } from "react";
import { Block, BlockWithTransactions } from "@ethersproject/abstract-provider";
import { JsonRpcProvider } from "@ethersproject/providers";
import { getAddress } from "@ethersproject/address";
import { Contract } from "@ethersproject/contracts";
import { BigNumber } from "@ethersproject/bignumber";
import { arrayify, hexDataSlice, isHexString } from "@ethersproject/bytes";
import { extract4Bytes } from "./use4Bytes";
2021-07-21 19:06:51 +00:00
import { getInternalOperations } from "./nodeFunctions";
2021-08-01 09:59:59 +00:00
import {
TokenMetas,
TokenTransfer,
TransactionData,
InternalOperation,
2021-08-02 02:53:36 +00:00
ProcessedTransaction,
OperationType,
2021-08-01 09:59:59 +00:00
} from "./types";
import erc20 from "./erc20.json";
const TRANSFER_TOPIC =
"0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef";
2021-07-17 18:00:08 +00:00
export interface ExtendedBlock extends Block {
2021-07-26 22:59:07 +00:00
blockReward: BigNumber;
unclesReward: BigNumber;
feeReward: BigNumber;
size: number;
sha3Uncles: string;
stateRoot: string;
totalDifficulty: BigNumber;
transactionCount: number;
2021-07-26 22:59:07 +00:00
}
export const readBlock = async (
provider: JsonRpcProvider,
blockNumberOrHash: string
) => {
let blockPromise: Promise<any>;
if (isHexString(blockNumberOrHash, 32)) {
// TODO: fix
blockPromise = provider.send("eth_getBlockByHash", [
blockNumberOrHash,
false,
]);
} else {
blockPromise = provider.send("ots_getBlockDetails", [blockNumberOrHash]);
}
const _rawBlock = await blockPromise;
const _block = provider.formatter.block(_rawBlock.block);
const _rawIssuance = _rawBlock.issuance;
const extBlock: ExtendedBlock = {
blockReward: provider.formatter.bigNumber(_rawIssuance.blockReward ?? 0),
unclesReward: provider.formatter.bigNumber(_rawIssuance.uncleReward ?? 0),
feeReward: provider.formatter.bigNumber(_rawBlock.totalFees),
size: provider.formatter.number(_rawBlock.block.size),
sha3Uncles: _rawBlock.block.sha3Uncles,
stateRoot: _rawBlock.block.stateRoot,
totalDifficulty: provider.formatter.bigNumber(
_rawBlock.block.totalDifficulty
),
transactionCount: provider.formatter.number(
_rawBlock.block.transactionCount
),
..._block,
};
return extBlock;
};
2021-08-02 02:53:36 +00:00
export const useBlockTransactions = (
provider: JsonRpcProvider | undefined,
2021-08-03 04:34:14 +00:00
blockNumber: number,
pageNumber: number,
pageSize: number
2021-08-03 03:45:07 +00:00
): [number | undefined, ProcessedTransaction[] | undefined] => {
const [totalTxs, setTotalTxs] = useState<number>();
2021-08-02 02:53:36 +00:00
const [txs, setTxs] = useState<ProcessedTransaction[]>();
useEffect(() => {
if (!provider) {
return;
}
const readBlock = async () => {
const result = await provider.send("ots_getBlockTransactions", [
blockNumber,
2021-08-03 04:34:14 +00:00
pageNumber,
pageSize,
2021-08-02 02:53:36 +00:00
]);
const _block = provider.formatter.blockWithTransactions(
result.fullblock
) as unknown as BlockWithTransactions;
const _receipts = result.receipts;
2021-08-02 02:53:36 +00:00
const rawTxs = _block.transactions
2021-08-02 02:53:36 +00:00
.map(
(t, i): ProcessedTransaction => ({
blockNumber: blockNumber,
timestamp: _block.timestamp,
miner: _block.miner,
idx: i,
hash: t.hash,
from: t.from,
to: t.to,
createdContractAddress: _receipts[i].contractAddress,
value: t.value,
fee:
t.type !== 2
? provider.formatter
.bigNumber(_receipts[i].gasUsed)
.mul(t.gasPrice!)
: provider.formatter
.bigNumber(_receipts[i].gasUsed)
.mul(t.maxPriorityFeePerGas!.add(_block.baseFeePerGas!)),
gasPrice:
t.type !== 2
? t.gasPrice!
: t.maxPriorityFeePerGas!.add(_block.baseFeePerGas!),
data: t.data,
status: provider.formatter.number(_receipts[i].status),
})
)
.reverse();
setTxs(rawTxs);
2021-08-03 03:45:07 +00:00
setTotalTxs(result.fullblock.transactionCount);
2021-08-02 02:53:36 +00:00
const checkTouchMinerAddr = await Promise.all(
rawTxs.map(async (res) => {
2021-08-02 02:53:36 +00:00
const ops = await getInternalOperations(provider, res.hash);
return (
ops.findIndex(
(op) =>
op.type === OperationType.TRANSFER &&
res.miner !== undefined &&
res.miner === getAddress(op.to)
2021-08-02 02:53:36 +00:00
) !== -1
);
})
);
const processedTxs = rawTxs.map(
2021-08-02 02:53:36 +00:00
(r, i): ProcessedTransaction => ({
...r,
internalMinerInteraction: checkTouchMinerAddr[i],
})
);
setTxs(processedTxs);
2021-08-02 02:53:36 +00:00
};
readBlock();
2021-08-03 04:34:14 +00:00
}, [provider, blockNumber, pageNumber, pageSize]);
2021-08-02 02:53:36 +00:00
2021-08-03 03:45:07 +00:00
return [totalTxs, txs];
2021-08-02 02:53:36 +00:00
};
2021-07-26 22:59:07 +00:00
export const useBlockData = (
provider: JsonRpcProvider | undefined,
2021-07-26 22:59:07 +00:00
blockNumberOrHash: string
) => {
const [block, setBlock] = useState<ExtendedBlock>();
useEffect(() => {
if (!provider) {
return;
}
const _readBlock = async () => {
const extBlock = await readBlock(provider, blockNumberOrHash);
2021-07-26 22:59:07 +00:00
setBlock(extBlock);
};
_readBlock();
2021-07-26 22:59:07 +00:00
}, [provider, blockNumberOrHash]);
return block;
};
2021-08-01 09:59:59 +00:00
export const useTxData = (
provider: JsonRpcProvider | undefined,
2021-08-01 09:59:59 +00:00
txhash: string
): TransactionData | undefined | null => {
const [txData, setTxData] = useState<TransactionData | undefined | null>();
2021-08-01 09:59:59 +00:00
useEffect(() => {
if (!provider) {
return;
}
const readTxData = async () => {
2021-08-01 09:59:59 +00:00
const [_response, _receipt] = await Promise.all([
provider.getTransaction(txhash),
provider.getTransactionReceipt(txhash),
]);
if (_response === null) {
setTxData(null);
return;
}
let _block: ExtendedBlock | undefined;
if (_response.blockNumber) {
_block = await readBlock(provider, _response.blockNumber.toString());
}
2021-08-01 09:59:59 +00:00
document.title = `Transaction ${_response.hash} | Otterscan`;
// Extract token transfers
const tokenTransfers: TokenTransfer[] = [];
if (_receipt) {
for (const l of _receipt.logs) {
if (l.topics.length !== 3) {
continue;
}
if (l.topics[0] !== TRANSFER_TOPIC) {
continue;
}
tokenTransfers.push({
token: l.address,
from: getAddress(hexDataSlice(arrayify(l.topics[1]), 12)),
to: getAddress(hexDataSlice(arrayify(l.topics[2]), 12)),
value: BigNumber.from(l.data),
});
2021-08-01 09:59:59 +00:00
}
}
// Extract token meta
const tokenMetas: TokenMetas = {};
for (const t of tokenTransfers) {
if (tokenMetas[t.token] !== undefined) {
2021-08-01 09:59:59 +00:00
continue;
}
const erc20Contract = new Contract(t.token, erc20, provider);
try {
const [name, symbol, decimals] = await Promise.all([
erc20Contract.name(),
erc20Contract.symbol(),
erc20Contract.decimals(),
]);
tokenMetas[t.token] = {
name,
symbol,
decimals,
};
} catch (err) {
tokenMetas[t.token] = null;
console.warn(`Couldn't get token ${t.token} metadata; ignoring`, err);
}
2021-08-01 09:59:59 +00:00
}
setTxData({
transactionHash: _response.hash,
from: _response.from,
to: _response.to,
2021-08-01 09:59:59 +00:00
value: _response.value,
tokenTransfers,
tokenMetas,
type: _response.type ?? 0,
maxFeePerGas: _response.maxFeePerGas,
maxPriorityFeePerGas: _response.maxPriorityFeePerGas,
gasPrice: _response.gasPrice!,
gasLimit: _response.gasLimit,
nonce: _response.nonce,
data: _response.data,
confirmedData:
_receipt === null
? undefined
: {
status: _receipt.status === 1,
blockNumber: _receipt.blockNumber,
transactionIndex: _receipt.transactionIndex,
blockBaseFeePerGas: _block!.baseFeePerGas,
blockTransactionCount: _block!.transactionCount,
confirmations: _receipt.confirmations,
timestamp: _block!.timestamp,
miner: _block!.miner,
createdContractAddress: _receipt.contractAddress,
fee: _response.gasPrice!.mul(_receipt.gasUsed),
gasUsed: _receipt.gasUsed,
logs: _receipt.logs,
},
2021-08-01 09:59:59 +00:00
});
};
readTxData();
2021-08-01 09:59:59 +00:00
}, [provider, txhash]);
return txData;
};
2021-07-21 19:06:51 +00:00
export const useInternalOperations = (
provider: JsonRpcProvider | undefined,
txData: TransactionData | undefined | null
2021-07-21 19:06:51 +00:00
): InternalOperation[] | undefined => {
const [intTransfers, setIntTransfers] = useState<InternalOperation[]>();
2021-07-17 18:00:08 +00:00
useEffect(() => {
const traceTransfers = async () => {
if (!provider || !txData || !txData.confirmedData) {
2021-07-17 18:00:08 +00:00
return;
}
2021-08-02 02:10:32 +00:00
const _transfers = await getInternalOperations(
provider,
txData.transactionHash
);
2021-07-19 23:49:54 +00:00
for (const t of _transfers) {
t.from = provider.formatter.address(t.from);
t.to = provider.formatter.address(t.to);
t.value = provider.formatter.bigNumber(t.value);
2021-07-17 18:58:33 +00:00
}
2021-07-19 23:49:54 +00:00
setIntTransfers(_transfers);
2021-07-17 18:00:08 +00:00
};
traceTransfers();
}, [provider, txData]);
2021-07-17 18:58:33 +00:00
return intTransfers;
2021-07-17 18:00:08 +00:00
};
2021-10-27 01:10:37 +00:00
export type TraceEntry = {
type: string;
depth: number;
from: string;
to: string;
value: BigNumber;
input: string;
};
export type TraceGroup = TraceEntry & {
children: TraceGroup[] | null;
};
export const useTraceTransaction = (
provider: JsonRpcProvider | undefined,
txHash: string
): TraceGroup[] | undefined => {
const [traceGroups, setTraceGroups] = useState<TraceGroup[] | undefined>();
useEffect(() => {
if (!provider) {
setTraceGroups(undefined);
return;
}
const traceTx = async () => {
const results = await provider.send("ots_traceTransaction", [txHash]);
// Implement better formatter
for (let i = 0; i < results.length; i++) {
results[i].from = provider.formatter.address(results[i].from);
results[i].to = provider.formatter.address(results[i].to);
2021-10-27 17:57:38 +00:00
results[i].value =
results[i].value === null
? null
: provider.formatter.bigNumber(results[i].value);
2021-10-27 01:10:37 +00:00
}
// Build trace tree
const buildTraceTree = (
flatList: TraceEntry[],
depth: number = 0
): TraceGroup[] => {
const entries: TraceGroup[] = [];
let children: TraceEntry[] | null = null;
for (let i = 0; i < flatList.length; i++) {
if (flatList[i].depth === depth) {
if (children !== null) {
const childrenTree = buildTraceTree(children, depth + 1);
const prev = entries.pop();
if (prev) {
prev.children = childrenTree;
entries.push(prev);
}
}
entries.push({
...flatList[i],
children: null,
});
children = null;
} else {
if (children === null) {
children = [];
}
children.push(flatList[i]);
}
}
if (children !== null) {
const childrenTree = buildTraceTree(children, depth + 1);
const prev = entries.pop();
if (prev) {
prev.children = childrenTree;
entries.push(prev);
}
}
return entries;
};
const traceTree = buildTraceTree(results);
setTraceGroups(traceTree);
};
traceTx();
}, [provider, txHash]);
return traceGroups;
};
/**
* Flatten a trace tree and extract and dedup 4byte function signatures
*/
export const useUniqueSignatures = (traces: TraceGroup[] | undefined) => {
const uniqueSignatures = useMemo(() => {
if (!traces) {
return undefined;
}
const sigs = new Set<string>();
let nextTraces: TraceGroup[] = [...traces];
while (nextTraces.length > 0) {
const traces = nextTraces;
nextTraces = [];
for (const t of traces) {
if (
t.type === "CALL" ||
t.type === "DELEGATECALL" ||
t.type === "STATICCALL" ||
t.type === "CALLCODE"
) {
const fourBytes = extract4Bytes(t.input);
if (fourBytes) {
sigs.add(fourBytes);
}
}
if (t.children) {
nextTraces.push(...t.children);
}
}
}
return [...sigs];
}, [traces]);
return uniqueSignatures;
};