Portal
🚧 Work-in-progress
HTTP proxy implementation using Node.js' http.createServer to accept connections and http(s).request to relay them to their destinations. Currently in use on @harvard-lil/scoop.
Philosophy
Portal uses standard Node.js networking components in order to provide a simple proxy with the following goals:
- No dependencies
- Interfaces that match existing Node.js conventions
- The ability to intercept raw traffic
Portal achieves this by using "mirror" streams that buffer the data from each socket, allowing Node.js' standard parsing mechanism to parse the data while making that same raw data available for modification before being passed forward in the proxy.
Configuration
The entrypoint for Portal is the createServer
function which, in addition to the options available to http.createServer
, also accepts the following:
-
clientOptions(request)
- a function which accepts the requesthttp.IncomingMessage
and returns an options object (orPromise
) to be passed tonew tls.TLSSocket
when the client socket is upgraded after an HTTPCONNECT
request. Most useful for dynamically generating akey
/cert
pair for the requested server name. -
serverOptions(request)
- a function which accepts the requesthttp.IncomingMessage
and returns an options object (orPromise
) to be passed tohttp(s).request
which will then be used to make requests to the destination. Most useful for setting SSL flags. -
requestTransformer(request)
- a function which accepts the requesthttp.IncomingMessage
and returns astream.Transform
instance (orPromise
) through which the incoming request data will be passed before being forwarded to its destination. -
responseTransformer(response, request)
- a function which accepts the response and requesthttp.IncomingMessages
and returns astream.Transform
instance (orPromise
) through which the incoming response data will be passed before being forwarded to its destination.
Events
The proxy server returned by createServer
emits all of the events available on http.Server
(ex: proxy.on('request')
). Additionally, it emits all of the events from http.ClientRequest
(ex: proxy.on('response')
) with the caveat that the upgrade
event is emitted as upgrade-client
in order to avoid a collision with the http.Server
event of the same name. Errors from both http.Server
and http.ClientRequest
are available via the 'error' event.
Example
import * as http from 'http'
import * as crypto from 'node:crypto'
import { TLSSocket } from 'tls'
import { Transform } from 'node:stream'
import { createServer } from './Portal.js'
const PORT = 1337
const HOST = '127.0.0.1'
const proxy = createServer({
requestTransformer: (request) => new Transform({
transform: (chunk, _encoding, callback) => {
console.log('Raw data to be passed in the request', chunk.toString())
callback(null, chunk)
}
}),
responseTransformer: (response, request) => new Transform({
transform: (chunk, _encoding, callback) => {
console.log('Raw data to be passed in the response', chunk.toString())
callback(null, chunk)
}
}),
clientOptions: async (request) => {
return {} // a custom key and cert could be returned here
},
serverOptions: async (request) => {
return {
// This flag allows legacy insecure renegotiation between OpenSSL and unpatched servers
// @see {@link https://stackoverflow.com/questions/74324019/allow-legacy-renegotiation-for-nodejs}
secureOptions: crypto.constants.SSL_OP_LEGACY_SERVER_CONNECT
}
}
})
proxy.on('request', (request) => {
console.log('Parsed request to observe', request.headers)
})
proxy.on('response', (response, request) => {
console.log('Parsed response to observe', response.headers)
})
proxy.on('error', (err) => {
console.log('Handle error', err)
})
proxy.listen(PORT, HOST)
/*
* Make an example request
*/
proxy.on('listening', () => {
const options = {
port: PORT,
host: HOST,
method: 'CONNECT',
path: 'example.com:443'
}
const req = http.request(options)
req.end()
req.on('connect', (res, socket, head) => {
const upgradedSocket = new TLSSocket(socket, {
rejectUnauthorized: false,
requestCert: false,
isServer: false
})
upgradedSocket.write('GET / HTTP/1.1\r\n' +
'Host: example.com:443\r\n' +
'Connection: close\r\n' +
'\r\n')
})
})