2021-09-15 17:43:36 +00:00
|
|
|
import { useState, useEffect, useMemo } from "react";
|
|
|
|
import { Interface } from "@ethersproject/abi";
|
|
|
|
import { TransactionData } from "./types";
|
2021-09-10 21:27:42 +00:00
|
|
|
import { sourcifyMetadata, SourcifySource, sourcifySourceFile } from "./url";
|
2021-09-06 00:08:06 +00:00
|
|
|
|
|
|
|
export type Metadata = {
|
|
|
|
version: string;
|
|
|
|
language: string;
|
|
|
|
compiler: {
|
|
|
|
version: string;
|
|
|
|
keccak256?: string | undefined;
|
|
|
|
};
|
|
|
|
sources: {
|
|
|
|
[filename: string]: {
|
|
|
|
keccak256: string;
|
|
|
|
content?: string | undefined;
|
|
|
|
urls?: string[];
|
|
|
|
license?: string;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
settings: {
|
|
|
|
remappings: string[];
|
|
|
|
optimizer?: {
|
|
|
|
enabled: boolean;
|
|
|
|
runs: number;
|
|
|
|
};
|
|
|
|
compilationTarget: {
|
|
|
|
[filename: string]: string;
|
|
|
|
};
|
|
|
|
libraries: {
|
|
|
|
[filename: string]: string;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
output: {
|
|
|
|
abi: any[];
|
|
|
|
userdocs: any[];
|
|
|
|
devdoc: any[];
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
export const useSourcify = (
|
|
|
|
checksummedAddress: string | undefined,
|
2021-09-06 21:32:11 +00:00
|
|
|
chainId: number | undefined,
|
2021-09-10 21:27:42 +00:00
|
|
|
source: SourcifySource
|
2021-09-06 00:08:06 +00:00
|
|
|
) => {
|
|
|
|
const [rawMetadata, setRawMetadata] = useState<Metadata | null | undefined>();
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
if (!checksummedAddress || chainId === undefined) {
|
|
|
|
return;
|
|
|
|
}
|
2021-09-10 21:27:42 +00:00
|
|
|
setRawMetadata(undefined);
|
2021-09-14 01:24:22 +00:00
|
|
|
|
|
|
|
const abortController = new AbortController();
|
2021-09-06 00:08:06 +00:00
|
|
|
const fetchMetadata = async () => {
|
|
|
|
try {
|
|
|
|
const contractMetadataURL = sourcifyMetadata(
|
|
|
|
checksummedAddress,
|
2021-09-06 21:32:11 +00:00
|
|
|
chainId,
|
2021-09-10 21:27:42 +00:00
|
|
|
source
|
2021-09-06 00:08:06 +00:00
|
|
|
);
|
2021-09-14 01:24:22 +00:00
|
|
|
const result = await fetch(contractMetadataURL, {
|
|
|
|
signal: abortController.signal,
|
|
|
|
});
|
2021-09-06 00:08:06 +00:00
|
|
|
if (result.ok) {
|
|
|
|
const _metadata = await result.json();
|
|
|
|
setRawMetadata(_metadata);
|
|
|
|
} else {
|
|
|
|
setRawMetadata(null);
|
|
|
|
}
|
|
|
|
} catch (err) {
|
|
|
|
console.error(err);
|
|
|
|
setRawMetadata(null);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
fetchMetadata();
|
2021-09-14 01:24:22 +00:00
|
|
|
|
|
|
|
return () => {
|
|
|
|
abortController.abort();
|
|
|
|
};
|
2021-09-10 21:27:42 +00:00
|
|
|
}, [checksummedAddress, chainId, source]);
|
2021-09-06 00:08:06 +00:00
|
|
|
|
|
|
|
return rawMetadata;
|
|
|
|
};
|
2021-09-06 05:09:12 +00:00
|
|
|
|
|
|
|
export const useContract = (
|
|
|
|
checksummedAddress: string,
|
|
|
|
networkId: number,
|
|
|
|
filename: string,
|
2021-09-06 21:32:11 +00:00
|
|
|
source: any,
|
2021-09-10 21:27:42 +00:00
|
|
|
sourcifySource: SourcifySource
|
2021-09-06 05:09:12 +00:00
|
|
|
) => {
|
|
|
|
const [content, setContent] = useState<string>(source.content);
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
if (source.content) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-09-14 01:24:22 +00:00
|
|
|
const abortController = new AbortController();
|
2021-09-06 05:09:12 +00:00
|
|
|
const readContent = async () => {
|
2021-09-07 03:44:28 +00:00
|
|
|
const normalizedFilename = filename.replaceAll(/[@:]/g, "_");
|
2021-09-06 05:09:12 +00:00
|
|
|
const url = sourcifySourceFile(
|
|
|
|
checksummedAddress,
|
|
|
|
networkId,
|
2021-09-06 21:32:11 +00:00
|
|
|
normalizedFilename,
|
2021-09-10 21:27:42 +00:00
|
|
|
sourcifySource
|
2021-09-06 05:09:12 +00:00
|
|
|
);
|
2021-09-14 01:24:22 +00:00
|
|
|
const res = await fetch(url, { signal: abortController.signal });
|
2021-09-06 05:09:12 +00:00
|
|
|
if (res.ok) {
|
|
|
|
const _content = await res.text();
|
|
|
|
setContent(_content);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
readContent();
|
2021-09-14 01:24:22 +00:00
|
|
|
|
|
|
|
return () => {
|
|
|
|
abortController.abort();
|
|
|
|
};
|
2021-09-10 21:27:42 +00:00
|
|
|
}, [checksummedAddress, networkId, filename, source.content, sourcifySource]);
|
2021-09-06 05:09:12 +00:00
|
|
|
|
|
|
|
return content;
|
|
|
|
};
|
2021-09-15 17:43:36 +00:00
|
|
|
|
|
|
|
export const useTransactionDescription = (
|
|
|
|
metadata: Metadata | null | undefined,
|
|
|
|
txData: TransactionData | null | undefined
|
|
|
|
) => {
|
|
|
|
const txDesc = useMemo(() => {
|
2021-09-18 18:40:19 +00:00
|
|
|
if (metadata === null) {
|
|
|
|
return null;
|
|
|
|
}
|
2021-09-15 17:43:36 +00:00
|
|
|
if (!metadata || !txData) {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
const abi = metadata.output.abi;
|
|
|
|
const intf = new Interface(abi as any);
|
|
|
|
return intf.parseTransaction({
|
|
|
|
data: txData.data,
|
|
|
|
value: txData.value,
|
|
|
|
});
|
|
|
|
}, [metadata, txData]);
|
|
|
|
|
|
|
|
return txDesc;
|
|
|
|
};
|