diff --git a/.prettierrc b/.prettierrc index 07a2cea..ac0fa89 100644 --- a/.prettierrc +++ b/.prettierrc @@ -9,4 +9,3 @@ "arrowParens": "always", "endOfLine": "lf" } - diff --git a/docs/test-specifications/token-service/tokenFreezeTransaction.md b/docs/test-specifications/token-service/tokenFreezeTransaction.md index 2ed7005..d77333c 100644 --- a/docs/test-specifications/token-service/tokenFreezeTransaction.md +++ b/docs/test-specifications/token-service/tokenFreezeTransaction.md @@ -54,18 +54,18 @@ The tests contained in this specification will assume that a valid account and a | Test no | Name | Input | Expected response | Implemented (Y/N) | |---------|---------------------------------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------|--------------------------------------------------------------------------------------------------|-------------------| -| 1 | Freezes a token on an account | tokenId=, accountId=, commonTransactionParams.signers=[] | The token freeze succeeds. | N | -| 2 | Freezes a token that doesn't exist on an account | tokenId="123.456.789", accountId= | The token freeze fails with an INVALID_TOKEN_ID response code from the network. | N | -| 3 | Freezes a token with an empty token ID on an account | tokenId="", accountId= | The token freeze fails with an SDK internal error. | N | -| 4 | Freezes a token with no token ID on an account | accountId= | The token freeze fails with an INVALID_TOKEN_ID response code from the network. | N | -| 5 | Freezes a deleted token on an account | tokenId=, accountId=, commonTransactionParams.signers=[] | The token freeze fails with an TOKEN_WAS_DELETED response code from the network. | N | -| 6 | Freezes a token on an account without signing with the token's freeze key | tokenId=, accountId= | The token freeze fails with an INVALID_SIGNATURE response code from the network. | N | -| 7 | Freezes a token but signs with the token's admin key | tokenId=, accountId=, commonTransactionParams.signers=[] | The token freeze fails with an INVALID_SIGNATURE response code from the network. | N | -| 8 | Freezes a token on an account but signs with an incorrect freeze key | tokenId=, accountId=, commonTransactionParams.signers=[] | The token freeze fails with an INVALID_SIGNATURE response code from the network. | N | -| 9 | Freezes a token with no freeze key on an account | tokenId=, accountId= | The token freeze fails with an TOKEN_HAS_NO_FREEZE_KEY response code from the network. | N | -| 10 | Freezes a token that is already frozen on an account | tokenId=, accountId=, commonTransactionParams.signers=[] | The token freeze succeeds. | N | -| 11 | Freezes a token on an account that is not associated with the token | tokenId=, accountId=, commonTransactionParams.signers=[] | The token freeze fails with an TOKEN_NOT_ASSOCIATED_WITH_ACCOUNT response code from the network. | N | -| 12 | Freezes a paused token on an account | tokenId=, accountId=, commonTransactionParams.signers=[] | The token freeze fails with an TOKEN_IS_PAUSED response code from the network. | N | +| 1 | Freezes a token on an account | tokenId=, accountId=, commonTransactionParams.signers=[] | The token freeze succeeds. | Y | +| 2 | Freezes a token that doesn't exist on an account | tokenId="123.456.789", accountId= | The token freeze fails with an INVALID_TOKEN_ID response code from the network. | Y | +| 3 | Freezes a token with an empty token ID on an account | tokenId="", accountId= | The token freeze fails with an SDK internal error. | Y | +| 4 | Freezes a token with no token ID on an account | accountId= | The token freeze fails with an INVALID_TOKEN_ID response code from the network. | Y | +| 5 | Freezes a deleted token on an account | tokenId=, accountId=, commonTransactionParams.signers=[] | The token freeze fails with an TOKEN_WAS_DELETED response code from the network. | Y | +| 6 | Freezes a token on an account without signing with the token's freeze key | tokenId=, accountId= | The token freeze fails with an INVALID_SIGNATURE response code from the network. | Y | +| 7 | Freezes a token but signs with the token's admin key | tokenId=, accountId=, commonTransactionParams.signers=[] | The token freeze fails with an INVALID_SIGNATURE response code from the network. | Y | +| 8 | Freezes a token on an account but signs with an incorrect freeze key | tokenId=, accountId=, commonTransactionParams.signers=[] | The token freeze fails with an INVALID_SIGNATURE response code from the network. | Y | +| 9 | Freezes a token with no freeze key on an account | tokenId=, accountId= | The token freeze fails with an TOKEN_HAS_NO_FREEZE_KEY response code from the network. | Y | +| 10 | Freezes a token that is already frozen on an account | tokenId=, accountId=, commonTransactionParams.signers=[] | The token freeze succeeds. | Y | +| 11 | Freezes a token on an account that is not associated with the token | tokenId=, accountId=, commonTransactionParams.signers=[] | The token freeze fails with an TOKEN_NOT_ASSOCIATED_WITH_ACCOUNT response code from the network. | Y | +| 12 | Freezes a paused token on an account | tokenId=, accountId=, commonTransactionParams.signers=[] | The token freeze fails with an TOKEN_IS_PAUSED response code from the network. | Y | #### JSON Request Example @@ -102,12 +102,12 @@ The tests contained in this specification will assume that a valid account and a - The ID of the account to freeze. -| Test no | Name | Input | Expected response | Implemented (Y/N) | -|---------|---------------------------------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------|--------------------------------------------------------------------------------------------------|-------------------| -| 1 | Freezes a token on an account that doesn't exist | tokenId=, accountId="123.456.789", commonTransactionParams.signers=[] | The token freeze fails with an INVALID_ACCOUNT_ID response code from the network. | N | -| 2 | Freezes a token on an account with an empty account ID | tokenId=, accountId="", commonTransactionParams.signers=[] | The token freeze fails with an SDK internal error. | N | -| 3 | Freezes a token on an account with no account ID | tokenId=, commonTransactionParams.signers=[] | The token freeze fails with an INVALID_ACCOUNT_ID response code from the network. | N | -| 4 | Freezes a token on a deleted account | tokenId=, accountId=, commonTransactionParams.signers=[] | The token freeze fails with an ACCOUNT_WAS_DELETED response code from the network. | N | +| Test no | Name | Input | Expected response | Implemented (Y/N) | +|---------|---------------------------------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------|-----------------------------------------------------------------------------------|-------------------| +| 1 | Freezes a token on an account that doesn't exist | tokenId=, accountId="123.456.789", commonTransactionParams.signers=[] | The token freeze fails with an INVALID_ACCOUNT_ID response code from the network. | Y | +| 2 | Freezes a token on an account with an empty account ID | tokenId=, accountId="", commonTransactionParams.signers=[] | The token freeze fails with an SDK internal error. | Y | +| 3 | Freezes a token on an account with no account ID | tokenId=, commonTransactionParams.signers=[] | The token freeze fails with an INVALID_ACCOUNT_ID response code from the network. | Y | +| 4 | Freezes a token on a deleted account | tokenId=, accountId=, commonTransactionParams.signers=[] | The token freeze fails with an ACCOUNT_DELETED response code from the network. | Y | #### JSON Request Example diff --git a/src/services/MirrorNodeClient.ts b/src/services/MirrorNodeClient.ts index 75e459e..07813e4 100644 --- a/src/services/MirrorNodeClient.ts +++ b/src/services/MirrorNodeClient.ts @@ -1,5 +1,5 @@ -import { fetchData } from "../utils/helpers/fetch-data"; -import { retryOnError } from "../utils/helpers/retry-on-error"; +import { fetchData } from "@helpers/fetch-data"; +import { retryOnError } from "@helpers/retry-on-error"; class MirrorNodeClient { private mirrorNodeRestUrl: string | undefined; @@ -25,6 +25,12 @@ class MirrorNodeClient { const url = `${this.mirrorNodeRestUrl}/api/v1/tokens/${tokenId}`; return retryOnError(async () => fetchData(url)); } + + // TODO: Get mirror node interface with OpenAPI + async getTokenRelationships(accountId: string): Promise { + const url = `${this.mirrorNodeRestUrl}/api/v1/accounts/${accountId}/tokens`; + return retryOnError(async () => fetchData(url)); + } } export default new MirrorNodeClient(); diff --git a/src/tests/token-service/test-token-freeze-transaction.ts b/src/tests/token-service/test-token-freeze-transaction.ts new file mode 100644 index 0000000..de4546e --- /dev/null +++ b/src/tests/token-service/test-token-freeze-transaction.ts @@ -0,0 +1,420 @@ +import { assert, expect } from "chai"; + +import { JSONRPCRequest } from "@services/Client"; +import mirrorNodeClient from "@services/MirrorNodeClient"; + +import { setOperator } from "@helpers/setup-tests"; +import { retryOnError } from "@helpers/retry-on-error"; +import { ErrorStatusCodes } from "@enums/error-status-codes"; + +/** + * Tests for TokenFreezeTransaction + */ +describe("TokenFreezeTransaction", function () { + // Tests should not take longer than 30 seconds to fully execute. + this.timeout(30000); + + // All tests require an account and a token to be created, and to have the two be associated. + let tokenId: string, + tokenFreezeKey: string, + tokenAdminKey: string, + tokenPauseKey: string, + accountId: string, + accountPrivateKey: string; + beforeEach(async function () { + await setOperator( + this, + process.env.OPERATOR_ACCOUNT_ID as string, + process.env.OPERATOR_ACCOUNT_PRIVATE_KEY as string, + ); + + tokenFreezeKey = ( + await JSONRPCRequest(this, "generateKey", { + type: "ed25519PrivateKey", + }) + ).key; + + tokenAdminKey = ( + await JSONRPCRequest(this, "generateKey", { + type: "ed25519PrivateKey", + }) + ).key; + + tokenPauseKey = ( + await JSONRPCRequest(this, "generateKey", { + type: "ecdsaSecp256k1PrivateKey", + }) + ).key; + + tokenId = ( + await JSONRPCRequest(this, "createToken", { + name: "testname", + symbol: "testsymbol", + treasuryAccountId: process.env.OPERATOR_ACCOUNT_ID, + adminKey: tokenAdminKey, + freezeKey: tokenFreezeKey, + pauseKey: tokenPauseKey, + commonTransactionParams: { + signers: [tokenAdminKey], + }, + }) + ).tokenId; + + accountPrivateKey = ( + await JSONRPCRequest(this, "generateKey", { + type: "ed25519PrivateKey", + }) + ).key; + + accountId = ( + await JSONRPCRequest(this, "createAccount", { + key: accountPrivateKey, + }) + ).accountId; + + await JSONRPCRequest(this, "associateToken", { + accountId, + tokenIds: [tokenId], + commonTransactionParams: { + signers: [accountPrivateKey], + }, + }); + }); + afterEach(async function () { + await JSONRPCRequest(this, "reset"); + }); + + async function verifyTokenFrozen(accountId: string, tokenId: string) { + // No way to get token associations via consensus node, so just query mirror node. + const mirrorNodeInfo = + await mirrorNodeClient.getTokenRelationships(accountId); + + let foundToken = false; + for (let i = 0; i < mirrorNodeInfo.tokens.length; i++) { + if (mirrorNodeInfo.tokens[i].token_id === tokenId) { + expect(mirrorNodeInfo.tokens[i].freeze_status).to.equal("FROZEN"); + foundToken = true; + break; + } + } + + if (!foundToken) { + assert.fail("Token ID not found"); + } + } + + describe("Token ID", function () { + it("(#1) Freezes a token on an account", async function () { + await JSONRPCRequest(this, "freezeToken", { + tokenId, + accountId, + commonTransactionParams: { + signers: [tokenFreezeKey], + }, + }); + + await retryOnError(async () => { + await verifyTokenFrozen(accountId, tokenId); + }); + }); + + it("(#2) Freezes a token that doesn't exist on an account", async function () { + try { + await JSONRPCRequest(this, "freezeToken", { + tokenId: "123.456.789", + accountId, + }); + } catch (err: any) { + assert.equal(err.data.status, "INVALID_TOKEN_ID"); + return; + } + + assert.fail("Should throw an error"); + }); + + it("(#3) Freezes a token with an empty token ID on an account", async function () { + try { + await JSONRPCRequest(this, "freezeToken", { + tokenId: "", + accountId, + }); + } catch (err: any) { + assert.equal( + err.code, + ErrorStatusCodes.INTERNAL_ERROR, + "Internal error", + ); + return; + } + + assert.fail("Should throw an error"); + }); + + it("(#4) Freezes a token with no token ID on an account", async function () { + try { + await JSONRPCRequest(this, "freezeToken", { + accountId, + }); + } catch (err: any) { + assert.equal(err.data.status, "INVALID_TOKEN_ID"); + return; + } + + assert.fail("Should throw an error"); + }); + + it("(#5) Freezes a deleted token on an account", async function () { + await JSONRPCRequest(this, "deleteToken", { + tokenId, + commonTransactionParams: { + signers: [tokenAdminKey], + }, + }); + + try { + await JSONRPCRequest(this, "freezeToken", { + tokenId, + accountId, + commonTransactionParams: { + signers: [tokenFreezeKey], + }, + }); + } catch (err: any) { + assert.equal(err.data.status, "TOKEN_WAS_DELETED"); + return; + } + + assert.fail("Should throw an error"); + }); + + it("(#6) Freezes a token on an account without signing with the token's freeze key", async function () { + try { + await JSONRPCRequest(this, "freezeToken", { + tokenId, + accountId, + }); + } catch (err: any) { + assert.equal(err.data.status, "INVALID_SIGNATURE"); + return; + } + + assert.fail("Should throw an error"); + }); + + it("(#7) Freezes a token but signs with the token's admin key", async function () { + try { + await JSONRPCRequest(this, "freezeToken", { + tokenId, + accountId, + commonTransactionParams: { + signers: [tokenAdminKey], + }, + }); + } catch (err: any) { + assert.equal(err.data.status, "INVALID_SIGNATURE"); + return; + } + + assert.fail("Should throw an error"); + }); + + it("(#8) Freezes a token on an account but signs with an incorrect freeze key", async function () { + const incorrectKey = ( + await JSONRPCRequest(this, "generateKey", { + type: "ed25519PrivateKey", + }) + ).key; + + try { + await JSONRPCRequest(this, "freezeToken", { + tokenId, + accountId, + commonTransactionParams: { + signers: [incorrectKey], + }, + }); + } catch (err: any) { + assert.equal(err.data.status, "INVALID_SIGNATURE"); + return; + } + + assert.fail("Should throw an error"); + }); + + it("(#9) Freezes a token with no freeze key on an account", async function () { + const tokenIdNoFreezeKey = ( + await JSONRPCRequest(this, "createToken", { + name: "testname", + symbol: "testsymbol", + treasuryAccountId: process.env.OPERATOR_ACCOUNT_ID, + }) + ).tokenId; + + try { + await JSONRPCRequest(this, "freezeToken", { + tokenId: tokenIdNoFreezeKey, + accountId, + }); + } catch (err: any) { + assert.equal(err.data.status, "TOKEN_HAS_NO_FREEZE_KEY"); + return; + } + + assert.fail("Should throw an error"); + }); + + it("(#10) Freezes a token that is already frozen on an account", async function () { + await JSONRPCRequest(this, "freezeToken", { + tokenId, + accountId, + commonTransactionParams: { + signers: [tokenFreezeKey], + }, + }); + + await JSONRPCRequest(this, "freezeToken", { + tokenId, + accountId, + commonTransactionParams: { + signers: [tokenFreezeKey], + }, + }); + + await retryOnError(async () => { + await verifyTokenFrozen(accountId, tokenId); + }); + }); + + it("(#11) Freezes a token on an account that is not associated with the token", async function () { + await JSONRPCRequest(this, "dissociateToken", { + accountId, + tokenIds: [tokenId], + commonTransactionParams: { + signers: [accountPrivateKey], + }, + }); + + try { + await JSONRPCRequest(this, "freezeToken", { + tokenId, + accountId, + commonTransactionParams: { + signers: [tokenFreezeKey], + }, + }); + } catch (err: any) { + assert.equal(err.data.status, "TOKEN_NOT_ASSOCIATED_TO_ACCOUNT"); + return; + } + + assert.fail("Should throw an error"); + }); + + it("(#12) Freezes a paused token on an account", async function () { + await JSONRPCRequest(this, "pauseToken", { + tokenId, + commonTransactionParams: { + signers: [tokenPauseKey], + }, + }); + + try { + await JSONRPCRequest(this, "freezeToken", { + tokenId, + accountId, + commonTransactionParams: { + signers: [tokenFreezeKey], + }, + }); + } catch (err: any) { + assert.equal(err.data.status, "TOKEN_IS_PAUSED"); + return; + } + + assert.fail("Should throw an error"); + }); + }); + + describe("Account ID", function () { + it("(#1) Freezes a token on an account that doesn't exist", async function () { + try { + await JSONRPCRequest(this, "freezeToken", { + tokenId, + accountId: "123.456.789", + commonTransactionParams: { + signers: [tokenFreezeKey], + }, + }); + } catch (err: any) { + assert.equal(err.data.status, "INVALID_ACCOUNT_ID"); + return; + } + + assert.fail("Should throw an error"); + }); + + it("(#2) Freezes a token on an account with an empty account ID", async function () { + try { + await JSONRPCRequest(this, "freezeToken", { + tokenId, + accountId: "", + commonTransactionParams: { + signers: [tokenFreezeKey], + }, + }); + } catch (err: any) { + assert.equal( + err.code, + ErrorStatusCodes.INTERNAL_ERROR, + "Internal error", + ); + return; + } + + assert.fail("Should throw an error"); + }); + + it("(#3) Freezes a token on an account with no account ID", async function () { + try { + await JSONRPCRequest(this, "freezeToken", { + tokenId, + commonTransactionParams: { + signers: [tokenFreezeKey], + }, + }); + } catch (err: any) { + assert.equal(err.data.status, "INVALID_ACCOUNT_ID"); + return; + } + + assert.fail("Should throw an error"); + }); + + it("(#4) Freezes a token on a deleted account", async function () { + await JSONRPCRequest(this, "deleteAccount", { + deleteAccountId: accountId, + transferAccountId: process.env.OPERATOR_ACCOUNT_ID, + commonTransactionParams: { + signers: [accountPrivateKey], + }, + }); + + try { + await JSONRPCRequest(this, "freezeToken", { + tokenId, + accountId, + commonTransactionParams: { + signers: [tokenFreezeKey], + }, + }); + } catch (err: any) { + assert.equal(err.data.status, "ACCOUNT_DELETED"); + return; + } + + assert.fail("Should throw an error"); + }); + }); + + return Promise.resolve(); +});