db-vendo-client/lib/request.js

126 lines
3.4 KiB
JavaScript
Raw Normal View History

2017-11-11 22:49:04 +01:00
'use strict'
2018-09-03 15:45:31 +02:00
const DEV = process.env.NODE_ENV === 'dev'
const DEBUG = /\bhafas-client\b/.test(process.env.DEBUG || '')
2018-09-03 15:31:20 +02:00
const {randomBytes} = require('crypto')
const createHash = require('create-hash')
const pick = require('lodash/pick')
2018-09-03 15:45:31 +02:00
const captureStackTrace = DEV ? require('capture-stack-trace') : () => {}
2019-02-05 19:07:19 +01:00
const {stringify} = require('qs')
2020-09-21 13:18:31 +02:00
const ProxyAgent = require('https-proxy-agent')
2017-11-11 22:49:04 +01:00
const Promise = require('pinkie-promise')
const {fetch} = require('fetch-ponyfill')({Promise})
const {addErrorInfo} = require('./errors')
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
const proxyAgent = proxyAddress ? new ProxyAgent(proxyAddress) : null
const id = randomBytes(6).toString('hex')
2018-08-08 18:40:35 +02:00
const randomizeUserAgent = (userAgent) => {
const i = Math.round(Math.random() * userAgent.length)
return userAgent.slice(0, i) + id + userAgent.slice(i)
}
const md5 = input => createHash('md5').update(input).digest()
const request = (ctx, userAgent, reqData) => {
const {profile, opt} = ctx
const body = profile.transformReqBody(ctx, {
2020-03-18 21:35:43 +01:00
// todo: is it `eng` actually?
// RSAG has `deu` instead of `de`
lang: opt.language || 'en',
svcReqL: [reqData]
2018-07-09 12:40:38 +02:00
})
Object.assign(body, pick(profile, [
'client', // client identification
'ext', // ?
'ver', // HAFAS protocol version
'auth', // static authentication
]))
2018-09-03 15:31:20 +02:00
if (DEBUG) console.error(JSON.stringify(body))
const req = profile.transformReq(ctx, {
2020-09-21 13:18:31 +02:00
agent: proxyAgent,
2017-11-11 22:49:04 +01:00
method: 'post',
// todo: CORS? referrer policy?
body: JSON.stringify(body),
headers: {
'Content-Type': 'application/json',
'Accept-Encoding': 'gzip, br, deflate',
2018-06-07 12:04:21 +02:00
'Accept': 'application/json',
2018-08-08 18:40:35 +02:00
'user-agent': randomizeUserAgent(userAgent)
2017-11-11 22:49:04 +01:00
},
redirect: 'follow',
query: {}
2017-11-11 22:49:04 +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.')
}
2021-01-14 20:30:48 +01:00
// Buffer.from(buf, 'hex') just returns buf
const salt = Buffer.from(profile.salt, 'hex')
if (profile.addChecksum) {
const checksum = md5(Buffer.concat([
Buffer.from(req.body, 'utf8'),
2021-01-14 20:30:48 +01:00
salt,
]))
req.query.checksum = checksum.toString('hex')
}
if (profile.addMicMac) {
const mic = md5(Buffer.from(req.body, 'utf8'))
req.query.mic = mic.toString('hex')
const micAsHex = Buffer.from(mic.toString('hex'), 'utf8')
2021-01-14 20:30:48 +01:00
const mac = md5(Buffer.concat([micAsHex, salt]))
req.query.mac = mac.toString('hex')
}
}
2018-03-02 23:57:29 +01:00
const url = profile.endpoint + '?' + stringify(req.query)
2017-11-11 22:49:04 +01:00
2018-01-23 01:24:37 +01:00
// Async stack traces are not supported everywhere yet, so we create our own.
const err = new Error()
2018-12-16 23:47:36 +01:00
err.isHafasError = true // todo: rename to `isHafasClientError`
err.request = req.body // todo: commit as bugfix
2018-03-02 23:57:29 +01:00
err.url = url
2018-01-23 01:24:37 +01:00
captureStackTrace(err)
2017-11-11 22:49:04 +01:00
return fetch(url, req)
.then((res) => {
2018-01-23 01:24:37 +01:00
err.statusCode = res.status
2017-11-11 22:49:04 +01:00
if (!res.ok) {
2018-01-23 01:24:37 +01:00
err.message = res.statusText
throw err
2017-11-11 22:49:04 +01:00
}
return res.json()
})
.then((b) => {
2018-09-03 15:31:20 +02:00
if (DEBUG) console.error(JSON.stringify(b))
2018-12-06 11:31:38 +01:00
if (b.err && b.err !== 'OK') {
addErrorInfo(err, b.err, b.errTxt, b.id)
2018-01-23 01:24:37 +01:00
throw err
}
if (!b.svcResL || !b.svcResL[0]) {
err.message = 'invalid response'
throw err
}
2017-11-11 22:49:04 +01:00
if (b.svcResL[0].err !== 'OK') {
addErrorInfo(err, b.svcResL[0].err, b.svcResL[0].errTxt, b.id)
2018-01-23 01:24:37 +01:00
throw err
2017-11-11 22:49:04 +01:00
}
const res = b.svcResL[0].res
return {
res,
common: profile.parseCommon({...ctx, res})
}
2017-11-11 22:49:04 +01:00
})
}
module.exports = request