2017-03-16 17:51:56 +00:00
|
|
|
import electron from 'electron';
|
|
|
|
import mkdirp from 'mkdirp';
|
2017-04-09 21:53:46 +00:00
|
|
|
import mimes from 'mime-types';
|
2017-03-16 17:51:56 +00:00
|
|
|
import {parse as urlParse} from 'url';
|
|
|
|
import {Curl} from 'node-libcurl';
|
2017-04-07 18:10:15 +00:00
|
|
|
import {join as pathJoin} from 'path';
|
2017-03-16 17:51:56 +00:00
|
|
|
import * as models from '../models';
|
|
|
|
import * as querystring from '../common/querystring';
|
|
|
|
import * as util from '../common/misc.js';
|
2017-04-05 05:48:04 +00:00
|
|
|
import {AUTH_BASIC, AUTH_DIGEST, AUTH_NTLM, CONTENT_TYPE_FORM_DATA, CONTENT_TYPE_FORM_URLENCODED, DEBOUNCE_MILLIS, getAppVersion} from '../common/constants';
|
2017-05-16 17:45:09 +00:00
|
|
|
import {describeByteSize, getSetCookieHeaders, hasAuthHeader, hasContentTypeHeader, hasUserAgentHeader, setDefaultProtocol} from '../common/misc';
|
2017-03-16 17:51:56 +00:00
|
|
|
import {getRenderedRequest} from '../common/render';
|
|
|
|
import fs from 'fs';
|
|
|
|
import * as db from '../common/database';
|
2017-03-28 22:45:23 +00:00
|
|
|
import * as CACerts from './cacert';
|
2017-03-23 22:10:42 +00:00
|
|
|
import {getAuthHeader} from './authentication';
|
2017-05-16 17:45:09 +00:00
|
|
|
import {cookiesFromJar, jarFromCookies} from '../common/cookies';
|
2017-03-16 17:51:56 +00:00
|
|
|
|
|
|
|
let cancelRequestFunction = null;
|
|
|
|
|
|
|
|
export function cancelCurrentRequest () {
|
|
|
|
if (typeof cancelRequestFunction === 'function') {
|
|
|
|
cancelRequestFunction();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-07 18:10:15 +00:00
|
|
|
export function _actuallySend (renderedRequest, workspace, settings) {
|
2017-03-16 17:51:56 +00:00
|
|
|
return new Promise(async resolve => {
|
|
|
|
function handleError (err, prefix = null) {
|
|
|
|
resolve({
|
|
|
|
url: renderedRequest.url,
|
|
|
|
parentId: renderedRequest._id,
|
|
|
|
error: prefix ? `${prefix}: ${err.message}` : err.message,
|
|
|
|
elapsedTime: 0,
|
2017-04-07 18:10:15 +00:00
|
|
|
statusMessage: 'Error',
|
|
|
|
settingSendCookies: renderedRequest.settingSendCookies,
|
|
|
|
settingStoreCookies: renderedRequest.settingStoreCookies
|
2017-03-16 17:51:56 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
2017-03-23 22:10:42 +00:00
|
|
|
// Initialize the curl handle
|
|
|
|
const curl = new Curl();
|
2017-03-29 23:09:28 +00:00
|
|
|
let timeline = [];
|
|
|
|
|
|
|
|
// Define helper to add base fields when responding
|
|
|
|
const respond = patch => {
|
|
|
|
resolve(Object.assign({
|
|
|
|
parentId: renderedRequest._id,
|
|
|
|
timeline: timeline,
|
|
|
|
settingSendCookies: renderedRequest.settingSendCookies,
|
|
|
|
settingStoreCookies: renderedRequest.settingStoreCookies
|
|
|
|
}, patch));
|
|
|
|
};
|
2017-03-23 22:10:42 +00:00
|
|
|
|
2017-04-07 18:10:15 +00:00
|
|
|
// Define helper to setOpt for better error handling
|
|
|
|
const setOpt = (opt, val, optional = false) => {
|
|
|
|
const name = Object.keys(Curl.option).find(name => Curl.option[name] === opt);
|
|
|
|
try {
|
|
|
|
curl.setOpt(opt, val);
|
|
|
|
} catch (err) {
|
|
|
|
if (!optional) {
|
|
|
|
throw new Error(`${err.message} (${opt} ${name || 'n/a'})`);
|
|
|
|
} else {
|
|
|
|
console.warn(`Failed to set optional Curl opt (${opt} ${name || 'n/a'})`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const setOptionalOpt = (opt, val) => setOpt(opt, val, true);
|
|
|
|
|
2017-03-16 17:51:56 +00:00
|
|
|
// Setup the cancellation logic
|
|
|
|
cancelRequestFunction = () => {
|
2017-03-29 23:09:28 +00:00
|
|
|
respond({
|
2017-04-06 02:30:28 +00:00
|
|
|
elapsedTime: curl.getInfo(Curl.info.TOTAL_TIME) * 1000,
|
|
|
|
bytesRead: curl.getInfo(Curl.info.SIZE_DOWNLOAD),
|
|
|
|
url: curl.getInfo(Curl.info.EFFECTIVE_URL),
|
2017-03-23 22:10:42 +00:00
|
|
|
statusMessage: 'Cancelled',
|
|
|
|
error: 'Request was cancelled'
|
|
|
|
});
|
2017-03-16 17:51:56 +00:00
|
|
|
|
2017-03-23 22:10:42 +00:00
|
|
|
// Kill it!
|
|
|
|
curl.close();
|
|
|
|
};
|
2017-03-16 17:51:56 +00:00
|
|
|
|
|
|
|
// Set all the basic options
|
2017-04-07 18:10:15 +00:00
|
|
|
setOpt(Curl.option.CUSTOMREQUEST, renderedRequest.method);
|
2017-04-20 01:37:40 +00:00
|
|
|
setOpt(Curl.option.NOBODY, renderedRequest.method.toLowerCase() === 'head' ? 1 : 0);
|
2017-04-07 18:10:15 +00:00
|
|
|
setOpt(Curl.option.FOLLOWLOCATION, settings.followRedirects);
|
|
|
|
setOpt(Curl.option.TIMEOUT_MS, settings.timeout); // 0 for no timeout
|
|
|
|
setOpt(Curl.option.VERBOSE, true); // True so debug function works
|
|
|
|
setOpt(Curl.option.NOPROGRESS, false); // False so progress function works
|
2017-04-20 01:37:40 +00:00
|
|
|
setOpt(Curl.option.ACCEPT_ENCODING, ''); // Auto decode everything
|
2017-03-28 22:45:23 +00:00
|
|
|
|
|
|
|
// Setup debug handler
|
2017-04-07 18:10:15 +00:00
|
|
|
setOpt(Curl.option.DEBUGFUNCTION, (infoType, content) => {
|
2017-03-29 02:21:49 +00:00
|
|
|
const name = Object.keys(Curl.info.debug).find(k => Curl.info.debug[k] === infoType);
|
|
|
|
|
2017-04-04 23:06:43 +00:00
|
|
|
if (
|
|
|
|
infoType === Curl.info.debug.SSL_DATA_IN ||
|
|
|
|
infoType === Curl.info.debug.SSL_DATA_OUT
|
|
|
|
) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-03-28 22:45:23 +00:00
|
|
|
// Ignore the possibly large data messages
|
2017-03-29 02:21:49 +00:00
|
|
|
if (infoType === Curl.info.debug.DATA_OUT) {
|
2017-04-09 21:53:46 +00:00
|
|
|
if (content.length < 1000) {
|
2017-03-29 02:21:49 +00:00
|
|
|
timeline.push({name, value: content});
|
|
|
|
} else {
|
|
|
|
timeline.push({name, value: `(${describeByteSize(content.length)} hidden)`});
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (infoType === Curl.info.debug.DATA_IN) {
|
2017-03-29 23:09:28 +00:00
|
|
|
timeline.push({
|
|
|
|
name: 'TEXT',
|
|
|
|
value: `Received ${describeByteSize(content.length)} chunk`
|
|
|
|
});
|
2017-03-28 22:45:23 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Don't show cookie setting because this will display every domain in the jar
|
|
|
|
if (infoType === Curl.info.debug.TEXT && content.indexOf('Added cookie') === 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
timeline.push({name, value: content});
|
|
|
|
|
|
|
|
return 0; // Must be here
|
|
|
|
});
|
2017-03-16 17:51:56 +00:00
|
|
|
|
2017-03-23 22:10:42 +00:00
|
|
|
// Set the headers (to be modified as we go)
|
|
|
|
const headers = [...renderedRequest.headers];
|
|
|
|
|
2017-03-16 17:51:56 +00:00
|
|
|
let lastPercent = 0;
|
2017-04-07 18:10:15 +00:00
|
|
|
// NOTE: This option was added in 7.32.0 so make it optional
|
|
|
|
setOptionalOpt(Curl.option.XFERINFOFUNCTION, (dltotal, dlnow, ultotal, ulnow) => {
|
2017-03-16 17:51:56 +00:00
|
|
|
if (dltotal === 0) {
|
2017-03-23 22:10:42 +00:00
|
|
|
return 0;
|
2017-03-16 17:51:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const percent = Math.round(dlnow / dltotal * 100);
|
|
|
|
if (percent !== lastPercent) {
|
2017-03-23 22:10:42 +00:00
|
|
|
// console.log('PROGRESS 2', `${percent}%`, ultotal, ulnow);
|
2017-03-16 17:51:56 +00:00
|
|
|
lastPercent = percent;
|
|
|
|
}
|
|
|
|
|
2017-03-23 22:10:42 +00:00
|
|
|
return 0;
|
2017-03-16 17:51:56 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
// Set the URL, including the query parameters
|
|
|
|
const qs = querystring.buildFromParams(renderedRequest.parameters);
|
|
|
|
const url = querystring.joinUrl(renderedRequest.url, qs);
|
2017-03-29 23:09:28 +00:00
|
|
|
const finalUrl = util.prepareUrlForSending(url, renderedRequest.settingEncodeUrl);
|
2017-04-07 18:10:15 +00:00
|
|
|
setOpt(Curl.option.URL, finalUrl);
|
2017-03-29 23:09:28 +00:00
|
|
|
timeline.push({name: 'TEXT', value: 'Preparing request to ' + finalUrl});
|
|
|
|
|
|
|
|
// log some things
|
|
|
|
if (renderedRequest.settingEncodeUrl) {
|
|
|
|
timeline.push({name: 'TEXT', value: 'Enable automatic URL encoding'});
|
|
|
|
} else {
|
|
|
|
timeline.push({name: 'TEXT', value: 'Disable automatic URL encoding'});
|
|
|
|
}
|
2017-03-16 17:51:56 +00:00
|
|
|
|
2017-04-07 18:10:15 +00:00
|
|
|
// SSL Validation
|
|
|
|
if (settings.validateSSL) {
|
|
|
|
timeline.push({name: 'TEXT', value: 'Enable SSL validation'});
|
|
|
|
} else {
|
|
|
|
setOpt(Curl.option.SSL_VERIFYHOST, 0);
|
|
|
|
setOpt(Curl.option.SSL_VERIFYPEER, 0);
|
|
|
|
timeline.push({name: 'TEXT', value: 'Disable SSL validation'});
|
|
|
|
}
|
|
|
|
|
2017-03-16 17:51:56 +00:00
|
|
|
// Setup CA Root Certificates if not on Mac. Thanks to libcurl, Mac will use
|
|
|
|
// certificates form the OS.
|
|
|
|
if (process.platform !== 'darwin') {
|
2017-03-28 22:45:23 +00:00
|
|
|
const basCAPath = pathJoin(electron.remote.app.getPath('temp'), 'insomnia');
|
|
|
|
const fullCAPath = pathJoin(basCAPath, CACerts.filename);
|
|
|
|
|
|
|
|
try {
|
|
|
|
fs.statSync(fullCAPath);
|
|
|
|
} catch (err) {
|
|
|
|
// Doesn't exist yet, so write it
|
|
|
|
mkdirp.sync(basCAPath);
|
|
|
|
fs.writeFileSync(fullCAPath, CACerts.blob);
|
|
|
|
console.log('[net] Set CA to', fullCAPath);
|
|
|
|
}
|
2017-03-16 17:51:56 +00:00
|
|
|
|
2017-04-07 18:10:15 +00:00
|
|
|
setOpt(Curl.option.CAINFO, fullCAPath);
|
2017-03-16 17:51:56 +00:00
|
|
|
}
|
|
|
|
|
2017-04-12 21:17:40 +00:00
|
|
|
// Set cookies from jar
|
2017-03-28 22:45:23 +00:00
|
|
|
if (renderedRequest.settingSendCookies) {
|
2017-03-29 23:09:28 +00:00
|
|
|
const cookies = renderedRequest.cookieJar.cookies || [];
|
|
|
|
for (const cookie of cookies) {
|
2017-03-28 22:45:23 +00:00
|
|
|
let expiresTimestamp = 0;
|
|
|
|
if (cookie.expires) {
|
|
|
|
const expiresDate = new Date(cookie.expires);
|
|
|
|
expiresTimestamp = Math.round(expiresDate.getTime() / 1000);
|
|
|
|
}
|
2017-05-16 17:45:09 +00:00
|
|
|
|
2017-04-07 18:10:15 +00:00
|
|
|
setOpt(Curl.option.COOKIELIST, [
|
2017-03-28 22:45:23 +00:00
|
|
|
cookie.httpOnly ? `#HttpOnly_${cookie.domain}` : cookie.domain,
|
2017-05-16 17:45:09 +00:00
|
|
|
cookie.hostOnly ? 'FALSE' : 'TRUE',
|
2017-03-28 22:45:23 +00:00
|
|
|
cookie.path,
|
|
|
|
cookie.secure ? 'TRUE' : 'FALSE',
|
|
|
|
expiresTimestamp,
|
|
|
|
cookie.key,
|
|
|
|
cookie.value
|
|
|
|
].join('\t'));
|
2017-03-16 17:51:56 +00:00
|
|
|
}
|
2017-03-29 23:09:28 +00:00
|
|
|
|
|
|
|
timeline.push({
|
|
|
|
name: 'TEXT',
|
|
|
|
value: 'Enable cookie sending with jar of ' +
|
|
|
|
`${cookies.length} cookie${cookies.length !== 1 ? 's' : ''}`
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
timeline.push({
|
|
|
|
name: 'TEXT',
|
|
|
|
value: 'Disable cookie sending due to user setting'
|
|
|
|
});
|
2017-03-16 17:51:56 +00:00
|
|
|
}
|
|
|
|
|
2017-03-28 22:45:23 +00:00
|
|
|
// Set proxy settings if we have them
|
|
|
|
if (settings.proxyEnabled) {
|
|
|
|
const {protocol} = urlParse(renderedRequest.url);
|
|
|
|
const {httpProxy, httpsProxy} = settings;
|
|
|
|
const proxyHost = protocol === 'https:' ? httpsProxy : httpProxy;
|
|
|
|
const proxy = proxyHost ? setDefaultProtocol(proxyHost) : null;
|
2017-03-29 23:09:28 +00:00
|
|
|
timeline.push({name: 'TEXT', value: `Enable network proxy for ${protocol}`});
|
2017-03-28 22:45:23 +00:00
|
|
|
if (proxy) {
|
2017-04-07 18:10:15 +00:00
|
|
|
setOpt(Curl.option.PROXY, proxy);
|
|
|
|
setOpt(Curl.option.PROXYAUTH, Curl.auth.ANY);
|
2017-03-28 22:45:23 +00:00
|
|
|
}
|
2017-04-07 18:10:15 +00:00
|
|
|
} else {
|
|
|
|
setOpt(Curl.option.PROXY, '');
|
2017-03-16 17:51:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Set client certs if needed
|
|
|
|
for (const certificate of workspace.certificates) {
|
|
|
|
const cHostWithProtocol = setDefaultProtocol(certificate.host, 'https:');
|
|
|
|
const {hostname: cHostname, port: cPort} = urlParse(cHostWithProtocol);
|
|
|
|
const {hostname, port} = urlParse(renderedRequest.url);
|
|
|
|
|
|
|
|
const assumedPort = parseInt(port) || 443;
|
|
|
|
const assumedCPort = parseInt(cPort) || 443;
|
|
|
|
|
|
|
|
// Exact host match (includes port)
|
|
|
|
if (cHostname === hostname && assumedCPort === assumedPort) {
|
|
|
|
const ensureFile = blobOrFilename => {
|
|
|
|
if (blobOrFilename.indexOf('/') === 0) {
|
|
|
|
return blobOrFilename;
|
|
|
|
} else {
|
|
|
|
// Legacy support. Certs used to be stored in blobs, so lets write it to
|
|
|
|
// the temp directory first.
|
|
|
|
// TODO: Delete this fallback eventually
|
|
|
|
const fullBase = pathJoin(electron.remote.app.getPath('temp'), 'insomnia');
|
|
|
|
mkdirp.sync(fullBase);
|
|
|
|
|
|
|
|
const name = `${renderedRequest._id}_${renderedRequest.modified}`;
|
|
|
|
const fullPath = pathJoin(fullBase, name);
|
|
|
|
fs.writeFileSync(fullPath, new Buffer(blobOrFilename, 'base64'));
|
|
|
|
|
|
|
|
return fullPath;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const {passphrase, cert, key, pfx} = certificate;
|
|
|
|
|
|
|
|
if (cert) {
|
2017-04-07 18:10:15 +00:00
|
|
|
setOpt(Curl.option.SSLCERT, ensureFile(cert));
|
|
|
|
setOpt(Curl.option.SSLCERTTYPE, 'PEM');
|
2017-03-29 23:09:28 +00:00
|
|
|
timeline.push({name: 'TEXT', value: 'Adding SSL PEM certificate'});
|
2017-03-16 17:51:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (pfx) {
|
2017-04-07 18:10:15 +00:00
|
|
|
setOpt(Curl.option.SSLCERT, ensureFile(pfx));
|
|
|
|
setOpt(Curl.option.SSLCERTTYPE, 'P12');
|
2017-03-29 23:09:28 +00:00
|
|
|
timeline.push({name: 'TEXT', value: 'Adding SSL P12 certificate'});
|
2017-03-16 17:51:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (key) {
|
2017-04-07 18:10:15 +00:00
|
|
|
setOpt(Curl.option.SSLKEY, ensureFile(key));
|
2017-03-29 23:09:28 +00:00
|
|
|
timeline.push({name: 'TEXT', value: 'Adding SSL KEY certificate'});
|
2017-03-16 17:51:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (passphrase) {
|
2017-04-07 18:10:15 +00:00
|
|
|
setOpt(Curl.option.KEYPASSWD, passphrase);
|
2017-03-16 17:51:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Build the body
|
2017-04-09 21:53:46 +00:00
|
|
|
let noBody = false;
|
2017-04-20 17:18:00 +00:00
|
|
|
const expectsBody = ['POST', 'PUT', 'PATCH'].includes(renderedRequest.method.toUpperCase());
|
2017-03-16 17:51:56 +00:00
|
|
|
if (renderedRequest.body.mimeType === CONTENT_TYPE_FORM_URLENCODED) {
|
|
|
|
const d = querystring.buildFromParams(renderedRequest.body.params || [], true);
|
2017-04-07 18:10:15 +00:00
|
|
|
setOpt(Curl.option.POSTFIELDS, d); // Send raw data
|
2017-03-16 17:51:56 +00:00
|
|
|
} else if (renderedRequest.body.mimeType === CONTENT_TYPE_FORM_DATA) {
|
|
|
|
const data = renderedRequest.body.params.map(param => {
|
|
|
|
if (param.type === 'file' && param.fileName) {
|
2017-04-12 21:17:40 +00:00
|
|
|
const type = mimes.lookup(param.fileName) || 'application/octet-stream';
|
|
|
|
return {name: param.name, file: param.fileName, type};
|
2017-03-16 17:51:56 +00:00
|
|
|
} else {
|
|
|
|
return {name: param.name, contents: param.value};
|
|
|
|
}
|
|
|
|
});
|
2017-04-07 18:10:15 +00:00
|
|
|
setOpt(Curl.option.HTTPPOST, data);
|
2017-03-16 17:51:56 +00:00
|
|
|
} else if (renderedRequest.body.fileName) {
|
2017-04-09 21:53:46 +00:00
|
|
|
const {size} = fs.statSync(renderedRequest.body.fileName);
|
|
|
|
headers.push({name: 'Content-Length', value: `${size}`});
|
|
|
|
const fd = fs.openSync(renderedRequest.body.fileName, 'r+');
|
2017-04-07 18:10:15 +00:00
|
|
|
setOpt(Curl.option.UPLOAD, 1);
|
|
|
|
setOpt(Curl.option.READDATA, fd);
|
2017-03-16 17:51:56 +00:00
|
|
|
const fn = () => fs.closeSync(fd);
|
|
|
|
curl.on('end', fn);
|
|
|
|
curl.on('error', fn);
|
2017-04-20 17:18:00 +00:00
|
|
|
} else if (typeof renderedRequest.body.mimeType === 'string' || expectsBody) {
|
2017-04-14 22:27:08 +00:00
|
|
|
setOpt(Curl.option.POSTFIELDS, renderedRequest.body.text || '');
|
2017-03-16 17:51:56 +00:00
|
|
|
} else {
|
|
|
|
// No body
|
2017-04-09 21:53:46 +00:00
|
|
|
noBody = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!noBody) {
|
|
|
|
// Don't chunk uploads
|
|
|
|
headers.push({name: 'Expect', value: ''});
|
|
|
|
headers.push({name: 'Transfer-Encoding', value: ''});
|
2017-03-16 17:51:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Build the body
|
|
|
|
const dataBuffers = [];
|
|
|
|
let dataBuffersLength = 0;
|
|
|
|
curl.on('data', chunk => {
|
|
|
|
dataBuffers.push(chunk);
|
|
|
|
dataBuffersLength += chunk.length;
|
|
|
|
});
|
|
|
|
|
2017-03-23 22:10:42 +00:00
|
|
|
// Handle Authorization header
|
2017-04-11 21:20:01 +00:00
|
|
|
if (!hasAuthHeader(headers) && !renderedRequest.authentication.disabled) {
|
2017-03-28 22:45:23 +00:00
|
|
|
if (renderedRequest.authentication.type === AUTH_BASIC) {
|
|
|
|
const {username, password} = renderedRequest.authentication;
|
2017-04-07 18:10:15 +00:00
|
|
|
setOpt(Curl.option.HTTPAUTH, Curl.auth.BASIC);
|
|
|
|
setOpt(Curl.option.USERNAME, username || '');
|
|
|
|
setOpt(Curl.option.PASSWORD, password || '');
|
2017-03-28 22:45:23 +00:00
|
|
|
} else if (renderedRequest.authentication.type === AUTH_DIGEST) {
|
|
|
|
const {username, password} = renderedRequest.authentication;
|
2017-04-07 18:10:15 +00:00
|
|
|
setOpt(Curl.option.HTTPAUTH, Curl.auth.DIGEST);
|
|
|
|
setOpt(Curl.option.USERNAME, username || '');
|
|
|
|
setOpt(Curl.option.PASSWORD, password || '');
|
2017-04-05 05:48:04 +00:00
|
|
|
} else if (renderedRequest.authentication.type === AUTH_NTLM) {
|
|
|
|
const {username, password} = renderedRequest.authentication;
|
2017-04-07 18:10:15 +00:00
|
|
|
setOpt(Curl.option.HTTPAUTH, Curl.auth.NTLM);
|
|
|
|
setOpt(Curl.option.USERNAME, username || '');
|
|
|
|
setOpt(Curl.option.PASSWORD, password || '');
|
2017-03-28 22:45:23 +00:00
|
|
|
} else {
|
|
|
|
const authHeader = await getAuthHeader(
|
|
|
|
renderedRequest._id,
|
|
|
|
renderedRequest.authentication
|
|
|
|
);
|
|
|
|
|
|
|
|
if (authHeader) {
|
|
|
|
headers.push(authHeader);
|
|
|
|
}
|
2017-03-23 22:10:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set User-Agent if it't not already in headers
|
2017-04-11 21:20:01 +00:00
|
|
|
if (!hasUserAgentHeader(headers)) {
|
2017-04-07 18:10:15 +00:00
|
|
|
setOpt(Curl.option.USERAGENT, `insomnia/${getAppVersion()}`);
|
2017-03-23 22:10:42 +00:00
|
|
|
}
|
2017-03-16 17:51:56 +00:00
|
|
|
|
2017-04-11 21:20:01 +00:00
|
|
|
// Prevent curl from adding default content-type header
|
|
|
|
if (!hasContentTypeHeader(headers)) {
|
|
|
|
headers.push({name: 'content-type', value: ''});
|
|
|
|
}
|
|
|
|
|
|
|
|
// NOTE: This is last because headers might be modified multiple times
|
|
|
|
const headerStrings = headers
|
|
|
|
.filter(h => h.name)
|
|
|
|
.map(h => `${(h.name || '').trim()}: ${h.value}`);
|
|
|
|
setOpt(Curl.option.HTTPHEADER, headerStrings);
|
|
|
|
|
2017-03-16 17:51:56 +00:00
|
|
|
// Handle the response ending
|
2017-05-16 17:45:09 +00:00
|
|
|
curl.on('end', async function (_1, _2, curlHeaders) {
|
2017-03-16 17:51:56 +00:00
|
|
|
// Headers are an array (one for each redirect)
|
|
|
|
curlHeaders = curlHeaders[curlHeaders.length - 1];
|
|
|
|
|
|
|
|
// Collect various things
|
2017-04-06 02:30:28 +00:00
|
|
|
const result = curlHeaders && curlHeaders.result;
|
|
|
|
const statusCode = result ? result.code : 0;
|
|
|
|
const statusMessage = result ? result.reason : 'Unknown';
|
2017-03-16 17:51:56 +00:00
|
|
|
|
|
|
|
// Collect the headers
|
|
|
|
const headers = [];
|
2017-04-07 18:10:15 +00:00
|
|
|
for (const name of curlHeaders ? Object.keys(curlHeaders) : []) {
|
2017-03-16 17:51:56 +00:00
|
|
|
if (typeof curlHeaders[name] === 'string') {
|
|
|
|
headers.push({name, value: curlHeaders[name]});
|
|
|
|
} else if (Array.isArray(curlHeaders[name])) {
|
|
|
|
for (const value of curlHeaders[name]) {
|
|
|
|
headers.push({name, value});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Calculate the content type
|
|
|
|
const contentTypeHeader = util.getContentTypeHeader(headers);
|
|
|
|
const contentType = contentTypeHeader ? contentTypeHeader.value : '';
|
|
|
|
|
|
|
|
// Update Cookie Jar
|
2017-05-16 17:45:09 +00:00
|
|
|
const setCookieHeaders = getSetCookieHeaders(headers);
|
|
|
|
if (renderedRequest.settingStoreCookies && setCookieHeaders.length) {
|
|
|
|
const jar = jarFromCookies(renderedRequest.cookieJar.cookies);
|
|
|
|
for (const header of getSetCookieHeaders(headers)) {
|
|
|
|
jar.setCookieSync(header.value, curl.getInfo(Curl.info.EFFECTIVE_URL));
|
|
|
|
}
|
2017-03-28 22:45:23 +00:00
|
|
|
|
2017-05-16 17:45:09 +00:00
|
|
|
const cookies = await cookiesFromJar(jar);
|
2017-03-29 23:09:28 +00:00
|
|
|
|
2017-05-16 17:45:09 +00:00
|
|
|
// Make sure domains are prefixed with dots (Curl does this)
|
|
|
|
for (const cookie of cookies) {
|
|
|
|
if (cookie.domain[0] !== '.') {
|
|
|
|
cookie.domain = `.${cookie.domain}`;
|
|
|
|
}
|
2017-03-29 23:09:28 +00:00
|
|
|
}
|
2017-05-16 17:45:09 +00:00
|
|
|
|
|
|
|
models.cookieJar.update(renderedRequest.cookieJar, {cookies});
|
|
|
|
|
|
|
|
const n = setCookieHeaders.length;
|
|
|
|
timeline.push({name: 'TEXT', value: `Saved ${n} cookie${n === 1 ? '' : 's'}`});
|
|
|
|
} else {
|
|
|
|
const n = setCookieHeaders.length;
|
|
|
|
timeline.push({name: 'TEXT', value: `Ignored ${n} cookie${n === 1 ? '' : 's'}`});
|
2017-03-16 17:51:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Handle the body
|
|
|
|
const encoding = 'base64';
|
|
|
|
const bodyBuffer = Buffer.concat(dataBuffers, dataBuffersLength);
|
|
|
|
const body = bodyBuffer.toString(encoding);
|
|
|
|
|
|
|
|
// Return the response data
|
2017-03-29 23:09:28 +00:00
|
|
|
respond({
|
2017-03-16 17:51:56 +00:00
|
|
|
headers,
|
|
|
|
encoding,
|
|
|
|
body,
|
|
|
|
contentType,
|
|
|
|
statusCode,
|
2017-04-06 02:30:28 +00:00
|
|
|
statusMessage,
|
|
|
|
elapsedTime: curl.getInfo(Curl.info.TOTAL_TIME) * 1000,
|
|
|
|
bytesRead: curl.getInfo(Curl.info.SIZE_DOWNLOAD),
|
|
|
|
url: curl.getInfo(Curl.info.EFFECTIVE_URL)
|
2017-03-16 17:51:56 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
// Close the request
|
|
|
|
this.close();
|
|
|
|
});
|
|
|
|
|
|
|
|
curl.on('error', function (err, code) {
|
|
|
|
let error = err + '';
|
|
|
|
let statusMessage = 'Error';
|
|
|
|
|
|
|
|
if (code === Curl.code.CURLE_ABORTED_BY_CALLBACK) {
|
|
|
|
error = 'Request aborted';
|
|
|
|
statusMessage = 'Abort';
|
|
|
|
}
|
|
|
|
|
2017-03-29 23:09:28 +00:00
|
|
|
respond({statusMessage, error});
|
2017-03-16 17:51:56 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
curl.perform();
|
|
|
|
} catch (err) {
|
|
|
|
handleError(err);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function send (requestId, environmentId) {
|
|
|
|
// First, lets wait for all debounces to finish
|
|
|
|
await util.delay(DEBOUNCE_MILLIS * 2);
|
|
|
|
|
|
|
|
const request = await models.request.getById(requestId);
|
|
|
|
const settings = await models.settings.getOrCreate();
|
|
|
|
|
2017-03-28 22:45:23 +00:00
|
|
|
// This may throw
|
|
|
|
const renderedRequest = await getRenderedRequest(request, environmentId);
|
2017-03-16 17:51:56 +00:00
|
|
|
|
|
|
|
// Get the workspace for the request
|
2017-04-21 04:30:52 +00:00
|
|
|
const ancestors = await db.withAncestors(request, [
|
|
|
|
models.requestGroup.type,
|
|
|
|
models.workspace.type
|
|
|
|
]);
|
2017-03-16 17:51:56 +00:00
|
|
|
const workspace = ancestors.find(doc => doc.type === models.workspace.type);
|
|
|
|
|
|
|
|
// Render succeeded so we're good to go!
|
2017-04-07 18:10:15 +00:00
|
|
|
return _actuallySend(renderedRequest, workspace, settings);
|
2017-03-16 17:51:56 +00:00
|
|
|
}
|