From 162e6adbf7221d23eac6545512818ce82af05519 Mon Sep 17 00:00:00 2001 From: yinyiqian1 Date: Thu, 4 Jul 2024 14:20:45 -0400 Subject: [PATCH 1/6] support DynamicNFT --- .../src/enums/definitions.json | 1 + .../src/models/transactions/NFTokenMint.ts | 5 + .../src/models/transactions/NFTokenModify.ts | 69 ++++++++++++ .../xrpl/src/models/transactions/index.ts | 1 + .../src/models/transactions/transaction.ts | 6 + .../transactions/nftokenModify.test.ts | 105 ++++++++++++++++++ .../xrpl/test/models/NFTokenModify.test.ts | 41 +++++++ 7 files changed, 228 insertions(+) create mode 100644 packages/xrpl/src/models/transactions/NFTokenModify.ts create mode 100644 packages/xrpl/test/integration/transactions/nftokenModify.test.ts create mode 100644 packages/xrpl/test/models/NFTokenModify.test.ts diff --git a/packages/ripple-binary-codec/src/enums/definitions.json b/packages/ripple-binary-codec/src/enums/definitions.json index 797be9ce21..435ab7baef 100644 --- a/packages/ripple-binary-codec/src/enums/definitions.json +++ b/packages/ripple-binary-codec/src/enums/definitions.json @@ -2974,6 +2974,7 @@ "DIDDelete": 50, "OracleSet": 51, "OracleDelete": 52, + "NFTokenModify":53, "EnableAmendment": 100, "SetFee": 101, "UNLModify": 102 diff --git a/packages/xrpl/src/models/transactions/NFTokenMint.ts b/packages/xrpl/src/models/transactions/NFTokenMint.ts index 2630a6b9c6..1bd7947929 100644 --- a/packages/xrpl/src/models/transactions/NFTokenMint.ts +++ b/packages/xrpl/src/models/transactions/NFTokenMint.ts @@ -38,6 +38,10 @@ export enum NFTokenMintFlags { * issuer. */ tfTransferable = 0x00000008, + /** + * If set, indicates that this NFT's URI can be modified. + */ + tfMutable = 0x00000010, } /** @@ -51,6 +55,7 @@ export interface NFTokenMintFlagsInterface extends GlobalFlags { tfOnlyXRP?: boolean tfTrustLine?: boolean tfTransferable?: boolean + tfMutable?: boolean } /** diff --git a/packages/xrpl/src/models/transactions/NFTokenModify.ts b/packages/xrpl/src/models/transactions/NFTokenModify.ts new file mode 100644 index 0000000000..b74c1e8857 --- /dev/null +++ b/packages/xrpl/src/models/transactions/NFTokenModify.ts @@ -0,0 +1,69 @@ +import { ValidationError } from '../../errors' +import { isHex } from '../utils' + +import { + BaseTransaction, + validateBaseTransaction, + isAccount, + isString, + validateOptionalField, + Account, +} from './common' + +/** + * The NFTokenModify transaction modifies an NFToken's URI + * if its tfMutable is set to true. + */ +export interface NFTokenModify extends BaseTransaction { + TransactionType: 'NFTokenModify' + /** + * Identifies the NFTokenID of the NFToken object that the + * offer references. + */ + NFTokenID: string + /** + * Indicates the AccountID of the account that owns the + * corresponding NFToken. + */ + Owner?: Account + /** + * URI that points to the data and/or metadata associated with the NFT. + * This field need not be an HTTP or HTTPS URL; it could be an IPFS URI, a + * magnet link, immediate data encoded as an RFC2379 "data" URL, or even an + * opaque issuer-specific encoding. The URI is NOT checked for validity, but + * the field is limited to a maximum length of 256 bytes. + * + * This field must be hex-encoded. You can use `convertStringToHex` to + * convert this field to the proper encoding. + * + * This field must not be an empty string. Omit it from the transaction or + * set to `undefined` value if you do not use it. + */ + URI?: string | null +} + +/** + * Verify the form and type of an NFTokenModify at runtime. + * + * @param tx - An NFTokenModify Transaction. + * @throws When the NFTokenModify is Malformed. + */ +export function validateNFTokenModify(tx: Record): void { + validateBaseTransaction(tx) + + if (tx.NFTokenID == null) { + throw new ValidationError('NFTokenModify: missing field NFTokenID') + } + + validateOptionalField(tx, 'Owner', isAccount) + validateOptionalField(tx, 'URI', isString) + + if (tx.URI !== undefined && typeof tx.URI === 'string') { + if (tx.URI === '') { + throw new ValidationError('NFTokenModify: URI must not be empty string') + } + if (!isHex(tx.URI)) { + throw new ValidationError('NFTokenModify: URI must be in hex format') + } + } +} diff --git a/packages/xrpl/src/models/transactions/index.ts b/packages/xrpl/src/models/transactions/index.ts index c7a8120758..6ca9cc94f4 100644 --- a/packages/xrpl/src/models/transactions/index.ts +++ b/packages/xrpl/src/models/transactions/index.ts @@ -52,6 +52,7 @@ export { NFTokenMintFlags, NFTokenMintFlagsInterface, } from './NFTokenMint' +export { NFTokenModify, validateNFTokenModify } from './NFTokenModify' export { OfferCancel } from './offerCancel' export { OfferCreateFlags, diff --git a/packages/xrpl/src/models/transactions/transaction.ts b/packages/xrpl/src/models/transactions/transaction.ts index 0ddc719539..cc8d09aab9 100644 --- a/packages/xrpl/src/models/transactions/transaction.ts +++ b/packages/xrpl/src/models/transactions/transaction.ts @@ -41,6 +41,7 @@ import { validateNFTokenCreateOffer, } from './NFTokenCreateOffer' import { NFTokenMint, validateNFTokenMint } from './NFTokenMint' +import { NFTokenModify, validateNFTokenModify } from './NFTokenModify' import { OfferCancel, validateOfferCancel } from './offerCancel' import { OfferCreate, validateOfferCreate } from './offerCreate' import { OracleDelete, validateOracleDelete } from './oracleDelete' @@ -120,6 +121,7 @@ export type SubmittableTransaction = | NFTokenCancelOffer | NFTokenCreateOffer | NFTokenMint + | NFTokenModify | OfferCancel | OfferCreate | OracleDelete @@ -326,6 +328,10 @@ export function validate(transaction: Record): void { validateNFTokenMint(tx) break + case 'NFTokenModify': + validateNFTokenModify(tx) + break + case 'OfferCancel': validateOfferCancel(tx) break diff --git a/packages/xrpl/test/integration/transactions/nftokenModify.test.ts b/packages/xrpl/test/integration/transactions/nftokenModify.test.ts new file mode 100644 index 0000000000..89c30fcc3f --- /dev/null +++ b/packages/xrpl/test/integration/transactions/nftokenModify.test.ts @@ -0,0 +1,105 @@ +import { assert } from 'chai' + +import { NFTokenModify } from '../../../dist/npm' +import { NFTokenMintFlags } from '../../../dist/npm/src' +import { + convertStringToHex, + getNFTokenID, + NFTokenMint, + TransactionMetadata, + TxRequest, +} from '../../../src' +import { hashSignedTx } from '../../../src/utils/hashes' +import serverUrl from '../serverUrl' +import { + setupClient, + teardownClient, + type XrplIntegrationTestContext, +} from '../setup' +import { testTransaction } from '../utils' + +// how long before each test case times out +const TIMEOUT = 20000 + +describe('NFTokenModify', function () { + let testContext: XrplIntegrationTestContext + + beforeEach(async () => { + testContext = await setupClient(serverUrl) + }) + afterEach(async () => teardownClient(testContext)) + + it( + 'modify NFToken URI', + async function () { + // Mint an NFToken with tfMutable flag and modify URI later + + const oldUri = convertStringToHex('https://www.google.com') + const newUri = convertStringToHex('https://www.ripple.com') + + const mutableMint: NFTokenMint = { + TransactionType: 'NFTokenMint', + Account: testContext.wallet.address, + Flags: NFTokenMintFlags.tfMutable, + URI: oldUri, + NFTokenTaxon: 0, + } + const response = await testTransaction( + testContext.client, + mutableMint, + testContext.wallet, + ) + assert.equal(response.type, 'response') + + const mutableTx: TxRequest = { + command: 'tx', + transaction: hashSignedTx(response.result.tx_blob), + } + const mutableTxResponse = await testContext.client.request(mutableTx) + + const mutableNFTokenID = + getNFTokenID( + mutableTxResponse.result.meta as TransactionMetadata, + ) ?? 'undefined' + + const accountNFTs = await testContext.client.request({ + command: 'account_nfts', + account: testContext.wallet.address, + }) + + assert.equal( + accountNFTs.result.account_nfts.find( + (nft) => nft.NFTokenID === mutableNFTokenID, + )?.URI, + oldUri, + ) + + const modifyTx: NFTokenModify = { + TransactionType: 'NFTokenModify', + Account: testContext.wallet.address, + NFTokenID: mutableNFTokenID, + URI: newUri, + } + + const modifyResponse = await testTransaction( + testContext.client, + modifyTx, + testContext.wallet, + ) + assert.equal(modifyResponse.type, 'response') + + const nfts = await testContext.client.request({ + command: 'account_nfts', + account: testContext.wallet.address, + }) + + assert.equal( + nfts.result.account_nfts.find( + (nft) => nft.NFTokenID === mutableNFTokenID, + )?.URI, + newUri, + ) + }, + TIMEOUT, + ) +}) diff --git a/packages/xrpl/test/models/NFTokenModify.test.ts b/packages/xrpl/test/models/NFTokenModify.test.ts new file mode 100644 index 0000000000..2f84793153 --- /dev/null +++ b/packages/xrpl/test/models/NFTokenModify.test.ts @@ -0,0 +1,41 @@ +import { assert } from 'chai' + +import { convertStringToHex, validate, ValidationError } from '../../src' + +const TOKEN_ID = + '00090032B5F762798A53D543A014CAF8B297CFF8F2F937E844B17C9E00000003' + +/** + * NFTokenModify Transaction Verification Testing. + * + * Providing runtime verification testing for each specific transaction type. + */ +describe('NFTokenModify', function () { + it(`verifies valid NFTokenModify`, function () { + const validNFTokenModify = { + TransactionType: 'NFTokenModify', + Account: 'rWYkbWkCeg8dP6rXALnjgZSjjLyih5NXm', + NFTokenID: TOKEN_ID, + Fee: '5000000', + Sequence: 2470665, + URI: convertStringToHex('http://xrpl.org'), + } as any + + assert.doesNotThrow(() => validate(validNFTokenModify)) + }) + + it(`throws w/ missing NFTokenID`, function () { + const invalid = { + TransactionType: 'NFTokenModify', + Account: 'rWYkbWkCeg8dP6rXALnjgZSjjLyih5NXm', + Fee: '5000000', + Sequence: 2470665, + } as any + + assert.throws( + () => validate(invalid), + ValidationError, + 'NFTokenModify: missing field NFTokenID', + ) + }) +}) From c51ca1cab41cf51b1645dcdf5e9fff9b78407261 Mon Sep 17 00:00:00 2001 From: yinyiqian1 Date: Thu, 4 Jul 2024 14:40:39 -0400 Subject: [PATCH 2/6] Update history.md --- packages/xrpl/HISTORY.md | 1 + 1 file changed, 1 insertion(+) diff --git a/packages/xrpl/HISTORY.md b/packages/xrpl/HISTORY.md index 2c534d0545..c70d0e0eb5 100644 --- a/packages/xrpl/HISTORY.md +++ b/packages/xrpl/HISTORY.md @@ -10,6 +10,7 @@ Subscribe to [the **xrpl-announce** mailing list](https://groups.google.com/g/xr ### Added * Add `nfts_by_issuer` clio-only API definition +* Add `NFTokenModify` transaction and add `tfMutable` flag in `NFTokenMint` ## 3.1.0 (2024-06-03) From a22924fbdc3e9245696bb7bf8d2a1fdcd9dae138 Mon Sep 17 00:00:00 2001 From: yinyiqian1 Date: Fri, 5 Jul 2024 15:31:33 -0400 Subject: [PATCH 3/6] use xrpl-codec-gen to generate definitions.json and modify ripple-binary-code HISTORY.md --- packages/ripple-binary-codec/HISTORY.md | 1 + packages/ripple-binary-codec/src/enums/definitions.json | 2 +- 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/packages/ripple-binary-codec/HISTORY.md b/packages/ripple-binary-codec/HISTORY.md index a0710d4eba..2cf4b39c8a 100644 --- a/packages/ripple-binary-codec/HISTORY.md +++ b/packages/ripple-binary-codec/HISTORY.md @@ -6,6 +6,7 @@ ### Added * Support for the Price Oracles amendment (XLS-47). +* Add `NFTokenModify` transaction and add `tfMutable` flag in `NFTokenMint` ### Fixed * Better error handling/error messages for serialization/deserialization errors. diff --git a/packages/ripple-binary-codec/src/enums/definitions.json b/packages/ripple-binary-codec/src/enums/definitions.json index 435ab7baef..473d6e74f4 100644 --- a/packages/ripple-binary-codec/src/enums/definitions.json +++ b/packages/ripple-binary-codec/src/enums/definitions.json @@ -2974,7 +2974,7 @@ "DIDDelete": 50, "OracleSet": 51, "OracleDelete": 52, - "NFTokenModify":53, + "NFTokenModify": 53, "EnableAmendment": 100, "SetFee": 101, "UNLModify": 102 From b46debbbe65519ea83a1108aee5816cd31200a57 Mon Sep 17 00:00:00 2001 From: yinyiqian1 Date: Fri, 5 Jul 2024 16:07:05 -0400 Subject: [PATCH 4/6] use validateRequiredField for NFTokenID check --- packages/xrpl/src/models/transactions/NFTokenModify.ts | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/packages/xrpl/src/models/transactions/NFTokenModify.ts b/packages/xrpl/src/models/transactions/NFTokenModify.ts index b74c1e8857..ac44f0afc8 100644 --- a/packages/xrpl/src/models/transactions/NFTokenModify.ts +++ b/packages/xrpl/src/models/transactions/NFTokenModify.ts @@ -8,6 +8,7 @@ import { isString, validateOptionalField, Account, + validateRequiredField, } from './common' /** @@ -51,10 +52,7 @@ export interface NFTokenModify extends BaseTransaction { export function validateNFTokenModify(tx: Record): void { validateBaseTransaction(tx) - if (tx.NFTokenID == null) { - throw new ValidationError('NFTokenModify: missing field NFTokenID') - } - + validateRequiredField(tx, 'NFTokenID', isString) validateOptionalField(tx, 'Owner', isAccount) validateOptionalField(tx, 'URI', isString) From 4bfc0fa335cb9cbdfd84b2e7a762e5b9c6bd8963 Mon Sep 17 00:00:00 2001 From: yinyiqian1 Date: Fri, 5 Jul 2024 16:09:12 -0400 Subject: [PATCH 5/6] move comment to a proper place --- .../xrpl/test/integration/transactions/nftokenModify.test.ts | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/packages/xrpl/test/integration/transactions/nftokenModify.test.ts b/packages/xrpl/test/integration/transactions/nftokenModify.test.ts index 89c30fcc3f..9c6d1b4019 100644 --- a/packages/xrpl/test/integration/transactions/nftokenModify.test.ts +++ b/packages/xrpl/test/integration/transactions/nftokenModify.test.ts @@ -29,11 +29,10 @@ describe('NFTokenModify', function () { }) afterEach(async () => teardownClient(testContext)) + // Mint an NFToken with tfMutable flag and modify URI later it( 'modify NFToken URI', async function () { - // Mint an NFToken with tfMutable flag and modify URI later - const oldUri = convertStringToHex('https://www.google.com') const newUri = convertStringToHex('https://www.ripple.com') From 6ca1b16c74dfc6124d8cdd0b7aac6dab29afd556 Mon Sep 17 00:00:00 2001 From: yinyiqian1 Date: Thu, 11 Jul 2024 10:46:40 -0400 Subject: [PATCH 6/6] Add some comment and modify integration test --- packages/xrpl/src/models/transactions/NFTokenModify.ts | 4 ++-- .../xrpl/test/integration/transactions/nftokenModify.test.ts | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/packages/xrpl/src/models/transactions/NFTokenModify.ts b/packages/xrpl/src/models/transactions/NFTokenModify.ts index ac44f0afc8..e67603fbf7 100644 --- a/packages/xrpl/src/models/transactions/NFTokenModify.ts +++ b/packages/xrpl/src/models/transactions/NFTokenModify.ts @@ -23,8 +23,8 @@ export interface NFTokenModify extends BaseTransaction { */ NFTokenID: string /** - * Indicates the AccountID of the account that owns the - * corresponding NFToken. + * Indicates the AccountID of the account that owns the corresponding NFToken. + * Can be omitted if the owner is the account submitting this transaction */ Owner?: Account /** diff --git a/packages/xrpl/test/integration/transactions/nftokenModify.test.ts b/packages/xrpl/test/integration/transactions/nftokenModify.test.ts index 9c6d1b4019..cd2b561fbc 100644 --- a/packages/xrpl/test/integration/transactions/nftokenModify.test.ts +++ b/packages/xrpl/test/integration/transactions/nftokenModify.test.ts @@ -34,7 +34,7 @@ describe('NFTokenModify', function () { 'modify NFToken URI', async function () { const oldUri = convertStringToHex('https://www.google.com') - const newUri = convertStringToHex('https://www.ripple.com') + const newUri = convertStringToHex('https://www.youtube.com') const mutableMint: NFTokenMint = { TransactionType: 'NFTokenMint',