Files
NodeBB/src/request.js

203 lines
5.4 KiB
JavaScript

'use strict';
const dns = require('dns').promises;
require('undici'); // keep this here, needed for SSRF (see `lookup()`)
const nconf = require('nconf');
const ipaddr = require('ipaddr.js');
const { CookieJar } = require('tough-cookie');
const fetchCookie = require('fetch-cookie').default;
const { version } = require('../package.json');
const plugins = require('./plugins');
const ttl = require('./cache/ttl');
const checkCache = ttl({
ttl: 1000 * 60 * 60, // 1 hour
});
let allowList = new Set();
let initialized = false;
exports.jar = function () {
return new CookieJar();
};
const userAgent = `NodeBB/${version.split('.').shift()}.x (${nconf.get('url')})`;
async function init() {
if (initialized) {
return;
}
allowList.add(nconf.get('url_parsed').host);
const { allowed } = await plugins.hooks.fire('filter:request.init', { allowed: allowList });
if (allowed instanceof Set) {
allowList = allowed;
}
initialized = true;
}
/**
* This method (alongside `check()`) guards against SSRF via DNS rebinding.
*
* - `check()` does a DNS lookup and ensures that all returned IPs do not belong to a reserved IP address space
* - `lookup()` provides additional logic that uses the cached DNS result from `check()`
* instead of doing another lookup (which is where DNS rebinding comes into play.)
* - For whatever reason `undici` needs to be required so that lookup can be overwritten properly.
*/
function lookup(hostname, options, callback) {
let { ok, lookup } = checkCache.get(hostname);
lookup = [...lookup];
if (!ok) {
throw new Error('lookup-failed');
}
if (!lookup) {
// trusted, do regular lookup
dns.lookup(hostname, options).then((addresses) => {
callback(null, addresses);
});
return;
}
// Lookup needs to behave asynchronously — https://github.com/nodejs/node/issues/28664
process.nextTick(() => {
if (options.all === true) {
callback(null, lookup);
} else {
const { address, family } = lookup.shift();
callback(null, address, family);
}
});
}
// Initialize fetch - somewhat hacky, but it's required for globalDispatcher to be available
async function call(url, method, { body, timeout, jar, ...config } = {}) {
const { ok } = await check(url);
if (!ok) {
throw new Error('[[error:reserved-ip-address]]');
}
let fetchImpl = fetch;
if (jar) {
fetchImpl = fetchCookie(fetch, jar);
}
const jsonTest = /application\/([a-z]+\+)?json/;
const opts = {
...config,
method,
headers: {
'content-type': 'application/json',
'user-agent': userAgent,
...config.headers,
},
};
if (timeout > 0) {
opts.signal = AbortSignal.timeout(timeout);
}
if (body && ['POST', 'PUT', 'PATCH', 'DEL', 'DELETE'].includes(method)) {
if (opts.headers['content-type'] && jsonTest.test(opts.headers['content-type'])) {
opts.body = JSON.stringify(body);
} else {
opts.body = body;
}
}
// Workaround for https://github.com/nodejs/undici/issues/1305
if (global[Symbol.for('undici.globalDispatcher.1')] !== undefined) {
class FetchAgent extends global[Symbol.for('undici.globalDispatcher.1')].constructor {
dispatch(opts, handler) {
delete opts.headers['sec-fetch-mode'];
return super.dispatch(opts, handler);
}
}
opts.dispatcher = new FetchAgent({
connect: { lookup },
});
}
const response = await fetchImpl(url, opts);
const { headers } = response;
const contentType = headers.get('content-type');
const isJSON = contentType && jsonTest.test(contentType);
let respBody = await response.text();
if (isJSON && respBody) {
try {
respBody = JSON.parse(respBody);
} catch (err) {
throw new Error('invalid json in response body', url);
}
}
return {
body: respBody,
response: {
ok: response.ok,
status: response.status,
statusCode: response.status,
statusText: response.statusText,
headers: Object.fromEntries(response.headers.entries()),
},
};
}
// Checks url to ensure it is not in reserved IP range (private, etc.)
async function check(url) {
await init();
const { host } = new URL(url);
const cached = checkCache.get(url);
if (cached !== undefined) {
return cached;
}
if (allowList.has(host)) {
const payload = { ok: true };
checkCache.set(host, payload);
return payload;
}
const addresses = new Set();
let lookup;
if (ipaddr.isValid(url)) {
addresses.add(url);
} else {
lookup = await dns.lookup(host, { all: true });
lookup.forEach(({ address, family }) => {
addresses.add({ address, family });
});
}
if (addresses.size < 1) {
return { ok: false };
}
// Every IP address that the host resolves to should be a unicast address
const ok = Array.from(addresses).every(({ address: ip }) => {
const parsed = ipaddr.parse(ip);
return parsed.range() === 'unicast';
});
const payload = { ok, lookup };
checkCache.set(host, payload);
return payload;
}
/*
const { body, response } = await request.get('someurl?foo=1&baz=2')
*/
exports.get = async (url, config) => call(url, 'GET', config);
exports.head = async (url, config) => call(url, 'HEAD', config);
exports.del = async (url, config) => call(url, 'DELETE', config);
exports.delete = exports.del;
exports.options = async (url, config) => call(url, 'OPTIONS', config);
/*
const { body, response } = await request.post('someurl', { body: { foo: 1, baz: 2}})
*/
exports.post = async (url, config) => call(url, 'POST', config);
exports.put = async (url, config) => call(url, 'PUT', config);
exports.patch = async (url, config) => call(url, 'PATCH', config);