Skip to content

Commit

Permalink
chore: code-split of the long (interceptor.ts) file into smaller pieces
Browse files Browse the repository at this point in the history
  • Loading branch information
peter-sanderson committed Jan 8, 2025
1 parent 31e1cf8 commit 0d4c182
Show file tree
Hide file tree
Showing 9 changed files with 359 additions and 324 deletions.
329 changes: 5 additions & 324 deletions packages/request-manager/src/interceptor.ts
Original file line number Diff line number Diff line change
@@ -1,330 +1,11 @@
import net from 'net';
import http from 'http';
import https from 'https';
import tls from 'tls';

import { getWeakRandomId } from '@trezor/utils';

import { TorIdentities } from './torIdentities';
import { InterceptorOptions } from './types';
import { createRequestPool } from './httpPool';

type InterceptorContext = InterceptorOptions & {
requestPool: ReturnType<typeof createRequestPool>;
torIdentities: TorIdentities;
};

const getIdentityName = (proxyAuthorization?: http.OutgoingHttpHeader) => {
const identity = Array.isArray(proxyAuthorization) ? proxyAuthorization[0] : proxyAuthorization;

// Only return identity name if it is explicitly defined.
return typeof identity === 'string' ? identity.match(/Basic (.*)/)?.[1] : undefined;
};

/** Should the request be blocked if Tor isn't enabled? */
const getIsTorRequired = (options: Readonly<http.RequestOptions>) =>
!!options.headers?.['Proxy-Authorization'];

const getIdentityForAgent = (options: Readonly<http.RequestOptions>) => {
if (options.headers?.['Proxy-Authorization']) {
// Use Proxy-Authorization header to define proxy identity
// https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Proxy-Authorization
return getIdentityName(options.headers['Proxy-Authorization']);
}
if (options.headers?.Upgrade === 'websocket') {
// Create random identity for each websocket connection
return `WebSocket/${options.host}/${getWeakRandomId(16)}`;
}
};

const isWhitelistedHost = (hostname: unknown, whitelist: string[] = ['127.0.0.1', 'localhost']) =>
typeof hostname === 'string' &&
whitelist.some(url => url === hostname || hostname.endsWith(url));

const interceptNetSocketConnect = (context: InterceptorContext) => {
// To avoid disclosure that the request was sent by trezor-suite
// remove headers added by underlying libs before they are sent to the server.
// 1. nodejs http always(!) adds "Connection: close" header
// https://github.com/nodejs/node/blob/e48763840625c037282681456ecd1e1cb034f636/lib/_http_outgoing.js#L508-L510
// 2. node-fetch always(!) adds "User-Agent", "Accept", "Connection"...
// https://github.com/node-fetch/node-fetch/blob/7b86e946b02dfdd28f4f8fca3d73a022cbb5ca1e/src/request.js#L226
const originalSocketWrite = net.Socket.prototype.write;
net.Socket.prototype.write = function (data, ...args) {
const overloadedHeaders: string[] = [];
if (typeof data === 'string' && /Allowed-Headers/gi.test(data)) {
const headers = data.split('\r\n');
const allowedHeaders = headers
.find(line => /^Allowed-Headers/i.test(line))
?.split(': ');

if (allowedHeaders) {
const allowedKeys = allowedHeaders[1].split(';');

headers.forEach(line => {
const [key, value] = line.split(': ');
if (key && value) {
if (allowedKeys.some(allowed => new RegExp(`^${allowed}`, 'i').test(key))) {
overloadedHeaders.push(line);
}
} else {
overloadedHeaders.push(line);
}
});

context.handler({
type: 'INTERCEPTED_HEADERS',
method: 'net.Socket.write',
details: overloadedHeaders.join(' '),
});
}
}

return originalSocketWrite.apply(this, [
overloadedHeaders.length > 0 ? overloadedHeaders.join('\r\n') : data,
...args,
] as unknown as Parameters<typeof originalSocketWrite>);
};

const originalSocketConnect = net.Socket.prototype.connect;

net.Socket.prototype.connect = function (...args) {
const [options, callback] = args;
let request: typeof options;
let details: string;
if (Array.isArray(options)) {
// Websocket in clear-net contains array where first element is SocketConnectOpts
[request] = options;
} else {
request = options;
}

if (typeof request === 'object') {
if ('port' in request) {
// TcpSocketConnectOpts
details = `${request.host}:${request.port}`;
} else {
// IpcSocketConnectOpts
details = request.path;
}
} else if (typeof request === 'string') {
details = request;
} else {
details = typeof callback === 'string' ? `${callback}:${request}` : request.toString();
}

context.handler({
type: 'INTERCEPTED_REQUEST',
method: 'net.Socket.connect',
details,
});

return originalSocketConnect.apply(
this,
args as unknown as Parameters<typeof originalSocketConnect>,
);
};
};

const interceptNetConnect = (context: InterceptorContext) => {
const originalConnect = net.connect;

net.connect = function (...args) {
const [options, callback] = args;

let details: string;
if (typeof options === 'object') {
if ('port' in options) {
// TcpNetConnectOpts
details = `${options.host}:${options.port}`;
} else {
// IpcNetConnectOpts
details = options.path;
}
} else if (typeof options === 'string') {
details = options;
} else {
details = typeof callback === 'string' ? `${callback}:${options}` : options.toString();
}

context.handler({
type: 'INTERCEPTED_REQUEST',
method: 'net.connect',
details,
});

return originalConnect.apply(this, args as Parameters<typeof net.connect>);
};
};

// http(s).request could have different arguments according to its types definition,
// but we only care when second argument (url) is object containing RequestOptions.
const overloadHttpRequest = (
context: InterceptorContext,
protocol: 'http' | 'https',
url: string | URL | http.RequestOptions,
options?: http.RequestOptions | ((r: http.IncomingMessage) => void),
callback?: unknown,
) => {
if (
!callback &&
typeof url === 'object' &&
'headers' in url &&
!isWhitelistedHost(url.hostname, context.notRequiredTorDomainsList) &&
(!options || typeof options === 'function')
) {
const isTorEnabled = context.getTorSettings().running;
const isTorRequired = getIsTorRequired(url);
const overloadedOptions = url;
const overloadedCallback = options;
const { host, path } = overloadedOptions;
let identity: string | undefined;

if (isTorEnabled) {
// Create proxy agent for the request (from Proxy-Authorization or default)
// get authorization data from request headers
identity = getIdentityForAgent(overloadedOptions) || 'default';
overloadedOptions.agent = context.torIdentities.getIdentity(
identity,
overloadedOptions.timeout,
protocol,
);
} else if (isTorRequired) {
// Block requests that explicitly requires TOR using Proxy-Authorization
if (context.allowTorBypass) {
context.handler({
type: 'INTERCEPTED_REQUEST',
method: 'http.request',
details: `Conditionally allowed request with Proxy-Authorization ${host}`,
});
} else {
context.handler({
type: 'INTERCEPTED_REQUEST',
method: 'http.request',
details: `Request blocked ${host}`,
});
throw new Error('Blocked request with Proxy-Authorization. TOR not enabled.');
}
}

context.handler({
type: 'INTERCEPTED_REQUEST',
method: 'http.request',
details: `${host}${path} with agent ${!!overloadedOptions.agent}`,
});

delete overloadedOptions.headers?.['Proxy-Authorization'];

// return tuple of params for original request
return [identity, overloadedOptions, overloadedCallback] as const;
}
};

const overloadWebsocketHandshake = (
context: InterceptorContext,
protocol: 'http' | 'https',
url: string | URL | http.RequestOptions,
options?: http.RequestOptions | ((r: http.IncomingMessage) => void),
callback?: unknown,
) => {
// @trezor/blockchain-link is adding an SocksProxyAgent to each connection
// related to https://github.com/trezor/trezor-suite/issues/7689
// this condition should be removed once suite will stop using TrezorConnect.setProxy
if (
typeof url === 'object' &&
isWhitelistedHost(url.host, context.notRequiredTorDomainsList) &&
'agent' in url
) {
delete url.agent;
}
if (
typeof url === 'object' &&
!isWhitelistedHost(url.host, context.notRequiredTorDomainsList) && // difference between overloadHttpRequest
'headers' in url &&
url.headers?.Upgrade === 'websocket'
) {
return overloadHttpRequest(context, protocol, url, options, callback);
}
};

const interceptHttp = (context: InterceptorContext) => {
const originalHttpRequest = http.request;

http.request = (...args) => {
const overload = overloadHttpRequest(context, 'http', ...args);
if (overload) {
const [identity, ...overloadedArgs] = overload;

return context.requestPool(originalHttpRequest(...overloadedArgs), identity);
}

// In cases that are not considered above we pass the args as they came.
return originalHttpRequest(...(args as Parameters<typeof http.request>));
};

const originalHttpGet = http.get;

http.get = (...args) => {
const overload = overloadWebsocketHandshake(context, 'http', ...args);
if (overload) {
const [identity, ...overloadedArgs] = overload;

return context.requestPool(originalHttpGet(...overloadedArgs), identity);
}

return originalHttpGet(...(args as Parameters<typeof https.get>));
};
};

const interceptHttps = (context: InterceptorContext) => {
const originalHttpsRequest = https.request;

https.request = (...args) => {
const overload = overloadHttpRequest(context, 'https', ...args);
if (overload) {
const [identity, ...overloadedArgs] = overload;

return context.requestPool(originalHttpsRequest(...overloadedArgs), identity);
}

// In cases that are not considered above we pass the args as they came.
return originalHttpsRequest(...(args as Parameters<typeof https.request>));
};

const originalHttpsGet = https.get;

https.get = (...args) => {
const overload = overloadWebsocketHandshake(context, 'https', ...args);
if (overload) {
const [identity, ...overloadedArgs] = overload;

return context.requestPool(originalHttpsGet(...overloadedArgs), identity);
}

return originalHttpsGet(...(args as Parameters<typeof https.get>));
};
};

const interceptTlsConnect = (context: InterceptorContext) => {
const originalTlsConnect = tls.connect;

tls.connect = (...args) => {
const [options] = args;
if (typeof options === 'object') {
context.handler({
type: 'INTERCEPTED_REQUEST',
method: 'tls.connect',
details: options.host || options.servername || 'unknown',
});

// allow untrusted/self-signed certificates for whitelisted domains (like https://*.sldev.cz)
options.rejectUnauthorized =
options.rejectUnauthorized ??
!isWhitelistedHost(options.host, context.notRequiredTorDomainsList);
}

return originalTlsConnect(...(args as Parameters<typeof tls.connect>));
};
};
import { interceptTlsConnect } from './interceptor/interceptTlsConnect';
import { interceptHttps } from './interceptor/interceptHttps';
import { interceptHttp } from './interceptor/interceptHttp';
import { interceptNetConnect } from './interceptor/interceptNetConnect';
import { interceptNetSocketConnect } from './interceptor/interceptNetSocketConnect';

export const createInterceptor = (interceptorOptions: InterceptorOptions) => {
const requestPool = createRequestPool(interceptorOptions);
Expand Down
35 changes: 35 additions & 0 deletions packages/request-manager/src/interceptor/interceptHttp.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
import http from 'http';
import https from 'https';

import { InterceptorContext } from './interceptorTypesAndUtils';
import { overloadHttpRequest } from './overloadHttpRequest';
import { overloadWebsocketHandshake } from './overloadWebsocketHandshake';

export const interceptHttp = (context: InterceptorContext) => {
const originalHttpRequest = http.request;

http.request = (...args) => {
const overload = overloadHttpRequest(context, 'http', ...args);
if (overload) {
const [identity, ...overloadedArgs] = overload;

return context.requestPool(originalHttpRequest(...overloadedArgs), identity);
}

// In cases that are not considered above we pass the args as they came.
return originalHttpRequest(...(args as Parameters<typeof http.request>));
};

const originalHttpGet = http.get;

http.get = (...args) => {
const overload = overloadWebsocketHandshake(context, 'http', ...args);
if (overload) {
const [identity, ...overloadedArgs] = overload;

return context.requestPool(originalHttpGet(...overloadedArgs), identity);
}

return originalHttpGet(...(args as Parameters<typeof https.get>));
};
};
Loading

0 comments on commit 0d4c182

Please sign in to comment.