mirror of
https://github.com/NodeBB/NodeBB.git
synced 2025-12-27 10:50:47 +01:00
96 lines
2.7 KiB
JavaScript
96 lines
2.7 KiB
JavaScript
'use strict';
|
|
|
|
const nconf = require('nconf');
|
|
const { CookieJar } = require('tough-cookie');
|
|
const fetchCookie = require('fetch-cookie').default;
|
|
const { version } = require('../package.json');
|
|
|
|
exports.jar = function () {
|
|
return new CookieJar();
|
|
};
|
|
|
|
const userAgent = `NodeBB/${version.split('.').shift()}.x (${nconf.get('url')})`;
|
|
|
|
// Initialize fetch - somewhat hacky, but it's required for globalDispatcher to be available
|
|
async function call(url, method, { body, timeout, jar, ...config } = {}) {
|
|
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();
|
|
}
|
|
|
|
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()),
|
|
},
|
|
};
|
|
}
|
|
|
|
/*
|
|
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);
|
|
|
|
|