2022-07-29 01:19:50 +02:00
|
|
|
import * as plugins from './smartrequest.plugins.js';
|
|
|
|
import * as interfaces from './smartrequest.interfaces.js';
|
2017-01-29 00:51:47 +01:00
|
|
|
|
2024-11-06 20:58:17 +01:00
|
|
|
export interface IExtendedIncomingMessage<T = any> extends plugins.http.IncomingMessage {
|
|
|
|
body: T;
|
2018-07-15 23:21:07 +02:00
|
|
|
}
|
2018-06-13 22:34:49 +02:00
|
|
|
|
2018-07-16 23:39:25 +02:00
|
|
|
const buildUtf8Response = (
|
2023-11-03 01:37:10 +01:00
|
|
|
incomingMessageArg: plugins.http.IncomingMessage,
|
2019-06-12 15:16:27 +02:00
|
|
|
autoJsonParse = true
|
2018-07-16 23:39:25 +02:00
|
|
|
): Promise<IExtendedIncomingMessage> => {
|
2019-08-22 12:38:11 +02:00
|
|
|
const done = plugins.smartpromise.defer<IExtendedIncomingMessage>();
|
2017-06-05 19:09:40 +02:00
|
|
|
// Continuously update stream with data
|
2018-08-14 01:47:54 +02:00
|
|
|
let body = '';
|
2020-08-24 12:01:38 +00:00
|
|
|
incomingMessageArg.on('data', (chunkArg) => {
|
2018-06-13 22:34:49 +02:00
|
|
|
body += chunkArg;
|
|
|
|
});
|
|
|
|
|
2019-08-22 12:38:55 +02:00
|
|
|
incomingMessageArg.on('end', () => {
|
2019-06-12 15:16:27 +02:00
|
|
|
if (autoJsonParse) {
|
|
|
|
try {
|
|
|
|
(incomingMessageArg as IExtendedIncomingMessage).body = JSON.parse(body);
|
|
|
|
} catch (err) {
|
|
|
|
(incomingMessageArg as IExtendedIncomingMessage).body = body;
|
|
|
|
}
|
|
|
|
} else {
|
2018-07-16 23:39:25 +02:00
|
|
|
(incomingMessageArg as IExtendedIncomingMessage).body = body;
|
2017-06-05 19:09:40 +02:00
|
|
|
}
|
2018-07-16 23:39:25 +02:00
|
|
|
done.resolve(incomingMessageArg as IExtendedIncomingMessage);
|
2018-06-13 22:34:49 +02:00
|
|
|
});
|
|
|
|
return done.promise;
|
|
|
|
};
|
2017-01-29 00:51:47 +01:00
|
|
|
|
2018-07-16 23:39:25 +02:00
|
|
|
/**
|
|
|
|
* determine wether a url is a unix sock
|
|
|
|
* @param urlArg
|
|
|
|
*/
|
|
|
|
const testForUnixSock = (urlArg: string): boolean => {
|
|
|
|
const unixRegex = /^(http:\/\/|https:\/\/|)unix:/;
|
|
|
|
return unixRegex.test(urlArg);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* determine socketPath and path for unixsock
|
|
|
|
*/
|
|
|
|
const parseSocketPathAndRoute = (stringToParseArg: string) => {
|
|
|
|
const parseRegex = /(.*):(.*)/;
|
|
|
|
const result = parseRegex.exec(stringToParseArg);
|
|
|
|
return {
|
|
|
|
socketPath: result[1],
|
2020-08-24 12:01:38 +00:00
|
|
|
path: result[2],
|
2018-08-14 01:47:54 +02:00
|
|
|
};
|
|
|
|
};
|
2018-07-16 23:39:25 +02:00
|
|
|
|
2019-08-22 12:40:19 +02:00
|
|
|
/**
|
|
|
|
* a custom http agent to make sure we can set custom keepAlive options for speedy subsequent calls
|
|
|
|
*/
|
2022-07-30 01:52:04 +02:00
|
|
|
const httpAgent = new plugins.agentkeepalive({
|
2022-07-29 01:19:50 +02:00
|
|
|
keepAlive: true,
|
2022-07-30 02:01:20 +02:00
|
|
|
maxFreeSockets: 10,
|
2022-07-30 18:43:10 +02:00
|
|
|
maxSockets: 100,
|
|
|
|
maxTotalSockets: 1000,
|
|
|
|
timeout: 60000,
|
2022-07-29 01:19:50 +02:00
|
|
|
});
|
2019-08-21 12:55:19 +02:00
|
|
|
|
2019-09-08 17:47:30 +02:00
|
|
|
/**
|
|
|
|
* a custom http agent to make sure we can set custom keepAlive options for speedy subsequent calls
|
|
|
|
*/
|
2022-07-30 18:43:10 +02:00
|
|
|
const httpAgentKeepAliveFalse = new plugins.agentkeepalive({
|
|
|
|
keepAlive: false,
|
2023-04-19 04:07:44 +02:00
|
|
|
timeout: 60000,
|
2019-09-08 17:47:30 +02:00
|
|
|
});
|
|
|
|
|
2019-08-22 12:40:19 +02:00
|
|
|
/**
|
|
|
|
* a custom https agent to make sure we can set custom keepAlive options for speedy subsequent calls
|
|
|
|
*/
|
2022-07-30 01:52:04 +02:00
|
|
|
const httpsAgent = new plugins.agentkeepalive.HttpsAgent({
|
2022-07-29 15:41:32 +02:00
|
|
|
keepAlive: true,
|
2022-07-30 02:01:20 +02:00
|
|
|
maxFreeSockets: 10,
|
2022-07-30 18:43:10 +02:00
|
|
|
maxSockets: 100,
|
|
|
|
maxTotalSockets: 1000,
|
2023-04-19 04:07:44 +02:00
|
|
|
timeout: 60000,
|
2022-07-29 15:41:32 +02:00
|
|
|
});
|
2019-08-21 12:55:19 +02:00
|
|
|
|
2019-09-08 17:47:30 +02:00
|
|
|
/**
|
|
|
|
* a custom https agent to make sure we can set custom keepAlive options for speedy subsequent calls
|
|
|
|
*/
|
2022-07-30 01:52:04 +02:00
|
|
|
const httpsAgentKeepAliveFalse = new plugins.agentkeepalive.HttpsAgent({
|
2022-07-30 18:43:10 +02:00
|
|
|
keepAlive: false,
|
2023-04-19 04:07:44 +02:00
|
|
|
timeout: 60000,
|
2019-09-08 17:47:30 +02:00
|
|
|
});
|
|
|
|
|
2018-06-13 22:34:49 +02:00
|
|
|
export let request = async (
|
2020-09-29 15:22:25 +00:00
|
|
|
urlArg: string,
|
2018-06-13 22:34:49 +02:00
|
|
|
optionsArg: interfaces.ISmartRequestOptions = {},
|
2019-09-29 00:56:56 +02:00
|
|
|
responseStreamArg: boolean = false,
|
|
|
|
requestDataFunc: (req: plugins.http.ClientRequest) => void = null
|
2018-07-16 23:39:25 +02:00
|
|
|
): Promise<IExtendedIncomingMessage> => {
|
2022-08-01 17:15:52 +02:00
|
|
|
const done = plugins.smartpromise.defer<IExtendedIncomingMessage>();
|
2018-07-16 23:39:25 +02:00
|
|
|
|
2019-06-12 15:16:27 +02:00
|
|
|
// merge options
|
|
|
|
const defaultOptions: interfaces.ISmartRequestOptions = {
|
2019-08-21 12:55:19 +02:00
|
|
|
// agent: agent,
|
2019-09-08 17:47:30 +02:00
|
|
|
autoJsonParse: true,
|
2020-08-24 12:01:38 +00:00
|
|
|
keepAlive: true,
|
2019-08-16 21:38:50 +02:00
|
|
|
};
|
2019-06-12 15:16:27 +02:00
|
|
|
|
|
|
|
optionsArg = {
|
|
|
|
...defaultOptions,
|
2020-08-24 12:01:38 +00:00
|
|
|
...optionsArg,
|
2019-08-16 21:38:50 +02:00
|
|
|
};
|
2019-06-12 15:16:27 +02:00
|
|
|
|
2018-07-16 23:39:25 +02:00
|
|
|
// parse url
|
2021-05-16 23:39:25 +00:00
|
|
|
const parsedUrl = plugins.smarturl.Smarturl.createFromUrl(urlArg, {
|
2022-02-15 18:57:42 +01:00
|
|
|
searchParams: optionsArg.queryParams || {},
|
2021-05-16 23:39:25 +00:00
|
|
|
});
|
2018-07-16 23:39:25 +02:00
|
|
|
optionsArg.hostname = parsedUrl.hostname;
|
|
|
|
if (parsedUrl.port) {
|
2019-08-22 12:38:11 +02:00
|
|
|
optionsArg.port = parseInt(parsedUrl.port, 10);
|
2018-07-16 23:39:25 +02:00
|
|
|
}
|
|
|
|
optionsArg.path = parsedUrl.path;
|
2021-05-16 23:39:25 +00:00
|
|
|
optionsArg.queryParams = parsedUrl.searchParams;
|
2018-07-16 23:39:25 +02:00
|
|
|
|
|
|
|
// determine if unixsock
|
2020-09-29 15:22:25 +00:00
|
|
|
if (testForUnixSock(urlArg)) {
|
2018-08-14 01:47:54 +02:00
|
|
|
const detailedUnixPath = parseSocketPathAndRoute(optionsArg.path);
|
2018-07-16 23:39:25 +02:00
|
|
|
optionsArg.socketPath = detailedUnixPath.socketPath;
|
|
|
|
optionsArg.path = detailedUnixPath.path;
|
2017-06-05 19:09:40 +02:00
|
|
|
}
|
2018-08-14 01:47:54 +02:00
|
|
|
|
2019-08-16 22:00:01 +02:00
|
|
|
// TODO: support tcp sockets
|
|
|
|
|
2023-04-19 04:07:44 +02:00
|
|
|
// lets determine agent
|
|
|
|
switch (true) {
|
|
|
|
case !!optionsArg.agent:
|
|
|
|
break;
|
|
|
|
case parsedUrl.protocol === 'https:' && optionsArg.keepAlive:
|
|
|
|
optionsArg.agent = httpsAgent;
|
|
|
|
break;
|
|
|
|
case parsedUrl.protocol === 'https:' && !optionsArg.keepAlive:
|
|
|
|
optionsArg.agent = httpsAgentKeepAliveFalse;
|
|
|
|
break;
|
|
|
|
case parsedUrl.protocol === 'http:' && optionsArg.keepAlive:
|
|
|
|
optionsArg.agent = httpAgent;
|
|
|
|
break;
|
|
|
|
case parsedUrl.protocol === 'http:' && !optionsArg.keepAlive:
|
|
|
|
optionsArg.agent = httpAgentKeepAliveFalse;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-07-16 23:39:25 +02:00
|
|
|
// lets determine the request module to use
|
|
|
|
const requestModule = (() => {
|
2019-09-08 17:47:30 +02:00
|
|
|
switch (true) {
|
2023-04-19 04:07:44 +02:00
|
|
|
case parsedUrl.protocol === 'https:':
|
2019-09-08 17:47:30 +02:00
|
|
|
return plugins.https;
|
2023-04-19 04:07:44 +02:00
|
|
|
case parsedUrl.protocol === 'http:':
|
2019-09-08 17:47:30 +02:00
|
|
|
return plugins.http;
|
|
|
|
}
|
2018-07-16 23:39:25 +02:00
|
|
|
})() as typeof plugins.https;
|
|
|
|
|
2020-09-29 15:20:40 +00:00
|
|
|
if (!requestModule) {
|
2020-09-29 15:22:25 +00:00
|
|
|
console.error(`The request to ${urlArg} is missing a viable protocol. Must be http or https`);
|
2020-09-29 15:20:40 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-07-16 23:39:25 +02:00
|
|
|
// lets perform the actual request
|
2023-04-19 14:38:28 +02:00
|
|
|
const requestToFire = requestModule.request(optionsArg, async (resArg) => {
|
|
|
|
if (optionsArg.hardDataCuttingTimeout) {
|
|
|
|
setTimeout(() => {
|
|
|
|
resArg.destroy();
|
|
|
|
done.reject(new Error('Request timed out'));
|
|
|
|
}, optionsArg.hardDataCuttingTimeout)
|
|
|
|
}
|
|
|
|
|
2019-09-29 00:56:56 +02:00
|
|
|
if (responseStreamArg) {
|
2023-04-19 14:38:28 +02:00
|
|
|
done.resolve(resArg as IExtendedIncomingMessage);
|
2019-09-28 21:33:13 +02:00
|
|
|
} else {
|
2023-04-19 14:38:28 +02:00
|
|
|
const builtResponse = await buildUtf8Response(resArg, optionsArg.autoJsonParse);
|
2019-09-28 21:33:13 +02:00
|
|
|
done.resolve(builtResponse);
|
|
|
|
}
|
|
|
|
});
|
2018-07-16 23:39:25 +02:00
|
|
|
|
|
|
|
// lets write the requestBody
|
|
|
|
if (optionsArg.requestBody) {
|
2019-09-28 21:33:13 +02:00
|
|
|
if (optionsArg.requestBody instanceof plugins.formData) {
|
2022-02-15 18:53:02 +01:00
|
|
|
optionsArg.requestBody.pipe(requestToFire).on('finish', (event: any) => {
|
2019-09-28 21:33:13 +02:00
|
|
|
requestToFire.end();
|
|
|
|
});
|
|
|
|
} else {
|
2018-08-14 01:47:54 +02:00
|
|
|
if (typeof optionsArg.requestBody !== 'string') {
|
2018-08-14 01:46:12 +02:00
|
|
|
optionsArg.requestBody = JSON.stringify(optionsArg.requestBody);
|
|
|
|
}
|
2019-08-22 12:38:11 +02:00
|
|
|
requestToFire.write(optionsArg.requestBody);
|
|
|
|
requestToFire.end();
|
2017-01-29 00:51:47 +01:00
|
|
|
}
|
2019-09-29 00:56:56 +02:00
|
|
|
} else if (requestDataFunc) {
|
|
|
|
requestDataFunc(requestToFire);
|
2018-07-19 23:22:11 +02:00
|
|
|
} else {
|
2019-08-22 12:38:11 +02:00
|
|
|
requestToFire.end();
|
2017-06-05 19:09:40 +02:00
|
|
|
}
|
2018-07-19 16:16:02 +02:00
|
|
|
|
|
|
|
// lets handle an error
|
2020-08-24 12:01:38 +00:00
|
|
|
requestToFire.on('error', (e) => {
|
2018-07-16 23:39:25 +02:00
|
|
|
console.error(e);
|
2022-08-01 17:10:11 +02:00
|
|
|
requestToFire.destroy();
|
2018-07-16 23:39:25 +02:00
|
|
|
});
|
2018-07-19 16:16:02 +02:00
|
|
|
|
2022-08-01 17:15:52 +02:00
|
|
|
const response = await done.promise;
|
|
|
|
response.on('error', (err) => {
|
|
|
|
console.log(err);
|
|
|
|
response.destroy();
|
2023-04-19 04:07:44 +02:00
|
|
|
});
|
2022-08-01 17:15:52 +02:00
|
|
|
return response;
|
2018-06-13 22:34:49 +02:00
|
|
|
};
|
2023-04-19 04:07:44 +02:00
|
|
|
|
|
|
|
export const safeGet = async (urlArg: string) => {
|
2023-04-19 14:38:28 +02:00
|
|
|
const agentToUse = urlArg.startsWith('http://') ? new plugins.http.Agent() : new plugins.https.Agent();
|
2023-04-19 04:07:44 +02:00
|
|
|
try {
|
|
|
|
const response = await request(urlArg, {
|
|
|
|
method: 'GET',
|
2023-04-19 14:24:43 +02:00
|
|
|
agent: agentToUse,
|
2023-04-19 04:07:44 +02:00
|
|
|
timeout: 5000,
|
2023-04-19 14:38:28 +02:00
|
|
|
hardDataCuttingTimeout: 5000,
|
2023-04-19 04:07:44 +02:00
|
|
|
autoJsonParse: false,
|
|
|
|
});
|
|
|
|
return response;
|
|
|
|
} catch (err) {
|
|
|
|
console.log(err);
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
};
|