Skip to content

Commit 71bf6a9

Browse files
authored
Add chain config for Linea (#1470)
* Can't as const here * Add linea chain to appconfig * Refactor wagmiClient to use resolver object, add linea env variables
1 parent 66133b5 commit 71bf6a9

File tree

6 files changed

+128
-63
lines changed

6 files changed

+128
-63
lines changed

apps/hyperdrive-trading/.env.sample

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,9 @@ VITE_GNOSIS_NODE_RPC_URL=
2020

2121
# Mainnet
2222
VITE_MAINNET_RPC_URL=
23+
24+
# Linea
25+
VITE_LINEA_RPC_URL=
2326
##################################################
2427

2528
# Error tracking via Rollbar
Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,11 @@
1-
import { gnosis, mainnet } from "viem/chains";
1+
import { gnosis, linea, mainnet } from "viem/chains";
22

33
/**
44
* Checks if the given chain ID corresponds to a mainnet chain.
55
* @param chainId - The chain ID to check.
66
* @returns True if the chain ID corresponds to a mainnet chain, false otherwise.
77
*/
88
export function isMainnetChain(chainId: number): boolean {
9-
const mainnetChainIds: number[] = [mainnet.id, gnosis.id];
9+
const mainnetChainIds: number[] = [mainnet.id, gnosis.id, linea.id];
1010
return mainnetChainIds.includes(chainId);
1111
}

apps/hyperdrive-trading/src/network/wagmiClient.ts

Lines changed: 92 additions & 47 deletions
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@ import { gnosisFork } from "src/chains/gnosisFork";
1111
import { CreateWalletFn } from "src/wallets/CreateWalletFn";
1212
import { capsuleWallet } from "src/wallets/capsule";
1313
import { Chain, http, Transport } from "viem";
14-
import { foundry, gnosis, mainnet, sepolia } from "wagmi/chains";
14+
import { foundry, gnosis, linea, mainnet, sepolia } from "wagmi/chains";
1515

1616
const {
1717
VITE_LOCALHOST_NODE_RPC_URL,
@@ -20,94 +20,139 @@ const {
2020
VITE_WALLET_CONNECT_PROJECT_ID,
2121
VITE_SEPOLIA_RPC_URL,
2222
VITE_MAINNET_RPC_URL,
23+
VITE_LINEA_RPC_URL,
2324
VITE_GNOSIS_FORK_NODE_RPC_URL,
2425
VITE_GNOSIS_FORK_CHAIN_ID,
2526
VITE_GNOSIS_NODE_RPC_URL,
2627
} = import.meta.env;
2728

28-
export const chains: Chain[] = [];
29-
const transports: Record<string, Transport> = {};
29+
interface WagmiClientConfig {
30+
rpcUrl: string;
31+
chain: Chain;
32+
wallets?: (CreateWalletFn | undefined)[];
33+
/**
34+
* used for fork chains only
35+
*/
36+
chainId?: number;
37+
}
38+
39+
/* To add a new chain to this file, follow these steps:
40+
41+
* 1. Import the new chain from "wagmi/chains" or define it in "src/chains" if it's a custom chain.
42+
* 2. Add a new property to the `chainConfigs` object with the chain's name as the key.
43+
* 3. Define the chain's configuration, including the `rpcUrl` and chain properties.
44+
* 4. Optionally, add a `wallets` property to specify the wallets that should be supported on this chain.
45+
* 5. If the chain is a fork chain, add a `chainId` property to specify the chain's ID.
3046
47+
* Example:
48+
* myChain: {
49+
* rpcUrl: VITE_MY_CHAIN_RPC_URL,
50+
* chain: myChain,
51+
* wallets: [capsuleWallet],
52+
* chainId: 123, // only required for fork chains
53+
* }
54+
*/
55+
56+
const chainConfigs: Record<string, WagmiClientConfig> = {
57+
mainnet: {
58+
rpcUrl: VITE_MAINNET_RPC_URL,
59+
chain: mainnet,
60+
wallets: [capsuleWallet],
61+
},
62+
gnosis: {
63+
rpcUrl: VITE_GNOSIS_NODE_RPC_URL,
64+
chain: gnosis,
65+
},
66+
linea: {
67+
rpcUrl: VITE_LINEA_RPC_URL,
68+
chain: linea,
69+
},
70+
foundry: {
71+
rpcUrl: VITE_LOCALHOST_NODE_RPC_URL,
72+
chain: foundry,
73+
},
74+
gnosisFork: {
75+
rpcUrl: VITE_GNOSIS_FORK_NODE_RPC_URL,
76+
chain: gnosisFork,
77+
chainId: VITE_GNOSIS_FORK_CHAIN_ID,
78+
},
79+
cloudChain: {
80+
rpcUrl: VITE_CUSTOM_CHAIN_NODE_RPC_URL,
81+
chain: cloudChain,
82+
chainId: VITE_CUSTOM_CHAIN_CHAIN_ID,
83+
},
84+
sepolia: {
85+
rpcUrl: VITE_SEPOLIA_RPC_URL,
86+
chain: sepolia,
87+
wallets: [capsuleWallet],
88+
},
89+
};
90+
91+
const chains: Chain[] = [];
92+
const transports: Record<string, Transport> = {};
93+
const customWallets: CreateWalletFn[] = [];
3194
const recommendedWallets = [
3295
injectedWallet,
3396
safeWallet,
3497
rainbowWallet,
3598
metaMaskWallet,
3699
];
37-
const customWallets: CreateWalletFn[] = [];
38100

39101
// WalletConnect
40102
if (VITE_WALLET_CONNECT_PROJECT_ID) {
41103
recommendedWallets.push(walletConnectWallet);
42104
}
43105

44-
// mainnet
106+
// Handle mainnet first to ensure it's always the first chain. This ensures
107+
// wagmi falls back to mainnet if the user isn't connected.
45108
if (VITE_MAINNET_RPC_URL) {
46109
chains.push(mainnet);
47110
transports[mainnet.id] = http(VITE_MAINNET_RPC_URL);
48111

49-
// TODO: push this into the custom wallets for local and cloudchain once
50-
// capsule support is verified
51-
if (capsuleWallet && !customWallets.includes(capsuleWallet)) {
112+
if (
113+
capsuleWallet &&
114+
chainConfigs["mainnet"].wallets?.includes(capsuleWallet)
115+
) {
52116
customWallets.push(capsuleWallet);
53117
}
54118
}
55119

56-
// Local docker anvil node
57-
if (VITE_LOCALHOST_NODE_RPC_URL && VITE_LOCALHOST_NODE_RPC_URL) {
58-
chains.push(foundry);
59-
transports[foundry.id] = http(VITE_LOCALHOST_NODE_RPC_URL);
60-
}
61-
62-
// Gnosis Fork
63-
if (VITE_GNOSIS_FORK_NODE_RPC_URL && VITE_GNOSIS_FORK_CHAIN_ID) {
64-
chains.push(gnosisFork);
65-
transports[gnosisFork.id] = http(VITE_GNOSIS_FORK_NODE_RPC_URL);
66-
}
120+
// Build the rest of the chains dynamically, excluding mainnet
121+
Object.values(chainConfigs).forEach(({ rpcUrl, chain, wallets }) => {
122+
// Exclude mainnet here
123+
if (chain.id === mainnet.id) {
124+
return;
125+
}
67126

68-
// CloudChain
69-
if (VITE_CUSTOM_CHAIN_NODE_RPC_URL && VITE_CUSTOM_CHAIN_CHAIN_ID) {
70-
chains.push(cloudChain);
71-
transports[cloudChain.id] = http(VITE_CUSTOM_CHAIN_NODE_RPC_URL);
72-
}
127+
chains.push(chain);
128+
transports[chain.id] = http(rpcUrl);
73129

74-
// Sepolia
75-
if (VITE_SEPOLIA_RPC_URL) {
76-
chains.push(sepolia);
77-
transports[sepolia.id] = http(VITE_SEPOLIA_RPC_URL);
78-
// TODO: push this into the custom wallets for local and cloudchain once
79-
// capsule support is verified
80-
if (capsuleWallet) {
81-
customWallets.push(capsuleWallet);
130+
// Add custom wallets if they exist
131+
if (wallets) {
132+
wallets.forEach((wallet) => {
133+
if (wallet && !customWallets.includes(wallet)) {
134+
customWallets.push(wallet);
135+
}
136+
});
82137
}
83-
}
84-
85-
// Gnosis
86-
if (VITE_GNOSIS_NODE_RPC_URL) {
87-
chains.push(gnosis);
88-
transports[gnosis.id] = http(VITE_GNOSIS_NODE_RPC_URL);
89-
}
138+
});
90139

140+
// Prepare wallet list
91141
const wallets: WalletList = [
92-
{
93-
groupName: "Other",
94-
wallets: recommendedWallets,
95-
},
142+
{ groupName: "Other", wallets: recommendedWallets },
96143
];
97-
// If Capsule is not configured in your .env, then don't include it in the
98-
// custom wallets
144+
99145
if (customWallets.length) {
100146
wallets.push({
101147
groupName: "Log In or Sign Up with Capsule",
102148
wallets: customWallets,
103149
});
104150
}
151+
105152
export const wagmiConfig = getDefaultConfig({
106153
appName: "Hyperdrive",
107154
projectId: VITE_WALLET_CONNECT_PROJECT_ID || "0",
108155
transports,
109-
// Viem's type for `chains` requires at least one item in the array, but since
110-
// we build the list up programmatically we must cast.
111156
chains: chains as [Chain, ...restChains: Chain[]],
112157
wallets,
113158
});

packages/hyperdrive-appconfig/.env.example

Lines changed: 2 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -9,4 +9,5 @@ GNOSIS_FORK_NODE_RPC_URL=
99

1010
SEPOLIA_NODE_RPC_URL=
1111
MAINNET_NODE_RPC_URL=
12-
GNOSIS_NODE_RPC_URL=
12+
GNOSIS_NODE_RPC_URL=
13+
LINEA_NODE_RPC_URL=

packages/hyperdrive-appconfig/src/chains/chains.ts

Lines changed: 28 additions & 13 deletions
Original file line numberDiff line numberDiff line change
@@ -1,14 +1,16 @@
11
import { cloudChain } from "src/chains/cloudChain";
22
import { gnosisFork } from "src/chains/gnosisFork";
3-
import { gnosis, mainnet, sepolia } from "viem/chains";
3+
import { gnosis, linea, mainnet, sepolia } from "viem/chains";
44

55
// Define Chain ID as a union of available chain keys
66
export type ChainId =
77
| (typeof gnosis)["id"]
88
| (typeof mainnet)["id"]
99
| (typeof cloudChain)["id"]
1010
| (typeof gnosisFork)["id"]
11-
| (typeof sepolia)["id"];
11+
| (typeof sepolia)["id"]
12+
| (typeof linea)["id"];
13+
1214
export interface ChainConfig {
1315
id: ChainId;
1416
name: string;
@@ -24,6 +26,7 @@ export interface ChainConfig {
2426
}
2527

2628
export const chains: Record<ChainId, ChainConfig> = {
29+
/* Mainnet chains */
2730
[mainnet.id]: {
2831
id: mainnet.id,
2932
name: "Mainnet",
@@ -33,6 +36,28 @@ export const chains: Record<ChainId, ChainConfig> = {
3336
iconUrl:
3437
"",
3538
},
39+
[gnosis.id]: {
40+
id: gnosis.id,
41+
name: "Gnosis",
42+
blockExplorerName: "Gnosisscan",
43+
blockExplorerUrl: "https://gnosisscan.io",
44+
earliestBlock: 35732205n,
45+
dailyAverageBlocks: 16605n,
46+
iconUrl:
47+
"",
48+
},
49+
[linea.id]: {
50+
id: linea.id,
51+
name: "Linea",
52+
blockExplorerName: "Lineascan",
53+
blockExplorerUrl: "https://lineascan.build/",
54+
earliestBlock: 9245278n, // block that the registry was deployed at
55+
dailyAverageBlocks: 43200n, // 2-second block times according to lineascan
56+
iconUrl:
57+
"",
58+
},
59+
60+
/* Testnets */
3661
[cloudChain.id]: {
3762
id: cloudChain.id,
3863
name: "Cloudchain",
@@ -51,16 +76,6 @@ export const chains: Record<ChainId, ChainConfig> = {
5176
iconUrl:
5277
"",
5378
},
54-
[gnosis.id]: {
55-
id: gnosis.id,
56-
name: "Gnosis",
57-
blockExplorerName: "Gnosisscan",
58-
blockExplorerUrl: "https://gnosisscan.io",
59-
earliestBlock: 35732205n,
60-
dailyAverageBlocks: 16605n,
61-
iconUrl:
62-
"",
63-
},
6479
[gnosisFork.id]: {
6580
id: gnosisFork.id,
6681
name: "Gnosis Fork",
@@ -70,4 +85,4 @@ export const chains: Record<ChainId, ChainConfig> = {
7085
iconUrl:
7186
"",
7287
},
73-
} as const;
88+
};

0 commit comments

Comments
 (0)