Skip to content

Commit

Permalink
Add support for Sourcify contract information lookup
Browse files Browse the repository at this point in the history
- Contract name, ABI and creation transaction hash (start block)
  from [Sourcify API](https://docs.sourcify.dev/docs/api/).
- Runs before the registry lookup and replaces default values
  (not interactive) if not provided by the user.
  This means priority for CLI parameters looks like:
  user submitted (env/CLI args) > Sourcify API > Default values > Registry fetch
  • Loading branch information
0237h committed Jan 13, 2025
1 parent 93d87a8 commit c5c9010
Show file tree
Hide file tree
Showing 3 changed files with 101 additions and 4 deletions.
55 changes: 55 additions & 0 deletions packages/cli/src/command-helpers/contracts.ts
Original file line number Diff line number Diff line change
Expand Up @@ -151,6 +151,61 @@ export class ContractService {
throw new Error(`Failed to fetch contract name for ${address}`);
}

async getFromSourcify(
ABICtor: typeof ABI,
networkId: string,
address: string,
): Promise<{ abi: ABI; startBlock: string; name: string } | null> {
try {
const network = this.registry.getNetworkById(networkId);
if (!network) throw new Error(`Invalid network ${networkId}`);

const chainId = network.caip2Id.split(':')[1];
if (!/^\d+$/.test(chainId))
throw new Error(`Invalid chainId, Sourcify API expects integer value, got '${chainId}'`);

const url = `https://sourcify.dev/server/files/any/${chainId}/${address}`;
const json:
| {
status: string;
files: { name: string; path: string; content: string }[];
}
| { error: string } = await (
await fetch(url).catch(error => {
throw new Error(`Sourcify API is unreachable: ${error}`);
})
).json();

if (json) {
if ('error' in json) throw new Error(`Sourcify API error: ${json.error}`);

let metadata: any = json.files.find(e => e.name === 'metadata.json')?.content;
if (!metadata) throw new Error('Contract is missing metadata');

const tx_hash = json.files.find(e => e.name === 'creator-tx-hash.txt')?.content;
if (!tx_hash) throw new Error('Contract is missing tx creation hash');

const tx = await this.fetchTransactionByHash(networkId, tx_hash);
if (!tx?.blockNumber)
throw new Error(`Can't fetch blockNumber from tx: ${JSON.stringify(tx)}`);

metadata = JSON.parse(metadata);
const contractName = Object.values(metadata.settings.compilationTarget)[0] as string;
return {
abi: new ABICtor(contractName, undefined, immutable.fromJS(metadata.output.abi)) as ABI,
startBlock: Number(tx.blockNumber).toString(),
name: contractName,
};
}

throw new Error(`No result: ${JSON.stringify(json)}`);
} catch (error) {
logger(`Failed to fetch from Sourcify: ${error}`);
}

return null;
}

private async fetchTransactionByHash(networkId: string, txHash: string) {
const urls = this.getRpcUrls(networkId);
if (!urls.length) {
Expand Down
16 changes: 14 additions & 2 deletions packages/cli/src/commands/add.ts
Original file line number Diff line number Diff line change
Expand Up @@ -80,12 +80,24 @@ export default class AddCommand extends Command {
if (isLocalHost) this.warn('`localhost` network detected, prompting user for inputs');
const registry = await loadRegistry();
const contractService = new ContractService(registry);
const sourcifyContractInfo = await contractService.getFromSourcify(
EthereumABI,
network,
address,
);

let startBlock = startBlockFlag ? parseInt(startBlockFlag).toString() : startBlockFlag;
let contractName = contractNameFlag || DEFAULT_CONTRACT_NAME;

let ethabi = null;
if (abi) {

if (sourcifyContractInfo) {
startBlock ??= sourcifyContractInfo.startBlock;
contractName =
contractName == DEFAULT_CONTRACT_NAME ? sourcifyContractInfo.name : contractName;
ethabi ??= sourcifyContractInfo.abi;
}

if (!ethabi && abi) {
ethabi = EthereumABI.load(contractName, abi);
} else {
try {
Expand Down
34 changes: 32 additions & 2 deletions packages/cli/src/commands/init.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ import fs from 'node:fs';
import os from 'node:os';
import path from 'node:path';
import { filesystem, print, prompt, system } from 'gluegun';
import immutable from 'immutable';
import { Args, Command, Flags } from '@oclif/core';
import { Network } from '@pinax/graph-networks-registry';
import { appendApiVersionForGraph } from '../command-helpers/compiler.js';
Expand Down Expand Up @@ -200,6 +201,11 @@ export default class InitCommand extends Command {
if ((fromContract || spkgPath) && protocol && subgraphName && directory && network && node) {
const registry = await loadRegistry();
const contractService = new ContractService(registry);
const sourcifyContractInfo = await contractService.getFromSourcify(
EthereumABI,
network,
fromContract!,
);

if (!protocolChoices.includes(protocol as ProtocolName)) {
this.error(
Expand All @@ -222,7 +228,13 @@ export default class InitCommand extends Command {
}
} else {
try {
abi = await contractService.getABI(ABI, network, fromContract!);
abi = sourcifyContractInfo
? new EthereumABI(
DEFAULT_CONTRACT_NAME,
undefined,
immutable.fromJS(sourcifyContractInfo.abi),
)
: await contractService.getABI(ABI, network, fromContract!);
} catch (e) {
this.exit(1);
}
Expand Down Expand Up @@ -448,7 +460,7 @@ async function processInitForm(
];
};

let network = networks[0];
let network: Network = networks[0];
let protocolInstance: Protocol = new Protocol('ethereum');
let isComposedSubgraph = false;
let isSubstreams = false;
Expand Down Expand Up @@ -611,6 +623,22 @@ async function processInitForm(
return address;
}

const sourcifyContractInfo = await contractService.getFromSourcify(
EthereumABI,
network.id,
address,
);
if (sourcifyContractInfo) {
initStartBlock ??= sourcifyContractInfo.startBlock;
initContractName ??= sourcifyContractInfo.name;
initAbi ??= sourcifyContractInfo.abi;
initDebugger.extend('processInitForm')(
"infoFromSourcify: '%s'/'%s'",
initStartBlock,
initContractName
);
}

// If ABI is not provided, try to fetch it from Etherscan API
if (protocolInstance.hasABIs() && !initAbi) {
abiFromApi = await retryWithPrompt(() =>
Expand All @@ -622,6 +650,8 @@ async function processInitForm(
),
);
initDebugger.extend('processInitForm')("abiFromEtherscan len: '%s'", abiFromApi?.name);
} else {
abiFromApi = initAbi;
}
// If startBlock is not provided, try to fetch it from Etherscan API
if (!initStartBlock) {
Expand Down

0 comments on commit c5c9010

Please sign in to comment.