Skip to content

Commit

Permalink
fix: add worker data fetcher service internal retry
Browse files Browse the repository at this point in the history
  • Loading branch information
vasyl-ivanchuk committed Mar 12, 2024
1 parent 9ed1cc0 commit d1ddb4d
Show file tree
Hide file tree
Showing 4 changed files with 51 additions and 11 deletions.
2 changes: 1 addition & 1 deletion packages/worker/src/config.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ describe("config", () => {
},
dataFetcher: {
url: "http://localhost:3040",
requestTimeout: 120_000,
requestTimeout: 60_000,
},
blocks: {
waitForBlocksInterval: 1000,
Expand Down
2 changes: 1 addition & 1 deletion packages/worker/src/config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -41,7 +41,7 @@ export default () => {
},
dataFetcher: {
url: DATA_FETCHER_URL || "http://localhost:3040",
requestTimeout: parseInt(DATA_FETCHER_REQUEST_TIMEOUT, 10) || 120_000,
requestTimeout: parseInt(DATA_FETCHER_REQUEST_TIMEOUT, 10) || 60_000,
},
blocks: {
waitForBlocksInterval: parseInt(WAIT_FOR_BLOCKS_INTERVAL, 10) || 1000,
Expand Down
41 changes: 33 additions & 8 deletions packages/worker/src/dataFetcher/dataFetcher.service.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,9 +5,10 @@ import { HttpService } from "@nestjs/axios";
import { ConfigService } from "@nestjs/config";
import { AxiosResponse, AxiosError } from "axios";
import * as rxjs from "rxjs";
import { setTimeout } from "node:timers/promises";
import { DataFetcherService } from "./dataFetcher.service";

jest.mock("timers/promises", () => ({
jest.mock("node:timers/promises", () => ({
setTimeout: jest.fn().mockResolvedValue(null),
}));

Expand Down Expand Up @@ -48,6 +49,7 @@ describe("DataFetcherService", () => {

describe("getBlockData", () => {
let pipeMock = jest.fn();
const blockData = { block: { number: 1 } };

beforeEach(() => {
pipeMock = jest.fn();
Expand All @@ -58,7 +60,6 @@ describe("DataFetcherService", () => {
});

it("returns block data", async () => {
const blockData = { block: { number: 1 } };
pipeMock.mockReturnValueOnce(
new rxjs.Observable((subscriber) => {
subscriber.next({
Expand All @@ -75,25 +76,49 @@ describe("DataFetcherService", () => {
expect(returnedBlockData).toEqual(blockData);
});

it("throws an error if the request fails with populated response details", async () => {
it("retries the request if the request fails with populated response details", async () => {
const error = new AxiosError("server error", "500", null, null, {
status: 500,
data: "error data",
} as AxiosResponse<string, any>);
pipeMock.mockImplementation((callback) => {
pipeMock.mockImplementationOnce((callback) => {
callback(error);
});

await expect(dataFetcherService.getBlockData(1)).rejects.toThrowError(error);
pipeMock.mockReturnValueOnce(
new rxjs.Observable((subscriber) => {
subscriber.next({
data: [blockData],
});
})
);

const returnedBlockData = await dataFetcherService.getBlockData(1);
expect(httpServiceMock.get).toBeCalledTimes(2);
expect(setTimeout).toBeCalledTimes(1);
expect(setTimeout).toBeCalledWith(1000);
expect(returnedBlockData).toEqual(blockData);
});

it("throws an error if the request fails without populated response details", async () => {
it("retries the request if the request fails without populated response details", async () => {
const error = new AxiosError("server error", "500");
pipeMock.mockImplementation((callback) => {
pipeMock.mockImplementationOnce((callback) => {
callback(error);
});

await expect(dataFetcherService.getBlockData(1)).rejects.toThrowError(error);
pipeMock.mockReturnValueOnce(
new rxjs.Observable((subscriber) => {
subscriber.next({
data: [blockData],
});
})
);

const returnedBlockData = await dataFetcherService.getBlockData(1);
expect(httpServiceMock.get).toBeCalledTimes(2);
expect(setTimeout).toBeCalledTimes(1);
expect(setTimeout).toBeCalledWith(1000);
expect(returnedBlockData).toEqual(blockData);
});
});
});
17 changes: 16 additions & 1 deletion packages/worker/src/dataFetcher/dataFetcher.service.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,9 @@ import { ConfigService } from "@nestjs/config";
import { catchError, firstValueFrom } from "rxjs";
import { AxiosError } from "axios";
import { BlockData } from "./types";
import { setTimeout } from "node:timers/promises";

const DATA_FETCHER_RETRY_TIMEOUT = 1000;

@Injectable()
export class DataFetcherService {
Expand All @@ -18,10 +21,22 @@ export class DataFetcherService {
}

public async getBlockData(blockNumber: number): Promise<BlockData> {
const blocksData = await this.getBlocksData(blockNumber, blockNumber);
const blocksData = await this.getBlocksDataRetryable(blockNumber, blockNumber);
return blocksData[0];
}

private async getBlocksDataRetryable(from: number, to: number): Promise<BlockData[]> {
try {
return await this.getBlocksData(from, to);
} catch {
this.logger.debug({
message: `Retrying to fetch data for blocks: [${from}, ${to}]`,
});
await setTimeout(DATA_FETCHER_RETRY_TIMEOUT);
return this.getBlocksDataRetryable(from, to);
}
}

private async getBlocksData(from: number, to: number): Promise<BlockData[]> {
const queryString = new URLSearchParams({
from: from.toString(),
Expand Down

0 comments on commit d1ddb4d

Please sign in to comment.