2017-07-22 00:55:34 +00:00
|
|
|
// @flow
|
2017-08-01 16:58:08 +00:00
|
|
|
import * as electron from 'electron';
|
2018-03-29 17:23:49 +00:00
|
|
|
import fs from 'fs';
|
|
|
|
import fsx from 'fs-extra';
|
2017-07-22 00:55:34 +00:00
|
|
|
import childProcess from 'child_process';
|
2018-10-17 20:25:15 +00:00
|
|
|
import { getTempDir, isDevelopment, isWindows, PLUGIN_PATH } from '../common/constants';
|
2017-08-01 16:58:08 +00:00
|
|
|
import mkdirp from 'mkdirp';
|
|
|
|
import path from 'path';
|
2017-07-22 00:55:34 +00:00
|
|
|
|
2020-11-30 23:38:06 +00:00
|
|
|
const YARN_DEPRECATED_WARN = /(?<keyword>warning)(?<dependencies>[^>:].+[>:])(?<issue>.+)/;
|
|
|
|
|
2018-08-04 18:52:35 +00:00
|
|
|
export default async function(lookupName: string): Promise<void> {
|
2017-07-22 00:55:34 +00:00
|
|
|
return new Promise(async (resolve, reject) => {
|
2017-08-01 16:58:08 +00:00
|
|
|
let info: Object = {};
|
2017-07-22 00:55:34 +00:00
|
|
|
try {
|
2018-08-04 18:52:35 +00:00
|
|
|
info = await _isInsomniaPlugin(lookupName);
|
|
|
|
|
|
|
|
// Get actual module name without version suffixes and things
|
|
|
|
const moduleName = info.name;
|
2017-07-22 00:55:34 +00:00
|
|
|
|
2018-03-29 17:23:49 +00:00
|
|
|
const pluginDir = path.join(PLUGIN_PATH, moduleName);
|
2017-08-01 16:58:08 +00:00
|
|
|
|
2018-03-29 17:23:49 +00:00
|
|
|
// Make plugin directory
|
|
|
|
mkdirp.sync(pluginDir);
|
2017-08-01 16:58:08 +00:00
|
|
|
|
2018-03-29 17:23:49 +00:00
|
|
|
// Download the module
|
|
|
|
const request = electron.remote.net.request(info.dist.tarball);
|
|
|
|
request.on('error', err => {
|
2018-08-04 18:52:35 +00:00
|
|
|
reject(new Error(`Failed to make plugin request ${info.dist.tarball}: ${err.message}`));
|
2018-03-28 20:32:51 +00:00
|
|
|
});
|
|
|
|
|
2018-08-04 18:52:35 +00:00
|
|
|
const { tmpDir } = await _installPluginToTmpDir(lookupName);
|
2018-03-29 17:23:49 +00:00
|
|
|
console.log(`[plugins] Moving plugin from ${tmpDir} to ${pluginDir}`);
|
|
|
|
|
|
|
|
// Move entire module to plugins folder
|
2018-06-25 17:42:50 +00:00
|
|
|
fsx.moveSync(path.join(tmpDir, moduleName), pluginDir, {
|
2018-12-12 17:36:11 +00:00
|
|
|
overwrite: true,
|
2018-06-25 17:42:50 +00:00
|
|
|
});
|
2018-03-29 17:23:49 +00:00
|
|
|
|
|
|
|
// Move each dependency into node_modules folder
|
|
|
|
const pluginModulesDir = path.join(pluginDir, 'node_modules');
|
|
|
|
mkdirp.sync(pluginModulesDir);
|
|
|
|
for (const name of fs.readdirSync(tmpDir)) {
|
|
|
|
const src = path.join(tmpDir, name);
|
|
|
|
if (name === moduleName || !fs.statSync(src).isDirectory()) {
|
|
|
|
continue;
|
2017-07-22 00:55:34 +00:00
|
|
|
}
|
2017-08-01 16:58:08 +00:00
|
|
|
|
2018-03-29 17:23:49 +00:00
|
|
|
const dest = path.join(pluginModulesDir, name);
|
2018-06-25 17:42:50 +00:00
|
|
|
fsx.moveSync(src, dest, { overwrite: true });
|
2018-03-29 17:23:49 +00:00
|
|
|
}
|
|
|
|
} catch (err) {
|
|
|
|
reject(err);
|
|
|
|
return;
|
|
|
|
}
|
2017-08-01 16:58:08 +00:00
|
|
|
|
2018-03-29 17:23:49 +00:00
|
|
|
resolve();
|
2017-07-22 00:55:34 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2018-08-04 18:52:35 +00:00
|
|
|
async function _isInsomniaPlugin(lookupName: string): Promise<Object> {
|
2017-07-22 00:55:34 +00:00
|
|
|
return new Promise((resolve, reject) => {
|
2020-04-09 17:32:19 +00:00
|
|
|
console.log('[plugins] Fetching module info from npm');
|
2017-11-04 20:53:40 +00:00
|
|
|
childProcess.execFile(
|
2018-07-11 23:07:20 +00:00
|
|
|
escape(process.execPath),
|
|
|
|
[
|
|
|
|
'--no-deprecation', // Because Yarn still uses `new Buffer()`
|
2019-10-07 18:37:42 +00:00
|
|
|
escape(_getYarnPath()),
|
2018-07-11 23:07:20 +00:00
|
|
|
'info',
|
2018-08-04 18:52:35 +00:00
|
|
|
lookupName,
|
2018-12-12 17:36:11 +00:00
|
|
|
'--json',
|
2018-07-11 23:07:20 +00:00
|
|
|
],
|
2017-11-04 20:53:40 +00:00
|
|
|
{
|
|
|
|
timeout: 5 * 60 * 1000,
|
|
|
|
maxBuffer: 1024 * 1024,
|
2018-07-11 23:07:20 +00:00
|
|
|
shell: true,
|
2017-11-04 20:53:40 +00:00
|
|
|
env: {
|
2018-06-25 17:42:50 +00:00
|
|
|
NODE_ENV: 'production',
|
2018-12-12 17:36:11 +00:00
|
|
|
ELECTRON_RUN_AS_NODE: 'true',
|
|
|
|
},
|
2018-06-25 17:42:50 +00:00
|
|
|
},
|
|
|
|
(err, stdout, stderr) => {
|
2017-11-04 20:53:40 +00:00
|
|
|
if (stderr) {
|
2019-04-18 00:50:03 +00:00
|
|
|
reject(new Error(`Yarn error ${stderr.toString()}`));
|
2017-07-22 00:55:34 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-11-04 20:53:40 +00:00
|
|
|
let yarnOutput;
|
|
|
|
try {
|
2019-04-18 00:50:03 +00:00
|
|
|
yarnOutput = JSON.parse(stdout.toString());
|
2019-04-13 21:39:30 +00:00
|
|
|
} catch (ex) {
|
|
|
|
// Output is not JSON. Check if yarn/electron terminated with non-zero exit code.
|
|
|
|
// In certain environments electron can exit with error even if output is OK.
|
2020-07-28 05:18:26 +00:00
|
|
|
// Parsing is attempted before checking exit code as workaround for false errors.
|
2019-04-13 21:39:30 +00:00
|
|
|
if (err) {
|
|
|
|
reject(new Error(`${lookupName} npm error: ${err.message}`));
|
|
|
|
} else {
|
|
|
|
reject(new Error(`Yarn response not JSON: ${ex.message}`));
|
|
|
|
}
|
2017-11-04 20:53:40 +00:00
|
|
|
return;
|
|
|
|
}
|
2017-08-01 16:58:08 +00:00
|
|
|
|
2017-11-04 20:53:40 +00:00
|
|
|
const data = yarnOutput.data;
|
|
|
|
if (!data.hasOwnProperty('insomnia')) {
|
2018-08-04 18:52:35 +00:00
|
|
|
reject(new Error(`"${lookupName}" not a plugin! Package missing "insomnia" attribute`));
|
2017-08-01 16:58:08 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-11-22 00:07:28 +00:00
|
|
|
console.log(`[plugins] Detected Insomnia plugin ${data.name}`);
|
2017-11-04 20:53:40 +00:00
|
|
|
|
2017-08-01 16:58:08 +00:00
|
|
|
resolve({
|
2017-11-04 20:53:40 +00:00
|
|
|
insomnia: data.insomnia,
|
|
|
|
name: data.name,
|
|
|
|
version: data.version,
|
2017-08-01 16:58:08 +00:00
|
|
|
dist: {
|
2017-11-04 20:53:40 +00:00
|
|
|
shasum: data.dist.shasum,
|
2018-12-12 17:36:11 +00:00
|
|
|
tarball: data.dist.tarball,
|
|
|
|
},
|
2017-08-01 16:58:08 +00:00
|
|
|
});
|
2018-12-12 17:36:11 +00:00
|
|
|
},
|
2017-07-22 00:55:34 +00:00
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
2017-11-26 20:45:40 +00:00
|
|
|
|
2018-08-04 18:52:35 +00:00
|
|
|
async function _installPluginToTmpDir(lookupName: string): Promise<{ tmpDir: string }> {
|
2018-03-29 17:23:49 +00:00
|
|
|
return new Promise((resolve, reject) => {
|
2018-08-04 18:52:35 +00:00
|
|
|
const tmpDir = path.join(getTempDir(), `${lookupName}-${Date.now()}`);
|
2018-03-29 17:23:49 +00:00
|
|
|
mkdirp.sync(tmpDir);
|
|
|
|
console.log(`[plugins] Installing plugin to ${tmpDir}`);
|
|
|
|
childProcess.execFile(
|
2018-07-11 23:07:20 +00:00
|
|
|
escape(process.execPath),
|
2018-03-29 17:23:49 +00:00
|
|
|
[
|
2018-07-11 23:07:20 +00:00
|
|
|
'--no-deprecation', // Because Yarn still uses `new Buffer()`
|
2019-10-07 18:37:42 +00:00
|
|
|
escape(_getYarnPath()),
|
2018-06-25 17:42:50 +00:00
|
|
|
'add',
|
2018-08-04 18:52:35 +00:00
|
|
|
lookupName,
|
2018-06-25 17:42:50 +00:00
|
|
|
'--modules-folder',
|
2019-09-30 20:13:03 +00:00
|
|
|
escape(tmpDir),
|
2018-06-25 17:42:50 +00:00
|
|
|
'--cwd',
|
2019-09-30 20:13:03 +00:00
|
|
|
escape(tmpDir),
|
2018-03-29 17:23:49 +00:00
|
|
|
'--no-lockfile',
|
|
|
|
'--production',
|
2018-12-12 17:36:11 +00:00
|
|
|
'--no-progress',
|
2018-03-29 17:23:49 +00:00
|
|
|
],
|
|
|
|
{
|
|
|
|
timeout: 5 * 60 * 1000,
|
|
|
|
maxBuffer: 1024 * 1024,
|
|
|
|
cwd: tmpDir,
|
2018-07-11 23:07:20 +00:00
|
|
|
shell: true, // Some package installs require a shell
|
2018-03-29 17:23:49 +00:00
|
|
|
env: {
|
2018-06-25 17:42:50 +00:00
|
|
|
NODE_ENV: 'production',
|
2018-12-12 17:36:11 +00:00
|
|
|
ELECTRON_RUN_AS_NODE: 'true',
|
|
|
|
},
|
2018-06-25 17:42:50 +00:00
|
|
|
},
|
|
|
|
(err, stdout, stderr) => {
|
2019-04-13 21:39:30 +00:00
|
|
|
// Check yarn/electron process exit code.
|
|
|
|
// In certain environments electron can exit with error even if the command was perfomed sucesfully.
|
|
|
|
// Checking for sucess message in output is a workaround for false errors.
|
2019-04-18 00:50:03 +00:00
|
|
|
if (err && !stdout.toString().includes('success')) {
|
2018-08-04 18:52:35 +00:00
|
|
|
reject(new Error(`${lookupName} install error: ${err.message}`));
|
2018-03-29 17:23:49 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-11-30 23:38:06 +00:00
|
|
|
if (stderr && !containsOnlyDeprecationWarnings(stderr)) {
|
2019-04-18 00:50:03 +00:00
|
|
|
reject(new Error(`Yarn error ${stderr.toString()}`));
|
2018-03-29 17:23:49 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-06-25 17:42:50 +00:00
|
|
|
resolve({ tmpDir });
|
2018-12-12 17:36:11 +00:00
|
|
|
},
|
2018-03-29 17:23:49 +00:00
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-11-30 23:38:06 +00:00
|
|
|
export function containsOnlyDeprecationWarnings(stderr) {
|
|
|
|
// Split on line breaks and remove falsy values (null, undefined, 0, -0, NaN, "", false)
|
|
|
|
const arr = stderr.split(/\r?\n/).filter(e => e);
|
|
|
|
// Retrieve all matching deprecated dependency warning
|
|
|
|
const warnings = arr.filter(e => isDeprecatedDependencies(e));
|
|
|
|
// Print each deprecation warnings to the console, so we don't hide them.
|
|
|
|
warnings.forEach(e => console.warn('[plugins] deprecation warning during installation: ', e));
|
|
|
|
// If they mismatch, it means there are warnings and errors
|
|
|
|
return warnings.length === arr.length;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Provided a string, it checks for the following message:<br>
|
|
|
|
* <<[warning] xxx > yyy > zzz: yyy<n is [no longer maintained] and [not recommended for usage] <br>
|
|
|
|
* due to the number of issues. Please, [upgrade your dependencies] to xxx>> <br>
|
|
|
|
* @param str The error message
|
|
|
|
* @returns {boolean} Returns true if it's a deprecated warning
|
|
|
|
*/
|
|
|
|
export function isDeprecatedDependencies(str: string): boolean {
|
|
|
|
// The issue contains the message as it is without the dependency list
|
|
|
|
const message = YARN_DEPRECATED_WARN.exec(str)?.groups.issue;
|
|
|
|
// Strict check, everything must be matched to be a false positive
|
|
|
|
// !! is not a mistake, makes it returns boolean instead of undefined on error
|
|
|
|
return !!(
|
|
|
|
message &&
|
|
|
|
message.includes('no longer maintained') &&
|
|
|
|
message.includes('not recommended for usage') &&
|
|
|
|
message.includes('upgrade your dependencies')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-06-25 17:42:50 +00:00
|
|
|
function _getYarnPath() {
|
|
|
|
const { app } = electron.remote || electron;
|
2017-11-29 23:14:44 +00:00
|
|
|
|
|
|
|
// TODO: This is brittle. Make finding this more robust.
|
|
|
|
if (isDevelopment()) {
|
|
|
|
return path.resolve(app.getAppPath(), './bin/yarn-standalone.js');
|
|
|
|
} else {
|
|
|
|
return path.resolve(app.getAppPath(), '../bin/yarn-standalone.js');
|
|
|
|
}
|
2017-11-26 20:45:40 +00:00
|
|
|
}
|
2018-07-11 23:07:20 +00:00
|
|
|
|
|
|
|
function escape(p) {
|
2018-10-17 20:25:15 +00:00
|
|
|
if (isWindows()) {
|
|
|
|
// Quote for Windows paths
|
|
|
|
return `"${p}"`;
|
|
|
|
} else {
|
2020-07-28 05:18:26 +00:00
|
|
|
// Escape whitespace and parenthesis with backslashes for Unix paths
|
|
|
|
return p.replace(/([\s()])/g, '\\$1');
|
2018-10-17 20:25:15 +00:00
|
|
|
}
|
2018-07-11 23:07:20 +00:00
|
|
|
}
|