diff --git a/packages/apps/src/__tests__/cross-config.spec.ts b/packages/apps/src/__tests__/cross-config.spec.ts index 6246f8acb..0fee6c2c2 100644 --- a/packages/apps/src/__tests__/cross-config.spec.ts +++ b/packages/apps/src/__tests__/cross-config.spec.ts @@ -1,7 +1,7 @@ -import { getChainsConfig } from "../utils/chain"; +import { getChainsConfig, getChainConfig } from "../utils/chain"; import type { ChainConfig } from "../types/chain"; -describe.each(getChainsConfig() as ChainConfig[])( +describe.each(getChainsConfig("all") as ChainConfig[])( "Should configure price for HelixLpBridge to cross native token", ({ network, tokens }) => { if (tokens.length) { @@ -17,3 +17,17 @@ describe.each(getChainsConfig() as ChainConfig[])( } }, ); + +describe.each(getChainsConfig("all") as ChainConfig[])("Should configure target token", ({ network, tokens }) => { + if (tokens.length) { + describe.each(tokens)(`Cross $symbol from ${network}`, (token) => { + if (token.cross.length) { + it.each(token.cross)(`to $target.network ($target.symbol)`, (cross) => { + const targetChain = getChainConfig(cross.target.network) as ChainConfig | undefined; + const targetToken = targetChain?.tokens.find((t) => t.symbol === cross.target.symbol); + expect(targetToken).not.toBeUndefined(); + }); + } + }); + } +}); diff --git a/packages/apps/src/__tests__/native-token-config.spec.ts b/packages/apps/src/__tests__/native-token-config.spec.ts new file mode 100644 index 000000000..cbd216ac9 --- /dev/null +++ b/packages/apps/src/__tests__/native-token-config.spec.ts @@ -0,0 +1,10 @@ +import { getChainsConfig } from "../utils/chain"; +import type { ChainConfig } from "../types/chain"; + +describe.each(getChainsConfig("all") as ChainConfig[])("Should configure native token", ({ network, tokens }) => { + it(`${network}`, () => { + if (tokens.length) { + expect(tokens.some((t) => t.type === "native")).not.toBeFalsy(); + } + }); +}); diff --git a/packages/apps/src/abi/lnaccess-controller.ts b/packages/apps/src/abi/lnaccess-controller.ts new file mode 100644 index 000000000..542430417 --- /dev/null +++ b/packages/apps/src/abi/lnaccess-controller.ts @@ -0,0 +1,182 @@ +const abi = [ + { + inputs: [ + { internalType: "address", name: "_dao", type: "address" }, + { internalType: "address", name: "_endpoint", type: "address" }, + ], + stateMutability: "nonpayable", + type: "constructor", + }, + { + anonymous: false, + inputs: [ + { indexed: false, internalType: "uint16", name: "lzRemoteChainId", type: "uint16" }, + { indexed: false, internalType: "bytes", name: "srcAddress", type: "bytes" }, + { indexed: false, internalType: "bool", name: "successed", type: "bool" }, + ], + name: "CallResult", + type: "event", + }, + { + anonymous: false, + inputs: [ + { indexed: false, internalType: "uint16", name: "lzRemoteChainId", type: "uint16" }, + { indexed: false, internalType: "bytes", name: "srcAddress", type: "bytes" }, + { indexed: false, internalType: "address", name: "remoteAppAddress", type: "address" }, + ], + name: "CallerUnMatched", + type: "event", + }, + { + inputs: [ + { internalType: "address", name: "appAddress", type: "address" }, + { internalType: "bool", name: "enable", type: "bool" }, + ], + name: "authoriseAppCaller", + outputs: [], + stateMutability: "nonpayable", + type: "function", + }, + { + inputs: [{ internalType: "address", name: "", type: "address" }], + name: "callerWhiteList", + outputs: [{ internalType: "bool", name: "", type: "bool" }], + stateMutability: "view", + type: "function", + }, + { + inputs: [], + name: "dao", + outputs: [{ internalType: "address", name: "", type: "address" }], + stateMutability: "view", + type: "function", + }, + { + inputs: [], + name: "endpoint", + outputs: [{ internalType: "contract ILayerZeroEndpoint", name: "", type: "address" }], + stateMutability: "view", + type: "function", + }, + { + inputs: [ + { internalType: "uint256", name: "_remoteChainId", type: "uint256" }, + { internalType: "bytes", name: "_message", type: "bytes" }, + ], + name: "fee", + outputs: [ + { internalType: "uint256", name: "nativeFee", type: "uint256" }, + { internalType: "uint256", name: "zroFee", type: "uint256" }, + ], + stateMutability: "view", + type: "function", + }, + { + inputs: [ + { internalType: "uint16", name: "_srcChainId", type: "uint16" }, + { internalType: "bytes", name: "_srcAddress", type: "bytes" }, + { internalType: "uint64", name: "", type: "uint64" }, + { internalType: "bytes", name: "_payload", type: "bytes" }, + ], + name: "lzReceive", + outputs: [], + stateMutability: "nonpayable", + type: "function", + }, + { + inputs: [], + name: "operator", + outputs: [{ internalType: "address", name: "", type: "address" }], + stateMutability: "view", + type: "function", + }, + { + inputs: [ + { internalType: "uint256", name: "_remoteChainId", type: "uint256" }, + { internalType: "address", name: "_remoteBridge", type: "address" }, + ], + name: "registerRemoteReceiver", + outputs: [], + stateMutability: "nonpayable", + type: "function", + }, + { + inputs: [ + { internalType: "uint256", name: "_remoteChainId", type: "uint256" }, + { internalType: "address", name: "_remoteBridge", type: "address" }, + ], + name: "registerRemoteSender", + outputs: [], + stateMutability: "nonpayable", + type: "function", + }, + { + inputs: [{ internalType: "bytes32", name: "", type: "bytes32" }], + name: "remoteAppReceivers", + outputs: [{ internalType: "address", name: "", type: "address" }], + stateMutability: "view", + type: "function", + }, + { + inputs: [{ internalType: "bytes32", name: "", type: "bytes32" }], + name: "remoteAppSenders", + outputs: [{ internalType: "address", name: "", type: "address" }], + stateMutability: "view", + type: "function", + }, + { + inputs: [{ internalType: "uint256", name: "", type: "uint256" }], + name: "remoteMessagers", + outputs: [ + { internalType: "uint16", name: "lzRemoteChainId", type: "uint16" }, + { internalType: "address", name: "messager", type: "address" }, + ], + stateMutability: "view", + type: "function", + }, + { + inputs: [ + { internalType: "uint256", name: "_remoteChainId", type: "uint256" }, + { internalType: "bytes", name: "_message", type: "bytes" }, + { internalType: "bytes", name: "_params", type: "bytes" }, + ], + name: "sendMessage", + outputs: [], + stateMutability: "payable", + type: "function", + }, + { + inputs: [{ internalType: "address", name: "_operator", type: "address" }], + name: "setOperator", + outputs: [], + stateMutability: "nonpayable", + type: "function", + }, + { + inputs: [ + { internalType: "uint256", name: "_appRemoteChainId", type: "uint256" }, + { internalType: "uint16", name: "_lzRemoteChainId", type: "uint16" }, + { internalType: "address", name: "_remoteMessager", type: "address" }, + ], + name: "setRemoteMessager", + outputs: [], + stateMutability: "nonpayable", + type: "function", + }, + { + inputs: [{ internalType: "address", name: "_dao", type: "address" }], + name: "transferOwnership", + outputs: [], + stateMutability: "nonpayable", + type: "function", + }, + { + inputs: [{ internalType: "uint16", name: "", type: "uint16" }], + name: "trustedRemotes", + outputs: [{ internalType: "bytes32", name: "", type: "bytes32" }], + stateMutability: "view", + type: "function", + }, +] as const; + +export default abi; diff --git a/packages/apps/src/bridges/lnbridge-default.ts b/packages/apps/src/bridges/lnbridge-default.ts index 12bcca403..253f9fd4b 100644 --- a/packages/apps/src/bridges/lnbridge-default.ts +++ b/packages/apps/src/bridges/lnbridge-default.ts @@ -1,4 +1,4 @@ -import { Address, TransactionReceipt } from "viem"; +import { Address, TransactionReceipt, bytesToHex } from "viem"; import { LnBridgeBase } from "./lnbridge-base"; import { ChainConfig, ChainID } from "@/types/chain"; import { Token } from "@/types/token"; @@ -165,4 +165,53 @@ export class LnBridgeDefault extends LnBridgeBase { return this.publicClient.waitForTransactionReceipt({ hash }); } } + + async getWithdrawFee() { + if (this.contract && this.sourceNativeToken && this.targetChain && this.sourcePublicClient) { + const bridgeAbi = (await import(`../abi/lnbridgev20-default`)).default; + const accessAbi = (await import(`../abi/lnaccess-controller`)).default; + const remoteChainId = BigInt(this.targetChain.id); + + const [sendService, _receiveService] = await this.sourcePublicClient.readContract({ + address: this.contract.sourceAddress, + abi: bridgeAbi, + functionName: "messagers", + args: [remoteChainId], + }); + const [nativeFee, _zroFee] = await this.sourcePublicClient.readContract({ + address: sendService, + abi: accessAbi, + functionName: "fee", + args: [remoteChainId, bytesToHex(Uint8Array.from([123]), { size: 500 })], + }); + + return { fee: nativeFee, token: this.sourceNativeToken }; + } + } + + async withdrawMargin(recipient: Address, amount: bigint, fee: bigint) { + await this.validateNetwork("source"); + + if ( + this.contract && + this.sourceToken && + this.targetToken && + this.targetChain && + this.publicClient && + this.walletClient + ) { + const abi = (await import(`../abi/lnbridgev20-default`)).default; + const remoteChainId = BigInt(this.targetChain.id); + + const hash = await this.walletClient.writeContract({ + address: this.contract.sourceAddress, + abi, + functionName: "requestWithdrawMargin", + args: [remoteChainId, this.sourceToken.address, this.targetToken.address, amount, recipient], + gas: this.getTxGasLimit(), + value: fee, + }); + return this.publicClient.waitForTransactionReceipt({ hash }); + } + } } diff --git a/packages/apps/src/components/relayer-manage-modal.tsx b/packages/apps/src/components/relayer-manage-modal.tsx index b87ef1b47..d36290635 100644 --- a/packages/apps/src/components/relayer-manage-modal.tsx +++ b/packages/apps/src/components/relayer-manage-modal.tsx @@ -11,6 +11,9 @@ import { formatBalance } from "@/utils/balance"; import { useRelayer } from "@/hooks/use-relayer"; import dynamic from "next/dynamic"; import { formatFeeRate, isValidFeeRate } from "@/utils/misc"; +import { TransactionReceipt } from "viem"; +import { Token } from "@/types/token"; +import { Subscription, from } from "rxjs"; type TabKey = "update" | "deposit" | "withdraw"; const Modal = dynamic(() => import("@/ui/modal"), { ssr: false }); @@ -35,6 +38,7 @@ export default function RelayerManageModal({ relayerInfo, isOpen, onClose, onSuc margin, baseFee, feeRate, + defaultBridge, setMargin, setBaseFee, setFeeRate, @@ -45,12 +49,22 @@ export default function RelayerManageModal({ relayerInfo, isOpen, onClose, onSuc setFeeAndRate, depositMargin, updateFeeAndMargin, + withdrawMargin, sourceApprove, targetApprove, } = useRelayer(); const [activeKey, setActiveKey] = useState["activeKey"]>("update"); const [height, setHeight] = useState(); const [busy, setBusy] = useState(false); + const [depositAmount, setDepositAmount] = useState<{ formatted: bigint; value: string }>({ + formatted: 0n, + value: "", + }); + const [withdrawAmount, setWithdrawAmount] = useState<{ formatted: bigint; value: string }>({ + formatted: 0n, + value: "", + }); + const [withdrawFee, setWithdrawFee] = useState<{ fee: bigint; token: Token }>(); const { chain } = useNetwork(); const { switchNetwork } = useSwitchNetwork(); @@ -97,6 +111,9 @@ export default function RelayerManageModal({ relayerInfo, isOpen, onClose, onSuc setSourceChain(_sourceChain); setTargetChain(_targetChain); setSourceToken(_sourceToken); + setWithdrawAmount({ formatted: 0n, value: "" }); + setDepositAmount({ formatted: 0n, value: "" }); + setActiveKey("update"); }, [ relayerInfo, setBaseFee, @@ -108,19 +125,35 @@ export default function RelayerManageModal({ relayerInfo, isOpen, onClose, onSuc setBridgeCategory, ]); + useEffect(() => { + let sub$$: Subscription | undefined; + if (defaultBridge && relayerInfo?.messageChannel === "layerzero") { + sub$$ = from(defaultBridge.getWithdrawFee()).subscribe({ + next: setWithdrawFee, + error: (err) => { + console.error(err); + setWithdrawFee(undefined); + }, + }); + } else { + setWithdrawFee(undefined); + } + return () => sub$$?.unsubscribe(); + }, [defaultBridge, relayerInfo]); + const okText = useMemo(() => { let text = "Confirm"; if (activeKey === "deposit") { if (bridgeCategory === "lnbridgev20-default") { if (chain?.id !== targetChain?.id) { text = "Switch Network"; - } else if (sourceToken?.type !== "native" && margin.formatted > (targetAllowance?.value || 0n)) { + } else if (sourceToken?.type !== "native" && depositAmount.formatted > (targetAllowance?.value || 0n)) { text = "Approve"; } } else if (bridgeCategory === "lnbridgev20-opposite") { if (chain?.id !== sourceChain?.id) { text = "Switch Network"; - } else if (sourceToken?.type !== "native" && margin.formatted > (sourceAllowance?.value || 0n)) { + } else if (sourceToken?.type !== "native" && depositAmount.formatted > (sourceAllowance?.value || 0n)) { text = "Approve"; } } @@ -130,7 +163,7 @@ export default function RelayerManageModal({ relayerInfo, isOpen, onClose, onSuc return text; }, [ chain, - margin, + depositAmount, activeKey, bridgeCategory, sourceChain, @@ -148,60 +181,56 @@ export default function RelayerManageModal({ relayerInfo, isOpen, onClose, onSuc isOpen={isOpen} onClose={onClose} onOk={async () => { + let receipt: TransactionReceipt | undefined; try { if (activeKey === "update") { if (chain?.id !== sourceChain?.id) { switchNetwork?.(sourceChain?.id); } else if (bridgeCategory === "lnbridgev20-default") { setBusy(true); - const receipt = await setFeeAndRate(baseFee.formatted, feeRate.formatted); - if (receipt?.status === "success") { - onSuccess(); - onClose(); - } + receipt = await setFeeAndRate(baseFee.formatted, feeRate.formatted); } else if (bridgeCategory === "lnbridgev20-opposite") { setBusy(true); - const receipt = await updateFeeAndMargin(margin.formatted, baseFee.formatted, feeRate.formatted); - if (receipt?.status === "success") { - onSuccess(); - onClose(); - } + receipt = await updateFeeAndMargin(margin.formatted, baseFee.formatted, feeRate.formatted); } } else if (activeKey === "deposit") { if (bridgeCategory === "lnbridgev20-default") { if (chain?.id !== targetChain?.id) { switchNetwork?.(targetChain?.id); - } else if (sourceToken?.type !== "native" && margin.formatted > (targetAllowance?.value || 0n)) { + } else if (sourceToken?.type !== "native" && depositAmount.formatted > (targetAllowance?.value || 0n)) { setBusy(true); - await targetApprove(margin.formatted); + await targetApprove(depositAmount.formatted); } else { setBusy(true); - const receipt = await depositMargin(margin.formatted); - if (receipt?.status === "success") { - onSuccess(); - onClose(); - } + receipt = await depositMargin(depositAmount.formatted); } } else if (bridgeCategory === "lnbridgev20-opposite") { if (chain?.id !== sourceChain?.id) { switchNetwork?.(sourceChain?.id); - } else if (sourceToken?.type !== "native" && margin.formatted > (sourceAllowance?.value || 0n)) { + } else if (sourceToken?.type !== "native" && depositAmount.formatted > (sourceAllowance?.value || 0n)) { setBusy(true); - await sourceApprove(margin.formatted); + await sourceApprove(depositAmount.formatted); } else { setBusy(true); - const receipt = await updateFeeAndMargin(margin.formatted, baseFee.formatted, feeRate.formatted); - if (receipt?.status === "success") { - onSuccess(); - onClose(); - } + receipt = await updateFeeAndMargin(depositAmount.formatted, baseFee.formatted, feeRate.formatted); } } + } else if (activeKey === "withdraw") { + if (chain?.id !== sourceChain?.id) { + switchNetwork?.(sourceChain?.id); + } else { + setBusy(true); + receipt = await withdrawMargin(withdrawAmount.formatted, withdrawFee?.fee ?? 0n); + } } } catch (err) { console.error(err); } finally { setBusy(false); + if (receipt?.status === "success") { + onSuccess(); + onClose(); + } } }} busy={busy} @@ -209,10 +238,21 @@ export default function RelayerManageModal({ relayerInfo, isOpen, onClose, onSuc disabledOk={ activeKey === "update" && !(baseFee.value && feeRate.value && isValidFeeRate(feeRate.formatted)) ? true - : activeKey === "deposit" && margin.formatted === 0n + : activeKey === "deposit" && depositAmount.formatted === 0n + ? true + : activeKey === "withdraw" && (withdrawAmount.formatted === 0n || !withdrawFee) ? true : false } + extra={ + activeKey === "withdraw" ? ( +
+ Powered by LayerZero & Helix +
+ ) : ( +
+ ) + } onCancel={onClose} > ), children: ( -
setHeight((prev) => node?.clientHeight || prev)}> +
@@ -262,7 +302,8 @@ export default function RelayerManageModal({ relayerInfo, isOpen, onClose, onSuc : undefined } suffix="symbol" - onChange={setMargin} + value={depositAmount} + onChange={setDepositAmount} /> ), @@ -287,11 +328,37 @@ export default function RelayerManageModal({ relayerInfo, isOpen, onClose, onSuc
), children: ( - - - +
setHeight((prev) => node?.clientHeight || prev)}> + + + + +
+ {withdrawFee ? ( + <> + {formatBalance(withdrawFee.fee, withdrawFee.token.decimals, { precision: 6 })} + {withdrawFee.token.symbol} + + ) : ( + + * Failed to get fee, withdraw is temporarily unavailable + + )} +
+
+
), - disabled: true, + disabled: relayerInfo?.messageChannel !== "layerzero", }, ]} activeKey={activeKey} diff --git a/packages/apps/src/config/chains/base-goerli.ts b/packages/apps/src/config/chains/base-goerli.ts index e151e1f46..7827484ff 100644 --- a/packages/apps/src/config/chains/base-goerli.ts +++ b/packages/apps/src/config/chains/base-goerli.ts @@ -28,6 +28,15 @@ export const baseGoerliChain: ChainConfig = { }, testnet: true, tokens: [ + { + decimals: 18, + symbol: "ETH", + name: "ETH", + type: "native", + address: "0x0000000000000000000000000000000000000000", + logo: "eth.svg", + cross: [], + }, { decimals: 18, symbol: "USDT", diff --git a/packages/apps/src/config/chains/mantle.ts b/packages/apps/src/config/chains/mantle.ts index 2df4b154f..6490795ea 100644 --- a/packages/apps/src/config/chains/mantle.ts +++ b/packages/apps/src/config/chains/mantle.ts @@ -27,6 +27,15 @@ export const mantleChain: ChainConfig = { }, }, tokens: [ + { + decimals: 18, + symbol: "MNT", + name: "MNT", + type: "native", + address: "0x0000000000000000000000000000000000000000", + logo: "mnt.svg", + cross: [], + }, { decimals: 6, symbol: "USDT", diff --git a/packages/apps/src/config/chains/pangolin.ts b/packages/apps/src/config/chains/pangolin.ts index f8f18d7d8..013d99cce 100644 --- a/packages/apps/src/config/chains/pangolin.ts +++ b/packages/apps/src/config/chains/pangolin.ts @@ -27,15 +27,5 @@ export const pangolinChain: ChainConfig = { }, }, testnet: true, - tokens: [ - { - decimals: 18, - symbol: "PRING", - name: "PRING", - type: "erc20", - address: "0x3F3eDBda6124462a09E071c5D90e072E0d5d4ed4", - logo: "ring.svg", - cross: [], - }, - ], + tokens: [], }; diff --git a/packages/apps/src/config/chains/polygon.ts b/packages/apps/src/config/chains/polygon.ts index beb92a162..28d4b04e9 100644 --- a/packages/apps/src/config/chains/polygon.ts +++ b/packages/apps/src/config/chains/polygon.ts @@ -27,6 +27,15 @@ export const polygonChain: ChainConfig = { }, }, tokens: [ + { + decimals: 18, + symbol: "MATIC", + name: "MATIC", + type: "native", + address: "0x0000000000000000000000000000000000000000", + logo: "matic.svg", + cross: [], + }, { decimals: 18, symbol: "RING", diff --git a/packages/apps/src/config/chains/scroll.ts b/packages/apps/src/config/chains/scroll.ts index 45675654d..167860040 100644 --- a/packages/apps/src/config/chains/scroll.ts +++ b/packages/apps/src/config/chains/scroll.ts @@ -27,6 +27,15 @@ export const scrollChain: ChainConfig = { }, }, tokens: [ + { + decimals: 18, + symbol: "ETH", + name: "ETH", + type: "native", + address: "0x0000000000000000000000000000000000000000", + logo: "eth.svg", + cross: [], + }, { decimals: 6, symbol: "USDT", diff --git a/packages/apps/src/config/chains/zksync.ts b/packages/apps/src/config/chains/zksync.ts index 551bea41c..c23059a83 100644 --- a/packages/apps/src/config/chains/zksync.ts +++ b/packages/apps/src/config/chains/zksync.ts @@ -27,6 +27,15 @@ export const zksyncChain: ChainConfig = { }, }, tokens: [ + { + decimals: 18, + symbol: "ETH", + name: "ETH", + type: "native", + address: "0x0000000000000000000000000000000000000000", + logo: "eth.svg", + cross: [], + }, { decimals: 6, symbol: "USDT", diff --git a/packages/apps/src/config/gql.ts b/packages/apps/src/config/gql.ts index 9451f340a..2f7c6613e 100644 --- a/packages/apps/src/config/gql.ts +++ b/packages/apps/src/config/gql.ts @@ -129,6 +129,7 @@ export const QUERY_LNRELAYERS = gql` cost profit heartbeatTimestamp + messageChannel } } } diff --git a/packages/apps/src/providers/relayer-provider.tsx b/packages/apps/src/providers/relayer-provider.tsx index f6539d713..8dbf8b119 100644 --- a/packages/apps/src/providers/relayer-provider.tsx +++ b/packages/apps/src/providers/relayer-provider.tsx @@ -50,6 +50,7 @@ interface RelayerCtx { depositMargin: (margin: bigint) => Promise; updateFeeAndMargin: (margin: bigint, baseFee: bigint, feeRate: number) => Promise; setFeeAndRate: (baseFee: bigint, feeRate: number) => Promise; + withdrawMargin: (amount: bigint, fee: bigint) => Promise; } const defaultValue: RelayerCtx = { @@ -79,6 +80,7 @@ const defaultValue: RelayerCtx = { depositMargin: async () => undefined, updateFeeAndMargin: async () => undefined, setFeeAndRate: async () => undefined, + withdrawMargin: async () => undefined, }; export const RelayerContext = createContext(defaultValue); @@ -229,13 +231,29 @@ export default function RelayerProvider({ children }: PropsWithChildren return receipt; } catch (err) { console.error(err); - notification.error({ title: "Transfer failed", description: (err as Error).message }); + notification.error({ title: "Update failed", description: (err as Error).message }); } } }, [address, oppositeBridge, sourceChain], ); + const withdrawMargin = useCallback( + async (amount: bigint, fee: bigint) => { + if (address && defaultBridge) { + try { + const receipt = await defaultBridge.withdrawMargin(address, amount, fee); + notifyTransaction(receipt, sourceChain); + return receipt; + } catch (err) { + console.error(err); + notification.error({ title: "Withdraw failed", description: (err as Error).message }); + } + } + }, + [address, defaultBridge, sourceChain], + ); + useEffect(() => { let sub$$: Subscription | undefined; @@ -299,6 +317,7 @@ export default function RelayerProvider({ children }: PropsWithChildren depositMargin, updateFeeAndMargin, setFeeAndRate, + withdrawMargin, }} > {children} diff --git a/packages/apps/src/types/graphql.ts b/packages/apps/src/types/graphql.ts index 63a6d95a2..368c59441 100644 --- a/packages/apps/src/types/graphql.ts +++ b/packages/apps/src/types/graphql.ts @@ -67,6 +67,8 @@ export interface RecordVariables { id: string; } +export type MessageChannel = "layerzero"; + interface Lnv20RelayInfo { id: string; nonce: string; @@ -88,6 +90,7 @@ interface Lnv20RelayInfo { cost?: string | null; profit?: string | null; heartbeatTimestamp?: number | null; + messageChannel?: MessageChannel | null; } export interface RelayersResponseData { @@ -123,6 +126,7 @@ export type LnRelayerInfo = Pick< | "cost" | "profit" | "heartbeatTimestamp" + | "messageChannel" >; export interface LnRelayersResponseData { diff --git a/packages/apps/src/ui/modal.tsx b/packages/apps/src/ui/modal.tsx index f3922d063..e21499b91 100644 --- a/packages/apps/src/ui/modal.tsx +++ b/packages/apps/src/ui/modal.tsx @@ -15,6 +15,7 @@ interface Props { disabledCancel?: boolean; disabledOk?: boolean; busy?: boolean; + extra?: ReactElement | null; onClose?: () => void; onCancel?: () => void; onOk?: () => void; @@ -32,6 +33,7 @@ export default function Modal({ disabledCancel, disabledOk, busy, + extra, onClose = () => undefined, onCancel, onOk, @@ -91,7 +93,10 @@ export default function Modal({ {/* footer */} {(onCancel || onOk) && ( <> -
+
+ {extra} +
+
{onCancel && ( diff --git a/packages/apps/src/utils/chain.ts b/packages/apps/src/utils/chain.ts index e6f13ab5c..f81b2d6fd 100644 --- a/packages/apps/src/utils/chain.ts +++ b/packages/apps/src/utils/chain.ts @@ -84,7 +84,7 @@ export function getChainConfig(chainIdOrNetwork?: ChainID | Network | null): Cha } } -export function getChainsConfig() { +export function getChainsConfig(env?: "all") { const all = [ arbitrumChain, arbitrumGoerliChain, @@ -107,7 +107,9 @@ export function getChainsConfig() { baseGoerliChain, ]; - if (isProduction()) { + if (env === "all") { + return all; + } else if (isProduction()) { return all.filter((c) => !c.hidden && !c.testnet); } else { return all.filter((c) => !c.hidden && !!c.testnet);