2021-07-01 18:21:40 +00:00
|
|
|
import React, { useEffect, useState, useMemo } from "react";
|
|
|
|
import { useParams, useLocation } from "react-router";
|
|
|
|
import { ethers } from "ethers";
|
|
|
|
import queryString from "query-string";
|
|
|
|
import { provider } from "./ethersconfig";
|
|
|
|
import StandardFrame from "./StandardFrame";
|
|
|
|
import StandardSubtitle from "./StandardSubtitle";
|
|
|
|
import ContentFrame from "./ContentFrame";
|
|
|
|
import PageControl from "./search/PageControl";
|
|
|
|
import ResultHeader from "./search/ResultHeader";
|
|
|
|
import PendingResults from "./search/PendingResults";
|
|
|
|
import TransactionItem from "./search/TransactionItem";
|
|
|
|
import BlockLink from "./components/BlockLink";
|
2021-07-03 00:40:02 +00:00
|
|
|
import { ProcessedTransaction, ENSReverseCache } from "./types";
|
2021-07-01 18:21:40 +00:00
|
|
|
import { PAGE_SIZE } from "./params";
|
|
|
|
import { useFeeToggler } from "./search/useFeeToggler";
|
|
|
|
|
|
|
|
type BlockParams = {
|
|
|
|
blockNumber: string;
|
|
|
|
};
|
|
|
|
|
|
|
|
type PageParams = {
|
|
|
|
p?: number;
|
|
|
|
};
|
|
|
|
|
|
|
|
const BlockTransactions: React.FC = () => {
|
|
|
|
const params = useParams<BlockParams>();
|
|
|
|
const location = useLocation<PageParams>();
|
|
|
|
const qs = queryString.parse(location.search);
|
|
|
|
let pageNumber = 1;
|
|
|
|
if (qs.p) {
|
|
|
|
try {
|
|
|
|
pageNumber = parseInt(qs.p as string);
|
|
|
|
} catch (err) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
const blockNumber = useMemo(
|
|
|
|
() => ethers.BigNumber.from(params.blockNumber),
|
|
|
|
[params.blockNumber]
|
|
|
|
);
|
|
|
|
|
|
|
|
const [txs, setTxs] = useState<ProcessedTransaction[]>();
|
|
|
|
useEffect(() => {
|
|
|
|
const readBlock = async () => {
|
|
|
|
const [_block, _receipts] = await Promise.all([
|
|
|
|
provider.getBlockWithTransactions(blockNumber.toNumber()),
|
|
|
|
provider.send("eth_getBlockReceipts", [blockNumber.toNumber()]),
|
|
|
|
]);
|
|
|
|
document.title = `Block #${_block.number} Transactions | Otterscan`;
|
|
|
|
|
|
|
|
setTxs(
|
|
|
|
_block.transactions
|
|
|
|
.map((t, i) => {
|
|
|
|
return {
|
|
|
|
blockNumber: blockNumber.toNumber(),
|
|
|
|
timestamp: _block.timestamp,
|
|
|
|
idx: i,
|
|
|
|
hash: t.hash,
|
|
|
|
from: t.from,
|
|
|
|
to: t.to,
|
|
|
|
value: t.value,
|
|
|
|
fee: t.gasLimit.mul(t.gasPrice!),
|
|
|
|
gasPrice: t.gasPrice!,
|
|
|
|
data: t.data,
|
|
|
|
status: _receipts[i].status,
|
|
|
|
};
|
|
|
|
})
|
|
|
|
.reverse()
|
|
|
|
);
|
|
|
|
};
|
|
|
|
readBlock();
|
|
|
|
}, [blockNumber]);
|
|
|
|
|
|
|
|
const page = useMemo(() => {
|
|
|
|
if (!txs) {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
const pageStart = (pageNumber - 1) * PAGE_SIZE;
|
|
|
|
return txs.slice(pageStart, pageStart + PAGE_SIZE);
|
|
|
|
}, [txs, pageNumber]);
|
|
|
|
const total = useMemo(() => txs?.length ?? 0, [txs]);
|
|
|
|
|
2021-07-03 00:40:02 +00:00
|
|
|
const [reverseCache, setReverseCache] = useState<ENSReverseCache>();
|
|
|
|
useEffect(() => {
|
|
|
|
if (!page) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const addrSet = new Set<string>();
|
|
|
|
for (const tx of page) {
|
|
|
|
if (tx.from) {
|
|
|
|
addrSet.add(tx.from);
|
|
|
|
}
|
|
|
|
if (tx.to) {
|
|
|
|
addrSet.add(tx.to);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const addresses = Array.from(addrSet);
|
|
|
|
|
|
|
|
const reverseResolve = async () => {
|
|
|
|
const solvers: Promise<string>[] = [];
|
|
|
|
for (const a of addresses) {
|
|
|
|
solvers.push(provider.lookupAddress(a));
|
|
|
|
}
|
|
|
|
|
|
|
|
const results = await Promise.all(solvers);
|
|
|
|
const cache: ENSReverseCache = {};
|
|
|
|
for (let i = 0; i < results.length; i++) {
|
|
|
|
if (results[i] === null) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
cache[addresses[i]] = results[i];
|
|
|
|
}
|
|
|
|
console.log("RESOLVED");
|
|
|
|
console.log(cache);
|
|
|
|
setReverseCache(cache);
|
|
|
|
};
|
|
|
|
reverseResolve();
|
|
|
|
}, [page]);
|
|
|
|
|
2021-07-01 18:21:40 +00:00
|
|
|
document.title = `Block #${blockNumber} Txns | Otterscan`;
|
|
|
|
|
|
|
|
const [feeDisplay, feeDisplayToggler] = useFeeToggler();
|
|
|
|
|
|
|
|
return (
|
|
|
|
<StandardFrame>
|
|
|
|
<StandardSubtitle>Transactions</StandardSubtitle>
|
|
|
|
<div className="pb-2 text-sm text-gray-500">
|
|
|
|
For Block <BlockLink blockTag={blockNumber.toNumber()} />
|
|
|
|
</div>
|
|
|
|
<ContentFrame>
|
|
|
|
<div className="flex justify-between items-baseline py-3">
|
|
|
|
<div className="text-sm text-gray-500">
|
|
|
|
{page === undefined ? (
|
|
|
|
<>Waiting for search results...</>
|
|
|
|
) : (
|
|
|
|
<>A total of {total} transactions found</>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
<PageControl
|
|
|
|
pageNumber={pageNumber}
|
|
|
|
pageSize={PAGE_SIZE}
|
|
|
|
total={total}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
<ResultHeader
|
|
|
|
feeDisplay={feeDisplay}
|
|
|
|
feeDisplayToggler={feeDisplayToggler}
|
|
|
|
/>
|
|
|
|
{page ? (
|
|
|
|
<>
|
|
|
|
{page.map((tx) => (
|
2021-07-03 00:40:02 +00:00
|
|
|
<TransactionItem
|
|
|
|
key={tx.hash}
|
|
|
|
tx={tx}
|
|
|
|
ensCache={reverseCache}
|
|
|
|
feeDisplay={feeDisplay}
|
|
|
|
/>
|
2021-07-01 18:21:40 +00:00
|
|
|
))}
|
|
|
|
<div className="flex justify-between items-baseline py-3">
|
|
|
|
<div className="text-sm text-gray-500">
|
|
|
|
A total of {total} transactions found
|
|
|
|
</div>
|
|
|
|
<PageControl
|
|
|
|
pageNumber={pageNumber}
|
|
|
|
pageSize={PAGE_SIZE}
|
|
|
|
total={total}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</>
|
|
|
|
) : (
|
|
|
|
<PendingResults />
|
|
|
|
)}
|
|
|
|
</ContentFrame>
|
|
|
|
</StandardFrame>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export default React.memo(BlockTransactions);
|