rsshub
Version:
Make RSS Great Again!
121 lines (102 loc) • 4.13 kB
text/typescript
import logger from '@/utils/logger';
import { config } from '@/config';
import undici, { Request, RequestInfo, RequestInit } from 'undici';
import proxy from '@/utils/proxy';
import { RateLimiterMemory, RateLimiterQueue } from 'rate-limiter-flexible';
import { useRegisterRequest } from 'node-network-devtools';
const limiter = new RateLimiterMemory({
points: 10,
duration: 1,
execEvenly: true,
});
const limiterQueue = new RateLimiterQueue(limiter, {
maxQueueSize: 4800,
});
export const useCustomHeader = (headers: Headers) => {
process.env.NODE_ENV === 'dev' &&
useRegisterRequest((req) => {
for (const [key, value] of headers.entries()) {
req.requestHeaders[key] = value;
}
return req;
});
};
const wrappedFetch: typeof undici.fetch = async (input: RequestInfo, init?: RequestInit) => {
const request = new Request(input, init);
const options: RequestInit = {};
logger.debug(`Outgoing request: ${request.method} ${request.url}`);
// ua
if (!request.headers.get('user-agent')) {
request.headers.set('user-agent', config.ua);
}
// accept
if (!request.headers.get('accept')) {
request.headers.set('accept', '*/*');
}
// referer
if (!request.headers.get('referer')) {
try {
const urlHandler = new URL(request.url);
request.headers.set('referer', urlHandler.origin);
} catch {
// ignore
}
}
let isRetry = false;
if (request.headers.get('x-prefer-proxy')) {
isRetry = true;
request.headers.delete('x-prefer-proxy');
}
config.enableRemoteDebugging && useCustomHeader(request.headers);
// proxy
if (!init?.dispatcher && (proxy.proxyObj.strategy !== 'on_retry' || isRetry)) {
const proxyRegex = new RegExp(proxy.proxyObj.url_regex);
let urlHandler;
try {
urlHandler = new URL(request.url);
} catch {
// ignore
}
if (proxyRegex.test(request.url) && request.url.startsWith('http') && !(urlHandler && urlHandler.host === proxy.proxyUrlHandler?.host)) {
const currentProxy = proxy.getCurrentProxy();
if (currentProxy) {
const dispatcher = proxy.getDispatcherForProxy(currentProxy);
if (dispatcher) {
options.dispatcher = dispatcher;
logger.debug(`Proxying request via ${currentProxy.uri}: ${request.url}`);
}
}
}
}
await limiterQueue.removeTokens(1);
const maxRetries = proxy.multiProxy?.allProxies.length || 1;
const attemptRequest = async (attempt: number): Promise<Response> => {
try {
return await undici.fetch(request, options);
} catch (error) {
if (options.dispatcher && proxy.multiProxy && attempt < maxRetries - 1) {
const currentProxy = proxy.getCurrentProxy();
if (currentProxy) {
logger.warn(`Request failed with proxy ${currentProxy.uri}, trying next proxy: ${error}`);
proxy.markProxyFailed(currentProxy.uri);
const nextProxy = proxy.getCurrentProxy();
if (nextProxy && nextProxy.uri !== currentProxy.uri) {
const nextDispatcher = proxy.getDispatcherForProxy(nextProxy);
if (nextDispatcher) {
options.dispatcher = nextDispatcher;
}
logger.debug(`Retrying request with proxy ${nextProxy.uri}: ${request.url}`);
return attemptRequest(attempt + 1);
} else {
logger.warn('No more proxies available, trying without proxy');
delete options.dispatcher;
return attemptRequest(attempt + 1);
}
}
}
throw error;
}
};
return attemptRequest(0);
};
export default wrappedFetch;