Notes/src/services/request.js

190 lines
5.8 KiB
JavaScript
Raw Normal View History

"use strict";
const utils = require('./utils');
2018-12-18 20:34:24 +01:00
const log = require('./log');
const url = require('url');
2020-03-25 11:28:44 +01:00
const syncOptions = require('./sync_options');
// this service provides abstraction over node's HTTP/HTTPS and electron net.client APIs
// this allows to support system proxy
function exec(opts) {
const client = getClient(opts);
2019-07-24 20:47:41 +02:00
// hack for cases where electron.net does not work but we don't want to set proxy
if (opts.proxy === 'noproxy') {
opts.proxy = null;
}
2021-01-10 21:56:40 +01:00
if (!opts.paging) {
opts.paging = {
pageCount: 1,
2021-01-11 22:48:51 +01:00
pageIndex: 0,
requestId: 'n/a'
2021-01-10 21:56:40 +01:00
};
}
2019-07-24 20:47:41 +02:00
const proxyAgent = getProxyAgent(opts);
2018-12-17 22:54:54 +01:00
const parsedTargetUrl = url.parse(opts.url);
2020-06-20 12:31:38 +02:00
return new Promise((resolve, reject) => {
try {
2021-01-11 22:48:51 +01:00
const headers = {
Cookie: (opts.cookieJar && opts.cookieJar.header) || "",
2021-01-10 21:56:40 +01:00
'Content-Type': opts.paging.pageCount === 1 ? 'application/json' : 'text/plain',
2021-01-11 22:48:51 +01:00
pageCount: opts.paging.pageCount,
pageIndex: opts.paging.pageIndex,
requestId: opts.paging.requestId
};
if (opts.auth) {
2021-02-07 21:50:34 +01:00
headers['trilium-cred'] = Buffer.from(opts.auth.username + ":" + opts.auth.password).toString('base64');
}
const request = client.request({
method: opts.method,
// url is used by electron net module
url: opts.url,
// 4 fields below are used by http and https node modules
2019-07-24 20:47:41 +02:00
protocol: parsedTargetUrl.protocol,
host: parsedTargetUrl.hostname,
port: parsedTargetUrl.port,
path: parsedTargetUrl.path,
timeout: opts.timeout, // works only for node.js client
2019-07-24 20:47:41 +02:00
headers,
agent: proxyAgent
});
request.on('error', err => reject(generateError(opts, err)));
request.on('response', response => {
2018-12-18 20:39:56 +01:00
if (![200, 201, 204].includes(response.statusCode)) {
reject(generateError(opts, response.statusCode + ' ' + response.statusMessage));
}
if (opts.cookieJar && response.headers['set-cookie']) {
opts.cookieJar.header = response.headers['set-cookie'];
}
let responseStr = '';
response.on('data', chunk => responseStr += chunk);
response.on('end', () => {
try {
const jsonObj = responseStr.trim() ? JSON.parse(responseStr) : null;
resolve(jsonObj);
}
catch (e) {
log.error("Failed to deserialize sync response: " + responseStr);
2018-12-18 20:39:56 +01:00
reject(generateError(opts, e.message));
}
});
});
2021-01-11 22:48:51 +01:00
request.end(opts.body);
}
catch (e) {
2018-12-18 20:39:56 +01:00
reject(generateError(opts, e.message));
}
2020-03-25 11:28:44 +01:00
});
}
2020-06-20 12:31:38 +02:00
function getImage(imageUrl) {
const proxyConf = syncOptions.getSyncProxy();
2020-03-25 11:28:44 +01:00
const opts = {
method: 'GET',
url: imageUrl,
proxy: proxyConf !== "noproxy" ? proxyConf : null
2020-03-25 11:28:44 +01:00
};
const client = getClient(opts);
const proxyAgent = getProxyAgent(opts);
const parsedTargetUrl = url.parse(opts.url);
2020-06-20 12:31:38 +02:00
return new Promise((resolve, reject) => {
2020-03-25 11:28:44 +01:00
try {
const request = client.request({
method: opts.method,
// url is used by electron net module
url: opts.url,
// 4 fields below are used by http and https node modules
protocol: parsedTargetUrl.protocol,
host: parsedTargetUrl.hostname,
port: parsedTargetUrl.port,
path: parsedTargetUrl.path,
timeout: opts.timeout, // works only for node client
2020-03-25 11:28:44 +01:00
headers: {},
agent: proxyAgent
});
request.on('error', err => reject(generateError(opts, err)));
request.on('abort', err => reject(generateError(opts, err)));
2020-03-25 11:28:44 +01:00
request.on('response', response => {
if (![200, 201, 204].includes(response.statusCode)) {
reject(generateError(opts, response.statusCode + ' ' + response.statusMessage));
}
const chunks = []
response.on('data', chunk => chunks.push(chunk));
response.on('end', () => resolve(Buffer.concat(chunks)));
});
request.end(undefined);
}
catch (e) {
reject(generateError(opts, e.message));
}
});
}
2019-07-24 20:47:41 +02:00
function getProxyAgent(opts) {
if (!opts.proxy) {
2020-03-25 11:28:44 +01:00
return null;
2019-07-24 20:47:41 +02:00
}
const {protocol} = url.parse(opts.url);
if (protocol === 'http:' || protocol === 'https:') {
const protoNoColon = protocol.substr(0, protocol.length - 1);
const AgentClass = require(protoNoColon + '-proxy-agent');
return new AgentClass(opts.proxy);
}
else {
return null;
}
}
function getClient(opts) {
2018-12-17 22:54:54 +01:00
// it's not clear how to explicitly configure proxy (as opposed to system proxy)
// so in that case we always use node's modules
if (utils.isElectron() && !opts.proxy) {
return require('electron').net;
}
else {
2019-07-24 20:47:41 +02:00
const {protocol} = url.parse(opts.url);
if (protocol === 'http:' || protocol === 'https:') {
return require(protocol.substr(0, protocol.length - 1));
}
else {
throw new Error(`Unrecognized protocol "${protocol}"`);
}
}
}
2018-12-18 20:39:56 +01:00
function generateError(opts, message) {
return new Error(`Request to ${opts.method} ${opts.url} failed, error: ${message}`);
}
module.exports = {
2020-03-25 11:28:44 +01:00
exec,
getImage
};