From f4d68cf8fceb807598550a8d185ebf8ddca315d4 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Kaan=20Uzdo=C4=9Fan?= Date: Tue, 11 Jul 2023 11:10:26 +0200 Subject: [PATCH] Use SourcifyChain class with providers --- packages/lib-sourcify/src/index.ts | 1 + .../lib-sourcify/src/lib/SourcifyChain.ts | 221 ++++++++++++++++++ packages/lib-sourcify/src/lib/types.ts | 10 +- packages/lib-sourcify/src/lib/verification.ts | 123 +--------- .../lib-sourcify/test/verification.spec.ts | 65 ++++-- .../SourcifyEventManager.ts | 1 - src/monitor/monitor.ts | 75 ++---- .../session-state/session-state.handlers.ts | 3 +- src/server/server.ts | 38 +-- src/sourcify-chains.ts | 29 ++- 10 files changed, 342 insertions(+), 224 deletions(-) create mode 100644 packages/lib-sourcify/src/lib/SourcifyChain.ts diff --git a/packages/lib-sourcify/src/index.ts b/packages/lib-sourcify/src/index.ts index 7a7c0fa6e..2c20204cd 100644 --- a/packages/lib-sourcify/src/index.ts +++ b/packages/lib-sourcify/src/index.ts @@ -3,6 +3,7 @@ import { setLogger, setLevel, ILogger } from './lib/logger'; export * from './lib/validation'; export * from './lib/verification'; export * from './lib/CheckedContract'; +export { default as SourcifyChain } from './lib/SourcifyChain'; export * from './lib/types'; export * from './lib/solidityCompiler'; export const setLibSourcifyLogger = setLogger; diff --git a/packages/lib-sourcify/src/lib/SourcifyChain.ts b/packages/lib-sourcify/src/lib/SourcifyChain.ts new file mode 100644 index 000000000..4ad7ba427 --- /dev/null +++ b/packages/lib-sourcify/src/lib/SourcifyChain.ts @@ -0,0 +1,221 @@ +import { + FetchRequest, + JsonRpcProvider, + Network, + TransactionResponse, + getAddress, +} from 'ethers'; +import { Chain, SourcifyChainExtension } from './types'; +import { logInfo, logWarn } from './logger'; + +const RPC_TIMEOUT = process.env.RPC_TIMEOUT + ? parseInt(process.env.RPC_TIMEOUT) + : 5000; + +// It is impossible to get the url from the Provider for logging purposes +interface JsonRpcProviderWithUrl extends JsonRpcProvider { + url?: string; +} + +// Need to define the rpc property explicitly as when a sourcifyChain is created with {...chain, sourcifyChainExtension}, Typescript throws with "Type '(string | FetchRequest)[]' is not assignable to type 'string[]'." For some reason the Chain.rpc is not getting overwritten by SourcifyChainExtension.rpc +export type SourcifyChainInstance = Omit & + Omit & { rpc: Array }; + +export default class SourcifyChain { + name: string; + title?: string | undefined; + chainId: number; + rpc: Array; + supported: boolean; + monitored: boolean; + contractFetchAddress?: string | undefined; + graphQLFetchAddress?: string | undefined; + txRegex?: string[] | undefined; + providers: JsonRpcProviderWithUrl[]; + + constructor(sourcifyChainObj: SourcifyChainInstance) { + this.name = sourcifyChainObj.name; + this.title = sourcifyChainObj.title; + this.chainId = sourcifyChainObj.chainId; + this.rpc = sourcifyChainObj.rpc; + this.supported = sourcifyChainObj.supported; + this.monitored = sourcifyChainObj.monitored; + this.contractFetchAddress = sourcifyChainObj.contractFetchAddress; + this.graphQLFetchAddress = sourcifyChainObj.graphQLFetchAddress; + this.txRegex = sourcifyChainObj.txRegex; + this.providers = []; + + if (!this.supported) return; // Don't create providers if chain is not supported + + if (!this?.rpc.length) + throw new Error( + 'No RPC provider was given for this chain with id ' + + this.chainId + + ' and name ' + + this.name + ); + + for (const rpc of this.rpc) { + let provider: JsonRpcProviderWithUrl | undefined; + const ethersNetwork = new Network(this.name, this.chainId); + if (typeof rpc === 'string') { + if (rpc.startsWith('http')) { + // Use staticNetwork to avoid sending unnecessary eth_chainId requests + provider = new JsonRpcProvider(rpc, ethersNetwork, { + staticNetwork: ethersNetwork, + }); + provider.url = rpc; + } else { + // Do not use WebSockets because of not being able to catch errors on websocket initialization. Most networks don't support WebSockets anyway. See https://github.com/ethers-io/ethers.js/discussions/2896 + // provider = new WebSocketProvider(rpc); + logInfo(`Won't create a WebSocketProvider for ${rpc}`); + } + } else { + provider = new JsonRpcProvider(rpc, ethersNetwork, { + staticNetwork: ethersNetwork, + }); + provider.url = rpc.url; + } + if (provider) { + this.providers.push(provider); + } + } + } + + rejectInMs = (ms: number, host?: string) => + new Promise((_resolve, reject) => { + setTimeout(() => reject(`RPC ${host} took too long to respond`), ms); + }); + + getTx = async (creatorTxHash: string) => { + // Try sequentially all providers + for (const provider of this.providers) { + try { + // Race the RPC call with a timeout + const tx = await Promise.race([ + provider.getTransaction(creatorTxHash), + this.rejectInMs(RPC_TIMEOUT, provider.url), + ]); + if (tx instanceof TransactionResponse) { + logInfo( + `Transaction ${creatorTxHash} fetched via ${provider.url} from chain ${this.chainId}` + ); + return tx; + } else { + throw new Error( + `Transaction ${creatorTxHash} not found on RPC ${provider.url} and chain ${this.chainId}` + ); + } + } catch (err) { + if (err instanceof Error) { + logWarn( + `Can't fetch the transaction ${creatorTxHash} from RPC ${provider.url} and chain ${this.chainId}\n ${err}` + ); + continue; + } else { + throw err; + } + } + } + throw new Error( + 'None of the RPCs responded fetching tx ' + + creatorTxHash + + ' on chain ' + + this.chainId + ); + }; + + /** + * Fetches the contract's deployed bytecode from SourcifyChain's rpc's. + * Tries to fetch sequentially if the first RPC is a local eth node. Fetches in parallel otherwise. + * + * @param {SourcifyChain} sourcifyChain - chain object with rpc's + * @param {string} address - contract address + */ + getBytecode = async (address: string): Promise => { + address = getAddress(address); + + // Request sequentially. Custom node is always before ALCHEMY so we don't waste resources if succeeds. + for (const provider of this.providers) { + try { + // Race the RPC call with a timeout + const bytecode = await Promise.race([ + provider.getCode(address), + this.rejectInMs(RPC_TIMEOUT, provider.url), + ]); + return bytecode; + } catch (err) { + if (err instanceof Error) { + logWarn( + `Can't fetch bytecode from RPC ${provider.url} and chain ${this.chainId}` + ); + continue; + } else { + throw err; + } + } + } + throw new Error( + 'None of the RPCs responded fetching bytecode for ' + + address + + ' on chain ' + + this.chainId + ); + }; + + getBlock = async (blockNumber: number, preFetchTxs = true) => { + // Request sequentially. Custom node is always before ALCHEMY so we don't waste resources if succeeds. + for (const provider of this.providers) { + try { + // Race the RPC call with a timeout + const block = await Promise.race([ + provider.getBlock(blockNumber, preFetchTxs), + this.rejectInMs(RPC_TIMEOUT, provider.url), + ]); + return block; + } catch (err) { + if (err instanceof Error) { + logWarn( + `Can't fetch block ${blockNumber} from RPC ${provider.url} and chain ${this.chainId}` + ); + continue; + } else { + throw err; + } + } + } + throw new Error( + 'None of the RPCs responded fetching block ' + + blockNumber + + ' on chain ' + + this.chainId + ); + }; + + getBlockNumber = async () => { + // Request sequentially. Custom node is always before ALCHEMY so we don't waste resources if succeeds. + for (const provider of this.providers) { + try { + // Race the RPC call with a timeout + const block = await Promise.race([ + provider.getBlockNumber(), + this.rejectInMs(RPC_TIMEOUT, provider.url), + ]); + return block; + } catch (err) { + if (err instanceof Error) { + logWarn( + `Can't fetch the current block number from RPC ${provider.url} and chain ${this.chainId}` + ); + continue; + } else { + throw err; + } + } + } + throw new Error( + 'None of the RPCs responded fetching the blocknumber on chain ' + + this.chainId + ); + }; +} diff --git a/packages/lib-sourcify/src/lib/types.ts b/packages/lib-sourcify/src/lib/types.ts index 0bddbf5b8..a0c220bdc 100644 --- a/packages/lib-sourcify/src/lib/types.ts +++ b/packages/lib-sourcify/src/lib/types.ts @@ -1,5 +1,6 @@ import { Abi } from 'abitype'; - +import { FetchRequest } from 'ethers'; +import SourcifyChain from './SourcifyChain'; export interface PathBuffer { path: string; buffer: Buffer; @@ -190,7 +191,7 @@ export type SourcifyChainExtension = { contractFetchAddress?: string; graphQLFetchAddress?: string; txRegex?: string[]; - rpc?: string[]; + rpc?: Array; }; // TODO: Double check against ethereum-lists/chains type @@ -202,14 +203,11 @@ export type Chain = { network?: string; networkId: number; nativeCurrency: Currency; - rpc: string[]; + rpc: Array; faucets?: string[]; infoURL?: string; }; -// a type that extends the Chain type -export type SourcifyChain = Chain & SourcifyChainExtension; - export type SourcifyChainMap = { [chainId: string]: SourcifyChain; }; diff --git a/packages/lib-sourcify/src/lib/verification.ts b/packages/lib-sourcify/src/lib/verification.ts index 1074e254f..31e2dfe11 100644 --- a/packages/lib-sourcify/src/lib/verification.ts +++ b/packages/lib-sourcify/src/lib/verification.ts @@ -6,22 +6,13 @@ import { Match, Metadata, RecompilationResult, - SourcifyChain, StringMap, } from './types'; import { decode as bytecodeDecode, splitAuxdata, } from '@ethereum-sourcify/bytecode-utils'; -import { - JsonRpcProvider, - Network, - TransactionResponse, - WebSocketProvider, - getAddress, - getCreateAddress, - keccak256, -} from 'ethers'; +import { getAddress, getCreateAddress, keccak256 } from 'ethers'; /* import { EVM } from '@ethereumjs/evm'; import { EEI } from '@ethereumjs/vm'; @@ -35,11 +26,8 @@ import { BigNumber } from '@ethersproject/bignumber'; import semverSatisfies from 'semver/functions/satisfies'; import { defaultAbiCoder as abiCoder, ParamType } from '@ethersproject/abi'; import { AbiConstructor } from 'abitype'; -import { logInfo, logWarn } from './logger'; - -const RPC_TIMEOUT = process.env.RPC_TIMEOUT - ? parseInt(process.env.RPC_TIMEOUT) - : 5000; +import { logInfo } from './logger'; +import SourcifyChain from './SourcifyChain'; export async function verifyDeployed( checkedContract: CheckedContract, @@ -69,7 +57,7 @@ export async function verifyDeployed( ); } - const deployedBytecode = await getBytecode(sourcifyChain, address); + const deployedBytecode = await sourcifyChain.getBytecode(address); // Can't match if there is no deployed bytecode if (!deployedBytecode) { @@ -394,7 +382,7 @@ export async function matchWithCreationTx( return; } - const creatorTx = await getTx(creatorTxHash, sourcifyChain); + const creatorTx = await sourcifyChain.getTx(creatorTxHash); const creatorTxData = creatorTx.data; // The reason why this uses `startsWith` instead of `===` is that creationTxData may contain constructor arguments at the end part. @@ -474,107 +462,6 @@ export async function matchWithCreationTx( match.creatorTxHash = creatorTxHash; } } -/** - * Fetches the contract's deployed bytecode from SourcifyChain's rpc's. - * Tries to fetch sequentially if the first RPC is a local eth node. Fetches in parallel otherwise. - * - * @param {SourcifyChain} sourcifyChain - chain object with rpc's - * @param {string} address - contract address - */ -export async function getBytecode( - sourcifyChain: SourcifyChain, - address: string -): Promise { - if (!sourcifyChain?.rpc.length) - throw new Error('No RPC provider was given for this chain.'); - address = getAddress(address); - - // Request sequentially. Custom node is always before ALCHEMY so we don't waste resources if succeeds. - for (const rpcURL of sourcifyChain.rpc) { - const ethersNetwork = new Network( - sourcifyChain.name, - sourcifyChain.chainId - ); - - const provider = rpcURL.startsWith('http') - ? // Use staticNetwork to avoid seding unnecessary eth_chainId requests - new JsonRpcProvider(rpcURL, ethersNetwork, { - staticNetwork: ethersNetwork, - }) - : new WebSocketProvider(rpcURL); - - try { - // Race the RPC call with a timeout - const bytecode = await Promise.race([ - provider.getCode(address), - rejectInMs(RPC_TIMEOUT, rpcURL), - ]); - return bytecode; - } catch (err) { - if (err instanceof Error) { - logWarn( - `Can't fetch bytecode from RPC ${rpcURL} and chain ${sourcifyChain.chainId}` - ); - continue; - } else { - throw err; - } - } - } - throw new Error('None of the RPCs responded'); -} - -async function getTx(creatorTxHash: string, sourcifyChain: SourcifyChain) { - if (!sourcifyChain?.rpc.length) - throw new Error('No RPC provider was given for this chain.'); - - for (const rpcURL of sourcifyChain.rpc) { - const ethersNetwork = new Network( - sourcifyChain.name, - sourcifyChain.chainId - ); - - const provider = rpcURL.startsWith('http') - ? // Use staticNetwork to avoid seding unnecessary eth_chainId requests - new JsonRpcProvider(rpcURL, ethersNetwork, { - staticNetwork: ethersNetwork, - }) - : new WebSocketProvider(rpcURL); - - try { - // Race the RPC call with a timeout - const tx = await Promise.race([ - provider.getTransaction(creatorTxHash), - rejectInMs(RPC_TIMEOUT, rpcURL), - ]); - if (tx instanceof TransactionResponse) { - logInfo( - `Transaction ${creatorTxHash} fetched via ${rpcURL} from chain ${sourcifyChain.chainId}` - ); - return tx; - } else { - throw new Error( - `Transaction ${creatorTxHash} not found on RPC ${rpcURL} and chain ${sourcifyChain.chainId}` - ); - } - } catch (err) { - if (err instanceof Error) { - logWarn( - `Can't fetch bytecode from RPC ${rpcURL} and chain ${sourcifyChain.chainId}` - ); - continue; - } else { - throw err; - } - } - } - throw new Error('None of the RPCs responded'); -} - -const rejectInMs = (ms: number, host: string) => - new Promise((_resolve, reject) => { - setTimeout(() => reject(`RPC ${host} took too long to respond`), ms); - }); export function addLibraryAddresses( template: string, diff --git a/packages/lib-sourcify/test/verification.spec.ts b/packages/lib-sourcify/test/verification.spec.ts index 8ac17b46c..d2401f97e 100644 --- a/packages/lib-sourcify/test/verification.spec.ts +++ b/packages/lib-sourcify/test/verification.spec.ts @@ -1,6 +1,6 @@ /* eslint-disable @typescript-eslint/no-var-requires */ import path from 'path'; -import { Metadata, SourcifyChain } from '../src/lib/types'; +import { Metadata } from '../src/lib/types'; import Ganache from 'ganache'; import { /* callContractMethodWithTx, */ @@ -13,6 +13,7 @@ import { import { describe, it, before } from 'mocha'; import { expect } from 'chai'; import { + SourcifyChain, calculateCreate2Address, /* getBytecode, @@ -24,8 +25,7 @@ import { verifyDeployed, } from '../src'; import fs from 'fs'; -import { JsonRpcProvider, JsonRpcSigner, Network } from 'ethers'; -// import { Match } from '@ethereum-sourcify/lib-sourcify'; +import { JsonRpcSigner } from 'ethers'; const ganacheServer = Ganache.server({ wallet: { totalAccounts: 1 }, @@ -35,7 +35,7 @@ const GANACHE_PORT = 8545; const UNUSED_ADDRESS = '0x1F98431c8aD98523631AE4a59f267346ea31F984'; // checksum valid -const sourcifyChainGanache: SourcifyChain = { +const ganacheChain = { name: 'ganache', shortName: 'ganache', chainId: 0, @@ -49,23 +49,14 @@ const sourcifyChainGanache: SourcifyChain = { monitored: false, supported: true, }; +const sourcifyChainGanache: SourcifyChain = new SourcifyChain(ganacheChain); -let localProvider: JsonRpcProvider; let signer: JsonRpcSigner; describe('lib-sourcify tests', () => { before(async () => { await ganacheServer.listen(GANACHE_PORT); - const ethersNetwork = new Network( - sourcifyChainGanache.rpc[0], - sourcifyChainGanache.chainId - ); - localProvider = new JsonRpcProvider( - `http://localhost:${GANACHE_PORT}`, - ethersNetwork, - { staticNetwork: ethersNetwork } - ); - signer = await localProvider.getSigner(); + signer = await sourcifyChainGanache.providers[0].getSigner(); }); describe('Verification tests', () => { @@ -438,6 +429,50 @@ describe('lib-sourcify tests', () => { }); describe('Unit tests', function () { + describe('SourcifyChain', () => { + it("Should fail to instantiate with empty rpc's", function () { + const emptyRpc = { ...ganacheChain, rpc: [] }; + try { + new SourcifyChain(emptyRpc); + throw new Error('Should have failed'); + } catch (err) { + if (err instanceof Error) { + expect(err.message).to.equal( + 'No RPC provider was given for this chain with id ' + + emptyRpc.chainId + + ' and name ' + + emptyRpc.name + ); + } else { + throw err; + } + } + }); + it('Should getBlock', async function () { + const block = await sourcifyChainGanache.getBlock(0); + expect(block?.number).equals(0); + }); + it('Should getBlockNumber', async function () { + const blockNumber = await sourcifyChainGanache.getBlockNumber(); + expect(blockNumber > 0); + }); + it('Should fail to get non-existing transaction', async function () { + try { + await sourcifyChainGanache.getTx( + '0x79ab5d59fcb70ca3f290aa39ed3f156a5c4b3897176aebd455cd20b6a30b107a' + ); + throw new Error('Should have failed'); + } catch (err) { + if (err instanceof Error) { + expect(err.message).to.equal( + 'None of the RPCs responded fetching tx 0x79ab5d59fcb70ca3f290aa39ed3f156a5c4b3897176aebd455cd20b6a30b107a on chain 0' + ); + } else { + throw err; + } + } + }); + }); it('Should calculateCreate2Address', async function () { expect( calculateCreate2Address( diff --git a/src/common/SourcifyEventManager/SourcifyEventManager.ts b/src/common/SourcifyEventManager/SourcifyEventManager.ts index 386aaa158..cc3ec2e72 100644 --- a/src/common/SourcifyEventManager/SourcifyEventManager.ts +++ b/src/common/SourcifyEventManager/SourcifyEventManager.ts @@ -7,7 +7,6 @@ interface Events extends GenericEvents { "Monitor.Error.CantStart": (e: { chainId: string; message: string }) => void; "Monitor.Started": (obj: { chainId: string; - providerURL: string; lastBlockNumber: number; startBlock: number; }) => void; diff --git a/src/monitor/monitor.ts b/src/monitor/monitor.ts index 3c93a01d6..c563862f0 100755 --- a/src/monitor/monitor.ts +++ b/src/monitor/monitor.ts @@ -1,11 +1,5 @@ import { SourceAddress } from "./util"; -import { - JsonRpcProvider, - Provider, - TransactionResponse, - WebSocketProvider, - getCreateAddress, -} from "ethers"; +import { TransactionResponse, getCreateAddress } from "ethers"; import SourceFetcher from "./source-fetcher"; import assert from "assert"; import { EventEmitter } from "stream"; @@ -29,7 +23,6 @@ const BLOCK_PAUSE_UPPER_LIMIT = parseInt(process.env.BLOCK_PAUSE_UPPER_LIMIT || "") || 30 * 1000; // default: 30 seconds const BLOCK_PAUSE_LOWER_LIMIT = parseInt(process.env.BLOCK_PAUSE_LOWER_LIMIT || "") || 0.5 * 1000; // default: 0.5 seconds -const PROVIDER_TIMEOUT = parseInt(process.env.PROVIDER_TIMEOUT || "") || 3000; function createsContract(tx: TransactionResponse): boolean { return !tx.to; @@ -40,7 +33,6 @@ function createsContract(tx: TransactionResponse): boolean { */ class ChainMonitor extends EventEmitter { private sourcifyChain: SourcifyChain; - private provider: Provider | undefined; private sourceFetcher: SourceFetcher; private verificationService: IVerificationService; private repositoryService: IRepositoryService; @@ -76,48 +68,18 @@ class ChainMonitor extends EventEmitter { const rawStartBlock = process.env[`MONITOR_START_${this.sourcifyChain.chainId}`]; - // iterate over RPCs to find a working one; log the search result - let found = false; - for (const providerURL of this.sourcifyChain.rpc) { - // const opts = { timeout: PROVIDER_TIMEOUT }; - // TODO: ethers provider does not support timeout https://github.com/ethers-io/ethers.js/issues/4122 - const provider = providerURL.startsWith("http") - ? // prettier vs. eslint indentation conflict here - /* eslint-disable indent*/ - new JsonRpcProvider(providerURL, { - name: this.sourcifyChain.name, - chainId: this.sourcifyChain.chainId, - }) - : new WebSocketProvider(providerURL, { - name: this.sourcifyChain.name, - chainId: this.sourcifyChain.chainId, - }); - /* eslint-enable indent*/ - try { - const lastBlockNumber = await provider.getBlockNumber(); - found = true; - - this.provider = provider; - - const startBlock = - rawStartBlock !== undefined - ? parseInt(rawStartBlock) - : lastBlockNumber; - - SourcifyEventManager.trigger("Monitor.Started", { - chainId: this.sourcifyChain.chainId.toString(), - providerURL, - lastBlockNumber, - startBlock, - }); - this.processBlock(startBlock); - break; - } catch (err) { - logger.info(`${providerURL} did not work: \n ${err}`); - } - } + try { + const lastBlockNumber = await this.sourcifyChain.getBlockNumber(); + const startBlock = + rawStartBlock !== undefined ? parseInt(rawStartBlock) : lastBlockNumber; - if (!found) { + SourcifyEventManager.trigger("Monitor.Started", { + chainId: this.sourcifyChain.chainId.toString(), + lastBlockNumber, + startBlock, + }); + this.processBlock(startBlock); + } catch (err) { SourcifyEventManager.trigger("Monitor.Error.CantStart", { chainId: this.sourcifyChain.chainId.toString(), message: "Couldn't find a working RPC node.", @@ -137,12 +99,7 @@ class ChainMonitor extends EventEmitter { }; private processBlock = (blockNumber: number) => { - if (!this.provider) - throw new Error( - `Can't process block ${blockNumber}. Provider not initialized` - ); - - this.provider + this.sourcifyChain .getBlock(blockNumber, true) .then((block) => { if (!block) { @@ -224,11 +181,9 @@ class ChainMonitor extends EventEmitter { if (retriesLeft-- <= 0) { return; } - if (!this.provider) - throw new Error(`Can't process bytecode. Provider not initialized`); - this.provider - .getCode(address) + this.sourcifyChain + .getBytecode(address) .then((bytecode) => { if (bytecode === "0x") { this.mySetTimeout( diff --git a/src/server/controllers/verification/session-state/session-state.handlers.ts b/src/server/controllers/verification/session-state/session-state.handlers.ts index fde4875b4..da503112c 100644 --- a/src/server/controllers/verification/session-state/session-state.handlers.ts +++ b/src/server/controllers/verification/session-state/session-state.handlers.ts @@ -14,7 +14,6 @@ import { import { PathBuffer, PathContent, - getBytecode, getIpfsGateway, isEmpty, performFetch, @@ -78,7 +77,7 @@ export async function addInputContractEndpoint(req: Request, res: Response) { const sourcifyChain = services.verification.supportedChainsMap[chainId]; - const bytecode = await getBytecode(sourcifyChain, address); + const bytecode = await sourcifyChain.getBytecode(address); const { ipfs: metadataIpfsCid } = bytecodeDecode(bytecode); diff --git a/src/server/server.ts b/src/server/server.ts index 20a0497ff..4dfd2dcba 100644 --- a/src/server/server.ts +++ b/src/server/server.ts @@ -197,21 +197,29 @@ export class Server { res.status(200).send("Alive and kicking!") ); this.app.get("/chains", (_req, res) => { - const sourcifyChains = sourcifyChainsArray.map(({ rpc, ...rest }) => { - // Don't show Alchemy & Infura IDs - rpc = rpc.map((url) => { - if (url.includes("alchemy")) - return url.replace(/\/[^/]*$/, "/{ALCHEMY_ID}"); - else if (url.includes("infura")) - return url.replace(/\/[^/]*$/, "/{INFURA_ID}"); - else return url; - }); - return { - ...rest, - rpc, - etherscanAPI: etherscanAPIs[rest.chainId]?.apiURL, - }; - }); + const sourcifyChains = sourcifyChainsArray.map( + ({ rpc, providers, ...rest }) => { + // Don't publish providers + // Don't show Alchemy & Infura IDs + rpc = rpc.map((url) => { + if (typeof url === "string") { + if (url.includes("alchemy")) + return url.replace(/\/[^/]*$/, "/{ALCHEMY_ID}"); + else if (url.includes("infura")) + return url.replace(/\/[^/]*$/, "/{INFURA_ID}"); + else return url; + } else { + // FetchRequest + return url.url; + } + }); + return { + ...rest, + rpc, + etherscanAPI: etherscanAPIs[rest.chainId]?.apiURL, + }; + } + ); res.status(200).json(sourcifyChains); }); diff --git a/src/sourcify-chains.ts b/src/sourcify-chains.ts index 05bfa6e7c..576e8a64d 100644 --- a/src/sourcify-chains.ts +++ b/src/sourcify-chains.ts @@ -11,6 +11,7 @@ import { import { etherscanAPIs } from "./config"; import { ValidationError } from "./common/errors"; import { logger } from "./common/loggerLoki"; +import { FetchRequest } from "ethers"; const allChains = chainsRaw as Chain[]; @@ -34,7 +35,7 @@ const AVALANCHE_SUBNET_SUFFIX = type ChainName = "eth" | "polygon" | "arb" | "opt"; const LOCAL_CHAINS: SourcifyChain[] = [ - { + new SourcifyChain({ name: "Ganache Localhost", shortName: "Ganache", chainId: 1337, @@ -46,8 +47,8 @@ const LOCAL_CHAINS: SourcifyChain[] = [ rpc: [`http://localhost:8545`], supported: true, monitored: true, - }, - { + }), + new SourcifyChain({ name: "Hardhat Network Localhost", shortName: "Hardhat Network", chainId: 31337, @@ -59,7 +60,7 @@ const LOCAL_CHAINS: SourcifyChain[] = [ rpc: [`http://localhost:8545`], supported: true, monitored: true, - }, + }), ]; interface SourcifyChainsExtensionsObject { @@ -78,12 +79,22 @@ function buildAlchemyAndCustomRpcURLs( chainName: ChainName, useOwn = false ) { - const rpcURLs: string[] = []; + const rpcURLs: SourcifyChain["rpc"] = []; if (useOwn) { const url = process.env[`NODE_URL_${chainSubName.toUpperCase()}`]; if (url) { - rpcURLs.push(url); + const ethersFetchReq = new FetchRequest(url); + ethersFetchReq.setHeader("Content-Type", "application/json"); + ethersFetchReq.setHeader( + "CF-Access-Client-Id", + process.env.CF_ACCESS_CLIENT_ID || "" + ); + ethersFetchReq.setHeader( + "CF-Access-Client-Secret", + process.env.CF_ACCESS_CLIENT_SECRET || "" + ); + rpcURLs.push(ethersFetchReq); } else { SourcifyEventManager.trigger("Core.Error", { message: `Environment variable NODE_URL_${chainSubName.toUpperCase()} not set!`, @@ -950,7 +961,11 @@ for (const i in allChains) { if (chainId in sourcifyChainsExtensions) { const sourcifyExtension = sourcifyChainsExtensions[chainId]; - const sourcifyChain = { ...chain, ...sourcifyExtension } as SourcifyChain; + // sourcifyExtension is spread later to overwrite chain values, rpc specifically + const sourcifyChain = new SourcifyChain({ + ...chain, + ...sourcifyExtension, + }); sourcifyChainsMap[chainId] = sourcifyChain; } }