From fc1202ee40d353a507aad8247c5ab69d7cebb39a Mon Sep 17 00:00:00 2001 From: Erdi Maden Date: Tue, 14 May 2024 13:58:14 -0500 Subject: [PATCH 01/10] Adding Address Class with tests and updating faucet_transaction test file name --- src/coinbase/address.ts | 87 +++++++++++++++++++ src/coinbase/tests/address_test.ts | 60 +++++++++++++ ...ion_test.ts => faucet_transaction_test.ts} | 1 - src/coinbase/types.ts | 17 ++++ 4 files changed, 164 insertions(+), 1 deletion(-) create mode 100644 src/coinbase/address.ts create mode 100644 src/coinbase/tests/address_test.ts rename src/coinbase/tests/{faucetTransaction_test.ts => faucet_transaction_test.ts} (95%) diff --git a/src/coinbase/address.ts b/src/coinbase/address.ts new file mode 100644 index 00000000..e8a30b50 --- /dev/null +++ b/src/coinbase/address.ts @@ -0,0 +1,87 @@ +import { Address as AddressModel } from "../client"; +import { InternalError } from "./errors"; +import { FaucetTransaction } from "./faucet_transaction"; +import { AddressClient } from "./types"; + +/** + * Class representing an Address in the Coinbase SDK. + */ +export class Address { + private model: AddressModel; + private client: AddressClient; + + /** + * Creates an instance of Address. + * @param {AddressModel} model - The address model data. + * @param {AddressClient} client - The API client to interact with address-related endpoints. + * @throws {InternalError} If the model or client is empty. + */ + constructor(model: AddressModel, client: AddressClient) { + if (!model) { + throw new InternalError("Address model cannot be empty"); + } + if (!client) { + throw new InternalError("Address client cannot be empty"); + } + this.model = model; + this.client = client; + } + + /** + * Requests faucet funds for the address. + * @returns {Promise} The faucet transaction object. + * @throws {InternalError} If the request does not return a transaction hash. + * @throws {Error} If the request fails. + */ + async faucet(): Promise { + try { + const response = await this.client.requestFaucetFunds( + this.model.wallet_id, + this.model.address_id, + ); + return new FaucetTransaction(response.data); + } catch (e) { + throw new Error(`Failed to request faucet funds`); + } + } + + /** + * Gets the address ID. + * @returns {string} The address ID. + */ + public getId(): string { + return this.model.address_id; + } + + /** + * Gets the network ID. + * @returns {string} The network ID. + */ + public getNetworkId(): string { + return this.model.network_id; + } + + /** + * Gets the public key. + * @returns {string} The public key. + */ + public getPublicKey(): string { + return this.model.public_key; + } + + /** + * Gets the wallet ID. + * @returns {string} The wallet ID. + */ + public getWalletId(): string { + return this.model.wallet_id; + } + + /** + * Returns a string representation of the address. + * @returns {string} A string representing the address. + */ + public toString(): string { + return `Coinbase:Address{addressId: '${this.model.address_id}', networkId: '${this.model.network_id}', walletId: '${this.model.wallet_id}'}`; + } +} diff --git a/src/coinbase/tests/address_test.ts b/src/coinbase/tests/address_test.ts new file mode 100644 index 00000000..f4a987f7 --- /dev/null +++ b/src/coinbase/tests/address_test.ts @@ -0,0 +1,60 @@ +import { AddressesApiFactory, Address as AddressModel } from "../../client"; +import { Address } from "./../address"; +import { FaucetTransaction } from "./../faucet_transaction"; + +import axios from "axios"; +import MockAdapter from "axios-mock-adapter"; + +const axiosMock = new MockAdapter(axios); + +const VALID_ADDRESS_MODEL: AddressModel = { + address_id: "mocked_address_id", + network_id: "mocked_network_id", + public_key: "mocked_public_key", + wallet_id: "mocked_wallet_id", +}; + +// Test suite for Address class +describe("Address", () => { + const client = AddressesApiFactory(); + + it("should create an Address instance", () => { + const address = new Address(VALID_ADDRESS_MODEL, client); + expect(address).toBeInstanceOf(Address); + expect(address.getId()).toBe("mocked_address_id"); + expect(address.getNetworkId()).toBe("mocked_network_id"); + expect(address.getPublicKey()).toBe("mocked_public_key"); + expect(address.getWalletId()).toBe("mocked_wallet_id"); + }); + + it("should throw an InternalError if model is not provided", () => { + expect(() => new Address(null!, client)).toThrow(`Address model cannot be empty`); + }); + + it("should throw an InternalError if client is not provided", () => { + expect(() => new Address(VALID_ADDRESS_MODEL, null!)).toThrow(`Address client cannot be empty`); + }); + + it("should request faucet funds and return a FaucetTransaction", async () => { + axiosMock.onPost().reply(200, { + transaction_hash: "mocked_transaction_hash", + }); + const address = new Address(VALID_ADDRESS_MODEL, client); + const faucetTransaction = await address.faucet(); + expect(faucetTransaction).toBeInstanceOf(FaucetTransaction); + expect(faucetTransaction.getTransactionHash()).toBe("mocked_transaction_hash"); + }); + + it("should throw an error if faucet request fails", async () => { + axiosMock.onPost().reply(400); + const address = new Address(VALID_ADDRESS_MODEL, client); + await expect(address.faucet()).rejects.toThrow("Failed to request faucet funds"); + }); + + it("should return the correct string representation", () => { + const address = new Address(VALID_ADDRESS_MODEL, client); + expect(address.toString()).toBe( + "Coinbase:Address{addressId: 'mocked_address_id', networkId: 'mocked_network_id', walletId: 'mocked_wallet_id'}", + ); + }); +}); diff --git a/src/coinbase/tests/faucetTransaction_test.ts b/src/coinbase/tests/faucet_transaction_test.ts similarity index 95% rename from src/coinbase/tests/faucetTransaction_test.ts rename to src/coinbase/tests/faucet_transaction_test.ts index 0dc9ad64..78b6dab8 100644 --- a/src/coinbase/tests/faucetTransaction_test.ts +++ b/src/coinbase/tests/faucet_transaction_test.ts @@ -1,4 +1,3 @@ -import { InternalError } from "./../errors"; import { FaucetTransaction } from "../faucet_transaction"; describe("FaucetTransaction tests", () => { diff --git a/src/coinbase/types.ts b/src/coinbase/types.ts index d5e0ad34..46395f7c 100644 --- a/src/coinbase/types.ts +++ b/src/coinbase/types.ts @@ -1,6 +1,23 @@ import { AxiosPromise, AxiosRequestConfig } from "axios"; import { User as UserModel } from "./../client/api"; +/** + * AddressAPI client type definition. + */ +export type AddressClient = { + /** + * Requests faucet funds for the address. + * @param {string} walletId - The wallet ID. + * @param {string} addressId - The address ID. + * @returns {Promise<{ data: { transaction_hash: string } }>} - The transaction hash + * @throws {Error} If the request fails. + */ + requestFaucetFunds( + walletId: string, + addressId: string, + ): Promise<{ data: { transaction_hash: string } }>; +}; + /** * UserAPI client type definition. */ From cfe1c38188d66e5f10c28d63118a245bd367bbdf Mon Sep 17 00:00:00 2001 From: Erdi Maden Date: Tue, 14 May 2024 15:43:50 -0500 Subject: [PATCH 02/10] Adding initial version of Wallet class --- src/coinbase/tests/wallet_test.ts | 71 +++++++++++++++++ src/coinbase/types.ts | 20 ++++- src/coinbase/wallet.ts | 123 ++++++++++++++++++++++++++++++ 3 files changed, 212 insertions(+), 2 deletions(-) create mode 100644 src/coinbase/tests/wallet_test.ts create mode 100644 src/coinbase/wallet.ts diff --git a/src/coinbase/tests/wallet_test.ts b/src/coinbase/tests/wallet_test.ts new file mode 100644 index 00000000..c290962a --- /dev/null +++ b/src/coinbase/tests/wallet_test.ts @@ -0,0 +1,71 @@ +import { HDKey } from "@scure/bip32"; +import * as bip39 from "bip39"; +import { Address } from "./../address"; +import { InternalError } from "./../errors"; +import { Wallet as WalletModel, Address as AddressModel } from "../../client"; +import { ApiClients } from "./../types"; +import { ethers } from "ethers"; +import { Wallet } from "../wallet"; +import MockAdapter from "axios-mock-adapter"; +import axios from "axios"; + +// Mock data +const VALID_WALLET_MODEL: WalletModel = { + id: "mocked_wallet_id", + network_id: "mocked_network_id", + default_address: { + address_id: "mocked_address_id", + network_id: "mocked_network_id", + public_key: "mocked_public_key", + wallet_id: "mocked_wallet_id", + }, +}; + +const VALID_ADDRESS_MODEL: AddressModel = { + address_id: "mocked_address_id", + network_id: "mocked_network_id", + public_key: "mocked_public_key", + wallet_id: "mocked_wallet_id", +}; + +// Mock ApiClients +const mockClient: ApiClients = { + address: { + getAddress: jest.fn().mockResolvedValue({ data: VALID_ADDRESS_MODEL }), + requestFaucetFunds: jest.fn(), + }, + user: { + getCurrentUser: jest.fn(), + }, +}; + +// Test suite for Wallet class +describe("Wallet", () => { + const seed = bip39.generateMnemonic(); + it("should create a Wallet instance", () => { + const wallet = new Wallet(VALID_WALLET_MODEL, mockClient, seed, 1); + expect(wallet).toBeInstanceOf(Wallet); + expect(wallet.getId()).toBe("mocked_wallet_id"); + expect(wallet.getNetworkId()).toBe("mocked_network_id"); + expect(wallet.defaultAddress()?.getId()).toBe("mocked_address_id"); + }); + + it("should return the correct string representation", () => { + const wallet = new Wallet(VALID_WALLET_MODEL, mockClient, seed, 0); + expect(wallet.toString()).toBe( + "Wallet{id: 'mocked_wallet_id', network_id: 'mocked_network_id'}", + ); + }); + + it("should throw an InternalError if address client is not provided", () => { + expect(() => new Wallet(VALID_WALLET_MODEL, undefined!)).toThrow( + "Address client cannot be empty", + ); + }); + + it("should generate and derive addresses based on the provided count", () => { + const addressCount = 3; + const wallet = new Wallet(VALID_WALLET_MODEL, mockClient, seed, addressCount); + expect(wallet["addresses"].length).toBe(addressCount); + }); +}); diff --git a/src/coinbase/types.ts b/src/coinbase/types.ts index 46395f7c..3040fbe6 100644 --- a/src/coinbase/types.ts +++ b/src/coinbase/types.ts @@ -1,5 +1,6 @@ -import { AxiosPromise, AxiosRequestConfig } from "axios"; -import { User as UserModel } from "./../client/api"; +import { AxiosPromise, AxiosRequestConfig, RawAxiosRequestConfig } from "axios"; +import { Address, User as UserModel } from "./../client/api"; +import { RequestArgs } from "../client/base"; /** * AddressAPI client type definition. @@ -16,6 +17,20 @@ export type AddressClient = { walletId: string, addressId: string, ): Promise<{ data: { transaction_hash: string } }>; + + /** + * Get address + * @summary Get address by onchain address + * @param {string} walletId The ID of the wallet the address belongs to. + * @param {string} addressId The onchain address of the address that is being fetched. + * @param {AxiosRequestConfig} [options] - Axios request options. + * @throws {RequiredError} + */ + getAddress( + walletId: string, + addressId: string, + options?: AxiosRequestConfig, + ): AxiosPromise
; }; /** @@ -41,4 +56,5 @@ export type ApiClients = { * @type {UserAPIClient} */ user?: UserAPIClient; + address?: AddressClient; }; diff --git a/src/coinbase/wallet.ts b/src/coinbase/wallet.ts new file mode 100644 index 00000000..9ff83548 --- /dev/null +++ b/src/coinbase/wallet.ts @@ -0,0 +1,123 @@ +import { HDKey } from "@scure/bip32"; +import * as bip39 from "bip39"; +import { Address } from "./address"; +import { InternalError } from "./errors"; +import { Wallet as WalletModel, Address as AddressModel } from "../client"; +import { ApiClients } from "./types"; +import { ethers } from "ethers"; + +/** + * Wallet class represents a wallet with address management. + */ +export class Wallet { + private model: WalletModel; + private client: ApiClients; + + private master: HDKey; + private addresses: Address[] = []; + private readonly addressPathPrefix = "m/44'/60'/0'/0"; + private addressIndex = 0; + + /** + * Creates an instance of Wallet. + * @param {WalletModel} model - The wallet model data. + * @param {ApiClients} client - The API client to interact with address-related endpoints. + * @param {string} seed - The seed to generate the wallet. + * @param {number} addressCount - The number of addresses to generate. + * @throws {InternalError} If the model or client is empty. + */ + constructor(model: WalletModel, client: ApiClients, seed: string = "", addressCount: number = 0) { + if (!model) { + throw new InternalError("Wallet model cannot be empty"); + } + if (!client?.address || !client?.user) { + throw new InternalError("Address client cannot be empty"); + } + this.model = model; + this.client = client; + + if (!seed) { + seed = bip39.generateMnemonic(); + } + + this.master = HDKey.fromMasterSeed(bip39.mnemonicToSeedSync(seed)); + for (let i = 0; i < addressCount; i++) { + this.deriveAddress(); + } + } + + /** + * Derives a new HDKey for the next address. + * @returns The derived HDKey. + */ + private deriveKey(): HDKey { + return this.master.deriveChild(this.addressIndex++); + } + + /** + * Derives a new address and caches it. + * @returns {Promise} + * @throws {InternalError} If address derivation or caching fails. + */ + private deriveAddress() { + try { + const key = this.deriveKey(); + const address = new ethers.Wallet(Buffer.from(key?.privateKey || "").toString("hex")).address; + if (!this.model.id) { + throw new InternalError("Wallet ID could not found"); + } + } catch (e) { + console.error("Error deriving address:", e); + throw new InternalError("Failed to derive address"); + } + } + + /** + * Caches an Address on the client-side and increments the address index. + * @param {AddressModel} address - The address to cache. + * @throws {InternalError} If the address is not provided. + * @returns {void} + */ + private cacheAddress(address: AddressModel): void { + this.addresses.push(new Address(address, this.client.address!)); + this.addressIndex++; + } + + /** + * Gets the network identifier. + * @returns The network identifier. + */ + public getNetworkId(): string { + return this.model.network_id; + } + + /** + * Gets the wallet identifier. + * @returns The wallet identifier. + */ + public getId(): string | undefined { + return this.model.id; + } + + /** + * Gets the default address of the wallet. + * @returns The default address. + */ + public defaultAddress(): Address | undefined { + if (!this.model.default_address) { + return undefined; + } + if (!this.client.address) { + throw new InternalError("Address client cannot be empty"); + } + return new Address(this.model.default_address, this.client.address); + } + + /** + * Converts the wallet instance to a string representation. + * @returns The string representation of the wallet. + */ + public toString(): string { + return `Wallet{id: '${this.model.id}', network_id: '${this.model.network_id}'}`; + } +} From a4e4fdb6948d2eec077a57bfaf458720cbb2508e Mon Sep 17 00:00:00 2001 From: Erdi Maden Date: Tue, 14 May 2024 21:57:26 -0500 Subject: [PATCH 03/10] Updating test cases --- src/coinbase/tests/wallet_test.ts | 19 +++++----------- src/coinbase/wallet.ts | 37 ++++++++++++------------------- 2 files changed, 19 insertions(+), 37 deletions(-) diff --git a/src/coinbase/tests/wallet_test.ts b/src/coinbase/tests/wallet_test.ts index c290962a..7d6250aa 100644 --- a/src/coinbase/tests/wallet_test.ts +++ b/src/coinbase/tests/wallet_test.ts @@ -1,13 +1,7 @@ -import { HDKey } from "@scure/bip32"; import * as bip39 from "bip39"; -import { Address } from "./../address"; -import { InternalError } from "./../errors"; -import { Wallet as WalletModel, Address as AddressModel } from "../../client"; -import { ApiClients } from "./../types"; -import { ethers } from "ethers"; +import { Address as AddressModel, Wallet as WalletModel } from "../../client"; import { Wallet } from "../wallet"; -import MockAdapter from "axios-mock-adapter"; -import axios from "axios"; +import { ApiClients } from "./../types"; // Mock data const VALID_WALLET_MODEL: WalletModel = { @@ -51,7 +45,7 @@ describe("Wallet", () => { }); it("should return the correct string representation", () => { - const wallet = new Wallet(VALID_WALLET_MODEL, mockClient, seed, 0); + const wallet = new Wallet(VALID_WALLET_MODEL, mockClient); expect(wallet.toString()).toBe( "Wallet{id: 'mocked_wallet_id', network_id: 'mocked_network_id'}", ); @@ -62,10 +56,7 @@ describe("Wallet", () => { "Address client cannot be empty", ); }); - - it("should generate and derive addresses based on the provided count", () => { - const addressCount = 3; - const wallet = new Wallet(VALID_WALLET_MODEL, mockClient, seed, addressCount); - expect(wallet["addresses"].length).toBe(addressCount); + it("should throw an InternalError if address client is not provided", () => { + expect(() => new Wallet(undefined!, mockClient)).toThrow("Wallet model cannot be empty"); }); }); diff --git a/src/coinbase/wallet.ts b/src/coinbase/wallet.ts index 9ff83548..f227cad0 100644 --- a/src/coinbase/wallet.ts +++ b/src/coinbase/wallet.ts @@ -1,13 +1,16 @@ import { HDKey } from "@scure/bip32"; import * as bip39 from "bip39"; +import { ethers } from "ethers"; +import { Address as AddressModel, Wallet as WalletModel } from "../client"; import { Address } from "./address"; import { InternalError } from "./errors"; -import { Wallet as WalletModel, Address as AddressModel } from "../client"; import { ApiClients } from "./types"; -import { ethers } from "ethers"; /** - * Wallet class represents a wallet with address management. + * A representation of a Wallet. Wallets come with a single default Address, but can expand to have a set of Addresses, + * each of which can hold a balance of one or more Assets. Wallets can create new Addresses, list their addresses, + * list their balances, and transfer Assets to other Addresses. Wallets should be created through User.createWallet or + * User.importWallet. */ export class Wallet { private model: WalletModel; @@ -19,10 +22,10 @@ export class Wallet { private addressIndex = 0; /** - * Creates an instance of Wallet. + * Returns a new Wallet object. Do not use this method directly. Instead, use User.createWallet or User.importWallet. * @param {WalletModel} model - The wallet model data. - * @param {ApiClients} client - The API client to interact with address-related endpoints. - * @param {string} seed - The seed to generate the wallet. + * @param {ApiClients} client - The API client to interact with the server. + * @param {string} seed - The seed to use for the Wallet. Expects a 32-byte hexadecimal with no 0x prefix. If not provided, a new seed will be generated. * @param {number} addressCount - The number of addresses to generate. * @throws {InternalError} If the model or client is empty. */ @@ -60,16 +63,8 @@ export class Wallet { * @throws {InternalError} If address derivation or caching fails. */ private deriveAddress() { - try { - const key = this.deriveKey(); - const address = new ethers.Wallet(Buffer.from(key?.privateKey || "").toString("hex")).address; - if (!this.model.id) { - throw new InternalError("Wallet ID could not found"); - } - } catch (e) { - console.error("Error deriving address:", e); - throw new InternalError("Failed to derive address"); - } + const key = this.deriveKey(); + const address = new ethers.Wallet(Buffer.from(key?.privateKey || "").toString("hex")).address; } /** @@ -104,13 +99,9 @@ export class Wallet { * @returns The default address. */ public defaultAddress(): Address | undefined { - if (!this.model.default_address) { - return undefined; - } - if (!this.client.address) { - throw new InternalError("Address client cannot be empty"); - } - return new Address(this.model.default_address, this.client.address); + return this.model.default_address + ? new Address(this.model.default_address, this.client.address!) + : undefined; } /** From e732b7f5492b9af9261db43f2c967c4219272147 Mon Sep 17 00:00:00 2001 From: Erdi Maden Date: Wed, 15 May 2024 21:21:42 -0500 Subject: [PATCH 04/10] - Updating JSDoc with the latest eslint rules - new eslint rules for JSDocs - base_sepolia usage replaced with Coinbase.networkList.BaseSepolia - Adding base Wallet class and test cases --- .eslintrc.json | 34 +++++++- package.json | 5 +- src/coinbase/address.ts | 7 ++ src/coinbase/api_error.ts | 6 +- src/coinbase/authenticator.ts | 10 ++- src/coinbase/coinbase.ts | 30 +++++-- src/coinbase/errors.ts | 24 +++++- src/coinbase/faucet_transaction.ts | 6 +- src/coinbase/tests/address_test.ts | 7 +- src/coinbase/tests/wallet_test.ts | 101 +++++++++++++----------- src/coinbase/types.ts | 59 +++++++------- src/coinbase/user.ts | 3 + src/coinbase/utils.ts | 15 ++++ src/coinbase/wallet.ts | 121 ++++++++++++++++++++--------- yarn.lock | 96 ++++++++++++++++++++++- 15 files changed, 393 insertions(+), 131 deletions(-) diff --git a/.eslintrc.json b/.eslintrc.json index 05df6a1d..2c50d540 100644 --- a/.eslintrc.json +++ b/.eslintrc.json @@ -3,7 +3,8 @@ "extends": [ "eslint:recommended", "plugin:@typescript-eslint/recommended", - "plugin:prettier/recommended" + "plugin:prettier/recommended", + "plugin:jsdoc/recommended" ], "plugins": ["@typescript-eslint", "prettier"], "env": { @@ -16,7 +17,34 @@ }, "rules": { "multiline-comment-style": ["error", "starred-block"], - "prettier/prettier": "error" + "prettier/prettier": "error", + "jsdoc/tag-lines": ["error", "any", { "startLines": 1 }], + "jsdoc/check-alignment": "error", + "jsdoc/no-undefined-types": "off", + "jsdoc/check-param-names": "error", + "jsdoc/check-tag-names": "error", + "jsdoc/check-types": "error", + "jsdoc/implements-on-classes": "error", + "jsdoc/require-description": "error", + "jsdoc/require-jsdoc": [ + "error", + { + "require": { + "FunctionDeclaration": true, + "MethodDefinition": true, + "ClassDeclaration": true, + "ArrowFunctionExpression": false, + "FunctionExpression": false + } + } + ], + "jsdoc/require-param": "error", + "jsdoc/require-param-description": "error", + "jsdoc/require-param-type": "off", + "jsdoc/require-returns": "error", + "jsdoc/require-returns-description": "error", + "jsdoc/require-returns-type": "off", + "jsdoc/require-hyphen-before-param-description": ["error", "always"] }, - "ignorePatterns": ["src/**/__tests__/**", "src/**/*.test.ts"] + "ignorePatterns": ["src/**/__tests__/**", "src/**/*.test.ts", "src/client/**"] } diff --git a/package.json b/package.json index 3f7dc00c..b0cc16f2 100644 --- a/package.json +++ b/package.json @@ -7,8 +7,8 @@ "main": "dist/index.js", "types": "dist/index.d.ts", "scripts": { - "lint": "eslint -c .eslintrc.json src/**/*.ts", - "lint-fix": "eslint -c .eslintrc.json src/**/*.ts --fix", + "lint": "eslint -c .eslintrc.json src/coinbase/*.ts", + "lint-fix": "eslint -c .eslintrc.json src/coinbase/*.ts --fix", "format": "prettier -c .prettierrc --write \"**/*.{ts,js,cjs,json,md}\"", "format-check": "prettier -c .prettierrc --check \"**/*.{ts,js,cjs,json,md}\"", "check": "tsc --noEmit", @@ -39,6 +39,7 @@ "axios-mock-adapter": "^1.22.0", "eslint": "^8.57.0", "eslint-config-prettier": "^9.1.0", + "eslint-plugin-jsdoc": "^48.2.5", "eslint-plugin-prettier": "^5.1.3", "jest": "^29.7.0", "mock-fs": "^5.2.0", diff --git a/src/coinbase/address.ts b/src/coinbase/address.ts index 524c3fb8..af080f41 100644 --- a/src/coinbase/address.ts +++ b/src/coinbase/address.ts @@ -12,6 +12,7 @@ export class Address { /** * Initializes a new Address instance. + * * @param {AddressModel} model - The address model data. * @param {AddressAPIClient} client - The API client to interact with address-related endpoints. * @throws {InternalError} If the model or client is empty. @@ -30,6 +31,7 @@ export class Address { /** * Requests faucet funds for the address. * Only supported on testnet networks. + * * @returns {Promise} The faucet transaction object. * @throws {InternalError} If the request does not return a transaction hash. * @throws {Error} If the request fails. @@ -44,6 +46,7 @@ export class Address { /** * Returns the address ID. + * * @returns {string} The address ID. */ public getId(): string { @@ -52,6 +55,7 @@ export class Address { /** * Returns the network ID. + * * @returns {string} The network ID. */ public getNetworkId(): string { @@ -60,6 +64,7 @@ export class Address { /** * Returns the public key. + * * @returns {string} The public key. */ public getPublicKey(): string { @@ -68,6 +73,7 @@ export class Address { /** * Returns the wallet ID. + * * @returns {string} The wallet ID. */ public getWalletId(): string { @@ -76,6 +82,7 @@ export class Address { /** * Returns a string representation of the address. + * * @returns {string} A string representing the address. */ public toString(): string { diff --git a/src/coinbase/api_error.ts b/src/coinbase/api_error.ts index 96d2683a..7fc89e2a 100644 --- a/src/coinbase/api_error.ts +++ b/src/coinbase/api_error.ts @@ -1,3 +1,4 @@ +/* eslint-disable jsdoc/require-jsdoc */ import { AxiosError } from "axios"; import { InternalError } from "./errors"; @@ -19,7 +20,8 @@ export class APIError extends AxiosError { /** * Initializes a new APIError object. - * @constructor + * + * @class * @param {AxiosError} error - The Axios error. */ constructor(error) { @@ -38,6 +40,7 @@ export class APIError extends AxiosError { /** * Creates a specific APIError based on the API error code. + * * @param {AxiosError} error - The underlying error object. * @returns {APIError} A specific APIError instance. */ @@ -100,6 +103,7 @@ export class APIError extends AxiosError { /** * Returns a String representation of the APIError. + * * @returns {string} a String representation of the APIError */ toString() { diff --git a/src/coinbase/authenticator.ts b/src/coinbase/authenticator.ts index 83dfcee5..525f9f1a 100644 --- a/src/coinbase/authenticator.ts +++ b/src/coinbase/authenticator.ts @@ -5,14 +5,16 @@ import { InvalidAPIKeyFormat } from "./errors"; const pemHeader = "-----BEGIN EC PRIVATE KEY-----"; const pemFooter = "-----END EC PRIVATE KEY-----"; -/* A class that builds JWTs for authenticating with the Coinbase Platform APIs. */ +/** + * A class that builds JWTs for authenticating with the Coinbase Platform APIs. + */ export class CoinbaseAuthenticator { private apiKey: string; private privateKey: string; /** * Initializes the Authenticator. - * @constructor + * * @param {string} apiKey - The API key name. * @param {string} privateKey - The private key associated with the API key. */ @@ -23,6 +25,7 @@ export class CoinbaseAuthenticator { /** * Middleware to intercept requests and add JWT to Authorization header. + * * @param {InternalAxiosRequestConfig} config - The request configuration. * @param {boolean} debugging - Flag to enable debugging. * @returns {Promise} The request configuration with the Authorization header added. @@ -44,6 +47,7 @@ export class CoinbaseAuthenticator { /** * Builds the JWT for the given API endpoint URL. + * * @param {string} url - URL of the API endpoint. * @param {string} method - HTTP method of the request. * @returns {Promise} JWT token. @@ -93,6 +97,7 @@ export class CoinbaseAuthenticator { /** * Extracts the PEM key from the given private key string. + * * @param {string} privateKeyString - The private key string. * @returns {string} The PEM key. * @throws {InvalidAPIKeyFormat} If the private key string is not in the correct format. @@ -109,6 +114,7 @@ export class CoinbaseAuthenticator { /** * Generates a random nonce for the JWT. + * * @returns {string} The generated nonce. */ private nonce(): string { diff --git a/src/coinbase/coinbase.ts b/src/coinbase/coinbase.ts index 00268f59..65da17f3 100644 --- a/src/coinbase/coinbase.ts +++ b/src/coinbase/coinbase.ts @@ -9,17 +9,29 @@ import { ApiClients } from "./types"; import { User } from "./user"; import { logApiResponse, registerAxiosInterceptors } from "./utils"; -// The Coinbase SDK. +/** + * The Coinbase SDK. + */ export class Coinbase { + /** + * The list of supported networks. + * + * @constant + */ + static networkList = { + BaseSepolia: "base_sepolia", + }; + apiClients: ApiClients = {}; /** * Initializes the Coinbase SDK. - * @constructor - * @param {string} apiKeyName - The API key name. - * @param {string} privateKey - The private key associated with the API key. - * @param {boolean} debugging - If true, logs API requests and responses to the console. - * @param {string} basePath - The base path for the API. + * + * @class + * @param apiKeyName - The API key name. + * @param privateKey - The private key associated with the API key. + * @param debugging - If true, logs API requests and responses to the console. + * @param basePath - The base path for the API. * @throws {InternalError} If the configuration is invalid. * @throws {InvalidAPIKeyFormat} If not able to create JWT token. */ @@ -51,7 +63,10 @@ export class Coinbase { /** * Reads the API key and private key from a JSON file and initializes the Coinbase SDK. - * @param {string} filePath - The path to the JSON file containing the API key and private key. + * + * @param filePath - The path to the JSON file containing the API key and private key. + * @param debugging - If true, logs API requests and responses to the console. + * @param basePath - The base path for the API. * @returns {Coinbase} A new instance of the Coinbase SDK. * @throws {InvalidAPIKeyFormat} If the file does not exist or the configuration values are missing/invalid. * @throws {InvalidConfiguration} If the configuration is invalid. @@ -86,6 +101,7 @@ export class Coinbase { /** * Returns User object for the default user. + * * @returns {User} The default user. * @throws {APIError} If the request fails. */ diff --git a/src/coinbase/errors.ts b/src/coinbase/errors.ts index 94d50f0a..af0431e6 100644 --- a/src/coinbase/errors.ts +++ b/src/coinbase/errors.ts @@ -1,12 +1,12 @@ /** * InvalidaAPIKeyFormat error is thrown when the API key format is invalid. - * @extends {Error} */ export class InvalidAPIKeyFormat extends Error { static DEFAULT_MESSAGE = "Invalid API key format"; /** * Initializes a new InvalidAPIKeyFormat instance. + * * @param message - The error message. */ constructor(message: string = InvalidAPIKeyFormat.DEFAULT_MESSAGE) { @@ -18,6 +18,26 @@ export class InvalidAPIKeyFormat extends Error { } } +/** + * ArgumentError is thrown when an argument is invalid. + */ +export class ArgumentError extends Error { + static DEFAULT_MESSAGE = "Argument Error"; + + /** + * Initializes a new ArgumentError instance. + * + * @param message - The error message. + */ + constructor(message: string = ArgumentError.DEFAULT_MESSAGE) { + super(message); + this.name = "ArgumentError"; + if (Error.captureStackTrace) { + Error.captureStackTrace(this, ArgumentError); + } + } +} + /** * InternalError is thrown when there is an internal error in the SDK. */ @@ -26,6 +46,7 @@ export class InternalError extends Error { /** * Initializes a new InternalError instance. + * * @param message - The error message. */ constructor(message: string = InternalError.DEFAULT_MESSAGE) { @@ -45,6 +66,7 @@ export class InvalidConfiguration extends Error { /** * Initializes a new InvalidConfiguration instance. + * * @param message - The error message. */ constructor(message: string = InvalidConfiguration.DEFAULT_MESSAGE) { diff --git a/src/coinbase/faucet_transaction.ts b/src/coinbase/faucet_transaction.ts index 877162b3..8ef0e226 100644 --- a/src/coinbase/faucet_transaction.ts +++ b/src/coinbase/faucet_transaction.ts @@ -10,7 +10,8 @@ export class FaucetTransaction { /** * Creates a new FaucetTransaction instance. * Do not use this method directly - instead, use Address.faucet(). - * @constructor + * + * @class * @param {FaucetTransactionModel} model - The FaucetTransaction model. * @throws {InternalError} If the model does not exist. */ @@ -23,6 +24,7 @@ export class FaucetTransaction { /** * Returns the transaction hash. + * * @returns {string} The transaction hash. */ public getTransactionHash(): string { @@ -31,6 +33,7 @@ export class FaucetTransaction { /** * Returns the link to the transaction on the blockchain explorer. + * * @returns {string} The link to the transaction on the blockchain explorer */ public getTransactionLink(): string { @@ -40,6 +43,7 @@ export class FaucetTransaction { /** * Returns a string representation of the FaucetTransaction. + * * @returns {string} A string representation of the FaucetTransaction. */ public toString(): string { diff --git a/src/coinbase/tests/address_test.ts b/src/coinbase/tests/address_test.ts index 34c1426e..b817e266 100644 --- a/src/coinbase/tests/address_test.ts +++ b/src/coinbase/tests/address_test.ts @@ -6,14 +6,15 @@ import { FaucetTransaction } from "./../faucet_transaction"; import MockAdapter from "axios-mock-adapter"; import { randomUUID } from "crypto"; import { APIError, FaucetLimitReachedError } from "../api_error"; -import { createAxiosMock } from "./utils"; +import { Coinbase } from "../coinbase"; import { InternalError } from "../errors"; +import { createAxiosMock } from "./utils"; const newEthAddress = ethers.Wallet.createRandom(); const VALID_ADDRESS_MODEL: AddressModel = { address_id: newEthAddress.address, - network_id: "base-sepolia", + network_id: Coinbase.networkList.BaseSepolia, public_key: newEthAddress.publicKey, wallet_id: randomUUID(), }; @@ -100,4 +101,4 @@ describe("Address", () => { `Coinbase:Address{addressId: '${VALID_ADDRESS_MODEL.address_id}', networkId: '${VALID_ADDRESS_MODEL.network_id}', walletId: '${VALID_ADDRESS_MODEL.wallet_id}'}`, ); }); -}); \ No newline at end of file +}); diff --git a/src/coinbase/tests/wallet_test.ts b/src/coinbase/tests/wallet_test.ts index 7d6250aa..8d8f9add 100644 --- a/src/coinbase/tests/wallet_test.ts +++ b/src/coinbase/tests/wallet_test.ts @@ -1,62 +1,75 @@ +import MockAdapter from "axios-mock-adapter"; import * as bip39 from "bip39"; -import { Address as AddressModel, Wallet as WalletModel } from "../../client"; +import { randomUUID } from "crypto"; +import { AddressesApiFactory, WalletsApiFactory } from "../../client"; +import { Coinbase } from "../coinbase"; +import { ArgumentError } from "../errors"; import { Wallet } from "../wallet"; -import { ApiClients } from "./../types"; +import { createAxiosMock } from "./utils"; -// Mock data -const VALID_WALLET_MODEL: WalletModel = { - id: "mocked_wallet_id", - network_id: "mocked_network_id", +const walletId = randomUUID(); +const VALID_WALLET_MODEL = { + id: randomUUID(), + network_id: Coinbase.networkList.BaseSepolia, default_address: { - address_id: "mocked_address_id", - network_id: "mocked_network_id", - public_key: "mocked_public_key", - wallet_id: "mocked_wallet_id", + wallet_id: walletId, + address_id: "0xdeadbeef", + public_key: "0x1234567890", + network_id: Coinbase.networkList.BaseSepolia, }, }; -const VALID_ADDRESS_MODEL: AddressModel = { - address_id: "mocked_address_id", - network_id: "mocked_network_id", - public_key: "mocked_public_key", - wallet_id: "mocked_wallet_id", -}; +describe("Wallet Class", () => { + let wallet, axiosMock; + const seed = bip39.generateMnemonic(); -// Mock ApiClients -const mockClient: ApiClients = { - address: { - getAddress: jest.fn().mockResolvedValue({ data: VALID_ADDRESS_MODEL }), - requestFaucetFunds: jest.fn(), - }, - user: { - getCurrentUser: jest.fn(), - }, -}; + const [axiosInstance, configuration, BASE_PATH] = createAxiosMock(); + const client = { + wallet: WalletsApiFactory(configuration, BASE_PATH, axiosInstance), + address: AddressesApiFactory(configuration, BASE_PATH, axiosInstance), + }; -// Test suite for Wallet class -describe("Wallet", () => { - const seed = bip39.generateMnemonic(); - it("should create a Wallet instance", () => { - const wallet = new Wallet(VALID_WALLET_MODEL, mockClient, seed, 1); - expect(wallet).toBeInstanceOf(Wallet); - expect(wallet.getId()).toBe("mocked_wallet_id"); - expect(wallet.getNetworkId()).toBe("mocked_network_id"); - expect(wallet.defaultAddress()?.getId()).toBe("mocked_address_id"); + beforeAll(async () => { + axiosMock = new MockAdapter(axiosInstance); + axiosMock.onPost().reply(200, VALID_WALLET_MODEL).onGet().reply(200, VALID_WALLET_MODEL); + wallet = await Wallet.init(VALID_WALLET_MODEL, client, seed, 2); }); - it("should return the correct string representation", () => { - const wallet = new Wallet(VALID_WALLET_MODEL, mockClient); + afterEach(() => { + axiosMock.reset(); + }); + + describe("should initializes a new Wallet", () => { + it("should return a Wallet instance", async () => { + expect(wallet).toBeInstanceOf(Wallet); + }); + it("should return the correct wallet ID", async () => { + expect(wallet.getId()).toBe(VALID_WALLET_MODEL.id); + }); + it("should return the correct network ID", async () => { + expect(wallet.getNetworkId()).toBe(Coinbase.networkList.BaseSepolia); + }); + it("should return the correct default address", async () => { + expect(wallet.defaultAddress()?.getId()).toBe(VALID_WALLET_MODEL.default_address.address_id); + }); + + it("should derive the correct number of addresses", async () => { + expect(wallet.addresses.length).toBe(2); + }); + }); + + it("should return the correct string representation", async () => { + const wallet = await Wallet.init(VALID_WALLET_MODEL, client); expect(wallet.toString()).toBe( - "Wallet{id: 'mocked_wallet_id', network_id: 'mocked_network_id'}", + `Wallet{id: '${VALID_WALLET_MODEL.id}', network_id: 'base_sepolia'}`, ); }); - it("should throw an InternalError if address client is not provided", () => { - expect(() => new Wallet(VALID_WALLET_MODEL, undefined!)).toThrow( - "Address client cannot be empty", - ); + it("should throw an ArgumentError when the API client is not provided", async () => { + await expect(Wallet.init(VALID_WALLET_MODEL, undefined!)).rejects.toThrow(ArgumentError); }); - it("should throw an InternalError if address client is not provided", () => { - expect(() => new Wallet(undefined!, mockClient)).toThrow("Wallet model cannot be empty"); + + it("should throw an ArgumentError when the wallet model is not provided", async () => { + await expect(Wallet.init(undefined!, client)).rejects.toThrow(ArgumentError); }); }); diff --git a/src/coinbase/types.ts b/src/coinbase/types.ts index 1525febd..f8d53d82 100644 --- a/src/coinbase/types.ts +++ b/src/coinbase/types.ts @@ -1,13 +1,36 @@ import { AxiosPromise, AxiosRequestConfig, RawAxiosRequestConfig } from "axios"; -import { Address, User as UserModel } from "./../client/api"; -import { RequestArgs } from "../client/base"; +import { + Address, + CreateWalletRequest, + User as UserModel, + Wallet as WalletModel, +} from "./../client/api"; + +/** + * WalletAPI client type definition. + */ +export type WalletAPIClient = { + /** + * Create a new wallet scoped to the user. + * + * @class + * @param {CreateWalletRequest} [createWalletRequest] - The wallet creation request. + * @param {RawAxiosRequestConfig} [options] - Axios request options. + * @throws {RequiredError} + */ + createWallet: ( + createWalletRequest?: CreateWalletRequest, + options?: RawAxiosRequestConfig, + ) => AxiosPromise; +}; /** * AddressAPI client type definition. */ -export type AddressClient = { +export type AddressAPIClient = { /** * Requests faucet funds for the address. + * * @param {string} walletId - The wallet ID. * @param {string} addressId - The address ID. * @returns {Promise<{ data: { transaction_hash: string } }>} - The transaction hash @@ -20,9 +43,10 @@ export type AddressClient = { /** * Get address + * * @summary Get address by onchain address - * @param {string} walletId The ID of the wallet the address belongs to. - * @param {string} addressId The onchain address of the address that is being fetched. + * @param {string} walletId - The ID of the wallet the address belongs to. + * @param {string} addressId - The onchain address of the address that is being fetched. * @param {AxiosRequestConfig} [options] - Axios request options. * @throws {RequiredError} */ @@ -33,29 +57,13 @@ export type AddressClient = { ): AxiosPromise
; }; -/** - * AddressAPI client type definition. - */ -export type AddressAPIClient = { - /** - * Requests faucet funds for the address. - * @param {string} walletId - The wallet ID. - * @param {string} addressId - The address ID. - * @returns {Promise<{ data: { transaction_hash: string } }>} - The transaction hash - * @throws {AxiosError} If the request fails. - */ - requestFaucetFunds( - walletId: string, - addressId: string, - ): Promise<{ data: { transaction_hash: string } }>; -}; - /** * UserAPI client type definition. */ export type UserAPIClient = { /** * Retrieves the current user. + * * @param {AxiosRequestConfig} [options] - Axios request options. * @returns {AxiosPromise} - A promise resolving to the User model. * @throws {AxiosError} If the request fails. @@ -68,10 +76,7 @@ export type UserAPIClient = { * Represents the set of API clients available in the SDK. */ export type ApiClients = { - /** - * The User API client. - * @type {UserAPIClient} - */ user?: UserAPIClient; - address?: AddressClient; + address?: AddressAPIClient; + wallet?: WalletAPIClient; }; diff --git a/src/coinbase/user.ts b/src/coinbase/user.ts index 59d063f1..c5fab06c 100644 --- a/src/coinbase/user.ts +++ b/src/coinbase/user.ts @@ -11,6 +11,7 @@ export class User { /** * Initializes a new User instance. + * * @param {UserModel} user - The user model. * @param {ApiClients} client - The API clients. */ @@ -21,6 +22,7 @@ export class User { /** * Returns the user's ID. + * * @returns {string} The user's ID. */ public getId(): string { @@ -29,6 +31,7 @@ export class User { /** * Returns a string representation of the User. + * * @returns {string} The string representation of the User. */ toString(): string { diff --git a/src/coinbase/utils.ts b/src/coinbase/utils.ts index 8e34b314..98f2fa04 100644 --- a/src/coinbase/utils.ts +++ b/src/coinbase/utils.ts @@ -4,8 +4,10 @@ import { APIError } from "./api_error"; /** * Prints Axios response to the console for debugging purposes. + * * @param response - The Axios response object. * @param debugging - Flag to enable or disable logging. + * @returns The Axios response object. */ export const logApiResponse = (response: AxiosResponse, debugging = false): AxiosResponse => { if (debugging) { @@ -25,6 +27,7 @@ export const logApiResponse = (response: AxiosResponse, debugging = false): Axio /** * Axios Request interceptor function type. + * * @param {InternalAxiosRequestConfig} value - The Axios request configuration. * @returns {InternalAxiosRequestConfig} The modified Axios request configuration. */ @@ -34,12 +37,14 @@ type RequestFunctionType = ( /** * Axios Response interceptor function type. + * * @param {AxiosResponse} value - The Axios response object. * @returns {AxiosResponse} The modified Axios response object. */ type ResponseFunctionType = (value: AxiosResponse) => AxiosResponse; /** + * Registers request and response interceptors to an Axios instance. * * @param {Axios} axiosInstance - The Axios instance to register the interceptors. * @param {RequestFunctionType} requestFn - The request interceptor function. @@ -55,3 +60,13 @@ export const registerAxiosInterceptors = ( return Promise.reject(APIError.fromError(error)); }); }; + +/** + * Converts a Uint8Array to a hex string. + * + * @param {Uint8Array} key - The key to convert. + * @returns {string} The converted hex string. + */ +export const convertStringToHex = (key: Uint8Array): string => { + return Buffer.from(key).toString("hex"); +}; diff --git a/src/coinbase/wallet.ts b/src/coinbase/wallet.ts index f227cad0..b43be90e 100644 --- a/src/coinbase/wallet.ts +++ b/src/coinbase/wallet.ts @@ -1,75 +1,118 @@ import { HDKey } from "@scure/bip32"; import * as bip39 from "bip39"; -import { ethers } from "ethers"; +import { ethers, Wallet as ETHWallet } from "ethers"; import { Address as AddressModel, Wallet as WalletModel } from "../client"; import { Address } from "./address"; -import { InternalError } from "./errors"; -import { ApiClients } from "./types"; +import { ArgumentError, InternalError } from "./errors"; +import { AddressAPIClient, WalletAPIClient } from "./types"; +import { convertStringToHex } from "./utils"; + +/** + * The Wallet API client types. + */ +type WalletClients = { + wallet: WalletAPIClient; + address: AddressAPIClient; +}; /** * A representation of a Wallet. Wallets come with a single default Address, but can expand to have a set of Addresses, * each of which can hold a balance of one or more Assets. Wallets can create new Addresses, list their addresses, - * list their balances, and transfer Assets to other Addresses. Wallets should be created through User.createWallet or - * User.importWallet. + * list their balances, and transfer Assets to other Addresses. Wallets should be created through User.createWallet or User.importWallet. */ export class Wallet { private model: WalletModel; - private client: ApiClients; + private client: WalletClients; private master: HDKey; private addresses: Address[] = []; private readonly addressPathPrefix = "m/44'/60'/0'/0"; private addressIndex = 0; + /** + * Private constructor to prevent direct instantiation outside of factory method. Use Wallet.init instead. + * + * @ignore + * @param model - The wallet model object. + * @param client - The API client to interact with the server. + * @param master - The HD master key. + * @hideconstructor + */ + private constructor(model: WalletModel, client: WalletClients, master: HDKey) { + this.model = model; + this.client = client; + this.master = master; + } + /** * Returns a new Wallet object. Do not use this method directly. Instead, use User.createWallet or User.importWallet. - * @param {WalletModel} model - The wallet model data. - * @param {ApiClients} client - The API client to interact with the server. - * @param {string} seed - The seed to use for the Wallet. Expects a 32-byte hexadecimal with no 0x prefix. If not provided, a new seed will be generated. - * @param {number} addressCount - The number of addresses to generate. - * @throws {InternalError} If the model or client is empty. + * + * @constructs Wallet + * @param model - The underlying Wallet model object + * @param client - The API client to interact with the server. + * @param seed - The seed to use for the Wallet. Expects a 32-byte hexadecimal with no 0x prefix. If not provided, a new seed will be generated. + * @param addressCount - The number of addresses already registered for the Wallet. + * @throws {ArgumentError} If the model or client is not provided. + * @throws {InternalError} - If address derivation or caching fails. + * @throws {APIError} - If the request fails. + * @returns A promise that resolves with the new Wallet object. */ - constructor(model: WalletModel, client: ApiClients, seed: string = "", addressCount: number = 0) { + public static async init( + model: WalletModel, + client: WalletClients, + seed: string = "", + addressCount: number = 0, + ): Promise { if (!model) { - throw new InternalError("Wallet model cannot be empty"); + throw new ArgumentError("Wallet model cannot be empty"); } - if (!client?.address || !client?.user) { - throw new InternalError("Address client cannot be empty"); + if (!client?.address || !client?.wallet) { + throw new ArgumentError("Address client cannot be empty"); } - this.model = model; - this.client = client; if (!seed) { seed = bip39.generateMnemonic(); } + const master = HDKey.fromMasterSeed(bip39.mnemonicToSeedSync(seed)); + const wallet = new Wallet(model, client, master); - this.master = HDKey.fromMasterSeed(bip39.mnemonicToSeedSync(seed)); for (let i = 0; i < addressCount; i++) { - this.deriveAddress(); + await wallet.deriveAddress(); } + + return wallet; } /** - * Derives a new HDKey for the next address. - * @returns The derived HDKey. + * Derives a key for an already registered Address in the Wallet. + * + * @returns The derived key. */ - private deriveKey(): HDKey { - return this.master.deriveChild(this.addressIndex++); + private deriveKey(): ETHWallet { + const derivedKey = this.master.derive(`${this.addressPathPrefix}/${this.addressIndex++}`); + if (!derivedKey?.privateKey) { + throw new InternalError("Failed to derive key"); + } + return new ethers.Wallet(convertStringToHex(derivedKey.privateKey)); } /** - * Derives a new address and caches it. - * @returns {Promise} - * @throws {InternalError} If address derivation or caching fails. + * Derives an already registered Address in the Wallet. + * + * @throws {InternalError} - If address derivation or caching fails. + * @throws {APIError} - If the request fails. + * @returns {Promise} A promise that resolves when the address is derived. */ - private deriveAddress() { + private async deriveAddress(): Promise { const key = this.deriveKey(); - const address = new ethers.Wallet(Buffer.from(key?.privateKey || "").toString("hex")).address; + const response = await this.client.address.getAddress(this.model.id!, key.address); + this.cacheAddress(response.data); } /** * Caches an Address on the client-side and increments the address index. - * @param {AddressModel} address - The address to cache. + * + * @param address - The AddressModel to cache. * @throws {InternalError} If the address is not provided. * @returns {void} */ @@ -79,24 +122,27 @@ export class Wallet { } /** - * Gets the network identifier. - * @returns The network identifier. + * Returns the Network ID of the Wallet. + * + * @returns The network ID. */ public getNetworkId(): string { return this.model.network_id; } /** - * Gets the wallet identifier. - * @returns The wallet identifier. + * Returns the wallet ID. + * + * @returns The wallet ID. */ public getId(): string | undefined { return this.model.id; } /** - * Gets the default address of the wallet. - * @returns The default address. + * Returns the default address of the Wallet. + * + * @returns The default address */ public defaultAddress(): Address | undefined { return this.model.default_address @@ -105,8 +151,9 @@ export class Wallet { } /** - * Converts the wallet instance to a string representation. - * @returns The string representation of the wallet. + * Returns a String representation of the Wallet. + * + * @returns a String representation of the Wallet */ public toString(): string { return `Wallet{id: '${this.model.id}', network_id: '${this.model.network_id}'}`; diff --git a/yarn.lock b/yarn.lock index 2857b342..9428b2ad 100644 --- a/yarn.lock +++ b/yarn.lock @@ -310,6 +310,18 @@ dependencies: "@jridgewell/trace-mapping" "0.3.9" +"@es-joy/jsdoccomment@~0.43.0": + version "0.43.0" + resolved "https://registry.yarnpkg.com/@es-joy/jsdoccomment/-/jsdoccomment-0.43.0.tgz#35c295cadd0a939d1a3a6cd1548f66ec76d38870" + integrity sha512-Q1CnsQrytI3TlCB1IVWXWeqUIPGVEKGaE7IbVdt13Nq/3i0JESAkQQERrfiQkmlpijl+++qyqPgaS31Bvc1jRQ== + dependencies: + "@types/eslint" "^8.56.5" + "@types/estree" "^1.0.5" + "@typescript-eslint/types" "^7.2.0" + comment-parser "1.4.1" + esquery "^1.5.0" + jsdoc-type-pratt-parser "~4.0.0" + "@eslint-community/eslint-utils@^4.2.0", "@eslint-community/eslint-utils@^4.4.0": version "4.4.0" resolved "https://registry.yarnpkg.com/@eslint-community/eslint-utils/-/eslint-utils-4.4.0.tgz#a23514e8fb9af1269d5f7788aa556798d61c6b59" @@ -745,6 +757,19 @@ dependencies: "@babel/types" "^7.20.7" +"@types/eslint@^8.56.5": + version "8.56.10" + resolved "https://registry.yarnpkg.com/@types/eslint/-/eslint-8.56.10.tgz#eb2370a73bf04a901eeba8f22595c7ee0f7eb58d" + integrity sha512-Shavhk87gCtY2fhXDctcfS3e6FdxWkCx1iUZ9eEUbh7rTqlZT0/IzOkCOVt0fCjcFuZ9FPYfuezTBImfHCDBGQ== + dependencies: + "@types/estree" "*" + "@types/json-schema" "*" + +"@types/estree@*", "@types/estree@^1.0.5": + version "1.0.5" + resolved "https://registry.yarnpkg.com/@types/estree/-/estree-1.0.5.tgz#a6ce3e556e00fd9895dd872dd172ad0d4bd687f4" + integrity sha512-/kYRxGDLWzHOB7q+wtSUQlFrtcdUccpfy+X+9iMBpHK8QLLhx2wIPYuS5DYtR9Wa/YlZAbIovy7qVdB1Aq6Lyw== + "@types/graceful-fs@^4.1.3": version "4.1.9" resolved "https://registry.yarnpkg.com/@types/graceful-fs/-/graceful-fs-4.1.9.tgz#2a06bc0f68a20ab37b3e36aa238be6abdf49e8b4" @@ -779,7 +804,7 @@ expect "^29.0.0" pretty-format "^29.0.0" -"@types/json-schema@^7.0.15": +"@types/json-schema@*", "@types/json-schema@^7.0.15": version "7.0.15" resolved "https://registry.yarnpkg.com/@types/json-schema/-/json-schema-7.0.15.tgz#596a1747233694d50f6ad8a7869fcb6f56cf5841" integrity sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA== @@ -883,6 +908,11 @@ resolved "https://registry.yarnpkg.com/@typescript-eslint/types/-/types-7.8.0.tgz#1fd2577b3ad883b769546e2d1ef379f929a7091d" integrity sha512-wf0peJ+ZGlcH+2ZS23aJbOv+ztjeeP8uQ9GgwMJGVLx/Nj9CJt17GWgWWoSmoRVKAX2X+7fzEnAjxdvK2gqCLw== +"@typescript-eslint/types@^7.2.0": + version "7.9.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/types/-/types-7.9.0.tgz#b58e485e4bfba055659c7e683ad4f5f0821ae2ec" + integrity sha512-oZQD9HEWQanl9UfsbGVcZ2cGaR0YT5476xfWE0oE5kQa2sNK2frxOlkeacLOTh9po4AlUT5rtkGyYM5kew0z5w== + "@typescript-eslint/typescript-estree@7.8.0": version "7.8.0" resolved "https://registry.yarnpkg.com/@typescript-eslint/typescript-estree/-/typescript-estree-7.8.0.tgz#b028a9226860b66e623c1ee55cc2464b95d2987c" @@ -997,6 +1027,11 @@ anymatch@^3.0.3: normalize-path "^3.0.0" picomatch "^2.0.4" +are-docs-informative@^0.0.2: + version "0.0.2" + resolved "https://registry.yarnpkg.com/are-docs-informative/-/are-docs-informative-0.0.2.tgz#387f0e93f5d45280373d387a59d34c96db321963" + integrity sha512-ixiS0nLNNG5jNQzgZJNoUpBKdo9yTYZMGJ+QgT2jmjR7G7+QHRCc4v6LQ3NgE7EBJq+o0ams3waJwkrlBom8Ig== + arg@^4.1.0: version "4.1.3" resolved "https://registry.yarnpkg.com/arg/-/arg-4.1.3.tgz#269fc7ad5b8e42cb63c896d5666017261c144089" @@ -1192,6 +1227,11 @@ buffer@^6.0.3: base64-js "^1.3.1" ieee754 "^1.2.1" +builtin-modules@^3.3.0: + version "3.3.0" + resolved "https://registry.yarnpkg.com/builtin-modules/-/builtin-modules-3.3.0.tgz#cae62812b89801e9656336e46223e030386be7b6" + integrity sha512-zhaCDicdLuWN5UbN5IMnFqNMhNfo919sH85y2/ea+5Yg9TsTkeZxpL+JLbp6cgYFS4sRLp3YV4S6yDuqVWHYOw== + callsites@^3.0.0: version "3.1.0" resolved "https://registry.yarnpkg.com/callsites/-/callsites-3.1.0.tgz#b3630abd8943432f54b3f0519238e33cd7df2f73" @@ -1294,6 +1334,11 @@ combined-stream@^1.0.8: dependencies: delayed-stream "~1.0.0" +comment-parser@1.4.1: + version "1.4.1" + resolved "https://registry.yarnpkg.com/comment-parser/-/comment-parser-1.4.1.tgz#bdafead37961ac079be11eb7ec65c4d021eaf9cc" + integrity sha512-buhp5kePrmda3vhc5B9t7pUQXAb2Tnd0qgpkIhPhkHXxJpiPJ11H0ZEU0oBpJ2QztSbzG/ZxMj/CHsYJqRHmyg== + concat-map@0.0.1: version "0.0.1" resolved "https://registry.yarnpkg.com/concat-map/-/concat-map-0.0.1.tgz#d8a96bd77fd68df7793a73036a3ba0d5405d477b" @@ -1452,6 +1497,21 @@ eslint-config-prettier@^9.1.0: resolved "https://registry.yarnpkg.com/eslint-config-prettier/-/eslint-config-prettier-9.1.0.tgz#31af3d94578645966c082fcb71a5846d3c94867f" integrity sha512-NSWl5BFQWEPi1j4TjVNItzYV7dZXZ+wP6I6ZhrBGpChQhZRUaElihE9uRRkcbRnNb76UMKDF3r+WTmNcGPKsqw== +eslint-plugin-jsdoc@^48.2.5: + version "48.2.5" + resolved "https://registry.yarnpkg.com/eslint-plugin-jsdoc/-/eslint-plugin-jsdoc-48.2.5.tgz#66ec712632852faa15065a094342786858f13c49" + integrity sha512-ZeTfKV474W1N9niWfawpwsXGu+ZoMXu4417eBROX31d7ZuOk8zyG66SO77DpJ2+A9Wa2scw/jRqBPnnQo7VbcQ== + dependencies: + "@es-joy/jsdoccomment" "~0.43.0" + are-docs-informative "^0.0.2" + comment-parser "1.4.1" + debug "^4.3.4" + escape-string-regexp "^4.0.0" + esquery "^1.5.0" + is-builtin-module "^3.2.1" + semver "^7.6.1" + spdx-expression-parse "^4.0.0" + eslint-plugin-prettier@^5.1.3: version "5.1.3" resolved "https://registry.yarnpkg.com/eslint-plugin-prettier/-/eslint-plugin-prettier-5.1.3.tgz#17cfade9e732cef32b5f5be53bd4e07afd8e67e1" @@ -1531,7 +1591,7 @@ esprima@^4.0.0: resolved "https://registry.yarnpkg.com/esprima/-/esprima-4.0.1.tgz#13b04cdb3e6c5d19df91ab6987a8695619b0aa71" integrity sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A== -esquery@^1.4.2: +esquery@^1.4.2, esquery@^1.5.0: version "1.5.0" resolved "https://registry.yarnpkg.com/esquery/-/esquery-1.5.0.tgz#6ce17738de8577694edd7361c57182ac8cb0db0b" integrity sha512-YQLXUplAwJgCydQ78IMJywZCceoqk1oH01OERdSAJc/7U2AylwjhSCLDEtqwg811idIS/9fIU5GjG73IgjKMVg== @@ -1895,6 +1955,13 @@ is-buffer@^2.0.5: resolved "https://registry.yarnpkg.com/is-buffer/-/is-buffer-2.0.5.tgz#ebc252e400d22ff8d77fa09888821a24a658c191" integrity sha512-i2R6zNFDwgEHJyQUtJEk0XFi1i0dPFn/oqjK3/vPCcDeJvW5NQ83V8QbicfF1SupOaB0h8ntgBC2YiE7dfyctQ== +is-builtin-module@^3.2.1: + version "3.2.1" + resolved "https://registry.yarnpkg.com/is-builtin-module/-/is-builtin-module-3.2.1.tgz#f03271717d8654cfcaf07ab0463faa3571581169" + integrity sha512-BSLE3HnV2syZ0FK0iMA/yUGplUeMmNz4AW5fnTunbCIqZi4vG3WjJT9FHMy5D69xmAYBHXQhJdALdpwVxV501A== + dependencies: + builtin-modules "^3.3.0" + is-core-module@^2.13.0: version "2.13.1" resolved "https://registry.yarnpkg.com/is-core-module/-/is-core-module-2.13.1.tgz#ad0d7532c6fea9da1ebdc82742d74525c6273384" @@ -2375,6 +2442,11 @@ js-yaml@^4.1.0: dependencies: argparse "^2.0.1" +jsdoc-type-pratt-parser@~4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/jsdoc-type-pratt-parser/-/jsdoc-type-pratt-parser-4.0.0.tgz#136f0571a99c184d84ec84662c45c29ceff71114" + integrity sha512-YtOli5Cmzy3q4dP26GraSOeAhqecewG04hoO8DY56CH4KJ9Fvv5qKWUCCo3HZob7esJQHCv6/+bnTy72xZZaVQ== + jsesc@^2.5.1: version "2.5.2" resolved "https://registry.yarnpkg.com/jsesc/-/jsesc-2.5.2.tgz#80564d2e483dacf6e8ef209650a67df3f0c283a4" @@ -2897,7 +2969,7 @@ semver@^6.3.0, semver@^6.3.1: resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.1.tgz#556d2ef8689146e46dcea4bfdd095f3434dffcb4" integrity sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA== -semver@^7.5.3, semver@^7.5.4, semver@^7.6.0: +semver@^7.5.3, semver@^7.5.4, semver@^7.6.0, semver@^7.6.1: version "7.6.2" resolved "https://registry.yarnpkg.com/semver/-/semver-7.6.2.tgz#1e3b34759f896e8f14d6134732ce798aeb0c6e13" integrity sha512-FNAIBWCx9qcRhoHcgcJ0gvU7SN1lYU2ZXuSfl04bSC5OpvDHFyJCjdNHomPXxjQlCBU67YW64PzY7/VIEH7F2w== @@ -2952,6 +3024,24 @@ source-map@^0.6.0, source-map@^0.6.1: resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.6.1.tgz#74722af32e9614e9c287a8d0bbde48b5e2f1a263" integrity sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g== +spdx-exceptions@^2.1.0: + version "2.5.0" + resolved "https://registry.yarnpkg.com/spdx-exceptions/-/spdx-exceptions-2.5.0.tgz#5d607d27fc806f66d7b64a766650fa890f04ed66" + integrity sha512-PiU42r+xO4UbUS1buo3LPJkjlO7430Xn5SVAhdpzzsPHsjbYVflnnFdATgabnLude+Cqu25p6N+g2lw/PFsa4w== + +spdx-expression-parse@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/spdx-expression-parse/-/spdx-expression-parse-4.0.0.tgz#a23af9f3132115465dac215c099303e4ceac5794" + integrity sha512-Clya5JIij/7C6bRR22+tnGXbc4VKlibKSVj2iHvVeX5iMW7s1SIQlqu699JkODJJIhh/pUu8L0/VLh8xflD+LQ== + dependencies: + spdx-exceptions "^2.1.0" + spdx-license-ids "^3.0.0" + +spdx-license-ids@^3.0.0: + version "3.0.17" + resolved "https://registry.yarnpkg.com/spdx-license-ids/-/spdx-license-ids-3.0.17.tgz#887da8aa73218e51a1d917502d79863161a93f9c" + integrity sha512-sh8PWc/ftMqAAdFiBu6Fy6JUOYjqDJBJvIhpfDMyHrr0Rbp5liZqd4TjtQ/RgfLjKFZb+LMx5hpml5qOWy0qvg== + sprintf-js@~1.0.2: version "1.0.3" resolved "https://registry.yarnpkg.com/sprintf-js/-/sprintf-js-1.0.3.tgz#04e6926f662895354f3dd015203633b857297e2c" From 264849be11d3df569130ffbbb50b7fa21e2a409c Mon Sep 17 00:00:00 2001 From: Erdi Maden Date: Wed, 15 May 2024 21:44:52 -0500 Subject: [PATCH 05/10] reverting eslint command changes --- .eslintrc.json | 2 +- package.json | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/.eslintrc.json b/.eslintrc.json index 2c50d540..38795cf9 100644 --- a/.eslintrc.json +++ b/.eslintrc.json @@ -46,5 +46,5 @@ "jsdoc/require-returns-type": "off", "jsdoc/require-hyphen-before-param-description": ["error", "always"] }, - "ignorePatterns": ["src/**/__tests__/**", "src/**/*.test.ts", "src/client/**"] + "ignorePatterns": ["src/**/__tests__/**", "src/**/*.test.ts"] } diff --git a/package.json b/package.json index b0cc16f2..6ee4598f 100644 --- a/package.json +++ b/package.json @@ -7,8 +7,8 @@ "main": "dist/index.js", "types": "dist/index.d.ts", "scripts": { - "lint": "eslint -c .eslintrc.json src/coinbase/*.ts", - "lint-fix": "eslint -c .eslintrc.json src/coinbase/*.ts --fix", + "lint": "eslint -c .eslintrc.json src/**/*.ts", + "lint-fix": "eslint -c .eslintrc.json src/**/*.ts --fix", "format": "prettier -c .prettierrc --write \"**/*.{ts,js,cjs,json,md}\"", "format-check": "prettier -c .prettierrc --check \"**/*.{ts,js,cjs,json,md}\"", "check": "tsc --noEmit", From 13e92111d752a255ea5364c1433bb8dcd29ad4c0 Mon Sep 17 00:00:00 2001 From: Erdi Maden Date: Wed, 15 May 2024 23:30:22 -0500 Subject: [PATCH 06/10] Updating typings --- src/coinbase/tests/wallet_test.ts | 2 +- src/coinbase/types.ts | 10 +++++----- src/coinbase/wallet.ts | 2 +- 3 files changed, 7 insertions(+), 7 deletions(-) diff --git a/src/coinbase/tests/wallet_test.ts b/src/coinbase/tests/wallet_test.ts index 8d8f9add..9ea52075 100644 --- a/src/coinbase/tests/wallet_test.ts +++ b/src/coinbase/tests/wallet_test.ts @@ -61,7 +61,7 @@ describe("Wallet Class", () => { it("should return the correct string representation", async () => { const wallet = await Wallet.init(VALID_WALLET_MODEL, client); expect(wallet.toString()).toBe( - `Wallet{id: '${VALID_WALLET_MODEL.id}', network_id: 'base_sepolia'}`, + `Wallet{id: '${VALID_WALLET_MODEL.id}', networkId: 'base_sepolia'}`, ); }); diff --git a/src/coinbase/types.ts b/src/coinbase/types.ts index f8d53d82..76c7954b 100644 --- a/src/coinbase/types.ts +++ b/src/coinbase/types.ts @@ -16,7 +16,7 @@ export type WalletAPIClient = { * @class * @param {CreateWalletRequest} [createWalletRequest] - The wallet creation request. * @param {RawAxiosRequestConfig} [options] - Axios request options. - * @throws {RequiredError} + * @throws {APIError} If the request fails. */ createWallet: ( createWalletRequest?: CreateWalletRequest, @@ -34,7 +34,7 @@ export type AddressAPIClient = { * @param {string} walletId - The wallet ID. * @param {string} addressId - The address ID. * @returns {Promise<{ data: { transaction_hash: string } }>} - The transaction hash - * @throws {Error} If the request fails. + * @throws {APIError} If the request fails. */ requestFaucetFunds( walletId: string, @@ -48,7 +48,7 @@ export type AddressAPIClient = { * @param {string} walletId - The ID of the wallet the address belongs to. * @param {string} addressId - The onchain address of the address that is being fetched. * @param {AxiosRequestConfig} [options] - Axios request options. - * @throws {RequiredError} + * @throws {APIError} If the request fails. */ getAddress( walletId: string, @@ -65,8 +65,8 @@ export type UserAPIClient = { * Retrieves the current user. * * @param {AxiosRequestConfig} [options] - Axios request options. - * @returns {AxiosPromise} - A promise resolving to the User model. - * @throws {AxiosError} If the request fails. + * @returns {AxiosPromise} - A promise resolvindg to the User model. + * @throws {APIError} If the request fails. */ getCurrentUser(options?: AxiosRequestConfig): AxiosPromise; }; diff --git a/src/coinbase/wallet.ts b/src/coinbase/wallet.ts index b43be90e..7ef4e9d9 100644 --- a/src/coinbase/wallet.ts +++ b/src/coinbase/wallet.ts @@ -156,6 +156,6 @@ export class Wallet { * @returns a String representation of the Wallet */ public toString(): string { - return `Wallet{id: '${this.model.id}', network_id: '${this.model.network_id}'}`; + return `Wallet{id: '${this.model.id}', networkId: '${this.model.network_id}'}`; } } From 1d9899c8a771afe461581ddc499c9f45335c99e0 Mon Sep 17 00:00:00 2001 From: Erdi Maden Date: Thu, 16 May 2024 11:19:59 -0500 Subject: [PATCH 07/10] updating types --- src/coinbase/types.ts | 23 +++++++++++------------ 1 file changed, 11 insertions(+), 12 deletions(-) diff --git a/src/coinbase/types.ts b/src/coinbase/types.ts index 76c7954b..123ce3ba 100644 --- a/src/coinbase/types.ts +++ b/src/coinbase/types.ts @@ -14,8 +14,8 @@ export type WalletAPIClient = { * Create a new wallet scoped to the user. * * @class - * @param {CreateWalletRequest} [createWalletRequest] - The wallet creation request. - * @param {RawAxiosRequestConfig} [options] - Axios request options. + * @param createdWalletRequest - The wallet creation request. + * @param options - Axios request options. * @throws {APIError} If the request fails. */ createWallet: ( @@ -31,9 +31,9 @@ export type AddressAPIClient = { /** * Requests faucet funds for the address. * - * @param {string} walletId - The wallet ID. - * @param {string} addressId - The address ID. - * @returns {Promise<{ data: { transaction_hash: string } }>} - The transaction hash + * @param walletId - The wallet ID. + * @param addressId - The address ID. + * @returns The transaction hash * @throws {APIError} If the request fails. */ requestFaucetFunds( @@ -42,12 +42,11 @@ export type AddressAPIClient = { ): Promise<{ data: { transaction_hash: string } }>; /** - * Get address + * Get address by onchain address * - * @summary Get address by onchain address - * @param {string} walletId - The ID of the wallet the address belongs to. - * @param {string} addressId - The onchain address of the address that is being fetched. - * @param {AxiosRequestConfig} [options] - Axios request options. + * @param walletId - The ID of the wallet the address belongs to. + * @param addressId - The onchain address of the address that is being fetched. + * @param options - Axios request options. * @throws {APIError} If the request fails. */ getAddress( @@ -64,8 +63,8 @@ export type UserAPIClient = { /** * Retrieves the current user. * - * @param {AxiosRequestConfig} [options] - Axios request options. - * @returns {AxiosPromise} - A promise resolvindg to the User model. + * @param options - Axios request options. + * @returns - A promise resolvindg to the User model. * @throws {APIError} If the request fails. */ getCurrentUser(options?: AxiosRequestConfig): AxiosPromise; From 7786b27975b7e5e7e316c97de17a23db1ee8ac6b Mon Sep 17 00:00:00 2001 From: Erdi Maden Date: Thu, 16 May 2024 11:23:21 -0500 Subject: [PATCH 08/10] Updating networkId --- src/coinbase/coinbase.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/coinbase/coinbase.ts b/src/coinbase/coinbase.ts index 65da17f3..6f25e92b 100644 --- a/src/coinbase/coinbase.ts +++ b/src/coinbase/coinbase.ts @@ -19,7 +19,7 @@ export class Coinbase { * @constant */ static networkList = { - BaseSepolia: "base_sepolia", + BaseSepolia: "base-sepolia", }; apiClients: ApiClients = {}; From 1f196fc01714a1c548176899f07ab2c824f6059a Mon Sep 17 00:00:00 2001 From: Erdi Maden Date: Thu, 16 May 2024 11:24:53 -0500 Subject: [PATCH 09/10] updating JSDocs --- src/coinbase/utils.ts | 18 +++++++++--------- 1 file changed, 9 insertions(+), 9 deletions(-) diff --git a/src/coinbase/utils.ts b/src/coinbase/utils.ts index 98f2fa04..b45ab341 100644 --- a/src/coinbase/utils.ts +++ b/src/coinbase/utils.ts @@ -28,8 +28,8 @@ export const logApiResponse = (response: AxiosResponse, debugging = false): Axio /** * Axios Request interceptor function type. * - * @param {InternalAxiosRequestConfig} value - The Axios request configuration. - * @returns {InternalAxiosRequestConfig} The modified Axios request configuration. + * @param value - The Axios request configuration. + * @returns The modified Axios request configuration. */ type RequestFunctionType = ( value: InternalAxiosRequestConfig, @@ -38,17 +38,17 @@ type RequestFunctionType = ( /** * Axios Response interceptor function type. * - * @param {AxiosResponse} value - The Axios response object. - * @returns {AxiosResponse} The modified Axios response object. + * @param value - The Axios response object. + * @returns The modified Axios response object. */ type ResponseFunctionType = (value: AxiosResponse) => AxiosResponse; /** * Registers request and response interceptors to an Axios instance. * - * @param {Axios} axiosInstance - The Axios instance to register the interceptors. - * @param {RequestFunctionType} requestFn - The request interceptor function. - * @param {ResponseFunctionType} responseFn - The response interceptor function. + * @param axiosInstance - The Axios instance to register the interceptors. + * @param requestFn - The request interceptor function. + * @param responseFn - The response interceptor function. */ export const registerAxiosInterceptors = ( axiosInstance: Axios, @@ -64,8 +64,8 @@ export const registerAxiosInterceptors = ( /** * Converts a Uint8Array to a hex string. * - * @param {Uint8Array} key - The key to convert. - * @returns {string} The converted hex string. + * @param key - The key to convert. + * @returns The converted hex string. */ export const convertStringToHex = (key: Uint8Array): string => { return Buffer.from(key).toString("hex"); From 228dab14a837d3e7156ad8a5ac5b3711de829177 Mon Sep 17 00:00:00 2001 From: Erdi Maden Date: Thu, 16 May 2024 11:26:59 -0500 Subject: [PATCH 10/10] Updating networkId in wallet test --- src/coinbase/tests/wallet_test.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/coinbase/tests/wallet_test.ts b/src/coinbase/tests/wallet_test.ts index 9ea52075..bcc64fd1 100644 --- a/src/coinbase/tests/wallet_test.ts +++ b/src/coinbase/tests/wallet_test.ts @@ -61,7 +61,7 @@ describe("Wallet Class", () => { it("should return the correct string representation", async () => { const wallet = await Wallet.init(VALID_WALLET_MODEL, client); expect(wallet.toString()).toBe( - `Wallet{id: '${VALID_WALLET_MODEL.id}', networkId: 'base_sepolia'}`, + `Wallet{id: '${VALID_WALLET_MODEL.id}', networkId: 'base-sepolia'}`, ); });