Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(bridge-ui): release token button #13203

Merged
merged 36 commits into from
Feb 24, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
36 commits
Select commit Hold shift + click to select a range
f45d817
feat(bridge-ui): adapt frontend to backend changes
shadab-taiko Feb 14, 2023
11e9ac0
fix(bridge-ui): fix tests
shadab-taiko Feb 14, 2023
6cd1487
Merge branch 'main' into a2-prep
cyberhorsey Feb 15, 2023
9d95b30
fix(bridge-ui): refactor
shadab-taiko Feb 15, 2023
11d4f22
Merge branch 'a2-prep' of github.com-taiko:taikochain/taiko-mono into…
shadab-taiko Feb 15, 2023
e243673
feat(bridge-ui): add custom erc20 support
shadab-taiko Feb 16, 2023
a08044f
fix(bridge-ui): fix tests
shadab-taiko Feb 16, 2023
30b59c7
Merge branch 'main' into custom-erc20
shadab-taiko Feb 16, 2023
95521a6
chore(bridge-ui): add env to README
shadab-taiko Feb 17, 2023
a030a93
feat(bridge-ui): fetch token details on user input
shadab-taiko Feb 17, 2023
89dc030
fix(bridge-ui): rename TokenStore to TokenService
shadab-taiko Feb 17, 2023
d85ab22
fix(bridge-ui): define domain method return types
shadab-taiko Feb 17, 2023
a4177c6
Merge branch 'main' into custom-erc20
shadab-taiko Feb 17, 2023
a723804
Merge branch 'main' into custom-erc20
shadab-taiko Feb 18, 2023
86ada77
fix(bridge-ui): add both chains while adding custom tokens
shadab-taiko Feb 18, 2023
9e885fe
fix(bridge-ui): remove accidental commit
shadab-taiko Feb 18, 2023
0f9e5e3
feat(bridge-ui): check if bridged token contract is deployed
shadab-taiko Feb 18, 2023
651355d
fix(brige-ui): test
shadab-taiko Feb 18, 2023
26581d2
Merge branch 'main' into custom-erc20
shadab-taiko Feb 20, 2023
5509402
Merge branch 'main' into custom-erc20
shadab-taiko Feb 20, 2023
250453e
Merge branch 'main' into release-token-buttons
shadab-taiko Feb 22, 2023
a851507
chore(bridge-ui): update abis
shadab-taiko Feb 22, 2023
3fe86be
feat(bridge-ui): create helper
shadab-taiko Feb 22, 2023
85c2312
fix(bridge-ui): review comments
shadab-taiko Feb 22, 2023
6eb496a
Merge branch 'main' of github.com-taiko:taikochain/taiko-mono into cu…
shadab-taiko Feb 22, 2023
f0f158c
Merge branch 'custom-erc20' into release-token-buttons
shadab-taiko Feb 22, 2023
a5d59b7
feat(bridge-ui): add release token for failed tx
shadab-taiko Feb 22, 2023
a1d8e67
fix(bridge-ui): fix message status in UI
shadab-taiko Feb 22, 2023
b09fa90
fix(bridge-ui): fix tests
shadab-taiko Feb 22, 2023
265bd98
Merge branch 'main' into release-token-buttons
shadab-taiko Feb 23, 2023
7680466
chore(bridge-ui): tests
shadab-taiko Feb 23, 2023
7be1ac2
chore(bridge-ui): refactor duplicate code
shadab-taiko Feb 23, 2023
c3ff1e9
Merge branch 'main' into release-token-buttons
shadab-taiko Feb 23, 2023
fbff9a0
feat(bridge-ui): check if failed tokens are released
shadab-taiko Feb 23, 2023
0b61b61
fix(bridge-ui): fix isEtherReleased check
shadab-taiko Feb 24, 2023
8af550d
Merge branch 'main' into release-token-buttons
shadab-taiko Feb 24, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion packages/bridge-ui/jest.config.js
Original file line number Diff line number Diff line change
Expand Up @@ -42,7 +42,7 @@ export default {
statements: 95,
branches: 72,
functions: 89,
lines: 96,
lines: 95,
},
},
modulePathIgnorePatterns: ["<rootDir>/public/build/"],
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,7 @@
message, and you must retry the processing yourself.
</li>
<li class="mb-2">
<strong>Failed</strong>: Your bridged asset is unable to be processed
<strong>Release</strong>: Your bridged asset is unable to be processed
and is available to you on the source chain.
</li>
</ul>
Expand Down
129 changes: 101 additions & 28 deletions packages/bridge-ui/src/components/Transaction.svelte
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@
import { ArrowTopRightOnSquare } from "svelte-heros-v2";
import { MessageStatus } from "../domain/message";
import { Contract, ethers } from "ethers";
import { bridges } from "../store/bridge";
import { bridges, chainIdToTokenVaultAddress } from "../store/bridge";
import { signer } from "../store/signer";
import {
pendingTransactions,
Expand All @@ -27,37 +27,43 @@
import { fetchSigner, switchNetwork } from "@wagmi/core";
import Bridge from "../constants/abi/Bridge";
import ButtonWithTooltip from "./ButtonWithTooltip.svelte";
import TokenVault from "../constants/abi/TokenVault";

export let transaction: BridgeTransaction;

export let fromChain: Chain;
export let toChain: Chain;
let loading: boolean;

let processable: boolean = false;
onMount(async () => {
processable = await isProcessable();
});

async function claim(bridgeTx: BridgeTransaction) {
if (fromChain.id !== bridgeTx.message.destChainId.toNumber()) {
const chain = chains[bridgeTx.message.destChainId.toNumber()];
await switchNetwork({
chainId: chain.id,
});
const provider = new ethers.providers.Web3Provider(window.ethereum);
await provider.send("eth_requestAccounts", []);
async function switchChainAndSetSigner(chain: Chain) {
await switchNetwork({
chainId: chain.id,
});
const provider = new ethers.providers.Web3Provider(window.ethereum);
await provider.send("eth_requestAccounts", []);

fromChainStore.set(chain);
if (chain === CHAIN_MAINNET) {
toChainStore.set(CHAIN_TKO);
} else {
toChainStore.set(CHAIN_MAINNET);
}
const wagmiSigner = await fetchSigner();
signer.set(wagmiSigner);
fromChainStore.set(chain);
if (chain === CHAIN_MAINNET) {
toChainStore.set(CHAIN_TKO);
} else {
toChainStore.set(CHAIN_MAINNET);
}
const wagmiSigner = await fetchSigner();
signer.set(wagmiSigner);
}

async function claim(bridgeTx: BridgeTransaction) {
try {
loading = true;
if (fromChain.id !== bridgeTx.message.destChainId.toNumber()) {
const chain = chains[bridgeTx.message.destChainId.toNumber()];
await switchChainAndSetSigner(chain)
}
const tx = await $bridges
.get(bridgeTx.message.data === "0x" ? BridgeType.ETH : BridgeType.ERC20)
.Claim({
Expand All @@ -79,6 +85,43 @@
} catch (e) {
console.log(e);
errorToast($_("toast.errorSendingTransaction"));
} finally {
loading = false;
}
}

async function releaseTokens(bridgeTx: BridgeTransaction) {
try {
loading = true;
if (fromChain.id !== bridgeTx.message.srcChainId.toNumber()) {
const chain = chains[bridgeTx.message.srcChainId.toNumber()];
await switchChainAndSetSigner(chain)
}
const tx = await $bridges
.get(bridgeTx.message.data === "0x" ? BridgeType.ETH : BridgeType.ERC20)
.ReleaseTokens({
signer: $signer,
message: bridgeTx.message,
msgHash: bridgeTx.msgHash,
destBridgeAddress:
chains[bridgeTx.message.destChainId.toNumber()].bridgeAddress,
srcBridgeAddress:
chains[bridgeTx.message.srcChainId.toNumber()].bridgeAddress,
destProvider: $providers.get(bridgeTx.message.destChainId.toNumber()),
srcTokenVaultAddress: $chainIdToTokenVaultAddress.get(bridgeTx.message.srcChainId.toNumber())
});

pendingTransactions.update((store) => {
store.push(tx);
return store;
});

successToast($_("toast.transactionSent"));
} catch (e) {
console.log(e);
errorToast($_("toast.errorSendingTransaction"));
} finally {
loading = false;
}
}

Expand Down Expand Up @@ -109,6 +152,29 @@
);

transaction.status = await contract.getMessageStatus(transaction.msgHash);
if(transaction.status === MessageStatus.Failed) {
if(transaction.message.data !== "0x") {
const srcTokenVaultContract = new ethers.Contract(
$chainIdToTokenVaultAddress.get(transaction.fromChainId),
TokenVault,
$providers.get(chains[transaction.message.srcChainId.toNumber()].id)
)
const {token, amount} = await srcTokenVaultContract.messageDeposits(transaction.msgHash);
if(token === ethers.constants.AddressZero && amount.eq(0)) {
transaction.status = MessageStatus.FailedReleased;
}
} else {
const srcBridgeContract = new ethers.Contract(
chains[transaction.fromChainId].bridgeAddress,
Bridge,
$providers.get(chains[transaction.message.srcChainId.toNumber()].id)
)
const isFailedMessageResolved = await srcBridgeContract.isEtherReleased(transaction.msgHash);
if(isFailedMessageResolved) {
transaction.status = MessageStatus.FailedReleased;
}
}
}
transaction = transaction;
if (transaction.status === MessageStatus.Done) clearInterval(interval);
}, 20 * 1000);
Expand All @@ -135,7 +201,7 @@
<span slot="buttonText">
{#if !processable}
Pending
{:else if !transaction.receipt && transaction.status === MessageStatus.New}
{:else if (!transaction.receipt && transaction.status === MessageStatus.New) || loading}
<div class="inline-block">
<LottiePlayer
src="/lottie/loader.json"
Expand All @@ -150,24 +216,31 @@
/>
</div>
{:else if transaction.receipt && transaction.status === MessageStatus.New}
<span
class="cursor-pointer border rounded p-1"
<button
class="cursor-pointer border rounded p-1 btn btn-sm border-white"
on:click={async () => await claim(transaction)}
>
Claim
</span>
</button>
{:else if transaction.status === MessageStatus.Retriable}
<span
class="cursor-pointer border rounded p-1"
on:click={async () => await claim(transaction)}
>
Retry
</span>
<button
class="cursor-pointer border rounded p-1 btn btn-sm border-white"
on:click={async () => await claim(transaction)}
>
Retry
</button>
{:else if transaction.status === MessageStatus.Failed}
<!-- todo: releaseTokens() on src bridge with proof from destBridge-->
Failed
<button
class="cursor-pointer border rounded p-1 btn btn-sm border-white"
on:click={async () => await releaseTokens(transaction)}
>
Release
</button>
{:else if transaction.status === MessageStatus.Done}
<span class="border border-transparent p-0">Claimed</span>
{:else if transaction.status === MessageStatus.FailedReleased}
<span class="border border-transparent p-0">Released</span>
{/if}
</span>
</ButtonWithTooltip>
Expand Down
19 changes: 19 additions & 0 deletions packages/bridge-ui/src/constants/abi/Bridge.ts
Original file line number Diff line number Diff line change
Expand Up @@ -414,6 +414,25 @@ export default [
"stateMutability": "view",
"type": "function"
},
{
"inputs": [
{
"internalType": "bytes32",
"name": "msgHash",
"type": "bytes32"
}
],
"name": "isEtherReleased",
"outputs": [
{
"internalType": "bool",
"name": "",
"type": "bool"
}
],
"stateMutability": "view",
"type": "function"
},
{
"inputs": [
{
Expand Down
76 changes: 35 additions & 41 deletions packages/bridge-ui/src/constants/abi/HeaderSync.ts
Original file line number Diff line number Diff line change
@@ -1,59 +1,53 @@
export default [
{
anonymous: false,
inputs: [
"anonymous": false,
"inputs": [
{
indexed: true,
internalType: "uint256",
name: "height",
type: "uint256",
"indexed": true,
"internalType": "uint256",
"name": "srcHeight",
"type": "uint256"
},
{
indexed: true,
internalType: "uint256",
name: "srcHeight",
type: "uint256",
},
{
indexed: false,
internalType: "bytes32",
name: "srcHash",
type: "bytes32",
},
"indexed": false,
"internalType": "bytes32",
"name": "srcHash",
"type": "bytes32"
}
],
name: "HeaderSynced",
type: "event",
"name": "HeaderSynced",
"type": "event"
},
{
inputs: [],
name: "getLatestSyncedHeader",
outputs: [
"inputs": [],
"name": "getLatestSyncedHeader",
"outputs": [
{
internalType: "bytes32",
name: "",
type: "bytes32",
},
"internalType": "bytes32",
"name": "",
"type": "bytes32"
}
],
stateMutability: "view",
type: "function",
"stateMutability": "view",
"type": "function"
},
{
inputs: [
"inputs": [
{
internalType: "uint256",
name: "number",
type: "uint256",
},
"internalType": "uint256",
"name": "number",
"type": "uint256"
}
],
name: "getSyncedHeader",
outputs: [
"name": "getSyncedHeader",
"outputs": [
{
internalType: "bytes32",
name: "",
type: "bytes32",
},
"internalType": "bytes32",
"name": "",
"type": "bytes32"
}
],
stateMutability: "view",
type: "function",
"stateMutability": "view",
"type": "function"
},
];
Loading