electron-forge
Version:
A complete tool for building modern Electron applications
416 lines (309 loc) • 29.5 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", {
value: true
});
var _keys = require('babel-runtime/core-js/object/keys');
var _keys2 = _interopRequireDefault(_keys);
var _promise = require('babel-runtime/core-js/promise');
var _promise2 = _interopRequireDefault(_promise);
var _regenerator = require('babel-runtime/regenerator');
var _regenerator2 = _interopRequireDefault(_regenerator);
var _getIterator2 = require('babel-runtime/core-js/get-iterator');
var _getIterator3 = _interopRequireDefault(_getIterator2);
var _assign = require('babel-runtime/core-js/object/assign');
var _assign2 = _interopRequireDefault(_assign);
var _bluebird = require('bluebird');
require('colors');
var _debug = require('debug');
var _debug2 = _interopRequireDefault(_debug);
var _nodeFetch = require('node-fetch');
var _nodeFetch2 = _interopRequireDefault(_nodeFetch);
var _fsExtra = require('fs-extra');
var _fsExtra2 = _interopRequireDefault(_fsExtra);
var _inquirer = require('inquirer');
var _inquirer2 = _interopRequireDefault(_inquirer);
var _nugget = require('nugget');
var _nugget2 = _interopRequireDefault(_nugget);
var _os = require('os');
var _os2 = _interopRequireDefault(_os);
var _path = require('path');
var _path2 = _interopRequireDefault(_path);
var _pify = require('pify');
var _pify2 = _interopRequireDefault(_pify);
var _semver = require('semver');
var _semver2 = _interopRequireDefault(_semver);
var _oraHandler = require('../util/ora-handler');
var _oraHandler2 = _interopRequireDefault(_oraHandler);
var _messages = require('../util/messages');
var _dmg = require('../installers/darwin/dmg');
var _dmg2 = _interopRequireDefault(_dmg);
var _zip = require('../installers/darwin/zip');
var _zip2 = _interopRequireDefault(_zip);
var _deb = require('../installers/linux/deb');
var _deb2 = _interopRequireDefault(_deb);
var _rpm = require('../installers/linux/rpm');
var _rpm2 = _interopRequireDefault(_rpm);
var _exe = require('../installers/win32/exe');
var _exe2 = _interopRequireDefault(_exe);
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
var d = (0, _debug2.default)('electron-forge:install');
var GITHUB_API = 'https://api.github.com';
/**
* @typedef {Object} InstallOptions
* @property {boolean} [interactive=false] Whether to use sensible defaults or prompt the user visually
* @property {boolean} [prerelease=false] Whether to install prerelease versions
* @property {string} repo The GitHub repository to install from, in the format owner/name
* @property {function} chooseAsset A function that must return the asset to use/install from a provided array of compatible GitHub assets
*/
/**
* Install an Electron application from GitHub. If you leave interactive as `false`, you MUST provide a `chooseAsset` function.
*
* @param {InstallOptions} providedOptions - Options for the install method
* @return {Promise} Will resolve when the install process is complete
*/
exports.default = function () {
var _ref = (0, _bluebird.coroutine)(_regenerator2.default.mark(function _callee3() {
var providedOptions = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
var _Object$assign, interactive, prerelease, repo, chooseAsset, latestRelease, possibleAssets, targetAsset, choices, _ref3, assetID, tmpdir, pathSafeRepo, filename, fullFilePath, nuggetOpts;
return _regenerator2.default.wrap(function _callee3$(_context3) {
while (1) {
switch (_context3.prev = _context3.next) {
case 0:
// eslint-disable-next-line prefer-const, no-unused-vars
_Object$assign = (0, _assign2.default)({
interactive: false,
prerelease: false
}, providedOptions), interactive = _Object$assign.interactive, prerelease = _Object$assign.prerelease, repo = _Object$assign.repo, chooseAsset = _Object$assign.chooseAsset;
_oraHandler2.default.interactive = interactive;
latestRelease = void 0;
possibleAssets = [];
_context3.next = 6;
return (0, _oraHandler2.default)('Searching for Application', function () {
var _ref2 = (0, _bluebird.coroutine)(_regenerator2.default.mark(function _callee(searchSpinner) {
var releases, sortedReleases, assets, installTargets;
return _regenerator2.default.wrap(function _callee$(_context) {
while (1) {
switch (_context.prev = _context.next) {
case 0:
if (!(!repo || repo.indexOf('/') === -1)) {
_context.next = 2;
break;
}
throw 'Invalid repository name, must be in the format owner/name';
case 2:
d('searching for repo:', repo);
releases = void 0;
_context.prev = 4;
_context.next = 7;
return (0, _nodeFetch2.default)(GITHUB_API + '/repos/' + repo + '/releases');
case 7:
_context.next = 9;
return _context.sent.json();
case 9:
releases = _context.sent;
_context.next = 14;
break;
case 12:
_context.prev = 12;
_context.t0 = _context['catch'](4);
case 14:
if (!(!releases || releases.message === 'Not Found' || !Array.isArray(releases))) {
_context.next = 16;
break;
}
throw 'Failed to find releases for repository "' + repo + '". Please check the name and try again.';
case 16:
releases = releases.filter(function (release) {
return !release.prerelease || prerelease;
});
sortedReleases = releases.sort(function (releaseA, releaseB) {
var tagA = releaseA.tag_name;
if (tagA.substr(0, 1) === 'v') tagA = tagA.substr(1);
var tagB = releaseB.tag_name;
if (tagB.substr(0, 1) === 'v') tagB = tagB.substr(1);
return _semver2.default.gt(tagB, tagA) ? 1 : -1;
});
latestRelease = sortedReleases[0];
searchSpinner.text = 'Searching for Releases'; // eslint-disable-line
assets = latestRelease.assets;
if (!(!assets || !Array.isArray(assets))) {
_context.next = 23;
break;
}
throw 'Could not find any assets for the latest release';
case 23:
installTargets = {
win32: [/\.exe$/],
darwin: [/OSX.*\.zip$/, /darwin.*\.zip$/, /macOS.*\.zip$/, /mac.*\.zip$/, /\.dmg$/],
linux: [/\.rpm$/, /\.deb$/]
};
possibleAssets = assets.filter(function (asset) {
var targetSuffixes = installTargets[process.platform];
var _iteratorNormalCompletion = true;
var _didIteratorError = false;
var _iteratorError = undefined;
try {
for (var _iterator = (0, _getIterator3.default)(targetSuffixes), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) {
var suffix = _step.value;
if (suffix.test(asset.name)) return true;
}
} catch (err) {
_didIteratorError = true;
_iteratorError = err;
} finally {
try {
if (!_iteratorNormalCompletion && _iterator.return) {
_iterator.return();
}
} finally {
if (_didIteratorError) {
throw _iteratorError;
}
}
}
return false;
});
if (!(possibleAssets.length === 0)) {
_context.next = 27;
break;
}
throw 'Failed to find any installable assets for target platform: ' + ('' + process.platform).cyan;
case 27:
case 'end':
return _context.stop();
}
}
}, _callee, undefined, [[4, 12]]);
}));
return function (_x2) {
return _ref2.apply(this, arguments);
};
}());
case 6:
(0, _messages.info)(interactive, 'Found latest release' + (prerelease ? ' (including prereleases)' : '') + ': ' + latestRelease.tag_name.cyan);
targetAsset = possibleAssets[0];
if (!(possibleAssets.length > 1)) {
_context3.next = 26;
break;
}
if (!chooseAsset) {
_context3.next = 15;
break;
}
_context3.next = 12;
return _promise2.default.resolve(chooseAsset(possibleAssets));
case 12:
targetAsset = _context3.sent;
_context3.next = 26;
break;
case 15:
if (!interactive) {
_context3.next = 25;
break;
}
choices = [];
possibleAssets.forEach(function (asset) {
choices.push({ name: asset.name, value: asset.id });
});
_context3.next = 20;
return _inquirer2.default.createPromptModule()({
type: 'list',
name: 'assetID',
message: 'Multiple potential assets found, please choose one from the list below:'.cyan,
choices: choices
});
case 20:
_ref3 = _context3.sent;
assetID = _ref3.assetID;
targetAsset = possibleAssets.find(function (asset) {
return asset.id === assetID;
});
_context3.next = 26;
break;
case 25:
throw 'expected a chooseAsset function to be provided but it was not';
case 26:
tmpdir = _path2.default.resolve(_os2.default.tmpdir(), 'forge-install');
pathSafeRepo = repo.replace(/[/\\]/g, '-');
filename = pathSafeRepo + '-' + latestRelease.tag_name + '-' + targetAsset.name;
fullFilePath = _path2.default.resolve(tmpdir, filename);
_context3.next = 32;
return _fsExtra2.default.pathExists(fullFilePath);
case 32:
_context3.t0 = !_context3.sent;
if (_context3.t0) {
_context3.next = 39;
break;
}
_context3.next = 36;
return _fsExtra2.default.stat(fullFilePath);
case 36:
_context3.t1 = _context3.sent.size;
_context3.t2 = targetAsset.size;
_context3.t0 = _context3.t1 !== _context3.t2;
case 39:
if (!_context3.t0) {
_context3.next = 45;
break;
}
_context3.next = 42;
return _fsExtra2.default.mkdirs(tmpdir);
case 42:
nuggetOpts = {
target: filename,
dir: tmpdir,
resume: true,
strictSSL: true
};
_context3.next = 45;
return (0, _pify2.default)(_nugget2.default)(targetAsset.browser_download_url, nuggetOpts);
case 45:
_context3.next = 47;
return (0, _oraHandler2.default)('Installing Application', function () {
var _ref4 = (0, _bluebird.coroutine)(_regenerator2.default.mark(function _callee2(installSpinner) {
var installActions, suffixFnIdent;
return _regenerator2.default.wrap(function _callee2$(_context2) {
while (1) {
switch (_context2.prev = _context2.next) {
case 0:
installActions = {
win32: {
'.exe': _exe2.default
},
darwin: {
'.zip': _zip2.default,
'.dmg': _dmg2.default
},
linux: {
'.deb': _deb2.default,
'.rpm': _rpm2.default
}
};
suffixFnIdent = (0, _keys2.default)(installActions[process.platform]).find(function (suffix) {
return targetAsset.name.endsWith(suffix);
});
_context2.next = 4;
return installActions[process.platform][suffixFnIdent](fullFilePath, installSpinner);
case 4:
case 'end':
return _context2.stop();
}
}
}, _callee2, undefined);
}));
return function (_x3) {
return _ref4.apply(this, arguments);
};
}());
case 47:
case 'end':
return _context3.stop();
}
}
}, _callee3, undefined);
}));
return function () {
return _ref.apply(this, arguments);
};
}();
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["api/install.js"],"names":["d","GITHUB_API","providedOptions","interactive","prerelease","repo","chooseAsset","latestRelease","possibleAssets","searchSpinner","indexOf","releases","json","message","Array","isArray","filter","release","sortedReleases","sort","releaseA","releaseB","tagA","tag_name","substr","tagB","gt","text","assets","installTargets","win32","darwin","linux","asset","targetSuffixes","process","platform","suffix","test","name","length","cyan","targetAsset","resolve","choices","forEach","push","value","id","createPromptModule","type","assetID","find","tmpdir","pathSafeRepo","replace","filename","fullFilePath","pathExists","stat","size","mkdirs","nuggetOpts","target","dir","resume","strictSSL","browser_download_url","installSpinner","installActions","suffixFnIdent","endsWith"],"mappings":"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","file":"api/install.js","sourcesContent":["import 'colors';\nimport debug from 'debug';\nimport fetch from 'node-fetch';\nimport fs from 'fs-extra';\nimport inquirer from 'inquirer';\nimport nugget from 'nugget';\nimport os from 'os';\nimport path from 'path';\nimport pify from 'pify';\nimport semver from 'semver';\n\nimport asyncOra from '../util/ora-handler';\nimport { info } from '../util/messages';\n\nimport darwinDMGInstaller from '../installers/darwin/dmg';\nimport darwinZipInstaller from '../installers/darwin/zip';\nimport linuxDebInstaller from '../installers/linux/deb';\nimport linuxRPMInstaller from '../installers/linux/rpm';\nimport win32ExeInstaller from '../installers/win32/exe';\n\nconst d = debug('electron-forge:install');\n\nconst GITHUB_API = 'https://api.github.com';\n\n/**\n * @typedef {Object} InstallOptions\n * @property {boolean} [interactive=false] Whether to use sensible defaults or prompt the user visually\n * @property {boolean} [prerelease=false] Whether to install prerelease versions\n * @property {string} repo The GitHub repository to install from, in the format owner/name\n * @property {function} chooseAsset A function that must return the asset to use/install from a provided array of compatible GitHub assets\n */\n\n/**\n * Install an Electron application from GitHub. If you leave interactive as `false`, you MUST provide a `chooseAsset` function.\n *\n * @param {InstallOptions} providedOptions - Options for the install method\n * @return {Promise} Will resolve when the install process is complete\n */\nexport default async (providedOptions = {}) => {\n  // eslint-disable-next-line prefer-const, no-unused-vars\n  let { interactive, prerelease, repo, chooseAsset } = Object.assign({\n    interactive: false,\n    prerelease: false,\n  }, providedOptions);\n  asyncOra.interactive = interactive;\n\n  let latestRelease;\n  let possibleAssets = [];\n\n  await asyncOra('Searching for Application', async (searchSpinner) => {\n    if (!repo || repo.indexOf('/') === -1) {\n      throw 'Invalid repository name, must be in the format owner/name';\n    }\n\n    d('searching for repo:', repo);\n    let releases;\n    try {\n      releases = await (await fetch(`${GITHUB_API}/repos/${repo}/releases`)).json();\n    } catch (err) {\n      // Ignore error\n    }\n\n    if (!releases || releases.message === 'Not Found' || !Array.isArray(releases)) {\n      throw `Failed to find releases for repository \"${repo}\".  Please check the name and try again.`;\n    }\n\n    releases = releases.filter(release => !release.prerelease || prerelease);\n\n    const sortedReleases = releases.sort((releaseA, releaseB) => {\n      let tagA = releaseA.tag_name;\n      if (tagA.substr(0, 1) === 'v') tagA = tagA.substr(1);\n      let tagB = releaseB.tag_name;\n      if (tagB.substr(0, 1) === 'v') tagB = tagB.substr(1);\n      return (semver.gt(tagB, tagA) ? 1 : -1);\n    });\n    latestRelease = sortedReleases[0];\n\n    searchSpinner.text = 'Searching for Releases'; // eslint-disable-line\n\n    const assets = latestRelease.assets;\n    if (!assets || !Array.isArray(assets)) {\n      throw 'Could not find any assets for the latest release';\n    }\n\n    const installTargets = {\n      win32: [/\\.exe$/],\n      darwin: [/OSX.*\\.zip$/, /darwin.*\\.zip$/, /macOS.*\\.zip$/, /mac.*\\.zip$/, /\\.dmg$/],\n      linux: [/\\.rpm$/, /\\.deb$/],\n    };\n\n    possibleAssets = assets.filter((asset) => {\n      const targetSuffixes = installTargets[process.platform];\n      for (const suffix of targetSuffixes) {\n        if (suffix.test(asset.name)) return true;\n      }\n      return false;\n    });\n\n    if (possibleAssets.length === 0) {\n      throw `Failed to find any installable assets for target platform: ${`${process.platform}`.cyan}`;\n    }\n  });\n\n  info(interactive, `Found latest release${prerelease ? ' (including prereleases)' : ''}: ${latestRelease.tag_name.cyan}`);\n\n  let targetAsset = possibleAssets[0];\n  if (possibleAssets.length > 1) {\n    if (chooseAsset) {\n      targetAsset = await Promise.resolve(chooseAsset(possibleAssets));\n    } else if (interactive) {\n      const choices = [];\n      possibleAssets.forEach((asset) => {\n        choices.push({ name: asset.name, value: asset.id });\n      });\n      const { assetID } = await inquirer.createPromptModule()({\n        type: 'list',\n        name: 'assetID',\n        message: 'Multiple potential assets found, please choose one from the list below:'.cyan,\n        choices,\n      });\n\n      targetAsset = possibleAssets.find(asset => asset.id === assetID);\n    } else {\n      throw 'expected a chooseAsset function to be provided but it was not';\n    }\n  }\n\n  const tmpdir = path.resolve(os.tmpdir(), 'forge-install');\n  const pathSafeRepo = repo.replace(/[/\\\\]/g, '-');\n  const filename = `${pathSafeRepo}-${latestRelease.tag_name}-${targetAsset.name}`;\n\n  const fullFilePath = path.resolve(tmpdir, filename);\n  if (!await fs.pathExists(fullFilePath) || (await fs.stat(fullFilePath)).size !== targetAsset.size) {\n    await fs.mkdirs(tmpdir);\n\n    const nuggetOpts = {\n      target: filename,\n      dir: tmpdir,\n      resume: true,\n      strictSSL: true,\n    };\n    await pify(nugget)(targetAsset.browser_download_url, nuggetOpts);\n  }\n\n  await asyncOra('Installing Application', async (installSpinner) => {\n    const installActions = {\n      win32: {\n        '.exe': win32ExeInstaller,\n      },\n      darwin: {\n        '.zip': darwinZipInstaller,\n        '.dmg': darwinDMGInstaller,\n      },\n      linux: {\n        '.deb': linuxDebInstaller,\n        '.rpm': linuxRPMInstaller,\n      },\n    };\n\n    const suffixFnIdent = Object.keys(installActions[process.platform]).find(suffix => targetAsset.name.endsWith(suffix));\n    await installActions[process.platform][suffixFnIdent](fullFilePath, installSpinner);\n  });\n};\n"]}