diff --git a/docs/test-specifications/token-service/TokenRevokeKycTransaction.md b/docs/test-specifications/token-service/TokenRevokeKycTransaction.md index 1de9990..333c53d 100644 --- a/docs/test-specifications/token-service/TokenRevokeKycTransaction.md +++ b/docs/test-specifications/token-service/TokenRevokeKycTransaction.md @@ -10,7 +10,7 @@ Each test within the test specification is linked to one of the properties withi https://docs.hedera.com/hedera/sdks-and-apis/sdks/token-service/disable-kyc-account-flag -**TokenGrantKyc protobufs:** +**TokenRevokeKyc protobufs:** https://github.com/hashgraph/hedera-protobufs/blob/main/services/token_revoke_kyc.proto @@ -54,19 +54,19 @@ The tests contained in this specification will assume that a valid account has b | Test no | Name | Input | Expected response | Implemented (Y/N) | |---------|-------------------------------------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------|----------------------------------------------------------------------------------------------------|-------------------| -| 1 | Revokes KYC of a token to an account | tokenId=, accountId=, commonTransactionParams.signers=[] | The token revokes KYC to the account. | N | -| 2 | Revokes KYC of a token that doesn't exist to an account | tokenId="123.456.789", accountId= | The token KYC revoke fails with an INVALID_TOKEN_ID response code from the network. | N | -| 3 | Revokes KYC of a token with an empty token ID to an account | tokenId="", accountId= | The token KYC revoke fails with an SDK internal error. | N | -| 4 | Revokes KYC of a token with no token ID to an account | accountId= | The token KYC revoke fails with an INVALID_TOKEN_ID response code from the network. | N | -| 5 | Revokes KYC of a deleted token to an account | tokenId=, accountId=, commonTransactionParams.signers=[] | The token KYC revoke fails with an TOKEN_WAS_DELETED response code from the network. | N | -| 6 | Revokes KYC of a token to an account without signing with the token's KYC key | tokenId=, accountId= | The token KYC revoke fails with an INVALID_SIGNATURE response code from the network. | N | -| 7 | Revokes KYC of a token to an account but signs with the the token's admin key | tokenId=, accountId=, commonTransactionParams.signers=[] | The token KYC revoke fails with an INVALID_SIGNATURE response code from the network. | N | -| 8 | Revokes KYC of a token to an account but signs with an incorrect private key | tokenId=, accountId=, commonTransactionParams.signers=[] | The token KYC revoke fails with an INVALID_SIGNATURE response code from the network. | N | -| 9 | Revokes KYC of a token with no KYC key to an account | tokenId=, accountId= | The token KYC revoke fails with an TOKEN_HAS_NO_KYC_KEY response code from the network. | N | -| 10 | Revokes KYC of a token to an account that doesn't have KYC | tokenId=, accountId=, commonTransactionParams.signers=[] | The token revokes KYC to the account. | N | -| 11 | Revokes KYC of a token to an account that is not associated with the token | tokenId=, accountId=, commonTransactionParams.signers=[] | The token KYC revoke fails with an TOKEN_NOT_ASSOCIATED_TO_ACCOUNT response code from the network. | N | -| 12 | Revokes KYC of a paused token to an account | tokenId=, accountId=, commonTransactionParams.signers=[] | The token KYC revoke fails with an TOKEN_IS_PAUSED response code from the network. | N | -| 13 | Revokes KYC of a token to a frozen account | tokenId=, accountId=, commonTransactionParams.signers=[] | The token KYC revoke fails with an ACCOUNT_FROZEN_FOR_TOKEN response code from the network. | N | +| 1 | Revokes KYC of a token to an account | tokenId=, accountId=, commonTransactionParams.signers=[] | The token revokes KYC to the account. | Y | +| 2 | Revokes KYC of a token that doesn't exist to an account | tokenId="123.456.789", accountId= | The token KYC revoke fails with an INVALID_TOKEN_ID response code from the network. | Y | +| 3 | Revokes KYC of a token with an empty token ID to an account | tokenId="", accountId= | The token KYC revoke fails with an SDK internal error. | Y | +| 4 | Revokes KYC of a token with no token ID to an account | accountId= | The token KYC revoke fails with an INVALID_TOKEN_ID response code from the network. | Y | +| 5 | Revokes KYC of a deleted token to an account | tokenId=, accountId=, commonTransactionParams.signers=[] | The token KYC revoke fails with an TOKEN_WAS_DELETED response code from the network. | Y | +| 6 | Revokes KYC of a token to an account without signing with the token's KYC key | tokenId=, accountId= | The token KYC revoke fails with an INVALID_SIGNATURE response code from the network. | Y | +| 7 | Revokes KYC of a token to an account but signs with the the token's admin key | tokenId=, accountId=, commonTransactionParams.signers=[] | The token KYC revoke fails with an INVALID_SIGNATURE response code from the network. | Y | +| 8 | Revokes KYC of a token to an account but signs with an incorrect private key | tokenId=, accountId=, commonTransactionParams.signers=[] | The token KYC revoke fails with an INVALID_SIGNATURE response code from the network. | Y | +| 9 | Revokes KYC of a token with no KYC key to an account | tokenId=, accountId= | The token KYC revoke fails with an TOKEN_HAS_NO_KYC_KEY response code from the network. | Y | +| 10 | Revokes KYC of a token to an account that doesn't have KYC | tokenId=, accountId=, commonTransactionParams.signers=[] | The token revokes KYC to the account. | Y | +| 11 | Revokes KYC of a token to an account that is not associated with the token | tokenId=, accountId=, commonTransactionParams.signers=[] | The token KYC revoke fails with an TOKEN_NOT_ASSOCIATED_TO_ACCOUNT response code from the network. | Y | +| 12 | Revokes KYC of a paused token to an account | tokenId=, accountId=, commonTransactionParams.signers=[] | The token KYC revoke fails with an TOKEN_IS_PAUSED response code from the network. | Y | +| 13 | Revokes KYC of a token to a frozen account | tokenId=, accountId=, commonTransactionParams.signers=[] | The token KYC revoke fails with an ACCOUNT_FROZEN_FOR_TOKEN response code from the network. | Y | #### JSON Request Example @@ -103,12 +103,12 @@ The tests contained in this specification will assume that a valid account has b - The ID of the account to which to revoke KYC. -| Test no | Name | Input | Expected response | Implemented (Y/N) | -|---------|---------------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------|----------------------------------------------------------------------------------------|-------------------| -| 1 | Revokes KYC of a token to an account that doesn't exist | tokenId=, accountId="123.456.789", commonTransactionParams.signers=[] | The token KYC revoke fails with an INVALID_ACCOUNT_ID response code from the network. | N | -| 2 | Revokes KYC of a token to an empty account ID | tokenId=, accountId="", commonTransactionParams.signers=[] | The token KYC revoke fails with an SDK internal error. | N | -| 3 | Revokes KYC of a token to an account with no account ID | tokenId=, commonTransactionParams.signers=[] | The token KYC revoke fails with an INVALID_TOKEN_ID response code from the network. | N | -| 4 | Revokes KYC of a token to a deleted account | tokenId=, accountId=, commonTransactionParams.signers=[] | The token KYC revoke fails with an ACCOUNT_WAS_DELETED response code from the network. | N | +| Test no | Name | Input | Expected response | Implemented (Y/N) | +|---------|---------------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------|---------------------------------------------------------------------------------------|-------------------| +| 1 | Revokes KYC of a token to an account that doesn't exist | tokenId=, accountId="123.456.789", commonTransactionParams.signers=[] | The token KYC revoke fails with an INVALID_ACCOUNT_ID response code from the network. | Y | +| 2 | Revokes KYC of a token to an empty account ID | tokenId=, accountId="", commonTransactionParams.signers=[] | The token KYC revoke fails with an SDK internal error. | Y | +| 3 | Revokes KYC of a token to an account with no account ID | tokenId=, commonTransactionParams.signers=[] | The token KYC revoke fails with an INVALID_ACCOUNT_ID response code from the network. | Y | +| 4 | Revokes KYC of a token to a deleted account | tokenId=, accountId=, commonTransactionParams.signers=[] | The token KYC revoke fails with an ACCOUNT_DELETED response code from the network. | Y | #### JSON Request Example diff --git a/src/tests/token-service/test-token-revoke-kyc-transaction.ts b/src/tests/token-service/test-token-revoke-kyc-transaction.ts new file mode 100644 index 0000000..2bfeaf4 --- /dev/null +++ b/src/tests/token-service/test-token-revoke-kyc-transaction.ts @@ -0,0 +1,461 @@ +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 TokenRevokeKycTransaction + */ +describe("TokenRevokeKycTransaction", 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, have the two be associated, and KYC granted. + let tokenId: string, + tokenFreezeKey: string, + tokenAdminKey: string, + tokenPauseKey: string, + tokenKycKey: 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; + + tokenKycKey = ( + await JSONRPCRequest(this, "generateKey", { + type: "ecdsaSecp256k1PrivateKey", + }) + ).key; + + tokenId = ( + await JSONRPCRequest(this, "createToken", { + name: "testname", + symbol: "testsymbol", + treasuryAccountId: process.env.OPERATOR_ACCOUNT_ID, + adminKey: tokenAdminKey, + kycKey: tokenKycKey, + 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], + }, + }); + + await JSONRPCRequest(this, "grantTokenKyc", { + tokenId, + accountId, + commonTransactionParams: { + signers: [tokenKycKey], + }, + }); + }); + afterEach(async function () { + await JSONRPCRequest(this, "reset"); + }); + + async function verifyTokenNoKyc(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].kyc_status).to.equal("REVOKED"); + foundToken = true; + break; + } + } + + if (!foundToken) { + assert.fail("Token ID not found"); + } + } + + describe("Token ID", function () { + it("(#1) Revokes KYC of a token to an account", async function () { + await JSONRPCRequest(this, "revokeTokenKyc", { + tokenId, + accountId, + commonTransactionParams: { + signers: [tokenKycKey], + }, + }); + + await retryOnError(async () => { + await verifyTokenNoKyc(accountId, tokenId); + }); + }); + + it("(#2) Revokes KYC of a token that doesn't exist to an account", async function () { + try { + await JSONRPCRequest(this, "revokeTokenKyc", { + 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) Revokes KYC of a token with an empty token ID to an account", async function () { + try { + await JSONRPCRequest(this, "revokeTokenKyc", { + tokenId: "", + accountId, + }); + } catch (err: any) { + assert.equal( + err.code, + ErrorStatusCodes.INTERNAL_ERROR, + "Internal error", + ); + return; + } + + assert.fail("Should throw an error"); + }); + + it("(#4) Revokes KYC of a token with no token ID to an account", async function () { + try { + await JSONRPCRequest(this, "revokeTokenKyc", { + accountId, + }); + } catch (err: any) { + assert.equal(err.data.status, "INVALID_TOKEN_ID"); + return; + } + + assert.fail("Should throw an error"); + }); + + it("(#5) Revokes KYC of a deleted token to an account", async function () { + await JSONRPCRequest(this, "deleteToken", { + tokenId, + commonTransactionParams: { + signers: [tokenAdminKey], + }, + }); + + try { + await JSONRPCRequest(this, "revokeTokenKyc", { + tokenId, + accountId, + commonTransactionParams: { + signers: [tokenKycKey], + }, + }); + } catch (err: any) { + assert.equal(err.data.status, "TOKEN_WAS_DELETED"); + return; + } + + assert.fail("Should throw an error"); + }); + + it("(#6) Revokes KYC of a token to an account without signing with the token's KYC key", async function () { + try { + await JSONRPCRequest(this, "revokeTokenKyc", { + tokenId, + accountId, + }); + } catch (err: any) { + assert.equal(err.data.status, "INVALID_SIGNATURE"); + return; + } + + assert.fail("Should throw an error"); + }); + + it("(#7) Revokes KYC of a token to an account but signs with the the token's admin key", async function () { + try { + await JSONRPCRequest(this, "revokeTokenKyc", { + tokenId, + accountId, + commonTransactionParams: { + signers: [tokenAdminKey], + }, + }); + } catch (err: any) { + assert.equal(err.data.status, "INVALID_SIGNATURE"); + return; + } + + assert.fail("Should throw an error"); + }); + + it("(#8) Revokes KYC of a token to an account but signs with an incorrect private key", async function () { + const incorrectPrivateKey = ( + await JSONRPCRequest(this, "generateKey", { + type: "ed25519PrivateKey", + }) + ).key; + + try { + await JSONRPCRequest(this, "revokeTokenKyc", { + tokenId, + accountId, + commonTransactionParams: { + signers: [incorrectPrivateKey], + }, + }); + } catch (err: any) { + assert.equal(err.data.status, "INVALID_SIGNATURE"); + return; + } + + assert.fail("Should throw an error"); + }); + + it("(#9) Revokes KYC of a token with no KYC key to an account", async function () { + const tokenIdNoKycKey = ( + await JSONRPCRequest(this, "createToken", { + name: "testname", + symbol: "testsymbol", + treasuryAccountId: process.env.OPERATOR_ACCOUNT_ID, + }) + ).tokenId; + + try { + await JSONRPCRequest(this, "revokeTokenKyc", { + tokenId: tokenIdNoKycKey, + accountId, + }); + } catch (err: any) { + assert.equal(err.data.status, "TOKEN_HAS_NO_KYC_KEY"); + return; + } + + assert.fail("Should throw an error"); + }); + + it("(#10) Revokes KYC of a token to an account that doesn't have KYC", async function () { + await JSONRPCRequest(this, "revokeTokenKyc", { + tokenId, + accountId, + commonTransactionParams: { + signers: [tokenKycKey], + }, + }); + + await JSONRPCRequest(this, "revokeTokenKyc", { + tokenId, + accountId, + commonTransactionParams: { + signers: [tokenKycKey], + }, + }); + + await retryOnError(async () => { + await verifyTokenNoKyc(accountId, tokenId); + }); + }); + + it("(#11) Revokes KYC of a token to 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, "revokeTokenKyc", { + tokenId, + accountId, + commonTransactionParams: { + signers: [tokenKycKey], + }, + }); + } catch (err: any) { + assert.equal(err.data.status, "TOKEN_NOT_ASSOCIATED_TO_ACCOUNT"); + return; + } + + assert.fail("Should throw an error"); + }); + + it("(#12) Revokes KYC of a paused token to an account", async function () { + await JSONRPCRequest(this, "pauseToken", { + tokenId, + commonTransactionParams: { + signers: [tokenPauseKey], + }, + }); + + try { + await JSONRPCRequest(this, "revokeTokenKyc", { + tokenId, + accountId, + commonTransactionParams: { + signers: [tokenKycKey], + }, + }); + } catch (err: any) { + assert.equal(err.data.status, "TOKEN_IS_PAUSED"); + return; + } + + assert.fail("Should throw an error"); + }); + + it("(#13) Revokes KYC of a token to a frozen account", async function () { + await JSONRPCRequest(this, "freezeToken", { + tokenId, + accountId, + commonTransactionParams: { + signers: [tokenFreezeKey], + }, + }); + + try { + await JSONRPCRequest(this, "revokeTokenKyc", { + tokenId, + accountId, + commonTransactionParams: { + signers: [tokenKycKey], + }, + }); + } catch (err: any) { + assert.equal(err.data.status, "ACCOUNT_FROZEN_FOR_TOKEN"); + return; + } + + assert.fail("Should throw an error"); + }); + }); + + describe("Account ID", function () { + it("(#1) Revokes KYC of a token to an account that doesn't exist", async function () { + try { + await JSONRPCRequest(this, "revokeTokenKyc", { + tokenId, + accountId: "123.456.789", + commonTransactionParams: { + signers: [tokenKycKey], + }, + }); + } catch (err: any) { + assert.equal(err.data.status, "INVALID_ACCOUNT_ID"); + return; + } + + assert.fail("Should throw an error"); + }); + + it("(#2) Revokes KYC of a token to an empty account ID", async function () { + try { + await JSONRPCRequest(this, "revokeTokenKyc", { + tokenId, + accountId: "", + commonTransactionParams: { + signers: [tokenKycKey], + }, + }); + } catch (err: any) { + assert.equal( + err.code, + ErrorStatusCodes.INTERNAL_ERROR, + "Internal error", + ); + return; + } + + assert.fail("Should throw an error"); + }); + + it("(#3) Revokes KYC of a token to an account with no account ID", async function () { + try { + await JSONRPCRequest(this, "revokeTokenKyc", { + tokenId, + commonTransactionParams: { + signers: [tokenKycKey], + }, + }); + } catch (err: any) { + assert.equal(err.data.status, "INVALID_ACCOUNT_ID"); + return; + } + + assert.fail("Should throw an error"); + }); + + it("(#4) Revokes KYC of a token to a deleted account", async function () { + await JSONRPCRequest(this, "deleteAccount", { + deleteAccountId: accountId, + transferAccountId: process.env.OPERATOR_ACCOUNT_ID, + commonTransactionParams: { + signers: [accountPrivateKey], + }, + }); + + try { + await JSONRPCRequest(this, "revokeTokenKyc", { + tokenId, + accountId, + commonTransactionParams: { + signers: [tokenKycKey], + }, + }); + } catch (err: any) { + assert.equal(err.data.status, "ACCOUNT_DELETED"); + return; + } + + assert.fail("Should throw an error"); + }); + }); + + return Promise.resolve(); +});