UNPKG

ethr-did-resolver

Version:
132 lines (117 loc) 5.26 kB
import { Contract, ContractFactory, JsonRpcProvider, Provider } from 'ethers' import { DEFAULT_REGISTRY_ADDRESS } from './helpers.js' import { deployments, EthrDidRegistryDeployment } from './config/deployments.js' import { EthereumDIDRegistry } from './config/EthereumDIDRegistry.js' const infuraNames: Record<string, string> = { polygon: 'matic', 'polygon:test': 'maticmum', aurora: 'aurora-mainnet', } const knownInfuraNames = ['mainnet', 'aurora', 'linea:goerli', 'sepolia'] /** * A configuration entry for an ethereum network * It should contain at least one of `name` or `chainId` AND one of `provider`, `web3`, or `rpcUrl` * * @example ```js * { name: 'development', registry: '0x9af37603e98e0dc2b855be647c39abe984fc2445', rpcUrl: 'http://127.0.0.1:8545/' } * { name: 'sepolia', chainId: 11155111, provider: new InfuraProvider('sepolia') } * { name: 'goerli', provider: new AlchemyProvider('goerli') } * { name: 'rsk:testnet', chainId: '0x1f', rpcUrl: 'https://public-node.testnet.rsk.co' } * ``` */ export interface ProviderConfiguration extends Omit<EthrDidRegistryDeployment, 'chainId'> { provider?: Provider | null chainId?: string | number | bigint // eslint-disable-next-line @typescript-eslint/no-explicit-any web3?: any } export interface MultiProviderConfiguration extends ProviderConfiguration { networks?: ProviderConfiguration[] } export interface InfuraConfiguration { infuraProjectId: string } export type ConfigurationOptions = MultiProviderConfiguration | InfuraConfiguration export type ConfiguredNetworks = Record<string, Contract> function configureNetworksWithInfura(projectId?: string): ConfiguredNetworks { if (!projectId) { return {} } const networks = knownInfuraNames .map((n) => { const existingDeployment = deployments.find((d) => d.name === n) if (existingDeployment && existingDeployment.name) { const infuraName = infuraNames[existingDeployment.name] || existingDeployment.name const rpcUrl = `https://${infuraName}.infura.io/v3/${projectId}` return { ...existingDeployment, rpcUrl } } }) .filter((conf) => !!conf) as ProviderConfiguration[] return configureNetworks({ networks }) } export function getContractForNetwork(conf: ProviderConfiguration): Contract { let provider: Provider = conf.provider || conf.web3?.currentProvider if (!provider) { if (conf.rpcUrl) { const chainIdRaw = conf.chainId ? conf.chainId : deployments.find((d) => d.name === conf.name)?.chainId const chainId = chainIdRaw ? BigInt(chainIdRaw) : chainIdRaw provider = new JsonRpcProvider(conf.rpcUrl, chainId || 'any') } else { throw new Error(`invalid_config: No web3 provider could be determined for network ${conf.name || conf.chainId}`) } } const contract = ContractFactory.fromSolidity(EthereumDIDRegistry) .attach(conf.registry || DEFAULT_REGISTRY_ADDRESS) .connect(provider) return contract as Contract } function configureNetwork(net: ProviderConfiguration): ConfiguredNetworks { const networks: ConfiguredNetworks = {} const chainId = net.chainId || deployments.find((d) => net.name && (d.name === net.name || d.description === net.name))?.chainId if (chainId) { if (net.name) { networks[net.name] = getContractForNetwork(net) } const id = typeof chainId === 'bigint' || typeof chainId === 'number' ? `0x${chainId.toString(16)}` : chainId networks[id] = getContractForNetwork(net) } else if (net.provider || net.web3 || net.rpcUrl) { networks[net.name || ''] = getContractForNetwork(net) } return networks } function configureNetworks(conf: MultiProviderConfiguration): ConfiguredNetworks { return { ...configureNetwork(conf), ...conf.networks?.reduce<ConfiguredNetworks>((networks, net) => { return { ...networks, ...configureNetwork(net) } }, {}), } } /** * Generates a configuration that maps ethereum network names and chainIDs to the respective ERC1056 contracts deployed * on them. * @returns a record of ERC1056 `Contract` instances * @param conf - configuration options for the resolver. An array of network details. * Each network entry should contain at least one of `name` or `chainId` AND one of `provider`, `web3`, or `rpcUrl` * For convenience, you can also specify an `infuraProjectId` which will create a mapping for all the networks * supported by https://infura.io. * @example ```js * [ * { name: 'development', registry: '0x9af37603e98e0dc2b855be647c39abe984fc2445', rpcUrl: 'http://127.0.0.1:8545/' }, * { name: 'goerli', chainId: 5, provider: new InfuraProvider('goerli') }, * { name: 'sepolia', provider: new AlchemyProvider('sepolia') }, * { name: 'rsk:testnet', chainId: '0x1f', rpcUrl: 'https://public-node.testnet.rsk.co' }, * ] * ``` */ export function configureResolverWithNetworks(conf: ConfigurationOptions = {}): ConfiguredNetworks { const networks = { ...configureNetworksWithInfura((<InfuraConfiguration>conf).infuraProjectId), ...configureNetworks(<MultiProviderConfiguration>conf), } if (Object.keys(networks).length === 0) { throw new Error('invalid_config: Please make sure to have at least one network') } return networks }