Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Nodes and tokens fetch from external api #1306

Merged
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 6 additions & 0 deletions config/config.devnet.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -94,6 +94,12 @@ features:
tps:
enabled: true
maxLookBehindNonces: 100
nodesFetch:
enabled: true
serviceUrl: 'https://devnet-api.multiversx.com'
tokensFetch:
enabled: true
serviceUrl: 'https://devnet-api.multiversx.com'
image:
width: 600
height: 600
Expand Down
8 changes: 7 additions & 1 deletion config/config.mainnet.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -97,7 +97,13 @@ features:
deadLetterQueueName: 'api-process-nfts-dlq'
tps:
enabled: true
maxLookBehindNonces: 100
maxLookBehindNonces: 100
nodesFetch:
enabled: true
serviceUrl: 'https://api.multiversx.com'
tokensFetch:
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

move the new features above dataApi
dataApi:
enabled: false
serviceUrl: 'https://data-api.multiversx.com'

enabled: true
serviceUrl: 'https://api.multiversx.com'
image:
width: 600
height: 600
Expand Down
6 changes: 6 additions & 0 deletions config/config.testnet.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -97,6 +97,12 @@ features:
tps:
enabled: true
maxLookBehindNonces: 100
nodesFetch:
enabled: true
serviceUrl: 'https://testnet-api.multiversx.com'
tokensFetch:
enabled: true
serviceUrl: 'https://testnet-api.multiversx.com'
image:
width: 600
height: 600
Expand Down
2 changes: 1 addition & 1 deletion docker-compose.yml
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ services:
environment:
- MYSQL_ROOT_PASSWORD=root
- MYSQL_DATABASE=api

GuticaStefan marked this conversation as resolved.
Show resolved Hide resolved
mongodbdatabase:
image: mongo:latest
environment:
Expand Down
26 changes: 26 additions & 0 deletions src/common/api-config/api.config.service.ts
Original file line number Diff line number Diff line change
Expand Up @@ -869,4 +869,30 @@ export class ApiConfigService {

return deepHistoryUrl;
}

isTokensFetchFeatureEnabled(): boolean {
return this.configService.get<boolean>('features.tokensFetch.enabled') ?? false;
}

getTokensFetchServiceUrl(): string {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

add some unit test for the new api config service methods in api config service spec file
isTokensFetchFeatureEnabled
getTokensFetchServiceUrl
isNodesFetchFeatureEnabled
getNodesFetchServiceUrl

const serviceUrl = this.configService.get<string>('features.tokensFetch.serviceUrl');
if (!serviceUrl) {
throw new Error('No tokens fetch service url present');
}

return serviceUrl;
}

isNodesFetchFeatureEnabled(): boolean {
return this.configService.get<boolean>('features.nodesFetch.enabled') ?? false;
}

getNodesFetchServiceUrl(): string {
const serviceUrl = this.configService.get<string>('features.nodesFetch.serviceUrl');
if (!serviceUrl) {
throw new Error('No nodes fetch service url present');
}

return serviceUrl;
}
}
21 changes: 20 additions & 1 deletion src/endpoints/nodes/node.service.ts
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ import { NodeAuction } from "./entities/node.auction";
import { NodeAuctionFilter } from "./entities/node.auction.filter";
import { Identity } from "../identities/entities/identity";
import { NodeSortAuction } from "./entities/node.sort.auction";
import { ApiService } from "@multiversx/sdk-nestjs-http";

@Injectable()
export class NodeService {
Expand All @@ -42,7 +43,8 @@ export class NodeService {
private readonly protocolService: ProtocolService,
private readonly keysService: KeysService,
@Inject(forwardRef(() => IdentitiesService))
private readonly identitiesService: IdentitiesService
private readonly identitiesService: IdentitiesService,
private readonly apiService: ApiService,
) { }

private getIssues(node: Node, version: string | undefined): string[] {
Expand Down Expand Up @@ -372,6 +374,10 @@ export class NodeService {
}

async getAllNodesRaw(): Promise<Node[]> {
if (this.apiConfigService.isNodesFetchFeatureEnabled()) {
return await this.getAllNodesFromApi();
}

const nodes = await this.getHeartbeatValidatorsAndQueue();

await this.applyNodeIdentities(nodes);
Expand All @@ -392,6 +398,19 @@ export class NodeService {
return nodes;
}

private async getAllNodesFromApi(): Promise<Node[]> {
try {
const { data } = await this.apiService.get(`${this.apiConfigService.getNodesFetchServiceUrl()}/nodes`, { params: { size: 10000 } });

return data;
} catch (error) {
this.logger.error('An unhandled error occurred when getting nodes from API');
this.logger.error(error);

throw error;
}
}

async processAuctions(nodes: Node[], auctions: Auction[]) {
const minimumAuctionStake = await this.stakeService.getMinimumAuctionStake(auctions);
const dangerZoneThreshold = BigInt(minimumAuctionStake) * BigInt(105) / BigInt(100);
Expand Down
18 changes: 17 additions & 1 deletion src/endpoints/tokens/token.service.ts
Original file line number Diff line number Diff line change
Expand Up @@ -720,8 +720,11 @@ export class TokenService {
}

async getAllTokensRaw(): Promise<TokenDetailed[]> {
this.logger.log(`Starting to fetch all tokens`);
if (this.apiConfigService.isTokensFetchFeatureEnabled()) {
return await this.getAllTokensFromApi();
}

this.logger.log(`Starting to fetch all tokens`);
const tokensProperties = await this.esdtService.getAllFungibleTokenProperties();
let tokens = tokensProperties.map(properties => ApiUtils.mergeObjects(new TokenDetailed(), properties));

Expand Down Expand Up @@ -906,6 +909,19 @@ export class TokenService {
);
}

private async getAllTokensFromApi(): Promise<TokenDetailed[]> {
try {
const { data } = await this.apiService.get(`${this.apiConfigService.getTokensFetchServiceUrl()}/tokens`, { params: { size: 10000 } });

return data;
} catch (error) {
this.logger.error('An unhandled error occurred when getting tokens from API');
this.logger.error(error);

throw error;
}
}

private async getTotalTransactions(token: TokenDetailed): Promise<{ count: number, lastUpdatedAt: number } | undefined> {
try {
const count = await this.transactionService.getTransactionCount(new TransactionFilter({ tokens: [token.identifier, ...token.assets?.extraTokens ?? []] }));
Expand Down
27 changes: 27 additions & 0 deletions src/test/unit/services/nodes.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ import { IdentitiesService } from "src/endpoints/identities/identities.service";
import { NodeAuctionFilter } from "src/endpoints/nodes/entities/node.auction.filter";
import * as fs from 'fs';
import * as path from 'path';
import { ApiService } from "@multiversx/sdk-nestjs-http";

describe('NodeService', () => {
let nodeService: NodeService;
Expand All @@ -27,6 +28,7 @@ describe('NodeService', () => {
let apiConfigService: ApiConfigService;
let gatewayService: GatewayService;
let identitiesService: IdentitiesService;
let apiService: ApiService;

beforeAll(async () => {
const moduleRef = await Test.createTestingModule({
Expand Down Expand Up @@ -58,6 +60,8 @@ describe('NodeService', () => {
getAuctionContractAddress: jest.fn(),
isNodeSyncProgressEnabled: jest.fn(),
isNodeEpochsLeftEnabled: jest.fn(),
isNodesFetchFeatureEnabled: jest.fn(),
getNodesFetchServiceUrl: jest.fn(),
},
},
{
Expand Down Expand Up @@ -108,6 +112,12 @@ describe('NodeService', () => {
getAllIdentities: jest.fn(),
},
},
{
provide: ApiService,
useValue: {
get: jest.fn(),
},
},
],
}).compile();

Expand All @@ -117,6 +127,7 @@ describe('NodeService', () => {
apiConfigService = moduleRef.get<ApiConfigService>(ApiConfigService);
gatewayService = moduleRef.get<GatewayService>(GatewayService);
identitiesService = moduleRef.get<IdentitiesService>(IdentitiesService);
apiService = moduleRef.get<ApiService>(ApiService);
});

beforeEach(() => { jest.restoreAllMocks(); });
Expand Down Expand Up @@ -427,6 +438,22 @@ describe('NodeService', () => {
});
});

describe('getAllNodes', () => {
it('should return values from external api', async () => {
const mockNodes = JSON.parse(fs.readFileSync(path.join(__dirname, '../../mocks/nodes.mock.json'), 'utf-8'));
nodeService['cacheService'].getOrSet = jest.fn().mockImplementation((_, callback) => callback());
jest.spyOn(apiConfigService, 'isNodesFetchFeatureEnabled').mockReturnValue(true);
jest.spyOn(apiConfigService, 'getNodesFetchServiceUrl').mockReturnValue('https://testnet-api.multiversx.com');
jest.spyOn(apiService, 'get').mockResolvedValueOnce({data: mockNodes});
const getHeartbeatValidatorsAndQueueSpy = jest.spyOn(nodeService, 'getHeartbeatValidatorsAndQueue');

const result = await nodeService.getAllNodes();
expect(result).toEqual(mockNodes);
expect(apiService.get).toHaveBeenCalledTimes(1);
expect(getHeartbeatValidatorsAndQueueSpy).not.toHaveBeenCalled();
});
});

describe('deleteOwnersForAddressInCache', () => {
it('should return an empty array if no cache entries are found for an address', async () => {
const address = 'erd1qqqqqqqqqqqqqpgqp699jngundfqw07d8jzkepucvpzush6k3wvqyc44rx';
Expand Down
19 changes: 19 additions & 0 deletions src/test/unit/services/tokens.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,8 @@ describe('Token Service', () => {
let collectionService: CollectionService;
let indexerService: IndexerService;
let assetsService: AssetsService;
let apiService: ApiService;
let apiConfigService: ApiConfigService;

beforeEach(async () => {
const moduleRef = await Test.createTestingModule({
Expand Down Expand Up @@ -86,6 +88,8 @@ describe('Token Service', () => {
provide: ApiConfigService,
useValue: {
getIsIndexerV3FlagActive: jest.fn(),
isTokensFetchFeatureEnabled: jest.fn(),
getTokensFetchServiceUrl: jest.fn(),
},
},
{
Expand Down Expand Up @@ -146,6 +150,8 @@ describe('Token Service', () => {
collectionService = moduleRef.get<CollectionService>(CollectionService);
indexerService = moduleRef.get<IndexerService>(IndexerService);
assetsService = moduleRef.get<AssetsService>(AssetsService);
apiService = moduleRef.get<ApiService>(ApiService);
apiConfigService = moduleRef.get<ApiConfigService>(ApiConfigService);
});

afterEach(() => {
Expand Down Expand Up @@ -612,6 +618,19 @@ describe('Token Service', () => {
},
];

it('should return values from external api', async () => {
tokenService['cachingService'].getOrSet = jest.fn().mockImplementation((_, callback) => callback());
jest.spyOn(apiConfigService, 'isTokensFetchFeatureEnabled').mockReturnValue(true);
jest.spyOn(apiConfigService, 'getTokensFetchServiceUrl').mockReturnValue('https://testnet-api.multiversx.com');
jest.spyOn(apiService, 'get').mockResolvedValueOnce({data: mockTokens});

const result = await tokenService.getAllTokens();
expect(result).toEqual(mockTokens);
expect(apiService.get).toHaveBeenCalledTimes(1);
expect(esdtService.getAllFungibleTokenProperties).not.toHaveBeenCalled();
expect(collectionService.getNftCollections).not.toHaveBeenCalled();
});

it('should return values from cache', async () => {
const cachedValueMock = jest.spyOn(tokenService['cachingService'], 'getOrSet').mockResolvedValue(mockTokens);

Expand Down
Loading