From c427bdd53e4462010108e9dbf3fe7bbffbddca6d Mon Sep 17 00:00:00 2001
From: doug <4741454+douglance@users.noreply.github.com>
Date: Tue, 12 Nov 2024 11:32:19 -0500
Subject: [PATCH] feat: adds support for fast confirmation times (#1901)
---
.../TransferPanel/TransferPanelSummary.tsx | 44 ++++++++++++
.../WithdrawalConfirmationDialog.tsx | 41 ++++++++++-
packages/arb-token-bridge-ui/src/constants.ts | 2 +
.../src/hooks/useTransferDuration.ts | 7 ++
.../src/util/WithdrawalUtils.ts | 68 +++++++++++++++++++
.../src/util/orbitChainsData.json | 4 +-
.../src/util/orbitChainsList.ts | 2 +
7 files changed, 164 insertions(+), 4 deletions(-)
diff --git a/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelSummary.tsx b/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelSummary.tsx
index 0a5200018c..a26be83a8b 100644
--- a/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelSummary.tsx
+++ b/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelSummary.tsx
@@ -1,4 +1,5 @@
import React, { useMemo } from 'react'
+import { InformationCircleIcon } from '@heroicons/react/24/outline'
import { twMerge } from 'tailwind-merge'
import { formatAmount } from '../../util/NumberUtils'
@@ -13,10 +14,12 @@ import { useNetworksRelationship } from '../../hooks/useNetworksRelationship'
import { NativeCurrencyPrice, useIsBridgingEth } from './NativeCurrencyPrice'
import { useAppState } from '../../state'
import { Loader } from '../common/atoms/Loader'
+import { Tooltip } from '../common/Tooltip'
import { isTokenNativeUSDC } from '../../util/TokenUtils'
import { NoteBox } from '../common/NoteBox'
import { DISABLED_CHAIN_IDS } from './useTransferReadiness'
import { useIsBatchTransferSupported } from '../../hooks/TransferPanel/useIsBatchTransferSupported'
+import { getConfirmationTime } from '../../util/WithdrawalUtils'
export type TransferPanelSummaryToken = {
symbol: string
@@ -259,6 +262,47 @@ export function TransferPanelSummary({ token }: TransferPanelSummaryProps) {
)}
+ {!isDepositMode &&
+ (isDestinationChainArbitrumOne ||
+ isDestinationChainArbitrumSepolia) && (
+
+
+
+ )}
)
}
+
+function ConfirmationTimeInfo({ chainId }: { chainId: number }) {
+ const {
+ confirmationTimeInReadableFormat,
+ confirmationTimeInReadableFormatShort,
+ isDefaultConfirmationTime
+ } = getConfirmationTime(chainId)
+ return (
+ <>
+ Confirmation time:
+
+
+ {confirmationTimeInReadableFormat}
+
+
+ {confirmationTimeInReadableFormatShort}
+
+ {!isDefaultConfirmationTime && (
+
+
+
+ )}
+
+ >
+ )
+}
diff --git a/packages/arb-token-bridge-ui/src/components/TransferPanel/WithdrawalConfirmationDialog.tsx b/packages/arb-token-bridge-ui/src/components/TransferPanel/WithdrawalConfirmationDialog.tsx
index 5b57633497..eda5f9edda 100644
--- a/packages/arb-token-bridge-ui/src/components/TransferPanel/WithdrawalConfirmationDialog.tsx
+++ b/packages/arb-token-bridge-ui/src/components/TransferPanel/WithdrawalConfirmationDialog.tsx
@@ -12,12 +12,16 @@ import { useAppState } from '../../state'
import { trackEvent } from '../../util/AnalyticsUtils'
import { getNetworkName, isNetwork } from '../../util/networks'
import { getFastBridges } from '../../util/fastBridges'
-import { CONFIRMATION_PERIOD_ARTICLE_LINK } from '../../constants'
+import {
+ CONFIRMATION_PERIOD_ARTICLE_LINK,
+ FAST_WITHDRAWAL_DOCS_ARTICLE_LINK
+} from '../../constants'
import { useNativeCurrency } from '../../hooks/useNativeCurrency'
import { useNetworks } from '../../hooks/useNetworks'
import { useNetworksRelationship } from '../../hooks/useNetworksRelationship'
import { SecurityGuaranteed, SecurityNotGuaranteed } from './SecurityLabels'
import { getWithdrawalConfirmationDate } from '../../hooks/useTransferDuration'
+import { getConfirmationTime } from '../../util/WithdrawalUtils'
function getCalendarUrl(
withdrawalDate: dayjs.Dayjs,
@@ -42,6 +46,8 @@ export function WithdrawalConfirmationDialog(
const { childChain, childChainProvider, parentChain } =
useNetworksRelationship(networks)
+ const { fastWithdrawalActive } = getConfirmationTime(childChain.id)
+
const [selectedIndex, setSelectedIndex] = useState(0)
const destinationNetworkName = getNetworkName(parentChain.id)
@@ -63,9 +69,14 @@ export function WithdrawalConfirmationDialog(
const [checkbox1Checked, setCheckbox1Checked] = useState(false)
const [checkbox2Checked, setCheckbox2Checked] = useState(false)
+ const [checkbox3Checked, setCheckbox3Checked] = useState(false)
const { isArbitrumOne } = isNetwork(childChain.id)
- const bothCheckboxesChecked = checkbox1Checked && checkbox2Checked
+
+ const allCheckboxesChecked =
+ checkbox1Checked &&
+ checkbox2Checked &&
+ (fastWithdrawalActive ? checkbox3Checked : true)
const estimatedConfirmationDate = getWithdrawalConfirmationDate({
createdAt: null,
@@ -81,6 +92,7 @@ export function WithdrawalConfirmationDialog(
setCheckbox1Checked(false)
setCheckbox2Checked(false)
+ setCheckbox3Checked(false)
setSelectedIndex(0)
}
@@ -91,7 +103,7 @@ export function WithdrawalConfirmationDialog(
className="max-w-[700px]"
title={`Move funds to ${destinationNetworkName}`}
actionButtonProps={{
- disabled: !bothCheckboxesChecked,
+ disabled: !allCheckboxesChecked,
hidden: isFastBridgesTab
}}
>
@@ -157,6 +169,29 @@ export function WithdrawalConfirmationDialog(
onChange={setCheckbox2Checked}
/>
+ {fastWithdrawalActive && (
+
+ I understand that ~{confirmationPeriod} is an estimate,
+ and it's possible the committee fails and it will
+ default back to the 8 days.{' '}
+ {
+ e.stopPropagation()
+ }}
+ >
+ Learn more.
+
+
+ }
+ checked={checkbox3Checked}
+ onChange={setCheckbox3Checked}
+ />
+ )}
+
diff --git a/packages/arb-token-bridge-ui/src/constants.ts b/packages/arb-token-bridge-ui/src/constants.ts
index 2e62500b05..85b03a5336 100644
--- a/packages/arb-token-bridge-ui/src/constants.ts
+++ b/packages/arb-token-bridge-ui/src/constants.ts
@@ -20,6 +20,8 @@ export const ETH_BALANCE_ARTICLE_LINK = `${SUPPORT_LINK_BASE}/hc/en-us/articles/
export const CONFIRMATION_PERIOD_ARTICLE_LINK = `${SUPPORT_LINK_BASE}/hc/en-us/articles/18213843096091`
+export const FAST_WITHDRAWAL_DOCS_ARTICLE_LINK = `${DOCS_DOMAIN}/run-arbitrum-node/arbos-releases/arbos31#additional-requirement-for-arbitrum-orbit-chains-who-wish-to-enable-fast-withdrawals`
+
export const ORBIT_QUICKSTART_LINK =
'https://docs.arbitrum.io/launch-orbit-chain/orbit-quickstart'
diff --git a/packages/arb-token-bridge-ui/src/hooks/useTransferDuration.ts b/packages/arb-token-bridge-ui/src/hooks/useTransferDuration.ts
index 771454a252..f8671e23e2 100644
--- a/packages/arb-token-bridge-ui/src/hooks/useTransferDuration.ts
+++ b/packages/arb-token-bridge-ui/src/hooks/useTransferDuration.ts
@@ -9,6 +9,7 @@ import {
getL1BlockTime,
isNetwork
} from '../util/networks'
+import { getConfirmationTime } from '../util/WithdrawalUtils'
const DEPOSIT_TIME_MINUTES = {
mainnet: 15,
@@ -121,6 +122,12 @@ export function getWithdrawalConfirmationDate({
// For new txs createdAt won't be defined yet, we default to the current time in that case
const createdAtDate = createdAt ? dayjs(createdAt) : dayjs()
+ const { confirmationTimeInSeconds, fastWithdrawalActive } =
+ getConfirmationTime(withdrawalFromChainId)
+ if (fastWithdrawalActive && confirmationTimeInSeconds) {
+ return createdAtDate.add(confirmationTimeInSeconds, 'second')
+ }
+
const blockNumberReferenceChainId = getBlockNumberReferenceChainIdByChainId({
chainId: withdrawalFromChainId
})
diff --git a/packages/arb-token-bridge-ui/src/util/WithdrawalUtils.ts b/packages/arb-token-bridge-ui/src/util/WithdrawalUtils.ts
index 2bf9334593..fe6f091928 100644
--- a/packages/arb-token-bridge-ui/src/util/WithdrawalUtils.ts
+++ b/packages/arb-token-bridge-ui/src/util/WithdrawalUtils.ts
@@ -9,6 +9,8 @@ import { BigNumber } from 'ethers'
import { GasEstimates } from '../hooks/arbTokenBridge.types'
import { Address } from './AddressUtils'
import { captureSentryErrorWithExtraData } from './SentryUtils'
+import { getBridgeUiConfigForChain } from './bridgeUiConfig'
+import { isNetwork } from './networks'
export async function withdrawInitTxEstimateGas({
amount,
@@ -105,3 +107,69 @@ export async function withdrawInitTxEstimateGas({
}
}
}
+
+const SECONDS_IN_MINUTE = 60
+const SECONDS_IN_HOUR = 3600
+const SECONDS_IN_DAY = 86400
+const DEFAULT_CONFIRMATION_TIME = 7 * SECONDS_IN_DAY
+const DEFAULT_FAST_WITHDRAWAL_TIME = SECONDS_IN_DAY
+const DEFAULT_TESTNET_CONFIRMATION_TIME = SECONDS_IN_HOUR
+
+function formatDuration(seconds: number, short = false): string {
+ if (seconds < SECONDS_IN_MINUTE) {
+ return `${seconds} ${short ? 'secs' : 'seconds'}`
+ }
+ if (seconds < SECONDS_IN_HOUR) {
+ return `${Math.round(seconds / SECONDS_IN_MINUTE)} ${
+ short ? 'mins' : 'minutes'
+ }`
+ }
+ if (seconds < SECONDS_IN_DAY) {
+ return `${Math.round(seconds / SECONDS_IN_HOUR)} hours`
+ }
+ return `${Math.round(seconds / SECONDS_IN_DAY)} days`
+}
+
+/**
+ * Calculate confirmation time for bridge transactions.
+ * @param {number} chainId - The ID of the parent chain.
+ */
+export function getConfirmationTime(chainId: number) {
+ const { fastWithdrawalTime, fastWithdrawalActive } =
+ getBridgeUiConfigForChain(chainId)
+ const isTestnet = isNetwork(chainId).isTestnet
+
+ const isDefaultConfirmationTime = !fastWithdrawalActive
+ const isDefaultFastWithdrawal = fastWithdrawalActive && !fastWithdrawalTime
+ const isCustomFastWithdrawal = fastWithdrawalActive && !!fastWithdrawalTime
+
+ let confirmationTimeInSeconds: number
+
+ if (isDefaultFastWithdrawal) {
+ confirmationTimeInSeconds = DEFAULT_FAST_WITHDRAWAL_TIME
+ } else if (isCustomFastWithdrawal) {
+ confirmationTimeInSeconds = fastWithdrawalTime / 1000
+ } else {
+ confirmationTimeInSeconds = isTestnet
+ ? DEFAULT_TESTNET_CONFIRMATION_TIME
+ : DEFAULT_CONFIRMATION_TIME
+ }
+
+ const confirmationTimeInReadableFormat = formatDuration(
+ confirmationTimeInSeconds
+ )
+ const confirmationTimeInReadableFormatShort = formatDuration(
+ confirmationTimeInSeconds,
+ true
+ )
+
+ return {
+ fastWithdrawalActive,
+ isDefaultConfirmationTime,
+ isDefaultFastWithdrawal,
+ isCustomFastWithdrawal,
+ confirmationTimeInSeconds,
+ confirmationTimeInReadableFormat,
+ confirmationTimeInReadableFormatShort
+ }
+}
diff --git a/packages/arb-token-bridge-ui/src/util/orbitChainsData.json b/packages/arb-token-bridge-ui/src/util/orbitChainsData.json
index 36738bab09..0c614f96ac 100644
--- a/packages/arb-token-bridge-ui/src/util/orbitChainsData.json
+++ b/packages/arb-token-bridge-ui/src/util/orbitChainsData.json
@@ -768,7 +768,9 @@
"symbol": "CHEESE",
"decimals": 18,
"logoUrl": "/images/CheeseChain_NativeTokenLogo.jpg"
- }
+ },
+ "fastWithdrawalTime": 900000,
+ "fastWithdrawalActive": true
}
},
{
diff --git a/packages/arb-token-bridge-ui/src/util/orbitChainsList.ts b/packages/arb-token-bridge-ui/src/util/orbitChainsList.ts
index 03602d9495..5888a68167 100644
--- a/packages/arb-token-bridge-ui/src/util/orbitChainsList.ts
+++ b/packages/arb-token-bridge-ui/src/util/orbitChainsList.ts
@@ -17,6 +17,8 @@ export type BridgeUiConfig = {
description?: string
}
nativeTokenData?: NativeCurrencyBase
+ fastWithdrawalTime?: number
+ fastWithdrawalActive?: boolean
}
export type OrbitChainConfig = ChainWithRpcUrl & {