diff --git a/CHANGELOG.md b/CHANGELOG.md index 26c1e3940..b3c734967 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -11,6 +11,7 @@ - support new fields in the `CreateServiceNervousSystem` proposal action `maximum_direct_participation_icp`, `minimum_direct_participation_icp` and `neurons_fund_participation`. - support new filter field `omit_large_fields` in `list_proposals`. - support new field `is_genesis` in `Neuron` and `NeuronInfo`. +- add support for `retrieve_btc_with_approval` in `@dfinity/ckbtc`. # 2023.10.02-1515Z diff --git a/packages/ckbtc/README.md b/packages/ckbtc/README.md index 154b13860..5583b61fc 100644 --- a/packages/ckbtc/README.md +++ b/packages/ckbtc/README.md @@ -78,7 +78,7 @@ Parameters: ### :factory: CkBTCMinterCanister -[:link: Source](https://github.com/dfinity/ic-js/tree/main/packages/ckbtc/src/minter.canister.ts#L33) +[:link: Source](https://github.com/dfinity/ic-js/tree/main/packages/ckbtc/src/minter.canister.ts#L35) #### Methods @@ -87,6 +87,7 @@ Parameters: - [updateBalance](#gear-updatebalance) - [getWithdrawalAccount](#gear-getwithdrawalaccount) - [retrieveBtc](#gear-retrievebtc) +- [retrieveBtcWithApproval](#gear-retrievebtcwithapproval) - [estimateWithdrawalFee](#gear-estimatewithdrawalfee) - [getMinterInfo](#gear-getminterinfo) @@ -96,7 +97,7 @@ Parameters: | -------- | ------------------------------------------------------------------------ | | `create` | `(options: CkBTCMinterCanisterOptions<_SERVICE>) => CkBTCMinterCanister` | -[:link: Source](https://github.com/dfinity/ic-js/tree/main/packages/ckbtc/src/minter.canister.ts#L34) +[:link: Source](https://github.com/dfinity/ic-js/tree/main/packages/ckbtc/src/minter.canister.ts#L36) ##### :gear: getBtcAddress @@ -114,7 +115,7 @@ Parameters: - `params.owner`: The owner for which the BTC address should be generated. If not provided, the `caller` will be use instead. - `params.subaccount`: An optional subaccount to compute the address. -[:link: Source](https://github.com/dfinity/ic-js/tree/main/packages/ckbtc/src/minter.canister.ts#L55) +[:link: Source](https://github.com/dfinity/ic-js/tree/main/packages/ckbtc/src/minter.canister.ts#L57) ##### :gear: updateBalance @@ -132,7 +133,7 @@ Parameters: - `params.owner`: The owner of the address. If not provided, the `caller` will be use instead. - `params.subaccount`: An optional subaccount of the address. -[:link: Source](https://github.com/dfinity/ic-js/tree/main/packages/ckbtc/src/minter.canister.ts#L74) +[:link: Source](https://github.com/dfinity/ic-js/tree/main/packages/ckbtc/src/minter.canister.ts#L76) ##### :gear: getWithdrawalAccount @@ -142,7 +143,7 @@ Returns the account to which the caller should deposit ckBTC before withdrawing | ---------------------- | ------------------------ | | `getWithdrawalAccount` | `() => Promise` | -[:link: Source](https://github.com/dfinity/ic-js/tree/main/packages/ckbtc/src/minter.canister.ts#L97) +[:link: Source](https://github.com/dfinity/ic-js/tree/main/packages/ckbtc/src/minter.canister.ts#L99) ##### :gear: retrieveBtc @@ -166,7 +167,33 @@ Parameters: - `params.address`: The bitcoin address. - `params.amount`: The ckBTC amount. -[:link: Source](https://github.com/dfinity/ic-js/tree/main/packages/ckbtc/src/minter.canister.ts#L116) +[:link: Source](https://github.com/dfinity/ic-js/tree/main/packages/ckbtc/src/minter.canister.ts#L118) + +##### :gear: retrieveBtcWithApproval + +Submits a request to convert ckBTC to BTC after making an ICRC-2 approval. + +# Note + +The BTC retrieval process is slow. Instead of synchronously waiting for a BTC transaction to settle, this method returns a request ([block_index]) that the caller can use to query the request status. + +# Preconditions + +The caller allowed the minter's principal to spend its funds using +[icrc2_approve] on the ckBTC ledger. + +| Method | Type | +| ------------------------- | ------------------------------------------------------------------------------------------------------------------------------------- | +| `retrieveBtcWithApproval` | `({ address, amount, fromSubaccount, }: { address: string; amount: bigint; fromSubaccount?: Uint8Array; }) => Promise` | + +Parameters: + +- `params.address`: The bitcoin address. +- `params.amount`: The ckBTC amount. +- `params.fromSubaccount`: An optional subaccount from which + the ckBTC should be transferred. + +[:link: Source](https://github.com/dfinity/ic-js/tree/main/packages/ckbtc/src/minter.canister.ts#L148) ##### :gear: estimateWithdrawalFee @@ -182,7 +209,7 @@ Parameters: - `params.certified`: query or update call - `params.amount`: The optional amount for which the fee should be estimated. -[:link: Source](https://github.com/dfinity/ic-js/tree/main/packages/ckbtc/src/minter.canister.ts#L135) +[:link: Source](https://github.com/dfinity/ic-js/tree/main/packages/ckbtc/src/minter.canister.ts#L179) ##### :gear: getMinterInfo @@ -197,7 +224,7 @@ Parameters: - `params`: The parameters to get the deposit fee. - `params.certified`: query or update call -[:link: Source](https://github.com/dfinity/ic-js/tree/main/packages/ckbtc/src/minter.canister.ts#L149) +[:link: Source](https://github.com/dfinity/ic-js/tree/main/packages/ckbtc/src/minter.canister.ts#L193) diff --git a/packages/ckbtc/src/errors/minter.errors.ts b/packages/ckbtc/src/errors/minter.errors.ts index 0ed19e4c5..8f7a7eb83 100644 --- a/packages/ckbtc/src/errors/minter.errors.ts +++ b/packages/ckbtc/src/errors/minter.errors.ts @@ -1,5 +1,9 @@ import { nonNullish } from "@dfinity/utils"; -import type { RetrieveBtcError, UpdateBalanceError } from "../../candid/minter"; +import type { + RetrieveBtcError, + RetrieveBtcWithApprovalError, + UpdateBalanceError, +} from "../../candid/minter"; export class MinterGenericError extends Error {} export class MinterTemporaryUnavailableError extends MinterGenericError {} @@ -12,9 +16,10 @@ export class MinterRetrieveBtcError extends MinterGenericError {} export class MinterMalformedAddressError extends MinterRetrieveBtcError {} export class MinterAmountTooLowError extends MinterRetrieveBtcError {} export class MinterInsufficientFundsError extends MinterRetrieveBtcError {} +export class MinterInsufficientAllowanceError extends MinterRetrieveBtcError {} const mapGenericError = ( - Err: UpdateBalanceError | RetrieveBtcError, + Err: UpdateBalanceError | RetrieveBtcError | RetrieveBtcWithApprovalError, ): MinterGenericError | undefined => { if ("GenericError" in Err) { const { @@ -79,3 +84,21 @@ export const createRetrieveBtcError = ( `Unsupported response type in minter.retrieveBtc ${JSON.stringify(Err)}`, ); }; + +export const createRetrieveBtcWithApprovalError = ( + Err: RetrieveBtcWithApprovalError, +): MinterGenericError => { + const error = mapGenericError(Err); + + if (nonNullish(error)) { + return error; + } + + if ("InsufficientAllowance" in Err) { + return new MinterInsufficientAllowanceError( + `${Err.InsufficientAllowance.allowance}`, + ); + } + + return createRetrieveBtcError(Err); +}; diff --git a/packages/ckbtc/src/minter.canister.spec.ts b/packages/ckbtc/src/minter.canister.spec.ts index ed6a5e2c8..482bf1a40 100644 --- a/packages/ckbtc/src/minter.canister.spec.ts +++ b/packages/ckbtc/src/minter.canister.spec.ts @@ -12,6 +12,7 @@ import { MinterAlreadyProcessingError, MinterAmountTooLowError, MinterGenericError, + MinterInsufficientAllowanceError, MinterInsufficientFundsError, MinterMalformedAddressError, MinterNoNewUtxosError, @@ -407,6 +408,186 @@ describe("ckBTC minter canister", () => { }); }); + describe("Retrieve BTC with approval", () => { + const success: RetrieveBtcOk = { + block_index: 1n, + }; + const ok = { Ok: success }; + + const params = { + address: bitcoinAddressMock, + amount: 123_000n, + }; + + it("should return Ok", async () => { + const service = mock>(); + service.retrieve_btc_with_approval.mockResolvedValue(ok); + + const canister = minter(service); + + const res = await canister.retrieveBtcWithApproval(params); + + expect(service.retrieve_btc_with_approval).toBeCalledTimes(1); + expect(service.retrieve_btc_with_approval).toBeCalledWith({ + ...params, + from_subaccount: [], + }); + expect(res).toEqual(success); + }); + + it("should return Ok with fromSubaccount", async () => { + const fromSubaccount = new Uint8Array([3, 4, 5]); + const service = mock>(); + service.retrieve_btc_with_approval.mockResolvedValue(ok); + + const canister = minter(service); + + const res = await canister.retrieveBtcWithApproval({ + ...params, + fromSubaccount, + }); + + expect(service.retrieve_btc_with_approval).toBeCalledTimes(1); + expect(service.retrieve_btc_with_approval).toBeCalledWith({ + ...params, + from_subaccount: [fromSubaccount], + }); + expect(res).toEqual(success); + }); + + it("should throw MinterGenericError", async () => { + const service = mock>(); + + const error = { + Err: { GenericError: { error_message: "message", error_code: 1n } }, + }; + service.retrieve_btc_with_approval.mockResolvedValue(error); + + const canister = minter(service); + + const call = () => canister.retrieveBtcWithApproval(params); + + await expect(call).rejects.toThrowError( + new MinterGenericError( + `${error.Err.GenericError.error_message} (${error.Err.GenericError.error_code})`, + ), + ); + }); + + it("should throw MinterTemporarilyUnavailable", async () => { + const service = mock>(); + + const error = { Err: { TemporarilyUnavailable: "unavailable" } }; + service.retrieve_btc_with_approval.mockResolvedValue(error); + + const canister = minter(service); + + const call = () => canister.retrieveBtcWithApproval(params); + + await expect(call).rejects.toThrowError( + new MinterTemporaryUnavailableError(error.Err.TemporarilyUnavailable), + ); + }); + + it("should throw MinterAlreadyProcessingError", async () => { + const service = mock>(); + + const error = { Err: { AlreadyProcessing: null } }; + service.retrieve_btc_with_approval.mockResolvedValue(error); + + const canister = minter(service); + + const call = () => canister.retrieveBtcWithApproval(params); + + await expect(call).rejects.toThrowError( + new MinterAlreadyProcessingError(), + ); + }); + + it("should throw MinterMalformedAddress", async () => { + const service = mock>(); + + const error = { Err: { MalformedAddress: "malformated" } }; + service.retrieve_btc_with_approval.mockResolvedValue(error); + + const canister = minter(service); + + const call = () => canister.retrieveBtcWithApproval(params); + + await expect(call).rejects.toThrowError( + new MinterMalformedAddressError(error.Err.MalformedAddress), + ); + }); + + it("should throw MinterAmountTooLowError", async () => { + const service = mock>(); + + const error = { Err: { AmountTooLow: 123n } }; + service.retrieve_btc_with_approval.mockResolvedValue(error); + + const canister = minter(service); + + const call = () => canister.retrieveBtcWithApproval(params); + + await expect(call).rejects.toThrowError( + new MinterAmountTooLowError(`${error.Err.AmountTooLow}`), + ); + }); + + it("should throw MinterInsufficientFundsError", async () => { + const service = mock>(); + + const error = { Err: { InsufficientFunds: { balance: 123n } } }; + service.retrieve_btc_with_approval.mockResolvedValue(error); + + const canister = minter(service); + + const call = () => canister.retrieveBtcWithApproval(params); + + await expect(call).rejects.toThrowError( + new MinterInsufficientFundsError( + `${error.Err.InsufficientFunds.balance}`, + ), + ); + }); + + it("should throw MinterInsufficientAllowanceError", async () => { + const service = mock>(); + + const error = { Err: { InsufficientAllowance: { allowance: 123n } } }; + service.retrieve_btc_with_approval.mockResolvedValue(error); + + const canister = minter(service); + + const call = () => canister.retrieveBtcWithApproval(params); + + await expect(call).rejects.toThrowError( + new MinterInsufficientAllowanceError( + `${error.Err.InsufficientAllowance.allowance}`, + ), + ); + }); + + it("should throw unsupported response", async () => { + const service = mock>(); + + const error = { Err: { Test: null } as unknown as RetrieveBtcError }; + service.retrieve_btc_with_approval.mockResolvedValue(error); + + const canister = minter(service); + + const call = () => canister.retrieveBtcWithApproval(params); + + await expect(call).rejects.toThrowError( + new MinterRetrieveBtcError( + `Unsupported response type in minter.retrieveBtc ${JSON.stringify( + error.Err, + )}`, + ), + ); + }); + }); + describe("Estimate Withdrawal Fee", () => { it("should return estimated fee", async () => { const result = { minter_fee: 123n, bitcoin_fee: 456n }; diff --git a/packages/ckbtc/src/minter.canister.ts b/packages/ckbtc/src/minter.canister.ts index 9e86c5838..ddf2c6ce9 100644 --- a/packages/ckbtc/src/minter.canister.ts +++ b/packages/ckbtc/src/minter.canister.ts @@ -14,6 +14,7 @@ import { idlFactory as certifiedIdlFactory } from "../candid/minter.certified.id import { idlFactory } from "../candid/minter.idl"; import { createRetrieveBtcError, + createRetrieveBtcWithApprovalError, createUpdateBalanceError, } from "./errors/minter.errors"; import type { CkBTCMinterCanisterOptions } from "./types/canister.options"; @@ -26,6 +27,7 @@ import type { import type { EstimateWithdrawalFee, RetrieveBtcResponse, + RetrieveBtcWithApprovalResponse, UpdateBalanceOk, UpdateBalanceResponse, } from "./types/minter.responses"; @@ -49,7 +51,7 @@ export class CkBTCMinterCanister extends Canister { * * @param {GetBTCAddressParams} params The parameters for which a BTC address should be resolved. * @param {Principal} params.owner The owner for which the BTC address should be generated. If not provided, the `caller` will be use instead. - * @param {Principal} params.subaccount An optional subaccount to compute the address. + * @param {Uint8Array} params.subaccount An optional subaccount to compute the address. * @returns {Promise} The BTC address of the given account. */ getBtcAddress = ({ @@ -125,6 +127,48 @@ export class CkBTCMinterCanister extends Canister { return response.Ok; }; + /** + * Submits a request to convert ckBTC to BTC after making an ICRC-2 approval. + * + * # Note + * + * The BTC retrieval process is slow. Instead of synchronously waiting for a BTC transaction to settle, this method returns a request ([block_index]) that the caller can use to query the request status. + * + * # Preconditions + * + * The caller allowed the minter's principal to spend its funds using + * [icrc2_approve] on the ckBTC ledger. + * + * @param {string} params.address The bitcoin address. + * @param {bigint} params.amount The ckBTC amount. + * @param {Uint8Array} params.fromSubaccount An optional subaccount from which + * the ckBTC should be transferred. + * @returns {Promise} The result or the operation. + */ + retrieveBtcWithApproval = async ({ + address, + amount, + fromSubaccount, + }: { + address: string; + amount: bigint; + fromSubaccount?: Uint8Array; + }): Promise => { + const response: RetrieveBtcWithApprovalResponse = await this.caller({ + certified: true, + }).retrieve_btc_with_approval({ + address, + amount, + from_subaccount: toNullable(fromSubaccount), + }); + + if ("Err" in response) { + throw createRetrieveBtcWithApprovalError(response.Err); + } + + return response.Ok; + }; + /** * Returns an estimation of the user's fee (in Satoshi) for a retrieve_btc request based on the current status of the Bitcoin network and the fee related to the minter. * diff --git a/packages/ckbtc/src/types/minter.responses.ts b/packages/ckbtc/src/types/minter.responses.ts index d3f1f1bdd..31d4c682e 100644 --- a/packages/ckbtc/src/types/minter.responses.ts +++ b/packages/ckbtc/src/types/minter.responses.ts @@ -1,6 +1,7 @@ import type { RetrieveBtcError, RetrieveBtcOk, + RetrieveBtcWithApprovalError, UpdateBalanceError, UtxoStatus, } from "../../candid/minter"; @@ -15,4 +16,8 @@ export type RetrieveBtcResponse = | { Ok: RetrieveBtcOk } | { Err: RetrieveBtcError }; +export type RetrieveBtcWithApprovalResponse = + | { Ok: RetrieveBtcOk } + | { Err: RetrieveBtcWithApprovalError }; + export type EstimateWithdrawalFee = { minter_fee: bigint; bitcoin_fee: bigint };