Skip to content

Commit 07a01e6

Browse files
Merge pull request #1185 from PolymeshAssociation/cherry-pick-lts-23
Cherry pick commits to be added to new lts v23
2 parents e81bddb + 0ef24a9 commit 07a01e6

File tree

7 files changed

+209
-105
lines changed

7 files changed

+209
-105
lines changed

src/api/entities/Asset/NonFungible/Nft.ts

+79-19
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,14 @@
11
import BigNumber from 'bignumber.js';
22

3-
import { Context, Entity, NftCollection, redeemNft } from '~/internal';
4-
import { NftMetadata, OptionalArgsProcedureMethod, RedeemNftParams } from '~/types';
3+
import { Context, Entity, NftCollection, PolymeshError, redeemNft } from '~/internal';
4+
import {
5+
DefaultPortfolio,
6+
ErrorCode,
7+
NftMetadata,
8+
NumberedPortfolio,
9+
OptionalArgsProcedureMethod,
10+
RedeemNftParams,
11+
} from '~/types';
512
import {
613
GLOBAL_BASE_IMAGE_URI_NAME,
714
GLOBAL_BASE_TOKEN_URI_NAME,
@@ -10,8 +17,11 @@ import {
1017
} from '~/utils/constants';
1118
import {
1219
bigNumberToU64,
20+
boolToBoolean,
1321
bytesToString,
1422
meshMetadataKeyToMetadataKey,
23+
meshPortfolioIdToPortfolio,
24+
portfolioToPortfolioId,
1525
stringToTicker,
1626
u64ToBigNumber,
1727
} from '~/utils/conversion';
@@ -104,26 +114,11 @@ export class Nft extends Entity<NftUniqueIdentifiers, HumanReadable> {
104114

105115
/**
106116
* Determine if the NFT exists on chain
107-
*
108-
* @note This method returns true, even if the token has been redeemed/burned
109117
*/
110118
public async exists(): Promise<boolean> {
111-
const {
112-
context,
113-
context: {
114-
polymeshApi: { query },
115-
},
116-
collection,
117-
id,
118-
} = this;
119-
const collectionId = await collection.getCollectionId();
120-
const rawCollectionId = bigNumberToU64(collectionId, context);
121-
122-
// note: "nextId" is actually the last used id
123-
const rawNextId = await query.nft.nextNFTId(rawCollectionId);
124-
const nextId = u64ToBigNumber(rawNextId);
119+
const owner = await this.getOwner();
125120

126-
return id.lte(nextId);
121+
return owner !== null;
127122
}
128123

129124
/**
@@ -198,6 +193,71 @@ export class Nft extends Entity<NftUniqueIdentifiers, HumanReadable> {
198193
return null;
199194
}
200195

196+
/**
197+
* Get owner of the NFT
198+
*
199+
* @note This method returns `null` if there is no existing holder for the token. This may happen even if the token has been redeemed/burned
200+
*/
201+
public async getOwner(): Promise<DefaultPortfolio | NumberedPortfolio | null> {
202+
const {
203+
collection: { ticker },
204+
id,
205+
context: {
206+
polymeshApi: {
207+
query: {
208+
nft: { nftOwner },
209+
},
210+
},
211+
},
212+
context,
213+
} = this;
214+
215+
const rawTicker = stringToTicker(ticker, context);
216+
const rawNftId = bigNumberToU64(id, context);
217+
218+
const owner = await nftOwner(rawTicker, rawNftId);
219+
220+
if (owner.isEmpty) {
221+
return null;
222+
}
223+
224+
return meshPortfolioIdToPortfolio(owner.unwrap(), context);
225+
}
226+
227+
/**
228+
* Check if the NFT is locked in any settlement instruction
229+
*
230+
* @throws if NFT has no owner (has been redeemed)
231+
*/
232+
public async isLocked(): Promise<boolean> {
233+
const {
234+
collection: { ticker },
235+
id,
236+
context: {
237+
polymeshApi: {
238+
query: { portfolio },
239+
},
240+
},
241+
context,
242+
} = this;
243+
244+
const owner = await this.getOwner();
245+
246+
if (!owner) {
247+
throw new PolymeshError({
248+
code: ErrorCode.DataUnavailable,
249+
message: 'NFT does not exists. The token may have been redeemed',
250+
});
251+
}
252+
253+
const rawLocked = await portfolio.portfolioLockedNFT(portfolioToPortfolioId(owner), [
254+
stringToTicker(ticker, context),
255+
bigNumberToU64(id, context),
256+
]);
257+
258+
return boolToBoolean(rawLocked);
259+
}
260+
201261
/**
202262
* @hidden
203263
*/

src/api/entities/Asset/__tests__/NonFungible/Nft.ts

+92-24
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,11 @@
11
import BigNumber from 'bignumber.js';
22
import { when } from 'jest-when';
33

4-
import { Context, Entity, Nft, PolymeshTransaction } from '~/internal';
4+
import { Context, Entity, Nft, PolymeshError, PolymeshTransaction } from '~/internal';
55
import { dsMockUtils, entityMockUtils, procedureMockUtils } from '~/testUtils/mocks';
6+
import { ErrorCode } from '~/types';
67
import { tuple } from '~/types/utils';
8+
import * as utilsConversionModule from '~/utils/conversion';
79

810
jest.mock(
911
'~/api/entities/Asset/NonFungible',
@@ -94,40 +96,23 @@ describe('Nft class', () => {
9496
});
9597

9698
describe('method: exists', () => {
97-
it('should return true when Nft Id is less than or equal to nextId for the collection', async () => {
99+
it('should return whether NFT exists or not', async () => {
98100
const ticker = 'TICKER';
99101
const context = dsMockUtils.getContextInstance();
100102
const id = new BigNumber(3);
101103
const nft = new Nft({ ticker, id }, context);
102104

103-
entityMockUtils.getNftCollectionInstance({
104-
getCollectionId: id,
105-
});
105+
const getOwnerSpy = jest.spyOn(nft, 'getOwner');
106106

107-
dsMockUtils.createQueryMock('nft', 'nextNFTId', {
108-
returnValue: new BigNumber(10),
109-
});
107+
getOwnerSpy.mockResolvedValueOnce(entityMockUtils.getDefaultPortfolioInstance());
110108

111-
const result = await nft.exists();
109+
let result = await nft.exists();
112110

113111
expect(result).toBe(true);
114-
});
115-
116-
it('should return false when Nft Id is greater than nextId for the collection', async () => {
117-
const ticker = 'TICKER';
118-
const context = dsMockUtils.getContextInstance();
119-
const id = new BigNumber(3);
120-
const nft = new Nft({ ticker, id }, context);
121112

122-
entityMockUtils.getNftCollectionInstance({
123-
getCollectionId: id,
124-
});
113+
getOwnerSpy.mockResolvedValueOnce(null);
125114

126-
dsMockUtils.createQueryMock('nft', 'nextNFTId', {
127-
returnValue: new BigNumber(1),
128-
});
129-
130-
const result = await nft.exists();
115+
result = await nft.exists();
131116

132117
expect(result).toBe(false);
133118
});
@@ -380,6 +365,89 @@ describe('Nft class', () => {
380365
});
381366
});
382367

368+
describe('method: getOwner', () => {
369+
const ticker = 'TEST';
370+
const id = new BigNumber(1);
371+
let context: Context;
372+
let nftOwnerMock: jest.Mock;
373+
let nft: Nft;
374+
375+
beforeEach(async () => {
376+
context = dsMockUtils.getContextInstance();
377+
nftOwnerMock = dsMockUtils.createQueryMock('nft', 'nftOwner');
378+
nft = new Nft({ ticker, id }, context);
379+
});
380+
381+
it('should return null if no owner exists', async () => {
382+
nftOwnerMock.mockResolvedValueOnce(dsMockUtils.createMockOption());
383+
384+
const result = await nft.getOwner();
385+
386+
expect(result).toBeNull();
387+
});
388+
389+
it('should return the owner of the NFT', async () => {
390+
const meshPortfolioIdToPortfolioSpy = jest.spyOn(
391+
utilsConversionModule,
392+
'meshPortfolioIdToPortfolio'
393+
);
394+
395+
const rawPortfolio = dsMockUtils.createMockPortfolioId({
396+
did: 'someDid',
397+
kind: dsMockUtils.createMockPortfolioKind({
398+
User: dsMockUtils.createMockU64(new BigNumber(1)),
399+
}),
400+
});
401+
402+
nftOwnerMock.mockResolvedValueOnce(dsMockUtils.createMockOption(rawPortfolio));
403+
const portfolio = entityMockUtils.getNumberedPortfolioInstance();
404+
405+
when(meshPortfolioIdToPortfolioSpy)
406+
.calledWith(rawPortfolio, context)
407+
.mockReturnValue(portfolio);
408+
409+
const result = await nft.getOwner();
410+
411+
expect(result).toBe(portfolio);
412+
});
413+
});
414+
415+
describe('method: isLocked', () => {
416+
const ticker = 'TEST';
417+
const id = new BigNumber(1);
418+
let context: Context;
419+
let nft: Nft;
420+
let ownerSpy: jest.SpyInstance;
421+
422+
beforeEach(async () => {
423+
context = dsMockUtils.getContextInstance();
424+
nft = new Nft({ ticker, id }, context);
425+
ownerSpy = jest.spyOn(nft, 'getOwner');
426+
});
427+
428+
it('should throw an error if NFT has no owner', () => {
429+
ownerSpy.mockResolvedValueOnce(null);
430+
431+
const error = new PolymeshError({
432+
code: ErrorCode.DataUnavailable,
433+
message: 'NFT does not exists. The token may have been redeemed',
434+
});
435+
return expect(nft.isLocked()).rejects.toThrow(error);
436+
});
437+
438+
it('should return whether NFT is locked in any settlement', async () => {
439+
ownerSpy.mockResolvedValue(entityMockUtils.getDefaultPortfolioInstance());
440+
441+
dsMockUtils.createQueryMock('portfolio', 'portfolioLockedNFT', {
442+
returnValue: dsMockUtils.createMockBool(true),
443+
});
444+
445+
const result = await nft.isLocked();
446+
447+
expect(result).toBe(true);
448+
});
449+
});
450+
383451
describe('method: toHuman', () => {
384452
it('should return a human readable version of the entity', () => {
385453
const context = dsMockUtils.getContextInstance();

src/api/entities/Instruction/__tests__/index.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -834,7 +834,7 @@ describe('Instruction class', () => {
834834
)
835835
.mockResolvedValue(expectedTransaction);
836836

837-
const tx = await instruction.executeManually({ id, skipAffirmationCheck: false });
837+
const tx = await instruction.executeManually({ skipAffirmationCheck: false });
838838

839839
expect(tx).toBe(expectedTransaction);
840840
});

0 commit comments

Comments
 (0)