egg/lib/core/httpclient.js

143 lines
4.4 KiB
JavaScript

'use strict';
const Agent = require('agentkeepalive');
const HttpsAgent = require('agentkeepalive').HttpsAgent;
const urllib = require('urllib');
const ms = require('humanize-ms');
const { FrameworkBaseError } = require('egg-errors');
class HttpClientError extends FrameworkBaseError {
get module() {
return 'httpclient';
}
}
class HttpClient extends urllib.HttpClient2 {
constructor(app) {
normalizeConfig(app);
const config = app.config.httpclient;
super({
app,
defaultArgs: config.request,
agent: new Agent(config.httpAgent),
httpsAgent: new HttpsAgent(config.httpsAgent),
});
this.app = app;
}
request(url, args, callback) {
if (typeof args === 'function') {
callback = args;
args = null;
}
args = args || {};
if (args.ctx && args.ctx.tracer) {
args.tracer = args.ctx.tracer;
} else {
args.tracer = args.tracer || this.app.tracer;
}
// the callback style
if (callback) {
this.app.deprecate('[httpclient] We now support async for this function, so callback isn\'t recommended.');
super.request(url, args)
.then(result => process.nextTick(() => callback(null, result.data, result.res)))
.catch(err => process.nextTick(() => callback(err)));
return;
}
// the Promise style
return super.request(url, args)
.catch(err => {
if (err.code === 'ENETUNREACH') {
throw HttpClientError.create(err.message, err.code);
}
throw err;
});
}
curl(url, args, callback) {
return this.request(url, args, callback);
}
requestThunk(url, args) {
this.app.deprecate('[httpclient] Please use `request()` instead of `requestThunk()`');
return callback => {
this.request(url, args, (err, data, res) => {
if (err) {
return callback(err);
}
callback(null, {
data,
status: res.status,
headers: res.headers,
res,
});
});
};
}
}
function normalizeConfig(app) {
const config = app.config.httpclient;
// compatibility
if (typeof config.keepAlive === 'boolean') {
config.httpAgent.keepAlive = config.keepAlive;
config.httpsAgent.keepAlive = config.keepAlive;
}
if (config.timeout) {
config.timeout = ms(config.timeout);
config.httpAgent.timeout = config.timeout;
config.httpsAgent.timeout = config.timeout;
}
// compatibility httpclient.freeSocketKeepAliveTimeout => httpclient.freeSocketTimeout
if (config.freeSocketKeepAliveTimeout && !config.freeSocketTimeout) {
config.freeSocketTimeout = config.freeSocketKeepAliveTimeout;
delete config.freeSocketKeepAliveTimeout;
}
if (config.freeSocketTimeout) {
config.freeSocketTimeout = ms(config.freeSocketTimeout);
config.httpAgent.freeSocketTimeout = config.freeSocketTimeout;
config.httpsAgent.freeSocketTimeout = config.freeSocketTimeout;
} else {
// compatibility agent.freeSocketKeepAliveTimeout
if (config.httpAgent.freeSocketKeepAliveTimeout && !config.httpAgent.freeSocketTimeout) {
config.httpAgent.freeSocketTimeout = config.httpAgent.freeSocketKeepAliveTimeout;
delete config.httpAgent.freeSocketKeepAliveTimeout;
}
if (config.httpsAgent.freeSocketKeepAliveTimeout && !config.httpsAgent.freeSocketTimeout) {
config.httpsAgent.freeSocketTimeout = config.httpsAgent.freeSocketKeepAliveTimeout;
delete config.httpsAgent.freeSocketKeepAliveTimeout;
}
}
if (typeof config.maxSockets === 'number') {
config.httpAgent.maxSockets = config.maxSockets;
config.httpsAgent.maxSockets = config.maxSockets;
}
if (typeof config.maxFreeSockets === 'number') {
config.httpAgent.maxFreeSockets = config.maxFreeSockets;
config.httpsAgent.maxFreeSockets = config.maxFreeSockets;
}
if (config.httpAgent.timeout < 30000) {
app.coreLogger.warn('[egg:httpclient] config.httpclient.httpAgent.timeout(%s) can\'t below 30000, auto reset to 30000',
config.httpAgent.timeout);
config.httpAgent.timeout = 30000;
}
if (config.httpsAgent.timeout < 30000) {
app.coreLogger.warn('[egg:httpclient] config.httpclient.httpsAgent.timeout(%s) can\'t below 30000, auto reset to 30000',
config.httpsAgent.timeout);
config.httpsAgent.timeout = 30000;
}
if (typeof config.request.timeout === 'string') {
config.request.timeout = ms(config.request.timeout);
}
}
module.exports = HttpClient;