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

feat: add optional fallbackNodeUrls property to RpcProvider #927

Closed
wants to merge 3 commits into from
Closed
Show file tree
Hide file tree
Changes from 2 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
21 changes: 21 additions & 0 deletions __tests__/rpcProvider.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -339,4 +339,25 @@ describeIfRpc('RPCProvider', () => {
expect(syncingStats).toMatchSchemaRef('GetSyncingStatsResponse');
});
});

describeIfRpc('Fallback node', () => {
beforeAll(() => {});
test('Ensure fallback node is used when base node fails', async () => {
const provider: RpcProvider = new RpcProvider({
nodeUrl: 'Incorrect URL',
fallbackNodeUrls: [process.env.TEST_RPC_URL!],
});
const blockNumber = await provider.getBlockNumber();
expect(typeof blockNumber).toBe('number');
});
});

test('Ensure fallback nodes are run until any of them succeeds', async () => {
const provider: RpcProvider = new RpcProvider({
nodeUrl: 'Incorrect URL',
fallbackNodeUrls: ['Another incorrect URL', process.env.TEST_RPC_URL!],
});
const blockNumber = await provider.getBlockNumber();
expect(typeof blockNumber).toBe('number');
});
});
36 changes: 32 additions & 4 deletions src/provider/rpc.ts
Original file line number Diff line number Diff line change
Expand Up @@ -80,8 +80,10 @@ export class RpcProvider implements ProviderInterface {

private chainId?: StarknetChainId;

public fallbackNodeUrls?: string[];

constructor(optionsOrProvider?: RpcProviderOptions) {
const { nodeUrl, retries, headers, blockIdentifier, chainId, rpcVersion } =
const { nodeUrl, retries, headers, blockIdentifier, chainId, rpcVersion, fallbackNodeUrls } =
optionsOrProvider || {};
if (Object.values(NetworkName).includes(nodeUrl as NetworkName)) {
// Network name provided for nodeUrl
Expand All @@ -101,16 +103,17 @@ export class RpcProvider implements ProviderInterface {
this.headers = { ...defaultOptions.headers, ...headers };
this.blockIdentifier = blockIdentifier || defaultOptions.blockIdentifier;
this.chainId = chainId; // setting to a non-null value skips making a request in getChainId()
this.fallbackNodeUrls = fallbackNodeUrls;
}

public fetch(method: string, params?: object, id: string | number = 0) {
public fetch(url: string, method: string, params?: object, id: string | number = 0) {
const rpcRequestBody: RPC.JRPC.RequestBody = {
id,
jsonrpc: '2.0',
method,
...(params && { params }),
};
return fetch(this.nodeUrl, {
return fetch(url, {
method: 'POST',
body: stringify(rpcRequestBody),
headers: this.headers as Record<string, string>,
Expand All @@ -137,11 +140,36 @@ export class RpcProvider implements ProviderInterface {
params?: RPC.Methods[T]['params']
): Promise<RPC.Methods[T]['result']> {
try {
const rawResult = await this.fetch(method, params);
const rawResult = await this.fetch(this.nodeUrl, method, params);
const { error, result } = await rawResult.json();
this.errorHandler(method, params, error);
return result as RPC.Methods[T]['result'];
} catch (error: any) {
if (this.fallbackNodeUrls) {
for (let i = 0; i < this.fallbackNodeUrls.length; i += 1) {
try {
// eslint-disable-next-line no-await-in-loop
const fallbackResult = await this.fetch(this.fallbackNodeUrls[i], method, params);
// eslint-disable-next-line no-await-in-loop
const { error: fallbackError, result } = await fallbackResult.json();
this.errorHandler(method, params, fallbackError);

// If a fallback node succeeds, update the primary and fallback URLs
const oldPrimaryUrl = this.nodeUrl;
this.nodeUrl = this.fallbackNodeUrls[i];
this.fallbackNodeUrls.splice(i, 1); // Remove the new primary from the fallback list
this.fallbackNodeUrls.push(oldPrimaryUrl); // Add the old primary to the end of the fallback list

return result as RPC.Methods[T]['result'];
} catch (fallbackError: any) {
if (i === this.fallbackNodeUrls.length - 1) {
this.errorHandler(method, params, fallbackError?.response?.data, fallbackError);
throw fallbackError;
}
}
}
}

this.errorHandler(method, params, error?.response?.data, error);
throw error;
}
Expand Down
1 change: 1 addition & 0 deletions src/types/provider/configuration.ts
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ export type RpcProviderOptions = {
blockIdentifier?: BlockIdentifier;
chainId?: StarknetChainId;
default?: boolean;
fallbackNodeUrls?: string[];
rpcVersion?: 'v0_5' | 'v0_6';
};

Expand Down