Skip to content

Commit

Permalink
fix: modified getContractResultWithRetry() to accept more general get…
Browse files Browse the repository at this point in the history
…Contract MN methods

Signed-off-by: Logan Nguyen <[email protected]>
  • Loading branch information
quiet-node committed Dec 22, 2024
1 parent 548cdeb commit d97250d
Show file tree
Hide file tree
Showing 4 changed files with 100 additions and 41 deletions.
45 changes: 32 additions & 13 deletions packages/relay/src/lib/clients/mirrorNodeClient.ts
Original file line number Diff line number Diff line change
Expand Up @@ -757,21 +757,40 @@ export class MirrorNodeClient {
* In some very rare cases the /contracts/results api is called before all the data is saved in
* the mirror node DB and `transaction_index` or `block_number` is returned as `undefined` or `block_hash` as `0x`.
* A single re-fetch is sufficient to resolve this problem.
* @param {string} transactionIdOrHash - The transaction ID or hash
* @param {RequestDetails} requestDetails - The request details for logging and tracking.
*
* @param {string} methodName - The name of the method used to fetch contract results.
* @param {any[]} args - The arguments to be passed to the specified method for fetching contract results.
* @param {RequestDetails} requestDetails - Details used for logging and tracking the request.
* @returns {Promise<any>} - A promise resolving to the fetched contract result, either on the first attempt or after a retry.
*/
public async getContractResultWithRetry(transactionIdOrHash: string, requestDetails: RequestDetails) {
const contractResult = await this.getContractResult(transactionIdOrHash, requestDetails);
if (
contractResult &&
!(
contractResult.transaction_index &&
contractResult.block_number &&
contractResult.block_hash != EthImpl.emptyHex
)
) {
return this.getContractResult(transactionIdOrHash, requestDetails);
public async getContractResultWithRetry(
methodName: string,
args: any[],
requestDetails: RequestDetails,
): Promise<any> {
const shortDelay = 500;
const contractResult = await this[methodName](...args);
const contractObjects = Array.isArray(contractResult) ? contractResult : [contractResult];

for (const contractObject of contractObjects) {
if (
contractObject &&
(contractObject.transaction_index == null ||
contractObject.block_number == null ||
contractObject.block_hash == EthImpl.emptyHex)
) {
if (this.logger.isLevelEnabled('debug')) {
this.logger.debug(

Check warning on line 783 in packages/relay/src/lib/clients/mirrorNodeClient.ts

View check run for this annotation

Codecov / codecov/patch

packages/relay/src/lib/clients/mirrorNodeClient.ts#L783

Added line #L783 was not covered by tests
`${requestDetails.formattedRequestId} Contract result contains undefined transaction_index, block_number, or block_hash set to 0x: transaction_hash:${contractObject.hash}, transaction_index:${contractObject.transaction_index}, block_number=${contractObject.block_number}, block_hash=${contractObject.block_hash}. Retrying after a delay of ${shortDelay} ms `,
);
}

// Backoff before repeating request
await new Promise((r) => setTimeout(r, shortDelay));
return await this[methodName](...args);
}
}

return contractResult;
}

Expand Down
17 changes: 13 additions & 4 deletions packages/relay/src/lib/eth.ts
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
/* -
/*-
*
* Hedera JSON RPC Relay
*
* Copyright (C) 2022-2024 Hedera Hashgraph, LLC
* Copyright (C) 2024 Hedera Hashgraph, LLC
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
Expand Down Expand Up @@ -2201,7 +2201,11 @@ export class EthImpl implements Eth {
this.logger.trace(`${requestIdPrefix} getTransactionByHash(hash=${hash})`, hash);
}

const contractResult = await this.mirrorNodeClient.getContractResultWithRetry(hash, requestDetails);
const contractResult = await this.mirrorNodeClient.getContractResultWithRetry(
this.mirrorNodeClient.getContractResult.name,
[hash, requestDetails],
requestDetails,
);
if (contractResult === null || contractResult.hash === undefined) {
// handle synthetic transactions
const syntheticLogs = await this.common.getLogsWithParams(
Expand Down Expand Up @@ -2265,7 +2269,12 @@ export class EthImpl implements Eth {
return cachedResponse;
}

const receiptResponse = await this.mirrorNodeClient.getContractResultWithRetry(hash, requestDetails);
const receiptResponse = await this.mirrorNodeClient.getContractResultWithRetry(
this.mirrorNodeClient.getContractResult.name,
[hash, requestDetails],
requestDetails,
);

if (receiptResponse === null || receiptResponse.hash === undefined) {
// handle synthetic transactions
const syntheticLogs = await this.common.getLogsWithParams(
Expand Down
19 changes: 12 additions & 7 deletions packages/relay/src/lib/services/debugService/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -18,18 +18,19 @@
*
*/

import { ConfigService } from '@hashgraph/json-rpc-config-service/dist/services';
import type { Logger } from 'pino';
import type { MirrorNodeClient } from '../../clients';
import type { IDebugService } from './IDebugService';
import type { CommonService } from '../ethService';

import { decodeErrorMessage, mapKeysAndValues, numberTo0x, strip0x } from '../../../formatters';
import type { MirrorNodeClient } from '../../clients';
import { IOpcode } from '../../clients/models/IOpcode';
import { IOpcodesResponse } from '../../clients/models/IOpcodesResponse';
import constants, { CallType, TracerType } from '../../constants';
import { predefined } from '../../errors/JsonRpcError';
import { EthImpl } from '../../eth';
import { IOpcodesResponse } from '../../clients/models/IOpcodesResponse';
import { IOpcode } from '../../clients/models/IOpcode';
import { ICallTracerConfig, IOpcodeLoggerConfig, ITracerConfig, RequestDetails } from '../../types';
import { ConfigService } from '@hashgraph/json-rpc-config-service/dist/services';
import type { CommonService } from '../ethService';
import type { IDebugService } from './IDebugService';

/**
* Represents a DebugService for tracing and debugging transactions and debugging
Expand Down Expand Up @@ -300,7 +301,11 @@ export class DebugService implements IDebugService {
try {
const [actionsResponse, transactionsResponse] = await Promise.all([
this.mirrorNodeClient.getContractsResultsActions(transactionHash, requestDetails),
this.mirrorNodeClient.getContractResultWithRetry(transactionHash, requestDetails),
this.mirrorNodeClient.getContractResultWithRetry(
this.mirrorNodeClient.getContractResult.name,
[transactionHash, requestDetails],
requestDetails,
),
]);
if (!actionsResponse || !transactionsResponse) {
throw predefined.RESOURCE_NOT_FOUND(`Failed to retrieve contract results for transaction ${transactionHash}`);
Expand Down
60 changes: 43 additions & 17 deletions packages/relay/tests/lib/mirrorNodeClient.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -19,23 +19,25 @@
*/

import { ConfigService } from '@hashgraph/json-rpc-config-service/dist/services';
import { expect } from 'chai';
import { Registry } from 'prom-client';
import { MirrorNodeClient } from '../../src/lib/clients';
import constants from '../../src/lib/constants';
import axios, { AxiosInstance } from 'axios';
import MockAdapter from 'axios-mock-adapter';
import { getRequestId, mockData, random20BytesAddress, withOverriddenEnvsInMochaTest } from '../helpers';
import pino from 'pino';
import { expect } from 'chai';
import { ethers } from 'ethers';
import pino from 'pino';
import { Registry } from 'prom-client';

import { MirrorNodeClientError, predefined } from '../../src';
import { MirrorNodeClient } from '../../src/lib/clients';
import constants from '../../src/lib/constants';
import { CacheService } from '../../src/lib/services/cacheService/cacheService';
import { getRequestId, mockData, random20BytesAddress, withOverriddenEnvsInMochaTest } from '../helpers';

const registry = new Registry();
import { MirrorNodeTransactionRecord, RequestDetails } from '../../src/lib/types';
import { SDKClientError } from '../../src/lib/errors/SDKClientError';
import { BigNumber } from 'bignumber.js';

import { SDKClientError } from '../../src/lib/errors/SDKClientError';
import { MirrorNodeTransactionRecord, RequestDetails } from '../../src/lib/types';

const logger = pino();
const noTransactions = '?transactions=false';
const requestDetails = new RequestDetails({ requestId: getRequestId(), ipAddress: '0.0.0.0' });
Expand Down Expand Up @@ -83,7 +85,7 @@ describe('MirrorNodeClient', async function () {

for (const code of nullResponseCodes) {
it(`returns null when ${code} is returned`, async () => {
let error = new Error('test error');
const error = new Error('test error');
error['response'] = 'test error';

const result = mirrorNodeInstance.handleError(
Expand All @@ -101,7 +103,7 @@ describe('MirrorNodeClient', async function () {
for (const code of errorRepsonseCodes) {
it(`throws an error when ${code} is returned`, async () => {
try {
let error = new Error('test error');
const error = new Error('test error');
error['response'] = 'test error';
mirrorNodeInstance.handleError(
error,
Expand Down Expand Up @@ -568,7 +570,11 @@ describe('MirrorNodeClient', async function () {
const hash = '0x4a563af33c4871b51a8b108aa2fe1dd5280a30dfb7236170ae5e5e7957eb6399';
mock.onGet(`contracts/results/${hash}`).reply(200, detailedContractResult);

const result = await mirrorNodeInstance.getContractResultWithRetry(hash, requestDetails);
const result = await mirrorNodeInstance.getContractResultWithRetry(
mirrorNodeInstance.getContractResult.name,
[hash, requestDetails],
requestDetails,
);
expect(result).to.exist;
expect(result.contract_id).equal(detailedContractResult.contract_id);
expect(result.to).equal(detailedContractResult.to);
Expand All @@ -585,7 +591,11 @@ describe('MirrorNodeClient', async function () {
mock.onGet(`contracts/results/${hash}`).replyOnce(200, { ...detailedContractResult, transaction_index: undefined });
mock.onGet(`contracts/results/${hash}`).reply(200, detailedContractResult);

const result = await mirrorNodeInstance.getContractResultWithRetry(hash, requestDetails);
const result = await mirrorNodeInstance.getContractResultWithRetry(
mirrorNodeInstance.getContractResult.name,
[hash, requestDetails],
requestDetails,
);
expect(result).to.exist;
expect(result.contract_id).equal(detailedContractResult.contract_id);
expect(result.to).equal(detailedContractResult.to);
Expand All @@ -601,7 +611,11 @@ describe('MirrorNodeClient', async function () {
.replyOnce(200, { ...detailedContractResult, transaction_index: undefined, block_number: undefined });
mock.onGet(`contracts/results/${hash}`).reply(200, detailedContractResult);

const result = await mirrorNodeInstance.getContractResultWithRetry(hash, requestDetails);
const result = await mirrorNodeInstance.getContractResultWithRetry(
mirrorNodeInstance.getContractResult.name,
[hash, requestDetails],
requestDetails,
);
expect(result).to.exist;
expect(result.contract_id).equal(detailedContractResult.contract_id);
expect(result.to).equal(detailedContractResult.to);
Expand All @@ -616,7 +630,11 @@ describe('MirrorNodeClient', async function () {
mock.onGet(`contracts/results/${hash}`).replyOnce(200, { ...detailedContractResult, block_number: undefined });
mock.onGet(`contracts/results/${hash}`).reply(200, detailedContractResult);

const result = await mirrorNodeInstance.getContractResultWithRetry(hash, requestDetails);
const result = await mirrorNodeInstance.getContractResultWithRetry(
mirrorNodeInstance.getContractResult.name,
[hash, requestDetails],
requestDetails,
);
expect(result).to.exist;
expect(result.contract_id).equal(detailedContractResult.contract_id);
expect(result.to).equal(detailedContractResult.to);
Expand All @@ -630,7 +648,11 @@ describe('MirrorNodeClient', async function () {
mock.onGet(`contracts/results/${hash}`).replyOnce(200, { ...detailedContractResult, block_hash: '0x' });
mock.onGet(`contracts/results/${hash}`).reply(200, detailedContractResult);

const result = await mirrorNodeInstance.getContractResultWithRetry(hash, requestDetails);
const result = await mirrorNodeInstance.getContractResultWithRetry(
mirrorNodeInstance.getContractResult.name,
[hash, requestDetails],
requestDetails,
);
expect(result).to.exist;
expect(result.block_hash).equal(detailedContractResult.block_hash);
expect(mock.history.get.length).to.eq(2);
Expand All @@ -646,7 +668,11 @@ describe('MirrorNodeClient', async function () {
});
mock.onGet(`contracts/results/${hash}`).reply(200, detailedContractResult);

const result = await mirrorNodeInstance.getContractResultWithRetry(hash, requestDetails);
const result = await mirrorNodeInstance.getContractResultWithRetry(
mirrorNodeInstance.getContractResult.name,
[hash, requestDetails],
requestDetails,
);
expect(result).to.exist;
expect(result.transaction_index).equal(detailedContractResult.transaction_index);
expect(result.block_number).equal(detailedContractResult.block_number);
Expand Down Expand Up @@ -1558,7 +1584,7 @@ describe('MirrorNodeClient', async function () {

it('should fetch contract for existing contract from cache on additional calls', async () => {
mock.onGet(contractPath).reply(200, mockData.contract);
let id = await mirrorNodeInstance.getContractId(evmAddress, requestDetails);
const id = await mirrorNodeInstance.getContractId(evmAddress, requestDetails);
expect(id).to.exist;
expect(id).to.be.equal(mockData.contract.contract_id);

Expand Down

0 comments on commit d97250d

Please sign in to comment.