insomnia/packages/insomnia-app/scripts/build.js
Gregory Schier 32a958ab95
View open source licenses in app (#1926)
* View opensource licenses

* Remove redundant check
2020-02-06 08:41:56 -05:00

248 lines
7.3 KiB
JavaScript

const packageJson = require('../package.json');
const childProcess = require('child_process');
const webpack = require('webpack');
const licenseChecker = require('license-checker');
const rimraf = require('rimraf');
const ncp = require('ncp').ncp;
const path = require('path');
const mkdirp = require('mkdirp');
const fs = require('fs');
const configRenderer = require('../webpack/webpack.config.production.babel');
const configMain = require('../webpack/webpack.config.electron.babel');
// Start build if ran from CLI
if (require.main === module) {
process.nextTick(async () => {
await module.exports.start();
});
}
module.exports.start = async function() {
console.log('[build] Starting build');
console.log('[build] npm: ' + childProcess.spawnSync('npm', ['--version']).stdout);
console.log('[build] node: ' + childProcess.spawnSync('node', ['--version']).stdout);
if (process.version.indexOf('v10.') !== 0) {
console.log('[build] Node v10.x.x is required to build');
process.exit(1);
}
// Remove folders first
console.log('[build] Removing existing directories');
await emptyDir('../build');
// Build the things
console.log('[build] Building license list');
await buildLicenseList('../', '../build/opensource-licenses.txt');
console.log('[build] Building Webpack renderer');
await buildWebpack(configRenderer);
console.log('[build] Building Webpack main');
await buildWebpack(configMain);
// Copy necessary files
console.log('[build] Copying files');
await copyFiles('../bin', '../build/');
await copyFiles('../app/static', '../build/static');
await copyFiles('../app/icons/', '../build/');
// Generate package.json
await generatePackageJson('../package.json', '../build/package.json');
// Install Node modules
console.log('[build] Installing dependencies');
await install('../build/');
console.log('[build] Complete!');
};
async function buildWebpack(config) {
return new Promise((resolve, reject) => {
const compiler = webpack(config);
compiler.run((err, stats) => {
if (err) {
reject(err);
} else if (stats.hasErrors()) {
reject(new Error('Failed to build webpack'));
console.log(stats.toJson().errors);
} else {
resolve();
}
});
});
}
async function emptyDir(relPath) {
return new Promise((resolve, reject) => {
const dir = path.resolve(__dirname, relPath);
rimraf(dir, err => {
if (err) {
reject(err);
} else {
mkdirp.sync(dir);
resolve();
}
});
});
}
async function copyFiles(relSource, relDest) {
return new Promise((resolve, reject) => {
const source = path.resolve(__dirname, relSource);
const dest = path.resolve(__dirname, relDest);
console.log(`[build] copy ${source} to ${dest}`);
ncp(source, dest, err => {
if (err) {
reject(err);
} else {
resolve();
}
});
});
}
async function buildLicenseList(relSource, relDest) {
return new Promise((resolve, reject) => {
const source = path.resolve(__dirname, relSource);
const dest = path.resolve(__dirname, relDest);
mkdirp.sync(path.dirname(dest));
licenseChecker.init({ start: source, production: true }, (err, packages) => {
if (err) {
return reject(err);
}
const out = [];
for (const pkgName of Object.keys(packages)) {
const { licenses, repository, publisher, email, licenseFile: lf } = packages[pkgName];
const licenseFile = (lf || '').includes('README') ? null : lf;
const txt = licenseFile ? fs.readFileSync(licenseFile) : '[no license file]';
const body = [
'-------------------------------------------------------------------------',
'',
`PACKAGE: ${pkgName}`,
licenses ? `LICENSES: ${licenses}` : null,
repository ? `REPOSITORY: ${repository}` : null,
publisher ? `PUBLISHER: ${publisher}` : null,
email ? `EMAIL: ${email}` : null,
'\n' + txt,
]
.filter(v => v !== null)
.join('\n');
out.push(`${body}\n\n`);
}
const header = [
'This application bundles the following third-party packages in ',
'accordance with the following licenses:',
'-------------------------------------------------------------------------',
'',
'',
].join('\n');
fs.writeFileSync(dest, header + out.join('\n\n'));
resolve();
});
});
}
async function install(relDir) {
return new Promise(resolve => {
const prefix = path.resolve(__dirname, relDir);
// // Link all plugins
// const plugins = path.resolve(__dirname, `../../../plugins`);
// for (const dir of fs.readdirSync(plugins)) {
// if (dir.indexOf('.') === 0) {
// continue;
// }
//
// console.log(`[build] Linking plugin ${dir}`);
// const p = path.join(plugins, dir);
// childProcess.spawnSync('npm', ['link', p], {
// cwd: prefix,
// shell: true,
// });
// }
// // Link all packages
// const packages = path.resolve(__dirname, `../../../packages`);
// for (const dir of fs.readdirSync(packages)) {
// // Don't like ourselves
// if (dir === packageJson.name) {
// continue;
// }
//
// if (dir.indexOf('.') === 0) {
// continue;
// }
//
// console.log(`[build] Linking local package ${dir}`);
// const p = path.join(packages, dir);
// childProcess.spawnSync('npm', ['link', p], { cwd: prefix, shell: true });
// }
const p = childProcess.spawn('npm', ['install', '--production', '--no-optional'], {
cwd: prefix,
shell: true,
});
p.stdout.on('data', data => {
console.log(data.toString());
});
p.stderr.on('data', data => {
console.log(data.toString());
});
p.on('exit', code => {
console.log('child process exited with code ' + code.toString());
resolve();
});
});
}
function generatePackageJson(relBasePkg, relOutPkg) {
// Read package.json's
const basePath = path.resolve(__dirname, relBasePkg);
const outPath = path.resolve(__dirname, relOutPkg);
const basePkg = JSON.parse(fs.readFileSync(basePath));
const appPkg = {
name: packageJson.app.name,
version: basePkg.app.version,
productName: basePkg.app.productName,
longName: basePkg.app.longName,
description: basePkg.description,
license: basePkg.license,
homepage: basePkg.homepage,
author: basePkg.author,
main: 'main.min.js',
dependencies: {},
};
for (const key of Object.keys(appPkg)) {
if (key === undefined) {
throw new Error(`[build] missing "app.${key}" from package.json`);
}
}
// Figure out which dependencies to pack
const allDependencies = Object.keys(basePkg.dependencies);
const packedDependencies = basePkg.packedDependencies;
const unpackedDependencies = allDependencies.filter(name => !packedDependencies.includes(name));
// Add dependencies
for (const name of unpackedDependencies) {
const version = basePkg.dependencies[name];
if (!version) {
throw new Error(`Failed to find packed dep "${name}" in dependencies`);
}
appPkg.dependencies[name] = version;
console.log(`[build] Adding native Node dep ${name}`);
}
fs.writeFileSync(outPath, JSON.stringify(appPkg, null, 2));
}