db-vendo-client/lib/request.js
2025-02-13 16:01:29 +01:00

178 lines
4.6 KiB
JavaScript

// import ProxyAgent from 'https-proxy-agent';
// import {isIP} from 'net';
// import {Agent as HttpsAgent} from 'https';
// import roundRobin from '@derhuerst/round-robin-scheduler';
import {stringify} from 'qs';
import {Request, fetch} from 'cross-fetch';
import {parse as parseContentType} from 'content-type';
import {HafasError} from './errors.js';
// const proxyAddress = process.env.HTTPS_PROXY || process.env.HTTP_PROXY || null;
// const localAddresses = process.env.LOCAL_ADDRESS || null;
// if (proxyAddress && localAddresses) {
// console.error('Both env vars HTTPS_PROXY/HTTP_PROXY and LOCAL_ADDRESS are not supported.');
// process.exit(1);
// }
// const plainAgent = new HttpsAgent({
// keepAlive: true,
// });
// let getAgent = () => plainAgent;
// if (proxyAddress) {
// const agent = new ProxyAgent(proxyAddress, {
// keepAlive: true,
// keepAliveMsecs: 10 * 1000, // 10s
// });
// getAgent = () => agent;
// } else if (localAddresses) {
// const agents = process.env.LOCAL_ADDRESS.split(',')
// .map((addr) => {
// const family = isIP(addr);
// if (family === 0) {
// throw new Error('invalid local address:' + addr);
// }
// return new HttpsAgent({
// localAddress: addr, family,
// keepAlive: true,
// });
// });
// const pool = roundRobin(agents);
// getAgent = () => pool.get();
// }
// const id = randomBytes(3)
// .toString('hex');
// const randomizeUserAgent = (userAgent) => {
// let ua = userAgent;
// for (
// let i = Math.round(5 + Math.random() * 5);
// i < ua.length;
// i += Math.round(5 + Math.random() * 5)
// ) {
// ua = ua.slice(0, i) + id + ua.slice(i);
// i += id.length;
// }
// return ua;
// };
const randomBytesHex = (nBytes = 8) => {
const array = new Uint8Array(nBytes);
crypto.getRandomValues(array);
return Array.from(array)
.map((byte) => byte.toString(16).padStart(2, '0'))
.join('');
};
const checkIfResponseIsOk = (_) => {
const {
body,
errProps: baseErrProps,
} = _;
const errProps = {
...baseErrProps,
};
if (body.id) {
errProps.hafasResponseId = body.id;
}
// Because we want more accurate stack traces, we don't construct the error here,
// but only return the constructor & error message.
const getError = (_) => {
// mutating here is ugly but pragmatic
if (_.fehlerNachricht.ueberschrift) {
errProps.hafasMessage = _.fehlerNachricht.ueberschrift;
}
if (_.fehlerNachricht.text) {
errProps.hafasDescription = _.fehlerNachricht.text;
}
return {
Error: HafasError,
message: errProps.hafasMessage || 'unknown error',
props: {code: _.fehlerNachricht.code},
};
};
if (body.fehlerNachricht) { // TODO better handling
const {Error: HafasError, message, props} = getError(body);
throw new HafasError(message, body.err, {...errProps, ...props});
}
};
const request = async (ctx, userAgent, reqData) => {
const {profile, opt} = ctx;
const endpoint = reqData.endpoint;
delete reqData.endpoint;
const rawReqBody = profile.transformReqBody(ctx, reqData.body);
const reqOptions = profile.transformReq(ctx, {
keepalive: true,
method: reqData.method,
// todo: CORS? referrer policy?
body: JSON.stringify(rawReqBody),
headers: {
'Content-Type': 'application/json',
'Accept-Encoding': 'gzip, br, deflate',
'Accept': 'application/json',
'Accept-Language': opt.language || profile.defaultLanguage || 'en',
'user-agent': userAgent,
...reqData.headers,
},
redirect: 'follow',
query: reqData.query,
});
let url = endpoint + (reqData.path || '');
if (reqOptions.query) {
url += '?' + stringify(reqOptions.query, {arrayFormat: 'brackets', encodeValuesOnly: true});
}
const reqId = randomBytesHex(3);
const fetchReq = new Request(url, reqOptions);
profile.logRequest(ctx, fetchReq, reqId);
const res = await fetch(url, reqOptions);
const errProps = {
// todo [breaking]: assign as non-enumerable property
request: fetchReq,
// todo [breaking]: assign as non-enumerable property
response: res,
url,
};
if (!res.ok) {
// todo [breaking]: make this a FetchError or a HafasClientError?
const err = new Error(res.statusText);
Object.assign(err, errProps);
throw err;
}
let cType = res.headers.get('content-type');
if (cType) {
const {type} = parseContentType(cType);
if (type !== reqOptions.headers['Accept']) {
throw new HafasError('invalid/unsupported response content-type: ' + cType, null, errProps);
}
}
const body = await res.text();
profile.logResponse(ctx, res, body, reqId);
const b = JSON.parse(body);
checkIfResponseIsOk({
body: b,
errProps,
});
return {
res: b,
common: {},
};
};
export {
checkIfResponseIsOk,
request,
};