mirror of
https://github.com/Kong/insomnia
synced 2024-11-08 23:00:30 +00:00
0a616fba6b
* 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
193 lines
5.6 KiB
JavaScript
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');
|
|
}
|
|
}
|