otterscan/src/sourcify/useSourcify.ts

206 lines
4.5 KiB
TypeScript
Raw Normal View History

2021-09-15 17:43:36 +00:00
import { useState, useEffect, useMemo } from "react";
import { Interface } from "@ethersproject/abi";
2021-12-14 00:03:13 +00:00
import { ErrorDescription } from "@ethersproject/abi/lib/interface";
2022-08-09 22:23:38 +00:00
import useSWRImmutable from "swr/immutable";
2021-11-25 18:50:59 +00:00
import { ChecksummedAddress, TransactionData } from "../types";
import { sourcifyMetadata, SourcifySource, sourcifySourceFile } from "../url";
2021-09-06 00:08:06 +00:00
export type UserMethod = {
notice?: string | undefined;
};
export type UserEvent = {
notice?: string | undefined;
};
2021-12-14 00:03:13 +00:00
export type UserError = [
{
notice?: string | undefined;
}
];
export type UserDoc = {
kind: "user";
version?: number | undefined;
notice?: string | undefined;
methods: Record<string, UserMethod>;
events: Record<string, UserEvent>;
2021-12-14 00:03:13 +00:00
errors?: Record<string, UserError> | undefined;
};
export type DevMethod = {
params?: Record<string, string>;
returns?: Record<string, string>;
};
2021-12-14 00:03:13 +00:00
export type DevError = [
{
params?: Record<string, string>;
}
];
export type DevDoc = {
kind: "dev";
version?: number | undefined;
methods: Record<string, DevMethod>;
2021-12-14 00:03:13 +00:00
errors?: Record<string, DevError> | undefined;
};
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[];
userdoc?: UserDoc | undefined;
devdoc?: DevDoc | undefined;
2021-09-06 00:08:06 +00:00
};
};
const sourcifyFetcher = async (url: string) => {
try {
const res = await fetch(url);
if (res.ok) {
return res.json();
}
return null;
} catch (err) {
console.warn(
`error while getting Sourcify metadata: url=${url} err=${err}`
);
return null;
}
};
2022-08-09 22:23:38 +00:00
export const useSourcifyMetadata = (
address: ChecksummedAddress | undefined,
2021-09-06 21:32:11 +00:00
chainId: number | undefined,
source: SourcifySource
): Metadata | null | undefined => {
2022-08-09 22:23:38 +00:00
const metadataURL = () =>
address === undefined || chainId === undefined
? null
: sourcifyMetadata(address, chainId, source);
const { data, error } = useSWRImmutable<Metadata>(
metadataURL,
sourcifyFetcher
2022-08-09 22:23:38 +00:00
);
if (error) {
return null;
}
return data;
2021-09-06 00:08:06 +00:00
};
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,
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,
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();
};
}, [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);
2021-09-22 06:02:21 +00:00
try {
return intf.parseTransaction({
data: txData.data,
value: txData.value,
});
} catch (err) {
console.warn("Couldn't find function signature", err);
return null;
}
2021-09-15 17:43:36 +00:00
}, [metadata, txData]);
return txDesc;
};
2021-12-14 00:03:13 +00:00
export const useError = (
metadata: Metadata | null | undefined,
output: string | null | undefined
): ErrorDescription | null | undefined => {
const err = useMemo(() => {
if (!metadata || !output) {
return undefined;
}
const abi = metadata.output.abi;
const intf = new Interface(abi as any);
try {
return intf.parseError(output);
} catch (err) {
console.warn("Couldn't find error signature", err);
return null;
}
}, [metadata, output]);
return err;
};