From d3651112f46d837e8d7c34fec77cd3ee1539975d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=B6ren?= Date: Fri, 23 Feb 2024 10:54:48 +0100 Subject: [PATCH] Allow native USDC `openWithPermit` function --- client/PublicRequestTypes.ts | 10 +++++----- package.json | 4 ++-- src/lib/RequestParser.ts | 24 ++++++++++++------------ src/lib/RequestTypes.ts | 8 ++++---- src/views/RefundSwap.vue | 2 +- src/views/RefundSwapSuccess.vue | 2 +- src/views/SetupSwap.vue | 11 +++++------ src/views/SetupSwapSuccess.vue | 24 ++++++++++++------------ yarn.lock | 26 +++++--------------------- 9 files changed, 47 insertions(+), 64 deletions(-) diff --git a/client/PublicRequestTypes.ts b/client/PublicRequestTypes.ts index 554f9315..e667ddbd 100644 --- a/client/PublicRequestTypes.ts +++ b/client/PublicRequestTypes.ts @@ -271,12 +271,12 @@ export interface BitcoinHtlcCreationInstructions { } export interface PolygonHtlcCreationInstructions extends RelayRequest { - type: 'USDC'; + type: 'USDC_MATIC'; /** * The sender's nonce in the token contract, required when calling the - * contract function `openWithApproval`. + * contract function `openWithPermit`. */ - approval?: { + permit?: { tokenNonce: number, }; } @@ -312,7 +312,7 @@ export interface BitcoinHtlcSettlementInstructions { } export interface PolygonHtlcSettlementInstructions extends RelayRequest { - type: 'USDC'; + type: 'USDC_MATIC'; amount: number; } @@ -360,7 +360,7 @@ export interface BitcoinHtlcRefundInstructions { } export interface PolygonHtlcRefundInstructions extends RelayRequest { - type: 'USDC'; + type: 'USDC_MATIC'; amount: number; } diff --git a/package.json b/package.json index 44faf1d5..4bc4bae6 100644 --- a/package.json +++ b/package.json @@ -21,9 +21,9 @@ "dependencies": { "@nimiq/browser-warning": "^1.1.1", "@nimiq/electrum-client": "https://github.com/nimiq/electrum-client#build", - "@nimiq/fastspot-api": "^1.7.0", + "@nimiq/fastspot-api": "https://github.com/nimiq/fastspot-api#4dd40d43495589cae79b8cf1116f9464fe1e0e21", "@nimiq/iqons": "^1.5.2", - "@nimiq/keyguard-client": "^1.5.4", + "@nimiq/keyguard-client": "https://gitpkg.vercel.app/nimiq/keyguard?scripts.postinstall=cd%20client%20%26%26%20.%2Fbuild-gitpkg.sh&785b0522b4ed85eb6e0e8407e52494ded25a1ab3", "@nimiq/ledger-api": "^2.3.0", "@nimiq/network-client": "^0.6.2", "@nimiq/oasis-api": "^1.1.1", diff --git a/src/lib/RequestParser.ts b/src/lib/RequestParser.ts index 509cee2f..62bf5510 100644 --- a/src/lib/RequestParser.ts +++ b/src/lib/RequestParser.ts @@ -527,11 +527,11 @@ export class RequestParser { // Validate and parse only what we use in the Hub - if (!['NIM', 'BTC', 'USDC', 'EUR'].includes(setupSwapRequest.fund.type)) { + if (!['NIM', 'BTC', 'USDC_MATIC', 'EUR'].includes(setupSwapRequest.fund.type)) { throw new Error('Funding type is not supported'); } - if (!['NIM', 'BTC', 'USDC', 'EUR'].includes(setupSwapRequest.redeem.type)) { + if (!['NIM', 'BTC', 'USDC_MATIC', 'EUR'].includes(setupSwapRequest.redeem.type)) { throw new Error('Redeeming type is not supported'); } @@ -562,9 +562,9 @@ export class RequestParser { throw new Error('When using the "slider" layout, `bitcoinAccount` must be provided'); } - const nimiqAddress = setupSwapRequest.fund.type === 'NIM' + const nimiqAddress = setupSwapRequest.fund.type === SwapAsset.NIM ? Nimiq.Address.fromAny(setupSwapRequest.fund.sender) - : setupSwapRequest.redeem.type === 'NIM' + : setupSwapRequest.redeem.type === SwapAsset.NIM ? Nimiq.Address.fromAny(setupSwapRequest.redeem.recipient) : undefined; if (nimiqAddress && !setupSwapRequest.nimiqAddresses.some( @@ -572,9 +572,9 @@ export class RequestParser { throw new Error('The address details of the NIM address doing the swap must be provided'); } - const polygonAddress = setupSwapRequest.fund.type === 'USDC' + const polygonAddress = setupSwapRequest.fund.type === SwapAsset.USDC_MATIC ? setupSwapRequest.fund.request.from - : setupSwapRequest.redeem.type === 'USDC' + : setupSwapRequest.redeem.type === SwapAsset.USDC_MATIC ? setupSwapRequest.redeem.request.from : undefined; if (polygonAddress && !setupSwapRequest.polygonAddresses.some( @@ -583,7 +583,7 @@ export class RequestParser { } } - if (setupSwapRequest.redeem.type === 'NIM') { + if (setupSwapRequest.redeem.type === SwapAsset.NIM) { if (!setupSwapRequest.redeem.validityStartHeight || setupSwapRequest.redeem.validityStartHeight < 1) { throw new Error( @@ -592,7 +592,7 @@ export class RequestParser { } } - if (setupSwapRequest.fund.type === 'NIM') { + if (setupSwapRequest.fund.type === SwapAsset.NIM) { if (!setupSwapRequest.fund.validityStartHeight || setupSwapRequest.fund.validityStartHeight < 1) { throw new Error(`Invalid validity start height: ${setupSwapRequest.fund.validityStartHeight}`); @@ -628,7 +628,7 @@ export class RequestParser { } : setupSwapRequest.fund.type === 'BTC' ? { ...setupSwapRequest.fund, type: SwapAsset[setupSwapRequest.fund.type], - } : setupSwapRequest.fund.type === 'USDC' ? { + } : setupSwapRequest.fund.type === 'USDC_MATIC' ? { ...setupSwapRequest.fund, type: SwapAsset[setupSwapRequest.fund.type], } : { // EUR @@ -646,7 +646,7 @@ export class RequestParser { } : setupSwapRequest.redeem.type === 'BTC' ? { ...setupSwapRequest.redeem, type: SwapAsset[setupSwapRequest.redeem.type], - } : setupSwapRequest.redeem.type === 'USDC' ? { + } : setupSwapRequest.redeem.type === 'USDC_MATIC' ? { ...setupSwapRequest.redeem, type: SwapAsset[setupSwapRequest.redeem.type], } : { // EUR @@ -664,8 +664,8 @@ export class RequestParser { // Only basic parsing and validation. Refund transaction specific data will be validated by the Keyguard // or subsequent Ledger transaction signing requests. - if (!['NIM', 'BTC', 'USDC'].includes(refundSwapRequest.refund.type)) { - throw new Error('Refunding object type must be "NIM", "BTC", or "USDC"'); + if (!['NIM', 'BTC', 'USDC_MATIC'].includes(refundSwapRequest.refund.type)) { + throw new Error('Refunding object type must be "NIM", "BTC", or "USDC_MATIC"'); } const parsedRefundSwapRequest: ParsedRefundSwapRequest = { diff --git a/src/lib/RequestTypes.ts b/src/lib/RequestTypes.ts index 7fa425da..3770a624 100644 --- a/src/lib/RequestTypes.ts +++ b/src/lib/RequestTypes.ts @@ -198,8 +198,8 @@ export interface ParsedSetupSwapRequest extends ParsedSimpleRequest { // htlcScript: Uint8Array, refundAddress: string, } | ({ - type: SwapAsset.USDC, - approval?: { + type: SwapAsset.USDC_MATIC, + permit?: { tokenNonce: number, }, } & RelayRequest) | { @@ -232,7 +232,7 @@ export interface ParsedSetupSwapRequest extends ParsedSimpleRequest { value: number, // Sats }; } | ({ - type: SwapAsset.USDC, + type: SwapAsset.USDC_MATIC, amount: number, } & RelayRequest) | { type: SwapAsset.EUR, @@ -311,7 +311,7 @@ export interface ParsedRefundSwapRequest extends ParsedSimpleRequest { }; refundAddress: string; // My address, must be refund address of HTLC } | ({ - type: SwapAsset.USDC, + type: SwapAsset.USDC_MATIC, amount: number, } & RelayRequest); } diff --git a/src/views/RefundSwap.vue b/src/views/RefundSwap.vue index e91697cc..4ed95063 100644 --- a/src/views/RefundSwap.vue +++ b/src/views/RefundSwap.vue @@ -122,7 +122,7 @@ export default class RefundSwap extends BitcoinSyncBaseView { this._signTransaction(signRequest); } - if (refundInfo.type === SwapAsset.USDC) { + if (refundInfo.type === SwapAsset.USDC_MATIC) { const signer = account.polygonAddresses.find((ai) => ai.address === refundInfo.request.from); if (!signer) { diff --git a/src/views/RefundSwapSuccess.vue b/src/views/RefundSwapSuccess.vue index 13f22709..1526efc7 100644 --- a/src/views/RefundSwapSuccess.vue +++ b/src/views/RefundSwapSuccess.vue @@ -55,7 +55,7 @@ export default class SignBtcTransactionSuccess extends Vue { return; } - if ('message' in this.keyguardResult && this.request.refund.type === SwapAsset.USDC) { + if ('message' in this.keyguardResult && this.request.refund.type === SwapAsset.USDC_MATIC) { const result: SignedPolygonTransaction = { message: this.keyguardResult.message, signature: this.keyguardResult.signature, diff --git a/src/views/SetupSwap.vue b/src/views/SetupSwap.vue index 949bffbd..ba698a40 100644 --- a/src/views/SetupSwap.vue +++ b/src/views/SetupSwap.vue @@ -11,7 +11,6 @@ import { WalletInfo } from '../lib/WalletInfo'; import { BtcAddressInfo } from '../lib/bitcoin/BtcAddressInfo'; import { SwapAsset } from '@nimiq/fastspot-api'; import { DEFAULT_KEY_PATH } from '../lib/Constants'; -import { threadId } from 'worker_threads'; // Import only types to avoid bundling of KeyguardClient in Ledger request if not required. // (But note that currently, the KeyguardClient is still always bundled in the RpcApi). @@ -183,7 +182,7 @@ export default class SetupSwap extends BitcoinSyncBaseView { }; } - if (this.request.fund.type === SwapAsset.USDC) { + if (this.request.fund.type === SwapAsset.USDC_MATIC) { const senderAddress = this.request.fund.request.from; const signer = this._account.polygonAddresses.find((ai) => ai.address === senderAddress); @@ -192,11 +191,11 @@ export default class SetupSwap extends BitcoinSyncBaseView { } fundingInfo = { - type: SwapAsset.USDC, + type: SwapAsset.USDC_MATIC, keyPath: signer.path, request: this.request.fund.request, relayData: this.request.fund.relayData, - approval: this.request.fund.approval, + permit: this.request.fund.permit, }; } @@ -255,7 +254,7 @@ export default class SetupSwap extends BitcoinSyncBaseView { }; } - if (this.request.redeem.type === SwapAsset.USDC) { + if (this.request.redeem.type === SwapAsset.USDC_MATIC) { const senderAddress = this.request.redeem.request.from; const signer = this._account.polygonAddresses.find((ai) => ai.address === senderAddress); @@ -264,7 +263,7 @@ export default class SetupSwap extends BitcoinSyncBaseView { } redeemingInfo = { - type: SwapAsset.USDC, + type: SwapAsset.USDC_MATIC, keyPath: signer.path, request: this.request.redeem.request, relayData: this.request.redeem.relayData, diff --git a/src/views/SetupSwapSuccess.vue b/src/views/SetupSwapSuccess.vue index 890e6b2b..a75b0b1a 100644 --- a/src/views/SetupSwapSuccess.vue +++ b/src/views/SetupSwapSuccess.vue @@ -84,7 +84,7 @@ export default class SetupSwapSuccess extends BitcoinSyncBaseView { case SwapAsset.BTC: refundAddress = this.request.fund.refundAddress; break; - case SwapAsset.USDC: + case SwapAsset.USDC_MATIC: refundAddress = this.request.fund.request.from; break; default: break; @@ -98,7 +98,7 @@ export default class SetupSwapSuccess extends BitcoinSyncBaseView { case SwapAsset.BTC: redeemAddress = this.request.redeem.output.address; break; - case SwapAsset.USDC: + case SwapAsset.USDC_MATIC: redeemAddress = this.request.redeem.request.from; break; case SwapAsset.EUR: @@ -228,8 +228,8 @@ export default class SetupSwapSuccess extends BitcoinSyncBaseView { } } - if (confirmedSwap.from.asset === SwapAsset.USDC || confirmedSwap.to.asset === SwapAsset.USDC) { - const contract = confirmedSwap.contracts[SwapAsset.USDC] as Contract; + if (confirmedSwap.from.asset === SwapAsset.USDC_MATIC || confirmedSwap.to.asset === SwapAsset.USDC_MATIC) { + const contract = confirmedSwap.contracts[SwapAsset.USDC_MATIC] as Contract; const htlc = contract.htlc as UsdcHtlcDetails; const contractData = { @@ -254,11 +254,11 @@ export default class SetupSwapSuccess extends BitcoinSyncBaseView { } hashRoot = contractData.hash; - if (confirmedSwap.from.asset === SwapAsset.USDC && refundAddress !== contractData.refundAddress) { + if (confirmedSwap.from.asset === SwapAsset.USDC_MATIC && refundAddress !== contractData.refundAddress) { this.$rpc.reject(new Error('Unknown HTLC refund address')); return; } - if (confirmedSwap.to.asset === SwapAsset.USDC && redeemAddress !== contractData.recipientAddress) { + if (confirmedSwap.to.asset === SwapAsset.USDC_MATIC && redeemAddress !== contractData.recipientAddress) { this.$rpc.reject(new Error('Unknown HTLC redeem address')); return; } @@ -304,8 +304,8 @@ export default class SetupSwapSuccess extends BitcoinSyncBaseView { }; } - if (this.request.fund.type === SwapAsset.USDC) { - const usdcHtlcData = confirmedSwap.contracts[SwapAsset.USDC]!.htlc as UsdcHtlcDetails; + if (this.request.fund.type === SwapAsset.USDC_MATIC) { + const usdcHtlcData = confirmedSwap.contracts[SwapAsset.USDC_MATIC]!.htlc as UsdcHtlcDetails; if (!usdcHtlcData.data) { // TODO: Create data with ethersJS @@ -313,7 +313,7 @@ export default class SetupSwapSuccess extends BitcoinSyncBaseView { } fundingHtlcInfo = { - type: SwapAsset.USDC, + type: SwapAsset.USDC_MATIC, htlcData: usdcHtlcData.data, }; } @@ -409,12 +409,12 @@ export default class SetupSwapSuccess extends BitcoinSyncBaseView { } } - if (this.request.redeem.type === SwapAsset.USDC) { - const usdcContract = confirmedSwap.contracts[SwapAsset.USDC] as Contract; + if (this.request.redeem.type === SwapAsset.USDC_MATIC) { + const usdcContract = confirmedSwap.contracts[SwapAsset.USDC_MATIC] as Contract; const usdcHtlcData = usdcContract.htlc as UsdcHtlcDetails; redeemingHtlcInfo = { - type: SwapAsset.USDC, + type: SwapAsset.USDC_MATIC, hash: confirmedSwap.hash, timeout: usdcContract.timeout, htlcId: usdcHtlcData.address, diff --git a/yarn.lock b/yarn.lock index 886ddbcf..51bff31e 100644 --- a/yarn.lock +++ b/yarn.lock @@ -1486,11 +1486,6 @@ resolved "https://registry.yarnpkg.com/@nimiq/core-web/-/core-web-1.5.3.tgz#f0a4de59394f210f2c2d9cda8ee35c716847d40e" integrity sha512-W66SS9n3ygYgD52r1GJr1WtYYOkcZsqdtMmDCEwDvkrmeARnHs2sAvj77Wt4PQG8JA7GwK5svIJr6rGccCaekw== -"@nimiq/core-web@1.5.8": - version "1.5.8" - resolved "https://registry.yarnpkg.com/@nimiq/core-web/-/core-web-1.5.8.tgz#da8abef84c6d293cbb9ca495a77f08daa08886b9" - integrity sha512-MNpFbGZetz2eZcHtJVa5tpmLjPf65mTcJBRQkOHS8kWE+f+Z++hdnwWUBQCEAph4oC6bsE5JSuU7VpwcogN7+w== - "@nimiq/core-web@^1.6.0", "@nimiq/core-web@^1.6.1": version "1.6.1" resolved "https://registry.yarnpkg.com/@nimiq/core-web/-/core-web-1.6.1.tgz#97cb5b43b257c7f6f6808ef603e9bf686377241f" @@ -1520,10 +1515,9 @@ dependencies: bitcoinjs-lib "^5.1.10" -"@nimiq/fastspot-api@^1.7.0": +"@nimiq/fastspot-api@https://github.com/nimiq/fastspot-api#4dd40d43495589cae79b8cf1116f9464fe1e0e21": version "1.7.0" - resolved "https://registry.yarnpkg.com/@nimiq/fastspot-api/-/fastspot-api-1.7.0.tgz#eea496f4898139c51971f0b177583c6cc6b06611" - integrity sha512-k7vSjYdjAiAQar+21v681Zrp9kN9uHMUw7/dyG2D/IrXqNPHkD133JyrzxPBga+myW411SUR4mtdPQRPQm3d/A== + resolved "https://github.com/nimiq/fastspot-api#4dd40d43495589cae79b8cf1116f9464fe1e0e21" "@nimiq/iqons@^1.5.2", "@nimiq/iqons@^1.6.0": version "1.6.0" @@ -1541,14 +1535,9 @@ btoa "^1.1.2" node-lmdb "^0.9.6" -"@nimiq/keyguard-client@^1.5.4": - version "1.5.4" - resolved "https://registry.yarnpkg.com/@nimiq/keyguard-client/-/keyguard-client-1.5.4.tgz#0842386e1c36e2b203cae94ac62ffafd48ee79fa" - integrity sha512-E4sQys66nSPCiMRm2NN9ibm0RZCA9VbyP7UApSEBShBCHkCCSYp1zjfuPm9+v4VKwA7yw+pEDZ5RE7Mlf2OdIQ== - dependencies: - "@nimiq/core-web" "1.5.8" - "@nimiq/rpc" "^0.3.0" - "@opengsn/common" "^2.2.5" +"@nimiq/keyguard-client@https://gitpkg.vercel.app/nimiq/keyguard?scripts.postinstall=cd%20client%20%26%26%20.%2Fbuild-gitpkg.sh&785b0522b4ed85eb6e0e8407e52494ded25a1ab3": + version "1.0.0" + resolved "https://gitpkg.vercel.app/nimiq/keyguard?scripts.postinstall=cd%20client%20%26%26%20.%2Fbuild-gitpkg.sh&785b0522b4ed85eb6e0e8407e52494ded25a1ab3#87e25cb600a9219ded03097a920321d54876ba60" "@nimiq/ledger-api@^2.3.0": version "2.3.0" @@ -1594,11 +1583,6 @@ resolved "https://registry.yarnpkg.com/@nimiq/rpc/-/rpc-0.1.5.tgz#53919b0a3a9abcdfebee0e865f4c663f72a8b8c2" integrity sha512-oTRThXzpbQOY8jz8h+2KXucWzW40nVfYBWROKXKBrSozhTG0nR+rzCbEm4ZyTC26b4RnmB6y4nYabUXi7gNWcA== -"@nimiq/rpc@^0.3.0": - version "0.3.0" - resolved "https://registry.yarnpkg.com/@nimiq/rpc/-/rpc-0.3.0.tgz#654c05acccc193b7d79fb09b2faf2114945ff872" - integrity sha512-je7fv+wP4nLEgTcZwu3FaGre22qkZ9AYGbStglVaJAxOH+3CvDnnOIa9IjGFaCEhtRQKRaQEvFqa5vN4IVnH+Q== - "@nimiq/rpc@^0.4.1": version "0.4.1" resolved "https://registry.yarnpkg.com/@nimiq/rpc/-/rpc-0.4.1.tgz#d5df1e426793afcdd8c407a2968442bbee874dbd"