forked from pelias/microservice-wrapper
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathservice.js
133 lines (106 loc) · 4.18 KB
/
service.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
'use strict';
const http = require('http');
const request = require('superagent');
const _ = require('lodash');
const ServiceConfiguration = require('./ServiceConfiguration');
function isDoNotTrack(headers) {
return _.has(headers, 'DNT') ||
_.has(headers, 'dnt') ||
_.has(headers, 'do_not_track');
}
// superagent doesn't exposed the assembled GET request, so synthesize it
function synthesizeUrl(serviceConfig, req, res) {
const parameters = _.map(serviceConfig.getParameters(req, res), (value, key) => {
return `${key}=${value}`;
}).join('&');
if (parameters) {
return encodeURI(`${serviceConfig.getUrl(req)}?${parameters}`);
} else {
return serviceConfig.getUrl(req);
}
}
module.exports = function setup(serviceConfig) {
if (!(serviceConfig instanceof ServiceConfiguration)) {
throw Error('serviceConfig should be an instance of ServiceConfiguration');
}
const logger = require( 'pelias-logger' ).get( serviceConfig.getName() );
if (!serviceConfig.isEnabled()) {
logger.warn(`${serviceConfig.getName()} service disabled`);
return (req, res, callback) => {
// only req was passed in so treat res as callback
if (_.isUndefined(callback)) {
callback = res;
}
// respond with an error to any call
callback(`${serviceConfig.getName()} service disabled`);
};
}
logger.info(`using ${serviceConfig.getName()} service at ${serviceConfig.getBaseUrl()}`);
// create one HTTP agent with keep alives enabled per service instance
const agent = new http.Agent({
keepAlive: true
});
return (req, res, callback) => {
// only req was passed in so treat res as callback
if (_.isUndefined(callback)) {
callback = res;
res = undefined;
}
const headers = serviceConfig.getHeaders(req, res) || {};
// save off do_not_track value for later check
const do_not_track = isDoNotTrack(req.headers);
let url_for_logging;
if (do_not_track) {
headers.dnt = '1';
url_for_logging = serviceConfig.getBaseUrl();
} else {
url_for_logging = synthesizeUrl(serviceConfig, req, res);
}
logger.debug(`${serviceConfig.getName()}: ${url_for_logging}`);
request
.get(serviceConfig.getUrl(req))
.set(headers)
.timeout(serviceConfig.getTimeout())
.retry(serviceConfig.getRetries())
.agent(agent)
.accept('json')
.query(serviceConfig.getParameters(req, res))
.on('error', (err) => {
if (err.status) {
// first handle case where a non-200 was returned
if (do_not_track) {
logger.error(`${url_for_logging} [do_not_track] returned status ${err.status}: ${err.response.text}`);
return callback(`${url_for_logging} [do_not_track] returned status ${err.status}: ${err.response.text}`);
} else {
logger.error(`${url_for_logging} returned status ${err.status}: ${err.response.text}`);
return callback(`${url_for_logging} returned status ${err.status}: ${err.response.text}`);
}
}
// handle case that something catastrophic happened while contacting the server
if (do_not_track) {
logger.error(`${url_for_logging} [do_not_track]: ${JSON.stringify(err)}`);
return callback(err);
} else {
logger.error(`${url_for_logging}: ${JSON.stringify(err)}`);
return callback(err);
}
})
.end((err, response) => {
// bail early if there's an error (shouldn't happen since it was already handled above)
if (err) {
return;
}
// if json was returned then just return it
if (response.type === 'application/json') {
return callback(null, response.body);
}
if (do_not_track) {
logger.error(`${url_for_logging} [do_not_track] could not parse response: ${response.text}`);
return callback(`${url_for_logging} [do_not_track] could not parse response: ${response.text}`);
} else {
logger.error(`${url_for_logging} could not parse response: ${response.text}`);
return callback(`${url_for_logging} could not parse response: ${response.text}`);
}
});
};
};