| 
									
										
										
										
											2017-11-11 22:49:04 +01:00
										 |  |  | 'use strict' | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-07 22:17:02 +01:00
										 |  |  | const ProxyAgent = require('https-proxy-agent') | 
					
						
							|  |  |  | const {isIP} = require('net') | 
					
						
							|  |  |  | const {Agent: HttpsAgent} = require('https') | 
					
						
							|  |  |  | const roundRobin = require('@derhuerst/round-robin-scheduler') | 
					
						
							| 
									
										
										
										
											2019-08-04 14:32:44 +02:00
										 |  |  | const {randomBytes} = require('crypto') | 
					
						
							| 
									
										
										
										
											2018-05-21 12:15:40 +02:00
										 |  |  | const createHash = require('create-hash') | 
					
						
							| 
									
										
										
										
											2019-02-05 19:07:19 +01:00
										 |  |  | const {stringify} = require('qs') | 
					
						
							| 
									
										
										
										
											2022-10-24 16:58:54 +02:00
										 |  |  | const {Request, fetch} = require('cross-fetch') | 
					
						
							| 
									
										
										
										
											2021-04-18 18:42:03 +02:00
										 |  |  | const {parse: parseContentType} = require('content-type') | 
					
						
							| 
									
										
										
										
											2022-05-03 23:21:44 +02:00
										 |  |  | const {HafasError} = require('./errors') | 
					
						
							|  |  |  | const checkIfResponseIsOk = require('./check-if-res-is-ok') | 
					
						
							| 
									
										
										
										
											2017-11-11 22:49:04 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-21 13:18:31 +02:00
										 |  |  | const proxyAddress = process.env.HTTPS_PROXY || process.env.HTTP_PROXY || null | 
					
						
							| 
									
										
										
										
											2021-02-07 22:17:02 +01:00
										 |  |  | 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) | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-12-09 18:53:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | const plainAgent = new HttpsAgent({ | 
					
						
							|  |  |  | 	keepAlive: true, | 
					
						
							|  |  |  | }) | 
					
						
							|  |  |  | let getAgent = () => plainAgent | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-07 22:17:02 +01:00
										 |  |  | if (proxyAddress) { | 
					
						
							| 
									
										
										
										
											2021-12-09 18:53:18 +01:00
										 |  |  | 	// todo: this doesn't honor `keepAlive: true`
 | 
					
						
							|  |  |  | 	// related:
 | 
					
						
							|  |  |  | 	// - https://github.com/TooTallNate/node-https-proxy-agent/pull/112
 | 
					
						
							|  |  |  | 	// - https://github.com/TooTallNate/node-agent-base/issues/5
 | 
					
						
							| 
									
										
										
										
											2021-02-07 22:17:02 +01:00
										 |  |  | 	const agent = new ProxyAgent(proxyAddress) | 
					
						
							|  |  |  | 	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) | 
					
						
							| 
									
										
										
										
											2021-12-09 18:53:18 +01:00
										 |  |  | 		return new HttpsAgent({ | 
					
						
							|  |  |  | 			localAddress: addr, family, | 
					
						
							|  |  |  | 			keepAlive: true, | 
					
						
							|  |  |  | 		}) | 
					
						
							| 
									
										
										
										
											2021-02-07 22:17:02 +01:00
										 |  |  | 	}) | 
					
						
							|  |  |  | 	const pool = roundRobin(agents) | 
					
						
							|  |  |  | 	getAgent = () => pool.get() | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-09-21 13:18:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-13 13:55:17 +01:00
										 |  |  | const id = randomBytes(3).toString('hex') | 
					
						
							| 
									
										
										
										
											2018-08-08 18:40:35 +02:00
										 |  |  | const randomizeUserAgent = (userAgent) => { | 
					
						
							| 
									
										
										
										
											2022-01-13 13:55:17 +01:00
										 |  |  | 	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 | 
					
						
							| 
									
										
										
										
											2018-08-08 18:40:35 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-21 12:15:40 +02:00
										 |  |  | const md5 = input => createHash('md5').update(input).digest() | 
					
						
							| 
									
										
										
										
											2018-01-15 00:45:05 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-03 17:51:52 +02:00
										 |  |  | const request = async (ctx, userAgent, reqData) => { | 
					
						
							| 
									
										
										
										
											2019-10-20 01:47:09 +02:00
										 |  |  | 	const {profile, opt} = ctx | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-03 17:51:52 +02:00
										 |  |  | 	const rawReqBody = profile.transformReqBody(ctx, { | 
					
						
							| 
									
										
										
										
											2020-03-18 21:35:43 +01:00
										 |  |  | 		// todo: is it `eng` actually?
 | 
					
						
							|  |  |  | 		// RSAG has `deu` instead of `de`
 | 
					
						
							| 
									
										
										
										
											2021-04-18 18:41:27 +02:00
										 |  |  | 		lang: opt.language || profile.defaultLanguage || 'en', | 
					
						
							| 
									
										
										
										
											2022-01-04 15:32:16 +01:00
										 |  |  | 		svcReqL: [reqData], | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		client: profile.client, // client identification
 | 
					
						
							|  |  |  | 		ext: profile.ext, // ?
 | 
					
						
							|  |  |  | 		ver: profile.ver, // HAFAS protocol version
 | 
					
						
							|  |  |  | 		auth: profile.auth, // static authentication
 | 
					
						
							| 
									
										
										
										
											2018-07-09 12:40:38 +02:00
										 |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2018-09-03 15:31:20 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-20 01:47:09 +02:00
										 |  |  | 	const req = profile.transformReq(ctx, { | 
					
						
							| 
									
										
										
										
											2021-02-07 22:17:02 +01:00
										 |  |  | 		agent: getAgent(), | 
					
						
							| 
									
										
										
										
											2017-11-11 22:49:04 +01:00
										 |  |  | 		method: 'post', | 
					
						
							|  |  |  | 		// todo: CORS? referrer policy?
 | 
					
						
							| 
									
										
										
										
											2022-05-03 17:51:52 +02:00
										 |  |  | 		body: JSON.stringify(rawReqBody), | 
					
						
							| 
									
										
										
										
											2017-11-11 22:49:04 +01:00
										 |  |  | 		headers: { | 
					
						
							|  |  |  | 			'Content-Type': 'application/json', | 
					
						
							| 
									
										
										
										
											2019-06-24 18:26:11 +02:00
										 |  |  | 			'Accept-Encoding': 'gzip, br, deflate', | 
					
						
							| 
									
										
										
										
											2018-06-07 12:04:21 +02:00
										 |  |  | 			'Accept': 'application/json', | 
					
						
							| 
									
										
										
										
											2021-12-29 17:01:14 +01:00
										 |  |  | 			'user-agent': profile.randomizeUserAgent | 
					
						
							|  |  |  | 				? randomizeUserAgent(userAgent) | 
					
						
							|  |  |  | 				: userAgent, | 
					
						
							| 
									
										
										
										
											2021-12-09 18:53:18 +01:00
										 |  |  | 			'connection': 'keep-alive', // prevent excessive re-connecting
 | 
					
						
							| 
									
										
										
										
											2017-11-11 22:49:04 +01:00
										 |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2019-06-24 18:26:11 +02:00
										 |  |  | 		redirect: 'follow', | 
					
						
							| 
									
										
										
										
											2018-01-15 00:45:05 +01:00
										 |  |  | 		query: {} | 
					
						
							| 
									
										
										
										
											2017-11-11 22:49:04 +01:00
										 |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-15 00:45:40 +01:00
										 |  |  | 	if (profile.addChecksum || profile.addMicMac) { | 
					
						
							| 
									
										
										
										
											2021-01-14 20:30:48 +01:00
										 |  |  | 		if (!Buffer.isBuffer(profile.salt) && 'string' !== typeof profile.salt) { | 
					
						
							|  |  |  | 			throw new TypeError('profile.salt must be a Buffer or a string.') | 
					
						
							| 
									
										
										
										
											2018-01-15 00:45:05 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-01-14 20:30:48 +01:00
										 |  |  | 		// Buffer.from(buf, 'hex') just returns buf
 | 
					
						
							|  |  |  | 		const salt = Buffer.from(profile.salt, 'hex') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-15 00:45:40 +01:00
										 |  |  | 		if (profile.addChecksum) { | 
					
						
							|  |  |  | 			const checksum = md5(Buffer.concat([ | 
					
						
							|  |  |  | 				Buffer.from(req.body, 'utf8'), | 
					
						
							| 
									
										
										
										
											2021-01-14 20:30:48 +01:00
										 |  |  | 				salt, | 
					
						
							| 
									
										
										
										
											2018-01-15 00:45:40 +01:00
										 |  |  | 			])) | 
					
						
							|  |  |  | 			req.query.checksum = checksum.toString('hex') | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (profile.addMicMac) { | 
					
						
							|  |  |  | 			const mic = md5(Buffer.from(req.body, 'utf8')) | 
					
						
							|  |  |  | 			req.query.mic = mic.toString('hex') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-19 15:47:41 +01:00
										 |  |  | 			const micAsHex = Buffer.from(mic.toString('hex'), 'utf8') | 
					
						
							| 
									
										
										
										
											2021-01-14 20:30:48 +01:00
										 |  |  | 			const mac = md5(Buffer.concat([micAsHex, salt])) | 
					
						
							| 
									
										
										
										
											2018-01-15 00:45:40 +01:00
										 |  |  | 			req.query.mac = mac.toString('hex') | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-01-15 00:45:05 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-07 01:29:36 +02:00
										 |  |  | 	const reqId = randomBytes(3).toString('hex') | 
					
						
							| 
									
										
										
										
											2018-03-02 23:57:29 +01:00
										 |  |  | 	const url = profile.endpoint + '?' + stringify(req.query) | 
					
						
							| 
									
										
										
										
											2022-10-06 14:22:46 +02:00
										 |  |  | 	const fetchReq = new Request(url, req) | 
					
						
							| 
									
										
										
										
											2022-10-07 01:29:36 +02:00
										 |  |  | 	profile.logRequest(ctx, fetchReq, reqId) | 
					
						
							| 
									
										
										
										
											2017-11-11 22:49:04 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-03 17:51:52 +02:00
										 |  |  | 	const res = await fetch(url, req) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const errProps = { | 
					
						
							| 
									
										
										
										
											2022-05-03 23:21:44 +02:00
										 |  |  | 		request: fetchReq, | 
					
						
							|  |  |  | 		response: res, | 
					
						
							|  |  |  | 		url, | 
					
						
							| 
									
										
										
										
											2022-05-03 17:51:52 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!res.ok) { | 
					
						
							| 
									
										
										
										
											2022-05-03 23:21:44 +02:00
										 |  |  | 		// todo [breaking]: make this a FetchError or a HafasClientError?
 | 
					
						
							| 
									
										
										
										
											2022-05-03 17:51:52 +02:00
										 |  |  | 		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 !== 'application/json') { | 
					
						
							| 
									
										
										
										
											2022-05-03 23:21:44 +02:00
										 |  |  | 			throw new HafasError('invalid/unsupported response content-type: ' + cType, null, errProps) | 
					
						
							| 
									
										
										
										
											2017-11-11 22:49:04 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-05-03 17:51:52 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-04-18 18:42:03 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-03 17:51:52 +02:00
										 |  |  | 	const body = await res.text() | 
					
						
							|  |  |  | 	profile.logResponse(ctx, res, body, reqId) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const b = JSON.parse(body) | 
					
						
							| 
									
										
										
										
											2022-05-03 23:21:44 +02:00
										 |  |  | 	checkIfResponseIsOk({ | 
					
						
							|  |  |  | 		body: b, | 
					
						
							|  |  |  | 		errProps, | 
					
						
							|  |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2018-09-03 15:31:20 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-03 17:51:52 +02:00
										 |  |  | 	const svcRes = b.svcResL[0].res | 
					
						
							|  |  |  | 	return { | 
					
						
							|  |  |  | 		res: svcRes, | 
					
						
							|  |  |  | 		common: profile.parseCommon({...ctx, res: svcRes}), | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-11-11 22:49:04 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | module.exports = request |