insomnia/packages/insomnia-app/app/plugins/install.js
Gregory Schier 0a616fba6b
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-17 17:50:03 -07:00

193 lines
5.6 KiB
JavaScript

// @flow
import * as electron from 'electron';
import fs from 'fs';
import fsx from 'fs-extra';
import childProcess from 'child_process';
import { getTempDir, isDevelopment, isWindows, PLUGIN_PATH } from '../common/constants';
import mkdirp from 'mkdirp';
import path from 'path';
export default async function(lookupName: string): Promise<void> {
return new Promise(async (resolve, reject) => {
let info: Object = {};
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): Promise<Object> {
return new Promise((resolve, reject) => {
console.log(`[plugins] Fetching module info from npm`);
childProcess.execFile(
escape(process.execPath),
[
'--no-deprecation', // Because Yarn still uses `new Buffer()`
_getYarnPath(),
'info',
lookupName,
'--json',
],
{
timeout: 5 * 60 * 1000,
maxBuffer: 1024 * 1024,
shell: true,
env: {
NODE_ENV: 'production',
ELECTRON_RUN_AS_NODE: 'true',
},
},
(err, stdout, stderr) => {
if (stderr) {
reject(new Error(`Yarn error ${stderr.toString()}`));
return;
}
let yarnOutput;
try {
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 attemted 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}`));
}
return;
}
const data = yarnOutput.data;
if (!data.hasOwnProperty('insomnia')) {
reject(new Error(`"${lookupName}" not a plugin! Package missing "insomnia" attribute`));
return;
}
console.log(`[plugins] Detected Insomnia plugin ${data.name}`);
resolve({
insomnia: data.insomnia,
name: data.name,
version: data.version,
dist: {
shasum: data.dist.shasum,
tarball: data.dist.tarball,
},
});
},
);
});
}
async function _installPluginToTmpDir(lookupName: string): Promise<{ tmpDir: string }> {
return new Promise((resolve, reject) => {
const tmpDir = path.join(getTempDir(), `${lookupName}-${Date.now()}`);
mkdirp.sync(tmpDir);
console.log(`[plugins] Installing plugin to ${tmpDir}`);
childProcess.execFile(
escape(process.execPath),
[
'--no-deprecation', // Because Yarn still uses `new Buffer()`
_getYarnPath(),
'add',
lookupName,
'--modules-folder',
tmpDir,
'--cwd',
tmpDir,
'--no-lockfile',
'--production',
'--no-progress',
],
{
timeout: 5 * 60 * 1000,
maxBuffer: 1024 * 1024,
cwd: tmpDir,
shell: true, // Some package installs require a shell
env: {
NODE_ENV: 'production',
ELECTRON_RUN_AS_NODE: 'true',
},
},
(err, stdout, stderr) => {
// 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.
if (err && !stdout.toString().includes('success')) {
reject(new Error(`${lookupName} install error: ${err.message}`));
return;
}
if (stderr) {
reject(new Error(`Yarn error ${stderr.toString()}`));
return;
}
resolve({ tmpDir });
},
);
});
}
function _getYarnPath() {
const { app } = electron.remote || electron;
// 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');
}
}
function escape(p) {
if (isWindows()) {
// Quote for Windows paths
return `"${p}"`;
} else {
// Escape with backslashes for Unix paths
return p.replace(/(\s+)/g, '\\$1');
}
}