insomnia/packages/insomnia-app/app/plugins/install.ts

247 lines
7.6 KiB
TypeScript
Raw Normal View History

2021-07-22 23:04:56 +00:00
import childProcess from 'child_process';
import * as electron from 'electron';
import fs from 'fs';
import fsx from 'fs-extra';
import mkdirp from 'mkdirp';
import path from 'path';
2021-07-22 23:04:56 +00:00
import { isDevelopment, isWindows, PLUGIN_PATH } from '../common/constants';
import { getTempDir } from '../common/electron-helpers';
const YARN_DEPRECATED_WARN = /(?<keyword>warning)(?<dependencies>[^>:].+[>:])(?<issue>.+)/;
interface InsomniaPlugin {
insomnia: any;
name: string;
version: string;
dist: {
shasum: string;
tarball: string;
};
}
export default async function(lookupName: string) {
return new Promise<void>(async (resolve, reject) => {
let info: InsomniaPlugin | null = null;
try {
info = await _isInsomniaPlugin(lookupName);
// Get actual module name without version suffixes and things
const moduleName = info.name;
const pluginDir = path.join(PLUGIN_PATH, moduleName);
// Make plugin directory
mkdirp.sync(pluginDir);
// Download the module
const request = electron.remote.net.request(info.dist.tarball);
request.on('error', err => {
reject(new Error(`Failed to make plugin request ${info?.dist.tarball}: ${err.message}`));
});
const { tmpDir } = await _installPluginToTmpDir(lookupName);
console.log(`[plugins] Moving plugin from ${tmpDir} to ${pluginDir}`);
// Move entire module to plugins folder
fsx.moveSync(
path.join(tmpDir, moduleName),
pluginDir,
{ overwrite: true },
);
// 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;
}
const dest = path.join(pluginModulesDir, name);
fsx.moveSync(src, dest, {
overwrite: true,
});
}
} catch (err) {
reject(err);
return;
}
resolve();
});
}
async function _isInsomniaPlugin(lookupName: string) {
return new Promise<InsomniaPlugin>((resolve, reject) => {
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',
lookupName,
'--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',
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) {
Version Control (beta) (#1439) * VCS proof of concept underway! * Stuff * Some things * Replace deprecated Electron makeSingleInstance * Rename `window` variables so not to be confused with window object * Don't unnecessarily update request when URL does not change * Regenerate package-lock * Fix tests + ESLint * Publish - insomnia-app@1.0.49 - insomnia-cookies@0.0.12 - insomnia-httpsnippet@1.16.18 - insomnia-importers@2.0.13 - insomnia-libcurl@0.0.23 - insomnia-prettify@0.1.7 - insomnia-url@0.1.6 - insomnia-xpath@1.0.9 - insomnia-plugin-base64@1.0.6 - insomnia-plugin-cookie-jar@1.0.8 - insomnia-plugin-core-themes@1.0.5 - insomnia-plugin-default-headers@1.1.9 - insomnia-plugin-file@1.0.7 - insomnia-plugin-hash@1.0.7 - insomnia-plugin-jsonpath@1.0.12 - insomnia-plugin-now@1.0.11 - insomnia-plugin-os@1.0.13 - insomnia-plugin-prompt@1.1.9 - insomnia-plugin-request@1.0.18 - insomnia-plugin-response@1.0.16 - insomnia-plugin-uuid@1.0.10 * Broken but w/e * Some tweaks * Big refactor. Create local snapshots and push done * POC merging and a lot of improvements * Lots of work done on initial UI/UX * Fix old tests * Atomic writes and size-based batches * Update StageEntry definition once again to be better * Factor out GraphQL query logic * Merge algorithm, history modal, other minor things * Fix test * Merge, checkout, revert w/ user changes now work * Force UI to refresh when switching branches changes active request * Rough draft pull() and some cleanup * E2EE stuff and some refactoring * Add ability to share project with team and fixed tests * VCS now created in root component and better remote project handling * Remove unused definition * Publish - insomnia-account@0.0.2 - insomnia-app@1.1.1 - insomnia-cookies@0.0.14 - insomnia-httpsnippet@1.16.20 - insomnia-importers@2.0.15 - insomnia-libcurl@0.0.25 - insomnia-prettify@0.1.9 - insomnia-sync@0.0.2 - insomnia-url@0.1.8 - insomnia-xpath@1.0.11 - insomnia-plugin-base64@1.0.8 - insomnia-plugin-cookie-jar@1.0.10 - insomnia-plugin-core-themes@1.0.7 - insomnia-plugin-file@1.0.9 - insomnia-plugin-hash@1.0.9 - insomnia-plugin-jsonpath@1.0.14 - insomnia-plugin-now@1.0.13 - insomnia-plugin-os@1.0.15 - insomnia-plugin-prompt@1.1.11 - insomnia-plugin-request@1.0.20 - insomnia-plugin-response@1.0.18 - insomnia-plugin-uuid@1.0.12 * Move some deps around * Fix Flow errors * Update package.json * Fix eslint errors * Fix tests * Update deps * bootstrap insomnia-sync * TRy fixing appveyor * Try something else * Bump lerna * try powershell * Try again * Fix imports * Fixed errors * sync types refactor * Show remote projects in workspace dropdown * Improved pulling of non-local workspaces * Loading indicators and some tweaks * Clean up sync staging modal * Some sync improvements: - No longer store stage - Upgrade Electron - Sync UI/UX improvements * Fix snyc tests * Upgraded deps and hot loader tweaks (it's broken for some reason) * Fix tests * Branches dialog, network refactoring, some tweaks * Fixed merging when other branch is empty * A bunch of small fixes from real testing * Fixed pull merge logic * Fix tests * Some bug fixes * A few small tweaks * Conflict resolution and other improvements * Fix tests * Add revert changes * Deal with duplicate projects per workspace * Some tweaks and accessibility improvements * Tooltip accessibility * Fix API endpoint * Fix tests * Remove jest dep from insomnia-importers
2019-04-18 00:50:03 +00:00
reject(new Error(`Yarn error ${stderr.toString()}`));
return;
}
2017-11-04 20:53:40 +00:00
let yarnOutput;
2017-11-04 20:53:40 +00:00
try {
Version Control (beta) (#1439) * VCS proof of concept underway! * Stuff * Some things * Replace deprecated Electron makeSingleInstance * Rename `window` variables so not to be confused with window object * Don't unnecessarily update request when URL does not change * Regenerate package-lock * Fix tests + ESLint * Publish - insomnia-app@1.0.49 - insomnia-cookies@0.0.12 - insomnia-httpsnippet@1.16.18 - insomnia-importers@2.0.13 - insomnia-libcurl@0.0.23 - insomnia-prettify@0.1.7 - insomnia-url@0.1.6 - insomnia-xpath@1.0.9 - insomnia-plugin-base64@1.0.6 - insomnia-plugin-cookie-jar@1.0.8 - insomnia-plugin-core-themes@1.0.5 - insomnia-plugin-default-headers@1.1.9 - insomnia-plugin-file@1.0.7 - insomnia-plugin-hash@1.0.7 - insomnia-plugin-jsonpath@1.0.12 - insomnia-plugin-now@1.0.11 - insomnia-plugin-os@1.0.13 - insomnia-plugin-prompt@1.1.9 - insomnia-plugin-request@1.0.18 - insomnia-plugin-response@1.0.16 - insomnia-plugin-uuid@1.0.10 * Broken but w/e * Some tweaks * Big refactor. Create local snapshots and push done * POC merging and a lot of improvements * Lots of work done on initial UI/UX * Fix old tests * Atomic writes and size-based batches * Update StageEntry definition once again to be better * Factor out GraphQL query logic * Merge algorithm, history modal, other minor things * Fix test * Merge, checkout, revert w/ user changes now work * Force UI to refresh when switching branches changes active request * Rough draft pull() and some cleanup * E2EE stuff and some refactoring * Add ability to share project with team and fixed tests * VCS now created in root component and better remote project handling * Remove unused definition * Publish - insomnia-account@0.0.2 - insomnia-app@1.1.1 - insomnia-cookies@0.0.14 - insomnia-httpsnippet@1.16.20 - insomnia-importers@2.0.15 - insomnia-libcurl@0.0.25 - insomnia-prettify@0.1.9 - insomnia-sync@0.0.2 - insomnia-url@0.1.8 - insomnia-xpath@1.0.11 - insomnia-plugin-base64@1.0.8 - insomnia-plugin-cookie-jar@1.0.10 - insomnia-plugin-core-themes@1.0.7 - insomnia-plugin-file@1.0.9 - insomnia-plugin-hash@1.0.9 - insomnia-plugin-jsonpath@1.0.14 - insomnia-plugin-now@1.0.13 - insomnia-plugin-os@1.0.15 - insomnia-plugin-prompt@1.1.11 - insomnia-plugin-request@1.0.20 - insomnia-plugin-response@1.0.18 - insomnia-plugin-uuid@1.0.12 * Move some deps around * Fix Flow errors * Update package.json * Fix eslint errors * Fix tests * Update deps * bootstrap insomnia-sync * TRy fixing appveyor * Try something else * Bump lerna * try powershell * Try again * Fix imports * Fixed errors * sync types refactor * Show remote projects in workspace dropdown * Improved pulling of non-local workspaces * Loading indicators and some tweaks * Clean up sync staging modal * Some sync improvements: - No longer store stage - Upgrade Electron - Sync UI/UX improvements * Fix snyc tests * Upgraded deps and hot loader tweaks (it's broken for some reason) * Fix tests * Branches dialog, network refactoring, some tweaks * Fixed merging when other branch is empty * A bunch of small fixes from real testing * Fixed pull merge logic * Fix tests * Some bug fixes * A few small tweaks * Conflict resolution and other improvements * Fix tests * Add revert changes * Deal with duplicate projects per workspace * Some tweaks and accessibility improvements * Tooltip accessibility * Fix API endpoint * Fix tests * Remove jest dep from insomnia-importers
2019-04-18 00:50:03 +00:00
yarnOutput = JSON.parse(stdout.toString());
} 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.
// Parsing is attempted before checking exit code as workaround for false errors.
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-11-04 20:53:40 +00:00
const data = yarnOutput.data;
2017-11-04 20:53:40 +00:00
if (!data.hasOwnProperty('insomnia')) {
reject(new Error(`"${lookupName}" not a plugin! Package missing "insomnia" attribute`));
return;
}
2017-11-22 00:07:28 +00:00
console.log(`[plugins] Detected Insomnia plugin ${data.name}`);
const insomniaPlugin: InsomniaPlugin = {
2017-11-04 20:53:40 +00:00
insomnia: data.insomnia,
name: data.name,
version: data.version,
dist: {
2017-11-04 20:53:40 +00:00
shasum: data.dist.shasum,
tarball: data.dist.tarball,
},
};
resolve(insomniaPlugin);
},
);
});
}
async function _installPluginToTmpDir(lookupName: string) {
return new Promise<{ tmpDir: string }>((resolve, reject) => {
const tmpDir = path.join(getTempDir(), `${lookupName}-${Date.now()}`);
mkdirp.sync(tmpDir);
console.log(`[plugins] Installing plugin to ${tmpDir}`);
childProcess.execFile(
2018-07-11 23:07:20 +00:00
escape(process.execPath),
[
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',
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),
'--no-lockfile',
'--production',
'--no-progress',
],
{
timeout: 5 * 60 * 1000,
maxBuffer: 1024 * 1024,
cwd: tmpDir,
shell: true,
// Some package installs require a shell
env: {
2018-06-25 17:42:50 +00:00
NODE_ENV: 'production',
ELECTRON_RUN_AS_NODE: 'true',
},
2018-06-25 17:42:50 +00:00
},
(err, stdout, stderr) => {
// Check yarn/electron process exit code.
2021-05-25 16:16:43 +00:00
// In certain environments electron can exit with error even if the command was performed successfully.
// Checking for success message in output is a workaround for false errors.
Version Control (beta) (#1439) * VCS proof of concept underway! * Stuff * Some things * Replace deprecated Electron makeSingleInstance * Rename `window` variables so not to be confused with window object * Don't unnecessarily update request when URL does not change * Regenerate package-lock * Fix tests + ESLint * Publish - insomnia-app@1.0.49 - insomnia-cookies@0.0.12 - insomnia-httpsnippet@1.16.18 - insomnia-importers@2.0.13 - insomnia-libcurl@0.0.23 - insomnia-prettify@0.1.7 - insomnia-url@0.1.6 - insomnia-xpath@1.0.9 - insomnia-plugin-base64@1.0.6 - insomnia-plugin-cookie-jar@1.0.8 - insomnia-plugin-core-themes@1.0.5 - insomnia-plugin-default-headers@1.1.9 - insomnia-plugin-file@1.0.7 - insomnia-plugin-hash@1.0.7 - insomnia-plugin-jsonpath@1.0.12 - insomnia-plugin-now@1.0.11 - insomnia-plugin-os@1.0.13 - insomnia-plugin-prompt@1.1.9 - insomnia-plugin-request@1.0.18 - insomnia-plugin-response@1.0.16 - insomnia-plugin-uuid@1.0.10 * Broken but w/e * Some tweaks * Big refactor. Create local snapshots and push done * POC merging and a lot of improvements * Lots of work done on initial UI/UX * Fix old tests * Atomic writes and size-based batches * Update StageEntry definition once again to be better * Factor out GraphQL query logic * Merge algorithm, history modal, other minor things * Fix test * Merge, checkout, revert w/ user changes now work * Force UI to refresh when switching branches changes active request * Rough draft pull() and some cleanup * E2EE stuff and some refactoring * Add ability to share project with team and fixed tests * VCS now created in root component and better remote project handling * Remove unused definition * Publish - insomnia-account@0.0.2 - insomnia-app@1.1.1 - insomnia-cookies@0.0.14 - insomnia-httpsnippet@1.16.20 - insomnia-importers@2.0.15 - insomnia-libcurl@0.0.25 - insomnia-prettify@0.1.9 - insomnia-sync@0.0.2 - insomnia-url@0.1.8 - insomnia-xpath@1.0.11 - insomnia-plugin-base64@1.0.8 - insomnia-plugin-cookie-jar@1.0.10 - insomnia-plugin-core-themes@1.0.7 - insomnia-plugin-file@1.0.9 - insomnia-plugin-hash@1.0.9 - insomnia-plugin-jsonpath@1.0.14 - insomnia-plugin-now@1.0.13 - insomnia-plugin-os@1.0.15 - insomnia-plugin-prompt@1.1.11 - insomnia-plugin-request@1.0.20 - insomnia-plugin-response@1.0.18 - insomnia-plugin-uuid@1.0.12 * Move some deps around * Fix Flow errors * Update package.json * Fix eslint errors * Fix tests * Update deps * bootstrap insomnia-sync * TRy fixing appveyor * Try something else * Bump lerna * try powershell * Try again * Fix imports * Fixed errors * sync types refactor * Show remote projects in workspace dropdown * Improved pulling of non-local workspaces * Loading indicators and some tweaks * Clean up sync staging modal * Some sync improvements: - No longer store stage - Upgrade Electron - Sync UI/UX improvements * Fix snyc tests * Upgraded deps and hot loader tweaks (it's broken for some reason) * Fix tests * Branches dialog, network refactoring, some tweaks * Fixed merging when other branch is empty * A bunch of small fixes from real testing * Fixed pull merge logic * Fix tests * Some bug fixes * A few small tweaks * Conflict resolution and other improvements * Fix tests * Add revert changes * Deal with duplicate projects per workspace * Some tweaks and accessibility improvements * Tooltip accessibility * Fix API endpoint * Fix tests * Remove jest dep from insomnia-importers
2019-04-18 00:50:03 +00:00
if (err && !stdout.toString().includes('success')) {
reject(new Error(`${lookupName} install error: ${err.message}`));
return;
}
if (stderr && !containsOnlyDeprecationWarnings(stderr)) {
Version Control (beta) (#1439) * VCS proof of concept underway! * Stuff * Some things * Replace deprecated Electron makeSingleInstance * Rename `window` variables so not to be confused with window object * Don't unnecessarily update request when URL does not change * Regenerate package-lock * Fix tests + ESLint * Publish - insomnia-app@1.0.49 - insomnia-cookies@0.0.12 - insomnia-httpsnippet@1.16.18 - insomnia-importers@2.0.13 - insomnia-libcurl@0.0.23 - insomnia-prettify@0.1.7 - insomnia-url@0.1.6 - insomnia-xpath@1.0.9 - insomnia-plugin-base64@1.0.6 - insomnia-plugin-cookie-jar@1.0.8 - insomnia-plugin-core-themes@1.0.5 - insomnia-plugin-default-headers@1.1.9 - insomnia-plugin-file@1.0.7 - insomnia-plugin-hash@1.0.7 - insomnia-plugin-jsonpath@1.0.12 - insomnia-plugin-now@1.0.11 - insomnia-plugin-os@1.0.13 - insomnia-plugin-prompt@1.1.9 - insomnia-plugin-request@1.0.18 - insomnia-plugin-response@1.0.16 - insomnia-plugin-uuid@1.0.10 * Broken but w/e * Some tweaks * Big refactor. Create local snapshots and push done * POC merging and a lot of improvements * Lots of work done on initial UI/UX * Fix old tests * Atomic writes and size-based batches * Update StageEntry definition once again to be better * Factor out GraphQL query logic * Merge algorithm, history modal, other minor things * Fix test * Merge, checkout, revert w/ user changes now work * Force UI to refresh when switching branches changes active request * Rough draft pull() and some cleanup * E2EE stuff and some refactoring * Add ability to share project with team and fixed tests * VCS now created in root component and better remote project handling * Remove unused definition * Publish - insomnia-account@0.0.2 - insomnia-app@1.1.1 - insomnia-cookies@0.0.14 - insomnia-httpsnippet@1.16.20 - insomnia-importers@2.0.15 - insomnia-libcurl@0.0.25 - insomnia-prettify@0.1.9 - insomnia-sync@0.0.2 - insomnia-url@0.1.8 - insomnia-xpath@1.0.11 - insomnia-plugin-base64@1.0.8 - insomnia-plugin-cookie-jar@1.0.10 - insomnia-plugin-core-themes@1.0.7 - insomnia-plugin-file@1.0.9 - insomnia-plugin-hash@1.0.9 - insomnia-plugin-jsonpath@1.0.14 - insomnia-plugin-now@1.0.13 - insomnia-plugin-os@1.0.15 - insomnia-plugin-prompt@1.1.11 - insomnia-plugin-request@1.0.20 - insomnia-plugin-response@1.0.18 - insomnia-plugin-uuid@1.0.12 * Move some deps around * Fix Flow errors * Update package.json * Fix eslint errors * Fix tests * Update deps * bootstrap insomnia-sync * TRy fixing appveyor * Try something else * Bump lerna * try powershell * Try again * Fix imports * Fixed errors * sync types refactor * Show remote projects in workspace dropdown * Improved pulling of non-local workspaces * Loading indicators and some tweaks * Clean up sync staging modal * Some sync improvements: - No longer store stage - Upgrade Electron - Sync UI/UX improvements * Fix snyc tests * Upgraded deps and hot loader tweaks (it's broken for some reason) * Fix tests * Branches dialog, network refactoring, some tweaks * Fixed merging when other branch is empty * A bunch of small fixes from real testing * Fixed pull merge logic * Fix tests * Some bug fixes * A few small tweaks * Conflict resolution and other improvements * Fix tests * Add revert changes * Deal with duplicate projects per workspace * Some tweaks and accessibility improvements * Tooltip accessibility * Fix API endpoint * Fix tests * Remove jest dep from insomnia-importers
2019-04-18 00:50:03 +00:00
reject(new Error(`Yarn error ${stderr.toString()}`));
return;
}
resolve({
tmpDir,
});
},
);
});
}
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) {
// 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');
}
}
2018-07-11 23:07:20 +00:00
function escape(p) {
if (isWindows()) {
// Quote for Windows paths
return `"${p}"`;
} else {
// Escape whitespace and parenthesis with backslashes for Unix paths
return p.replace(/([\s()])/g, '\\$1');
}
2018-07-11 23:07:20 +00:00
}