diff --git a/smart-contracts/assembly/contracts/NFT/NFT-example.ts b/smart-contracts/assembly/contracts/NFT/NFT-example.ts index d4ea73a..b9eaecc 100644 --- a/smart-contracts/assembly/contracts/NFT/NFT-example.ts +++ b/smart-contracts/assembly/contracts/NFT/NFT-example.ts @@ -15,7 +15,7 @@ import { Args, boolToByte, stringToBytes, - u256ToBytes, + u64ToBytes, } from '@massalabs/as-types'; import { _approve, @@ -34,6 +34,9 @@ import { setOwner, onlyOwner } from '../utils/ownership'; import { Context, isDeployingContract } from '@massalabs/massa-as-sdk'; +const NAME = 'MASSA_NFT'; +const SYMBOL = 'NFT'; + /** * @param binaryArgs - serialized strings representing the name and the symbol of the NFT * @@ -44,14 +47,9 @@ import { Context, isDeployingContract } from '@massalabs/massa-as-sdk'; * * Finally, it sets the owner of the contract to the caller of the constructor. */ -export function constructor(binaryArgs: StaticArray): void { +export function constructor(_: StaticArray): void { assert(isDeployingContract()); - const args = new Args(binaryArgs); - const name = args.nextString().expect('name argument is missing or invalid'); - const symbol = args - .nextString() - .expect('symbol argument is missing or invalid'); - _constructor(name, symbol); + _constructor(NAME, SYMBOL); setOwner(new Args().add(Context.caller().toString()).serialize()); } @@ -66,7 +64,7 @@ export function symbol(): string { /** * * @param binaryArgs - serialized string representing the address whose balance we want to check - * @returns a serialized u256 representing the balance of the address + * @returns a serialized u64 representing the balance of the address * @remarks As we can see, instead of checking the storage directly, * we call the _balanceOf function from the NFT-internals. */ @@ -75,31 +73,31 @@ export function balanceOf(binaryArgs: StaticArray): StaticArray { const address = args .nextString() .expect('address argument is missing or invalid'); - return u256ToBytes(_balanceOf(address)); + return u64ToBytes(_balanceOf(address)); } /** * - * @param binaryArgs - serialized u256 representing the tokenId whose owner we want to check + * @param binaryArgs - serialized u64 representing the tokenId whose owner we want to check * @returns a serialized string representing the address of owner of the tokenId */ export function ownerOf(binaryArgs: StaticArray): StaticArray { const args = new Args(binaryArgs); const tokenId = args - .nextU256() + .nextU64() .expect('tokenId argument is missing or invalid'); return stringToBytes(_ownerOf(tokenId)); } /** * - * @param binaryArgs - serialized u256 representing the tokenId whose approved address we want to check + * @param binaryArgs - serialized u64 representing the tokenId whose approved address we want to check * @returns a serialized string representing the address of the approved address of the tokenId */ export function getApproved(binaryArgs: StaticArray): StaticArray { const args = new Args(binaryArgs); const tokenId = args - .nextU256() + .nextU64() .expect('tokenId argument is missing or invalid'); return stringToBytes(_getApproved(tokenId)); } @@ -132,7 +130,7 @@ export function approve(binaryArgs: StaticArray): void { const args = new Args(binaryArgs); const to = args.nextString().expect('to argument is missing or invalid'); const tokenId = args - .nextU256() + .nextU64() .expect('tokenId argument is missing or invalid'); _approve(to, tokenId); } @@ -165,7 +163,7 @@ export function transferFrom(binaryArgs: StaticArray): void { const from = args.nextString().expect('from argument is missing or invalid'); const to = args.nextString().expect('to argument is missing or invalid'); const tokenId = args - .nextU256() + .nextU64() .expect('tokenId argument is missing or invalid'); _transferFrom(from, to, tokenId); } @@ -190,14 +188,14 @@ export function mint(binaryArgs: StaticArray): void { const args = new Args(binaryArgs); const to = args.nextString().expect('to argument is missing or invalid'); const tokenId = args - .nextU256() + .nextU64() .expect('tokenId argument is missing or invalid'); _update(to, tokenId, ''); } /** * - * @param binaryArgs - serialized u256 representing the tokenId to burn + * @param binaryArgs - serialized u64 representing the tokenId to burn * * @remarks This function is not part of the ERC721 standard. * It serves as an example of how to use the NFT-internals functions to implement custom features. @@ -211,7 +209,7 @@ export function mint(binaryArgs: StaticArray): void { export function burn(binaryArgs: StaticArray): void { const args = new Args(binaryArgs); const tokenId = args - .nextU256() + .nextU64() .expect('tokenId argument is missing or invalid'); _update('', tokenId, ''); } diff --git a/smart-contracts/assembly/contracts/NFT/NFT-internals.ts b/smart-contracts/assembly/contracts/NFT/NFT-internals.ts index e863a05..6579fbc 100644 --- a/smart-contracts/assembly/contracts/NFT/NFT-internals.ts +++ b/smart-contracts/assembly/contracts/NFT/NFT-internals.ts @@ -19,16 +19,14 @@ import { stringToBytes, - bytesToU256, + bytesToU64, bytesToString, boolToByte, byteToBool, - u256ToBytes, + u64ToBytes, } from '@massalabs/as-types'; import { Storage, Context } from '@massalabs/massa-as-sdk'; -import { u256 } from 'as-bignum/assembly'; - export const NAME_KEY: StaticArray = [0x01]; export const SYMBOL_KEY: StaticArray = [0x02]; @@ -62,16 +60,16 @@ export function balanceKey(address: string): StaticArray { * @param tokenId - the tokenID of the owner * @returns the key of the owner in the storage for the given tokenId */ -export function ownerKey(tokenId: u256): StaticArray { - return OWNER_KEY_PREFIX.concat(u256ToBytes(tokenId)); +export function ownerKey(tokenId: u64): StaticArray { + return OWNER_KEY_PREFIX.concat(u64ToBytes(tokenId)); } /** * @param tokenId - the tokenID of the approved token * @returns the key of the allowance in the storage for the given owner and spender */ -function allowanceKey(tokenId: u256): StaticArray { - return ALLOWANCE_KEY_PREFIX.concat(u256ToBytes(tokenId)); +function allowanceKey(tokenId: u64): StaticArray { + return ALLOWANCE_KEY_PREFIX.concat(u64ToBytes(tokenId)); } /** @@ -94,9 +92,9 @@ function operatorAllowanceKey( * * @param owner - An address for whom to query the balance */ -export function _balanceOf(owner: string): u256 { +export function _balanceOf(owner: string): u64 { const key = balanceKey(owner); - return Storage.has(key) ? bytesToU256(Storage.get(key)) : u256.Zero; + return Storage.has(key) ? bytesToU64(Storage.get(key)) : 0; } /** @@ -105,7 +103,7 @@ export function _balanceOf(owner: string): u256 { * @param tokenId - The identifier for an NFT * @returns the address of the owner of the NFT or an empty string if the NFT is not owned */ -export function _ownerOf(tokenId: u256): string { +export function _ownerOf(tokenId: u64): string { const key = ownerKey(tokenId); return Storage.has(key) ? bytesToString(Storage.get(key)) : ''; } @@ -134,7 +132,7 @@ export function _symbol(): string { * @remarks If approved is the zero address, the function will clear the approval for the NFT by deleting the key. * */ -export function _approve(approved: string, tokenId: u256): void { +export function _approve(approved: string, tokenId: u64): void { assert(_isAuthorized(Context.caller().toString(), tokenId), 'Unauthorized'); const key = allowanceKey(tokenId); approved != '' @@ -149,7 +147,7 @@ export function _approve(approved: string, tokenId: u256): void { * @param tokenId - Id of the NFT * @returns Address of the approved owner of the NFT or an empty string if no address is approved. */ -export function _getApproved(tokenId: u256): string { +export function _getApproved(tokenId: u64): string { const key = allowanceKey(tokenId); return Storage.has(key) ? bytesToString(Storage.get(key)) : ''; } @@ -161,7 +159,7 @@ export function _getApproved(tokenId: u256): string { * @param tokenId - tokenId of the token * @returns true if the operator is approved, false if not */ -export function _isApproved(operator: string, tokenId: u256): bool { +export function _isApproved(operator: string, tokenId: u64): bool { const allowKey = allowanceKey(tokenId); return Storage.has(allowKey) ? bytesToString(Storage.get(allowKey)) == operator @@ -202,7 +200,7 @@ export function _isApprovedForAll(owner: string, operator: string): bool { * 2. The operator has been approved by the owner * 3. The operator has been approved for all NFTs by the owner */ -export function _isAuthorized(operator: string, tokenId: u256): bool { +export function _isAuthorized(operator: string, tokenId: u64): bool { return ( _ownerOf(tokenId) == operator || _isApproved(operator, tokenId) || @@ -227,7 +225,7 @@ export function _isAuthorized(operator: string, tokenId: u256): bool { * For example if you were to wrap this helper in a `transfer` function, * you should check that the caller is the owner of the token, and then call the _update function. */ -export function _update(to: string, tokenId: u256, auth: string): string { +export function _update(to: string, tokenId: u64, auth: string): string { const from = _ownerOf(tokenId); assert(to != from, 'The from and to addresses are the same'); if (auth != '') { @@ -237,19 +235,19 @@ export function _update(to: string, tokenId: u256, auth: string): string { // clear the approval _approve('', tokenId); // update the balance of the from - const fromBalance = bytesToU256(Storage.get(balanceKey(from))); - assert(fromBalance > u256.Zero, 'Insufficient balance'); - Storage.set(balanceKey(from), u256ToBytes(fromBalance - u256.One)); + const fromBalance = bytesToU64(Storage.get(balanceKey(from))); + assert(fromBalance > 0, 'Insufficient balance'); + Storage.set(balanceKey(from), u64ToBytes(fromBalance - 1)); } if (to != '') { const toBalanceKey = balanceKey(to); // update the balance of the to if (Storage.has(toBalanceKey)) { - const toBalance = bytesToU256(Storage.get(toBalanceKey)); - assert(toBalance < u256.Max, 'Balance overflow'); - Storage.set(toBalanceKey, u256ToBytes(toBalance + u256.One)); + const toBalance = bytesToU64(Storage.get(toBalanceKey)); + assert(toBalance < u64.MAX_VALUE, 'Balance overflow'); + Storage.set(toBalanceKey, u64ToBytes(toBalance + 1)); } else { - Storage.set(toBalanceKey, u256ToBytes(u256.One)); + Storage.set(toBalanceKey, u64ToBytes(1)); } // update the owner of the token Storage.set(ownerKey(tokenId), stringToBytes(to.toString())); @@ -271,7 +269,7 @@ export function _update(to: string, tokenId: u256, auth: string): string { * If the caller is not the owner, it must be an authorized operator to execute the function. * **/ -export function _transferFrom(from: string, to: string, tokenId: u256): void { +export function _transferFrom(from: string, to: string, tokenId: u64): void { assert( _isAuthorized(Context.caller().toString(), tokenId), 'Unauthorized caller', diff --git a/smart-contracts/assembly/contracts/NFT/NFTEnumerable-example.ts b/smart-contracts/assembly/contracts/NFT/NFTEnumerable-example.ts index 4e57664..9e0b80c 100644 --- a/smart-contracts/assembly/contracts/NFT/NFTEnumerable-example.ts +++ b/smart-contracts/assembly/contracts/NFT/NFTEnumerable-example.ts @@ -22,7 +22,7 @@ * * [TOTAL_SUPPLY_KEY] = totalSupply * - `TOTAL_SUPPLY_KEY`: A constant key for the total supply of tokens. - * - `totalSupply`: A `u256` value representing the total number of tokens in existence. + * - `totalSupply`: A `u64` value representing the total number of tokens in existence. * * - **Owned Tokens:** * @@ -56,7 +56,7 @@ import { Args, boolToByte, stringToBytes, - u256ToBytes, + u64ToBytes, } from '@massalabs/as-types'; import { _approve, @@ -75,6 +75,9 @@ import { import { setOwner, onlyOwner } from '../utils/ownership'; import { Context, isDeployingContract } from '@massalabs/massa-as-sdk'; +const NAME = 'MASSA_NFT'; +const SYMBOL = 'NFT'; + /** * @param binaryArgs - serialized strings representing the name and the symbol of the NFT * @@ -85,14 +88,9 @@ import { Context, isDeployingContract } from '@massalabs/massa-as-sdk'; * * Finally, it sets the owner of the contract to the caller of the constructor. */ -export function constructor(binaryArgs: StaticArray): void { +export function constructor(_: StaticArray): void { assert(isDeployingContract()); - const args = new Args(binaryArgs); - const name = args.nextString().expect('name argument is missing or invalid'); - const symbol = args - .nextString() - .expect('symbol argument is missing or invalid'); - _constructor(name, symbol); + _constructor(NAME, SYMBOL); setOwner(new Args().add(Context.caller().toString()).serialize()); } @@ -107,38 +105,38 @@ export function symbol(): string { /** * * @param binaryArgs - serialized string representing the address whose balance we want to check - * @returns a serialized u256 representing the balance of the address + * @returns a serialized u64 representing the balance of the address */ export function balanceOf(binaryArgs: StaticArray): StaticArray { const args = new Args(binaryArgs); const address = args .nextString() .expect('address argument is missing or invalid'); - return u256ToBytes(_balanceOf(address)); + return u64ToBytes(_balanceOf(address)); } /** * - * @param binaryArgs - serialized u256 representing the tokenId whose owner we want to check + * @param binaryArgs - serialized u64 representing the tokenId whose owner we want to check * @returns a serialized string representing the address of owner of the tokenId */ export function ownerOf(binaryArgs: StaticArray): StaticArray { const args = new Args(binaryArgs); const tokenId = args - .nextU256() + .nextU64() .expect('tokenId argument is missing or invalid'); return stringToBytes(_ownerOf(tokenId)); } /** * - * @param binaryArgs - serialized u256 representing the tokenId whose approved address we want to check + * @param binaryArgs - serialized u64 representing the tokenId whose approved address we want to check * @returns a serialized string representing the address of the approved address of the tokenId */ export function getApproved(binaryArgs: StaticArray): StaticArray { const args = new Args(binaryArgs); const tokenId = args - .nextU256() + .nextU64() .expect('tokenId argument is missing or invalid'); return stringToBytes(_getApproved(tokenId)); } @@ -171,7 +169,7 @@ export function approve(binaryArgs: StaticArray): void { const args = new Args(binaryArgs); const to = args.nextString().expect('to argument is missing or invalid'); const tokenId = args - .nextU256() + .nextU64() .expect('tokenId argument is missing or invalid'); _approve(to, tokenId); } @@ -203,7 +201,7 @@ export function transferFrom(binaryArgs: StaticArray): void { const from = args.nextString().expect('from argument is missing or invalid'); const to = args.nextString().expect('to argument is missing or invalid'); const tokenId = args - .nextU256() + .nextU64() .expect('tokenId argument is missing or invalid'); _transferFrom(from, to, tokenId); } @@ -219,14 +217,14 @@ export function mint(binaryArgs: StaticArray): void { const args = new Args(binaryArgs); const to = args.nextString().expect('to argument is missing or invalid'); const tokenId = args - .nextU256() + .nextU64() .expect('tokenId argument is missing or invalid'); _update(to, tokenId, ''); } /** * - * @param binaryArgs - serialized u256 representing the tokenId to burn + * @param binaryArgs - serialized u64 representing the tokenId to burn * * @remarks This function is not part of the ERC721 standard. * It serves as an example of how to use the NFT-enumerable-internals functions to implement custom features. @@ -234,17 +232,17 @@ export function mint(binaryArgs: StaticArray): void { export function burn(binaryArgs: StaticArray): void { const args = new Args(binaryArgs); const tokenId = args - .nextU256() + .nextU64() .expect('tokenId argument is missing or invalid'); _update('', tokenId, ''); } /** * Returns the total number of tokens. - * @returns a serialized u256 representing the total supply + * @returns a serialized u64 representing the total supply */ export function totalSupply(_: StaticArray): StaticArray { - return u256ToBytes(_totalSupply()); + return u64ToBytes(_totalSupply()); } /** diff --git a/smart-contracts/assembly/contracts/NFT/NFTEnumerable-internals.ts b/smart-contracts/assembly/contracts/NFT/NFTEnumerable-internals.ts index 8af1de1..f8f80aa 100644 --- a/smart-contracts/assembly/contracts/NFT/NFTEnumerable-internals.ts +++ b/smart-contracts/assembly/contracts/NFT/NFTEnumerable-internals.ts @@ -5,8 +5,8 @@ */ import { Context, Storage } from '@massalabs/massa-as-sdk'; -import { u256 } from 'as-bignum/assembly'; -import { bytesToU256, stringToBytes, u256ToBytes } from '@massalabs/as-types'; + +import { bytesToU64, stringToBytes, u64ToBytes } from '@massalabs/as-types'; import { _isAuthorized, _ownerOf, @@ -27,7 +27,7 @@ export const OWNED_TOKENS_KEY: StaticArray = stringToBytes('ownedTokens'); */ export function _constructor(name: string, symbol: string): void { _constructorBase(name, symbol); - Storage.set(TOTAL_SUPPLY_KEY, u256ToBytes(u256.Zero)); + Storage.set(TOTAL_SUPPLY_KEY, u64ToBytes(0)); } /* -------------------------------------------------------------------------- */ @@ -37,22 +37,22 @@ export function _constructor(name: string, symbol: string): void { /** * Returns the total number of tokens in existence. */ -export function _totalSupply(): u256 { - return bytesToU256(Storage.get(TOTAL_SUPPLY_KEY)); +export function _totalSupply(): u64 { + return bytesToU64(Storage.get(TOTAL_SUPPLY_KEY)); } /** * Increases the total supply by the given delta. * @param delta - The amount to increase the total supply by. * - * @throws Will throw an error if the addition of delta to currentSupply exceeds u256.Max. + * @throws Will throw an error if the addition of delta to currentSupply exceeds u64.Max. */ -export function _increaseTotalSupply(delta: u256): void { +export function _increaseTotalSupply(delta: u64): void { const currentSupply = _totalSupply(); - const maxAllowedDelta = u256.sub(u256.Max, currentSupply); - assert(u256.le(delta, maxAllowedDelta), 'Total supply overflow'); - const newSupply = u256.add(currentSupply, delta); - Storage.set(TOTAL_SUPPLY_KEY, u256ToBytes(newSupply)); + const maxAllowedDelta = u64.MAX_VALUE - currentSupply; + assert(delta <= maxAllowedDelta, 'Total supply overflow'); + const newSupply = currentSupply + delta; + Storage.set(TOTAL_SUPPLY_KEY, u64ToBytes(newSupply)); } /** @@ -61,11 +61,11 @@ export function _increaseTotalSupply(delta: u256): void { * * @throws Will throw an error if `delta` exceeds the current total supply, causing an underflow. */ -export function _decreaseTotalSupply(delta: u256): void { +export function _decreaseTotalSupply(delta: u64): void { const currentSupply = _totalSupply(); - assert(u256.le(delta, currentSupply), 'Total supply underflow'); - const newSupply = u256.sub(currentSupply, delta); - Storage.set(TOTAL_SUPPLY_KEY, u256ToBytes(newSupply)); + assert(delta <= currentSupply, 'Total supply underflow'); + const newSupply = currentSupply - delta; + Storage.set(TOTAL_SUPPLY_KEY, u64ToBytes(newSupply)); } /* -------------------------------------------------------------------------- */ @@ -85,9 +85,9 @@ export function _getOwnedTokensKeyPrefix(owner: string): StaticArray { * @param owner - The owner's address. * @param tokenId - The token ID to add. */ -function _addTokenToOwnerEnumeration(owner: string, tokenId: u256): void { - const key = _getOwnedTokensKeyPrefix(owner).concat(u256ToBytes(tokenId)); - Storage.set(key, u256ToBytes(tokenId)); +function _addTokenToOwnerEnumeration(owner: string, tokenId: u64): void { + const key = _getOwnedTokensKeyPrefix(owner).concat(u64ToBytes(tokenId)); + Storage.set(key, u64ToBytes(tokenId)); } /** @@ -95,8 +95,8 @@ function _addTokenToOwnerEnumeration(owner: string, tokenId: u256): void { * @param owner - The owner's address. * @param tokenId - The token ID to remove. */ -function _removeTokenFromOwnerEnumeration(owner: string, tokenId: u256): void { - const key = _getOwnedTokensKeyPrefix(owner).concat(u256ToBytes(tokenId)); +function _removeTokenFromOwnerEnumeration(owner: string, tokenId: u64): void { + const key = _getOwnedTokensKeyPrefix(owner).concat(u64ToBytes(tokenId)); Storage.del(key); } @@ -110,13 +110,13 @@ function _removeTokenFromOwnerEnumeration(owner: string, tokenId: u256): void { * @param tokenId - The token ID. * @param auth - The address authorized to perform the update. */ -export function _update(to: string, tokenId: u256, auth: string): void { +export function _update(to: string, tokenId: u64, auth: string): void { const previousOwner = _updateBase(to, tokenId, auth); // Mint if (previousOwner == '') { _addTokenToOwnerEnumeration(to, tokenId); - _increaseTotalSupply(u256.One); + _increaseTotalSupply(1); } else { // Transfer if (to != '' && to != previousOwner) { @@ -126,7 +126,7 @@ export function _update(to: string, tokenId: u256, auth: string): void { // Burn else if (to == '') { _removeTokenFromOwnerEnumeration(previousOwner, tokenId); - _decreaseTotalSupply(u256.One); + _decreaseTotalSupply(1); } } } @@ -141,7 +141,7 @@ export function _update(to: string, tokenId: u256, auth: string): void { * @param to - The new owner's address. * @param tokenId - The token ID to transfer. */ -export function _transferFrom(from: string, to: string, tokenId: u256): void { +export function _transferFrom(from: string, to: string, tokenId: u64): void { assert( _isAuthorized(Context.caller().toString(), tokenId), 'Unauthorized caller', diff --git a/smart-contracts/assembly/contracts/NFT/__tests__/NFT-example.spec.ts b/smart-contracts/assembly/contracts/NFT/__tests__/NFT-example.spec.ts index dd145d7..bf5c90e 100644 --- a/smart-contracts/assembly/contracts/NFT/__tests__/NFT-example.spec.ts +++ b/smart-contracts/assembly/contracts/NFT/__tests__/NFT-example.spec.ts @@ -20,11 +20,11 @@ import { } from '../NFT-example'; import { Args, + NoArg, byteToBool, bytesToString, - bytesToU256, + bytesToU64, } from '@massalabs/as-types'; -import { u256 } from 'as-bignum/assembly'; const NFTName = 'MASSA_NFT'; const NFTSymbol = 'NFT'; @@ -35,7 +35,7 @@ const to = 'AU178qZCfaNXkz9tQiXJcVfAEnYGJ27UoNtFFJh3BiT8jTfY8P2D'; const approved = 'AU1sF3HSa7fcBoE12bE1Eq2ohKqcRPBHuNRmdqAMfw8WEkHCU3aF'; const newOwner = 'AU12F7y3PWpw72XcwhSksJztRiTSqAvLxaLacP2qDYhNUEfEXuG4T'; const zeroAddress = ''; -const tokenId = u256.One; +const tokenId: u64 = 1; function switchUser(user: string): void { changeCallStack(user + ' , ' + tokenAddress); @@ -45,7 +45,7 @@ beforeEach(() => { resetStorage(); switchUser(contractOwner); setDeployContext(contractOwner); - constructor(new Args().add(NFTName).add(NFTSymbol).serialize()); + constructor(NoArg.serialize()); }); describe('Initialization', () => { @@ -67,9 +67,9 @@ describe('Minting', () => { expect(bytesToString(ownerOf(new Args().add(tokenId).serialize()))).toBe( to, ); - expect( - bytesToU256(balanceOf(new Args().add(to).serialize())), - ).toStrictEqual(u256.One); + expect(bytesToU64(balanceOf(new Args().add(to).serialize()))).toStrictEqual( + 1, + ); }); throws('Minting from not owner should fail', () => { switchUser(from); @@ -84,23 +84,23 @@ describe('Minting', () => { }); test('Mint multiple tokens to an address', () => { mint(new Args().add(to).add(tokenId).serialize()); - mint(new Args().add(to).add(new u256(2)).serialize()); - expect( - bytesToU256(balanceOf(new Args().add(to).serialize())), - ).toStrictEqual(new u256(2)); + mint(new Args().add(to).add(u64(2)).serialize()); + expect(bytesToU64(balanceOf(new Args().add(to).serialize()))).toStrictEqual( + 2, + ); }); test('Mint multiple tokens to different addresses', () => { mint(new Args().add(to).add(tokenId).serialize()); - mint(new Args().add(from).add(new u256(2)).serialize()); - expect( - bytesToU256(balanceOf(new Args().add(to).serialize())), - ).toStrictEqual(u256.One); - expect( - bytesToString(ownerOf(new Args().add(new u256(2)).serialize())), - ).toBe(from); + mint(new Args().add(from).add(u64(2)).serialize()); + expect(bytesToU64(balanceOf(new Args().add(to).serialize()))).toStrictEqual( + 1, + ); + expect(bytesToString(ownerOf(new Args().add(u64(2)).serialize()))).toBe( + from, + ); expect( - bytesToU256(balanceOf(new Args().add(from).serialize())), - ).toStrictEqual(u256.One); + bytesToU64(balanceOf(new Args().add(from).serialize())), + ).toStrictEqual(1); expect(bytesToString(ownerOf(new Args().add(tokenId).serialize()))).toBe( to, ); diff --git a/smart-contracts/assembly/contracts/NFT/__tests__/NFT-internals.spec.ts b/smart-contracts/assembly/contracts/NFT/__tests__/NFT-internals.spec.ts index 390be02..34c89be 100644 --- a/smart-contracts/assembly/contracts/NFT/__tests__/NFT-internals.spec.ts +++ b/smart-contracts/assembly/contracts/NFT/__tests__/NFT-internals.spec.ts @@ -4,7 +4,6 @@ import { setDeployContext, } from '@massalabs/massa-as-sdk'; -import { u256 } from 'as-bignum/assembly'; import { _approve, _balanceOf, @@ -27,7 +26,7 @@ const to = 'AU178qZCfaNXkz9tQiXJcVfAEnYGJ27UoNtFFJh3BiT8jTfY8P2D'; const approved = 'AU1sF3HSa7fcBoE12bE1Eq2ohKqcRPBHuNRmdqAMfw8WEkHCU3aF'; const newOwner = 'AU12F7y3PWpw72XcwhSksJztRiTSqAvLxaLacP2qDYhNUEfEXuG4T'; const zeroAddress = ''; -const tokenId = u256.One; +const tokenId: u64 = 1; const NFTName = 'MASSA_NFT'; const NFTSymbol = 'NFT'; @@ -125,10 +124,10 @@ describe('Transferring NFTs', () => { expect(ownerOfToken).toBe(newOwner); const balanceOfNewOwner = _balanceOf(newOwner); - expect(balanceOfNewOwner).toBe(u256.One); + expect(balanceOfNewOwner).toBe(1); const balanceOfOldOwner = _balanceOf(from); - expect(balanceOfOldOwner).toBe(u256.Zero); + expect(balanceOfOldOwner).toBe(0); }); throws('Transferring a non-existent token should fail', () => { _transferFrom(from, to, tokenId); diff --git a/smart-contracts/assembly/contracts/NFT/__tests__/NFTEnumerable-example.spec.ts b/smart-contracts/assembly/contracts/NFT/__tests__/NFTEnumerable-example.spec.ts index 866f72e..8c20109 100644 --- a/smart-contracts/assembly/contracts/NFT/__tests__/NFTEnumerable-example.spec.ts +++ b/smart-contracts/assembly/contracts/NFT/__tests__/NFTEnumerable-example.spec.ts @@ -6,11 +6,11 @@ import { import { Args, + NoArg, byteToBool, bytesToString, - bytesToU256, + bytesToU64, } from '@massalabs/as-types'; -import { u256 } from 'as-bignum/assembly'; import { approve, balanceOf, @@ -38,13 +38,7 @@ const from = 'AU12CzoKEASaeBHnxGLnHDG2u73dLzWWfgvW6bc4L1UfMA5Uc5Fg7'; const to = 'AU178qZCfaNXkz9tQiXJcVfAEnYGJ27UoNtFFJh3BiT8jTfY8P2D'; const approved = 'AU1sF3HSa7fcBoE12bE1Eq2ohKqcRPBHuNRmdqAMfw8WEkHCU3aF'; const zeroAddress = ''; -const tokenIds = [ - u256.One, - u256.fromU32(2), - u256.fromU32(3), - u256.fromU32(4), - u256.fromU32(5), -]; +const tokenIds: u64[] = [1, 2, 3, 4, 5]; function switchUser(user: string): void { changeCallStack(user + ' , ' + tokenAddress); @@ -54,58 +48,58 @@ beforeEach(() => { resetStorage(); switchUser(contractOwner); setDeployContext(contractOwner); - constructor(new Args().add(NFTName).add(NFTSymbol).serialize()); + constructor(NoArg.serialize()); }); describe('NFT Enumerable Contract', () => { describe('Initialization', () => { - test('should return correct name and symbol', () => { + it('should return correct name and symbol', () => { expect(name()).toBe(NFTName); expect(symbol()).toBe(NFTSymbol); }); - test('should return correct contract owner', () => { + it('should return correct contract owner', () => { expect(bytesToString(ownerAddress([]))).toBe(contractOwner); }); }); describe('Minting', () => { - test('should mint a token to an address', () => { + it('should mint a token to an address', () => { mint(new Args().add(to).add(tokenIds[0]).serialize()); expect( bytesToString(ownerOf(new Args().add(tokenIds[0]).serialize())), ).toBe(to); expect( - bytesToU256(balanceOf(new Args().add(to).serialize())), - ).toStrictEqual(u256.One); - expect(bytesToU256(totalSupply([]))).toStrictEqual(u256.One); + bytesToU64(balanceOf(new Args().add(to).serialize())), + ).toStrictEqual(1); + expect(bytesToU64(totalSupply([]))).toStrictEqual(1); }); - test('should mint multiple tokens to different addresses', () => { + it('should mint multiple tokens to different addresses', () => { mint(new Args().add(to).add(tokenIds[0]).serialize()); mint(new Args().add(from).add(tokenIds[1]).serialize()); expect( - bytesToU256(balanceOf(new Args().add(to).serialize())), - ).toStrictEqual(u256.One); + bytesToU64(balanceOf(new Args().add(to).serialize())), + ).toStrictEqual(1); expect( - bytesToU256(balanceOf(new Args().add(from).serialize())), - ).toStrictEqual(u256.One); - expect(bytesToU256(totalSupply([]))).toStrictEqual(u256.fromU32(2)); + bytesToU64(balanceOf(new Args().add(from).serialize())), + ).toStrictEqual(1); + expect(bytesToU64(totalSupply([]))).toStrictEqual(u64(2)); }); - test('should not mint to zero address', () => { + it('should mint to an invalid address', () => { expect(() => { mint(new Args().add(zeroAddress).add(tokenIds[0]).serialize()); }).toThrow('Unauthorized to'); }); - test('should not mint an already existing tokenId', () => { + it('should not mint an already existing tokenId', () => { mint(new Args().add(to).add(tokenIds[0]).serialize()); expect(() => { mint(new Args().add(to).add(tokenIds[0]).serialize()); }).toThrow('Token already minted'); }); - test('should not allow non-owner to mint tokens', () => { + it('should not allow non-owner to mint tokens', () => { switchUser(from); expect(() => { mint(new Args().add(to).add(tokenIds[0]).serialize()); @@ -114,7 +108,7 @@ describe('NFT Enumerable Contract', () => { }); describe('Approval', () => { - test('should approve a token for an address', () => { + it('should approve a token for an address', () => { mint(new Args().add(from).add(tokenIds[0]).serialize()); switchUser(from); approve(new Args().add(approved).add(tokenIds[0]).serialize()); @@ -123,7 +117,7 @@ describe('NFT Enumerable Contract', () => { ).toBe(approved); }); - test('should set approval for all', () => { + it('should set approval for all', () => { mint(new Args().add(from).add(tokenIds[0]).serialize()); switchUser(from); setApprovalForAll(new Args().add(approved).add(true).serialize()); @@ -134,7 +128,7 @@ describe('NFT Enumerable Contract', () => { ).toBe(true); }); - test('should revoke approval for all', () => { + it('should revoke approval for all', () => { mint(new Args().add(from).add(tokenIds[0]).serialize()); switchUser(from); setApprovalForAll(new Args().add(approved).add(true).serialize()); @@ -146,7 +140,7 @@ describe('NFT Enumerable Contract', () => { ).toBe(false); }); - test('should not approve token not owned', () => { + it('should not approve token not owned', () => { mint(new Args().add(from).add(tokenIds[0]).serialize()); switchUser(approved); expect(() => { @@ -156,7 +150,7 @@ describe('NFT Enumerable Contract', () => { }); describe('Transfers', () => { - test('should transfer token from owner', () => { + it('should transfer token from owner', () => { mint(new Args().add(from).add(tokenIds[0]).serialize()); switchUser(from); transferFrom(new Args().add(from).add(to).add(tokenIds[0]).serialize()); @@ -164,14 +158,14 @@ describe('NFT Enumerable Contract', () => { bytesToString(ownerOf(new Args().add(tokenIds[0]).serialize())), ).toBe(to); expect( - bytesToU256(balanceOf(new Args().add(to).serialize())), - ).toStrictEqual(u256.One); + bytesToU64(balanceOf(new Args().add(to).serialize())), + ).toStrictEqual(1); expect( - bytesToU256(balanceOf(new Args().add(from).serialize())), - ).toStrictEqual(u256.Zero); + bytesToU64(balanceOf(new Args().add(from).serialize())), + ).toStrictEqual(0); }); - test('should transfer approved token', () => { + it('should transfer approved token', () => { mint(new Args().add(from).add(tokenIds[0]).serialize()); switchUser(from); approve(new Args().add(approved).add(tokenIds[0]).serialize()); @@ -182,7 +176,7 @@ describe('NFT Enumerable Contract', () => { ).toBe(to); }); - test('should transfer token using approval for all', () => { + it('should transfer token using approval for all', () => { mint(new Args().add(from).add(tokenIds[0]).serialize()); switchUser(from); setApprovalForAll(new Args().add(approved).add(true).serialize()); @@ -193,7 +187,7 @@ describe('NFT Enumerable Contract', () => { ).toBe(to); }); - test('should not transfer token without approval', () => { + it('should not transfer token without approval', () => { mint(new Args().add(from).add(tokenIds[0]).serialize()); switchUser(approved); expect(() => { @@ -203,7 +197,7 @@ describe('NFT Enumerable Contract', () => { }); describe('Burning', () => { - test('should burn a token', () => { + it('should burn a token', () => { mint(new Args().add(from).add(tokenIds[0]).serialize()); switchUser(from); burn(new Args().add(tokenIds[0]).serialize()); @@ -212,10 +206,10 @@ describe('NFT Enumerable Contract', () => { [], ); - expect(bytesToU256(totalSupply([]))).toStrictEqual(u256.Zero); + expect(bytesToU64(totalSupply([]))).toStrictEqual(0); }); - test('should burn a token with approval', () => { + it('should burn a token with approval', () => { mint(new Args().add(from).add(tokenIds[0]).serialize()); switchUser(from); approve(new Args().add(approved).add(tokenIds[0]).serialize()); @@ -224,10 +218,10 @@ describe('NFT Enumerable Contract', () => { expect(ownerOf(new Args().add(tokenIds[0]).serialize())).toStrictEqual( [], ); - expect(bytesToU256(totalSupply([]))).toStrictEqual(u256.Zero); + expect(bytesToU64(totalSupply([]))).toStrictEqual(0); }); - test('should burn a token using approval for all', () => { + it('should burn a token using approval for all', () => { mint(new Args().add(from).add(tokenIds[0]).serialize()); switchUser(from); setApprovalForAll(new Args().add(approved).add(true).serialize()); @@ -236,10 +230,10 @@ describe('NFT Enumerable Contract', () => { expect(ownerOf(new Args().add(tokenIds[0]).serialize())).toStrictEqual( [], ); - expect(bytesToU256(totalSupply([]))).toStrictEqual(u256.Zero); + expect(bytesToU64(totalSupply([]))).toStrictEqual(0); }); - test('should not burn token without approval', () => { + it('should not burn token without approval', () => { mint(new Args().add(from).add(tokenIds[0]).serialize()); switchUser(to); expect(() => { @@ -249,15 +243,15 @@ describe('NFT Enumerable Contract', () => { }); describe('Enumeration', () => { - test('should return correct total supply', () => { - expect(bytesToU256(totalSupply([]))).toStrictEqual(u256.Zero); + it('should return correct total supply', () => { + expect(bytesToU64(totalSupply([]))).toStrictEqual(0); mint(new Args().add(from).add(tokenIds[0]).serialize()); - expect(bytesToU256(totalSupply([]))).toStrictEqual(u256.One); + expect(bytesToU64(totalSupply([]))).toStrictEqual(1); mint(new Args().add(to).add(tokenIds[1]).serialize()); - expect(bytesToU256(totalSupply([]))).toStrictEqual(u256.fromU32(2)); + expect(bytesToU64(totalSupply([]))).toStrictEqual(2); }); - test('should return correct tokens owned by an address', () => { + it('should return correct tokens owned by an address', () => { // Assuming we have an exported function to get owned tokens mint(new Args().add(from).add(tokenIds[0]).serialize()); mint(new Args().add(from).add(tokenIds[1]).serialize()); @@ -275,7 +269,7 @@ describe('NFT Enumerable Contract', () => { expect(toTokens).toContainEqual(tokenIds[2]); }); - test('should update owned tokens after transfer', () => { + it('should update owned tokens after transfer', () => { mint(new Args().add(from).add(tokenIds[0]).serialize()); mint(new Args().add(from).add(tokenIds[1]).serialize()); switchUser(from); @@ -292,7 +286,7 @@ describe('NFT Enumerable Contract', () => { expect(toTokens).toContainEqual(tokenIds[0]); }); - test('should update owned tokens after burn', () => { + it('should update owned tokens after burn', () => { mint(new Args().add(from).add(tokenIds[0]).serialize()); mint(new Args().add(from).add(tokenIds[1]).serialize()); switchUser(from); @@ -304,7 +298,7 @@ describe('NFT Enumerable Contract', () => { expect(fromTokens).toContainEqual(tokenIds[1]); // Total supply should be updated - expect(bytesToU256(totalSupply([]))).toStrictEqual(u256.One); + expect(bytesToU64(totalSupply([]))).toStrictEqual(1); }); }); }); diff --git a/smart-contracts/assembly/contracts/NFT/__tests__/NFTEnumerable-internals.spec.ts b/smart-contracts/assembly/contracts/NFT/__tests__/NFTEnumerable-internals.spec.ts index c89d903..d20e474 100644 --- a/smart-contracts/assembly/contracts/NFT/__tests__/NFTEnumerable-internals.spec.ts +++ b/smart-contracts/assembly/contracts/NFT/__tests__/NFTEnumerable-internals.spec.ts @@ -1,5 +1,4 @@ import { resetStorage, setDeployContext } from '@massalabs/massa-as-sdk'; -import { u256 } from 'as-bignum/assembly'; import { _update, _balanceOf, @@ -16,26 +15,20 @@ const caller = 'A12UBnqTHDQALpocVBnkPNy7y5CndUJQTLutaVDDFgMJcq5kQiKq'; const owner1 = caller; const owner2 = 'AU178qZCfaNXkz9tQiXJcVfAEnYGJ27UoNtFFJh3BiT8jTfY8P2D'; const zeroAddress = ''; -const tokenIds = [ - u256.fromU32(1), - u256.fromU32(2), - u256.fromU32(3), - u256.fromU32(4), - u256.fromU32(5), -]; +const tokenIds: u64[] = [1, 2, 3, 4, 5]; const NFTName = 'MASSA_NFT'; const NFTSymbol = 'NFT'; -function mint(to: string, tokenId: u256): void { +function mint(to: string, tokenId: u64): void { _update(to, tokenId, zeroAddress); } -function transfer(from: string, to: string, tokenId: u256): void { +function transfer(from: string, to: string, tokenId: u64): void { _update(to, tokenId, from); } -function burn(owner: string, tokenId: u256): void { +function burn(owner: string, tokenId: u64): void { _update(zeroAddress, tokenId, owner); } @@ -48,59 +41,59 @@ describe('NFT Enumerable Internals', () => { describe('Initialization', () => { it('should have zero total supply initially', () => { - expect(_totalSupply()).toStrictEqual(u256.Zero); + expect(_totalSupply()).toStrictEqual(0); }); }); describe('Total Supply Management', () => { it('should update total supply when token is minted', () => { mint(owner1, tokenIds[0]); - expect(_totalSupply()).toStrictEqual(u256.One); + expect(_totalSupply()).toStrictEqual(1); }); it('should update total supply when token is burned', () => { mint(owner1, tokenIds[0]); - expect(_totalSupply()).toStrictEqual(u256.One); + expect(_totalSupply()).toStrictEqual(1); burn(owner1, tokenIds[0]); - expect(_totalSupply()).toStrictEqual(u256.Zero); + expect(_totalSupply()).toStrictEqual(0); }); - it('should not allow total supply to exceed u256.Max', () => { - // Set total supply to u256.Max - 1 - const nearMaxSupply = u256.sub(u256.Max, u256.One); - _increaseTotalSupply(u256.sub(u256.Max, u256.One)); + it('should not allow total supply to exceed u64.Max', () => { + // Set total supply to u64.Max - 1 + const nearMaxSupply = u64.MAX_VALUE - 1; + _increaseTotalSupply(nearMaxSupply); expect(_totalSupply()).toStrictEqual(nearMaxSupply); - // Mint one more token should succeed (totalSupply = u256.Max) + // Mint one more token should succeed (totalSupply = u64.Max) mint(owner1, tokenIds[0]); - expect(_totalSupply()).toStrictEqual(u256.Max); + expect(_totalSupply()).toStrictEqual(u64.MAX_VALUE); // Minting another token should fail due to overflow expect(() => { - _increaseTotalSupply(u256.One); + _increaseTotalSupply(1); }).toThrow('Total supply overflow'); // Ensure your contract throws this exact error }); it('should not allow total supply to underflow', () => { // Ensure total supply is zero - expect(_totalSupply()).toStrictEqual(u256.Zero); + expect(_totalSupply()).toStrictEqual(0); // Attempt to decrease supply by 1 should fail expect(() => { - _decreaseTotalSupply(u256.One); + _decreaseTotalSupply(1); }).toThrow('Total supply underflow'); // Ensure your contract throws this exact error // Set total supply to 1 - _increaseTotalSupply(u256.One); - expect(_totalSupply()).toStrictEqual(u256.One); + _increaseTotalSupply(1); + expect(_totalSupply()).toStrictEqual(1); // Decrease supply by 1 should succeed - _decreaseTotalSupply(u256.One); - expect(_totalSupply()).toStrictEqual(u256.Zero); + _decreaseTotalSupply(1); + expect(_totalSupply()).toStrictEqual(0); // Attempt to decrease supply by another 1 should fail expect(() => { - _decreaseTotalSupply(u256.One); + _decreaseTotalSupply(1); }).toThrow('Total supply underflow'); // Ensure your contract throws this exact error }); }); @@ -111,8 +104,8 @@ describe('NFT Enumerable Internals', () => { mint(owner1, tokenIds[1]); mint(owner2, tokenIds[2]); - expect(_balanceOf(owner1)).toStrictEqual(u256.fromU32(2)); - expect(_balanceOf(owner2)).toStrictEqual(u256.One); + expect(_balanceOf(owner1)).toStrictEqual(2); + expect(_balanceOf(owner2)).toStrictEqual(1); const owner1Tokens = getOwnedTokens(owner1); expect(owner1Tokens.length).toBe(2); @@ -130,8 +123,8 @@ describe('NFT Enumerable Internals', () => { transfer(owner1, owner2, tokenIds[0]); - expect(_balanceOf(owner1)).toStrictEqual(u256.One); - expect(_balanceOf(owner2)).toStrictEqual(u256.One); + expect(_balanceOf(owner1)).toStrictEqual(1); + expect(_balanceOf(owner2)).toStrictEqual(1); // Verify ownership expect(_ownerOf(tokenIds[0])).toStrictEqual(owner2); @@ -154,8 +147,8 @@ describe('NFT Enumerable Internals', () => { transfer(owner1, owner2, tokenIds[0]); expect(_ownerOf(tokenIds[0])).toStrictEqual(owner2); - expect(_balanceOf(owner1)).toStrictEqual(u256.Zero); - expect(_balanceOf(owner2)).toStrictEqual(u256.One); + expect(_balanceOf(owner1)).toStrictEqual(0); + expect(_balanceOf(owner2)).toStrictEqual(1); // Verify owned tokens const owner1Tokens = getOwnedTokens(owner1); @@ -173,8 +166,8 @@ describe('NFT Enumerable Internals', () => { transfer(owner1, owner2, tokenIds[0]); expect(_ownerOf(tokenIds[0])).toStrictEqual(owner2); - expect(_balanceOf(owner1)).toStrictEqual(u256.Zero); - expect(_balanceOf(owner2)).toStrictEqual(u256.fromU32(2)); + expect(_balanceOf(owner1)).toStrictEqual(0); + expect(_balanceOf(owner2)).toStrictEqual(2); // Verify owned tokens const owner1Tokens = getOwnedTokens(owner1); @@ -194,8 +187,8 @@ describe('NFT Enumerable Internals', () => { burn(owner1, tokenIds[0]); - expect(_balanceOf(owner1)).toStrictEqual(u256.One); - expect(_totalSupply()).toStrictEqual(u256.One); + expect(_balanceOf(owner1)).toStrictEqual(1); + expect(_totalSupply()).toStrictEqual(1); // Verify that accessing the burned token's owner returns zero address expect(_ownerOf(tokenIds[0])).toStrictEqual(zeroAddress); @@ -213,8 +206,8 @@ describe('NFT Enumerable Internals', () => { burn(owner1, tokenIds[0]); burn(owner1, tokenIds[1]); - expect(_balanceOf(owner1)).toStrictEqual(u256.Zero); - expect(_totalSupply()).toStrictEqual(u256.Zero); + expect(_balanceOf(owner1)).toStrictEqual(0); + expect(_totalSupply()).toStrictEqual(0); // Verify owned tokens const owner1Tokens = getOwnedTokens(owner1); @@ -236,11 +229,11 @@ describe('NFT Enumerable Internals', () => { burn(owner1, tokenIds[0]); // Verify total supply - expect(_totalSupply()).toStrictEqual(u256.fromU32(2)); + expect(_totalSupply()).toStrictEqual(2); // Verify balances - expect(_balanceOf(owner1)).toStrictEqual(u256.Zero); - expect(_balanceOf(owner2)).toStrictEqual(u256.fromU32(2)); + expect(_balanceOf(owner1)).toStrictEqual(0); + expect(_balanceOf(owner2)).toStrictEqual(2); // Verify ownership expect(_ownerOf(tokenIds[1])).toStrictEqual(owner2); @@ -293,7 +286,7 @@ describe('NFT Enumerable Internals', () => { mint(owner1, tokenIds[0]); burn(owner1, tokenIds[0]); - expect(_totalSupply()).toStrictEqual(u256.Zero); + expect(_totalSupply()).toStrictEqual(0); // Check token ownership has been cleared expect(_ownerOf(tokenIds[0])).toStrictEqual(zeroAddress); @@ -303,14 +296,14 @@ describe('NFT Enumerable Internals', () => { expect(owner1Tokens.length).toBe(0); }); - it('should mint a token with id=u256.Max', () => { - mint(owner1, u256.Max); - expect(_totalSupply()).toStrictEqual(u256.One); - expect(_balanceOf(owner1)).toStrictEqual(u256.One); - expect(_ownerOf(u256.Max)).toStrictEqual(owner1); + it('should mint a token with id=u64.Max', () => { + mint(owner1, u64.MAX_VALUE); + expect(_totalSupply()).toStrictEqual(1); + expect(_balanceOf(owner1)).toStrictEqual(1); + expect(_ownerOf(u64.MAX_VALUE)).toStrictEqual(owner1); const owner1Tokens = getOwnedTokens(owner1); expect(owner1Tokens.length).toBe(1); - expect(owner1Tokens).toContainEqual(u256.Max); + expect(owner1Tokens).toContainEqual(u64.MAX_VALUE); }); }); }); diff --git a/smart-contracts/assembly/contracts/NFT/__tests__/helpers.ts b/smart-contracts/assembly/contracts/NFT/__tests__/helpers.ts index e4a0312..a2a20ea 100644 --- a/smart-contracts/assembly/contracts/NFT/__tests__/helpers.ts +++ b/smart-contracts/assembly/contracts/NFT/__tests__/helpers.ts @@ -1,6 +1,5 @@ -import { bytesToU256 } from '@massalabs/as-types'; +import { bytesToU64 } from '@massalabs/as-types'; import { getKeys, Storage } from '@massalabs/massa-as-sdk'; -import { u256 } from 'as-bignum/assembly'; import { _getOwnedTokensKeyPrefix } from '../NFTEnumerable-internals'; /** @@ -8,15 +7,15 @@ import { _getOwnedTokensKeyPrefix } from '../NFTEnumerable-internals'; * * @param owner - The address of the owner. * - * @returns An array of u256 representing the tokens owned by the address. + * @returns An array of u64 representing the tokens owned by the address. * */ -export function getOwnedTokens(owner: string): u256[] { - const tokens: u256[] = []; +export function getOwnedTokens(owner: string): u64[] { + const tokens: u64[] = []; const keys = getKeys(_getOwnedTokensKeyPrefix(owner)); for (let i = 0; i < keys.length; i++) { - tokens.push(bytesToU256(Storage.get(keys[i]))); + tokens.push(bytesToU64(Storage.get(keys[i]))); } return tokens; }