mirror of
https://github.com/Foundry376/Mailspring.git
synced 2024-12-26 10:00:50 +08:00
202 lines
7.1 KiB
JavaScript
202 lines
7.1 KiB
JavaScript
#!/usr/bin/env node
|
|
/* eslint global-require: 0 */
|
|
/* eslint quote-props: 0 */
|
|
const path = require('path');
|
|
const https = require('https');
|
|
const fs = require('fs');
|
|
const rimraf = require('rimraf');
|
|
const targz = require('targz');
|
|
const { safeExec } = require('./utils/child-process-wrapper.js');
|
|
const { execSync } = require('child_process');
|
|
|
|
const appDependencies = require('../app/package.json').dependencies;
|
|
const rootDependencies = require('../package.json').dependencies;
|
|
const npmElectronTarget = rootDependencies.electron;
|
|
const npmEnvs = {
|
|
system: process.env,
|
|
electron: Object.assign({}, process.env, {
|
|
npm_config_target: npmElectronTarget,
|
|
npm_config_arch: process.arch,
|
|
npm_config_target_arch: process.arch,
|
|
npm_config_disturl: 'https://electronjs.org/headers',
|
|
npm_config_runtime: 'electron',
|
|
npm_config_build_from_source: true,
|
|
}),
|
|
};
|
|
|
|
function npm(cmd, options) {
|
|
const { cwd, env } = Object.assign({ cwd: '.', env: 'system' }, options);
|
|
|
|
return new Promise((resolve, reject) => {
|
|
console.log(
|
|
`\n-- Running npm ${cmd} in ${cwd} with ${env} config (arch=${npmEnvs[env].npm_config_target_arch}) --`
|
|
);
|
|
|
|
safeExec(
|
|
`npm ${cmd}`,
|
|
{
|
|
cwd: path.resolve(__dirname, '..', cwd),
|
|
env: npmEnvs[env],
|
|
},
|
|
(err, stdout) => {
|
|
return err ? reject(err) : resolve(stdout);
|
|
}
|
|
);
|
|
});
|
|
}
|
|
|
|
function getMailsyncURL(callback) {
|
|
const distKey = `${process.platform}-${process.arch}`;
|
|
const distDir = {
|
|
'darwin-x64': 'osx',
|
|
'darwin-arm64': 'osx',
|
|
'win32-x64': 'win-ia32', // At this time, Mailsync is still 32-bit
|
|
'win32-ia32': 'win-ia32',
|
|
'linux-x64': 'linux',
|
|
'linux-ia32': null,
|
|
}[distKey];
|
|
|
|
if (!distDir) {
|
|
console.error(
|
|
`\nSorry, a Mailspring Mailsync build for your machine (${distKey}) is not yet available.`
|
|
);
|
|
return;
|
|
}
|
|
|
|
const out = execSync('git submodule status ./mailsync');
|
|
const [_, hash] = /[\+-]([A-Za-z0-9]{8})/.exec(out.toString());
|
|
callback(
|
|
`https://mailspring-builds.s3.amazonaws.com/mailsync/${hash}/${distDir}/mailsync.tar.gz`
|
|
);
|
|
}
|
|
|
|
function downloadMailsync() {
|
|
getMailsyncURL(distS3URL => {
|
|
https.get(distS3URL, response => {
|
|
if (response.statusCode === 200) {
|
|
response.pipe(fs.createWriteStream(`app/mailsync.tar.gz`));
|
|
response.on('end', () => {
|
|
console.log(
|
|
`\nDownloaded Mailsync prebuilt binary from ${distS3URL} to ./app/mailsync.tar.gz.`
|
|
);
|
|
targz.decompress(
|
|
{
|
|
src: `app/mailsync.tar.gz`,
|
|
dest: 'app/',
|
|
},
|
|
err => {
|
|
if (!err) {
|
|
console.log(`\nUnpackaged Mailsync into ./app.`);
|
|
} else {
|
|
console.error(`\nEncountered an error unpacking: ${err}`);
|
|
}
|
|
}
|
|
);
|
|
});
|
|
} else {
|
|
console.error(
|
|
`Sorry, an error occurred while fetching the Mailspring Mailsync build for your machine\n(${distS3URL})\n`
|
|
);
|
|
if (process.env.CI) {
|
|
throw new Error('Mailsync build not available.');
|
|
}
|
|
response.pipe(process.stderr);
|
|
response.on('end', () => console.error('\n'));
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
// For speed, we cache app/node_modules. However, we need to
|
|
// be sure to do a full rebuild of native node modules when the
|
|
// Electron version changes. To do this we check a marker file.
|
|
const appPath = path.resolve(__dirname, '..', 'app');
|
|
const appModulesPath = path.resolve(appPath, 'node_modules');
|
|
const cacheVersionPath = path.join(appModulesPath, '.postinstall-target-version');
|
|
const cacheElectronTarget =
|
|
fs.existsSync(cacheVersionPath) && fs.readFileSync(cacheVersionPath).toString();
|
|
|
|
if (cacheElectronTarget !== npmElectronTarget) {
|
|
console.log(
|
|
`\n-- Clearing app/node_modules (${cacheElectronTarget} !== ${npmElectronTarget}) --`
|
|
);
|
|
rimraf.sync(appModulesPath);
|
|
}
|
|
|
|
// Audit is emitted with npm ls, no need to run it on EVERY command which is an odd default
|
|
|
|
async function sqliteMissingUsleep() {
|
|
return new Promise(resolve => {
|
|
const sqliteLibDir = path.join(appModulesPath, 'better-sqlite3', 'build', 'Release');
|
|
safeExec(
|
|
`nm '${sqliteLibDir}/sqlite3.a' | grep usleep`,
|
|
{ ignoreStderr: true },
|
|
(err, resp) => {
|
|
resolve(resp === '');
|
|
}
|
|
);
|
|
});
|
|
}
|
|
|
|
async function run() {
|
|
// run `npm install` in ./app with Electron NPM config
|
|
await npm(`install --no-audit`, { cwd: './app', env: 'electron' });
|
|
|
|
// run `npm dedupe` in ./app with Electron NPM config
|
|
await npm(`dedupe --no-audit`, { cwd: './app', env: 'electron' });
|
|
|
|
// run `npm ls` in ./app - detects missing peer dependencies, etc.
|
|
await npm(`ls`, { cwd: './app', env: 'electron' });
|
|
|
|
// rebuild sqlite3 using our custom amalgamation, which has USLEEP enabled
|
|
if (process.platform === 'win32' || (await sqliteMissingUsleep())) {
|
|
// remove the existing build so NPM can't see that it's already present
|
|
rimraf.sync(path.join(appModulesPath, 'better-sqlite3'));
|
|
// install the module pointing to our local sqlite source with custom #DEFINEs set
|
|
const amalgamationPath = path.join(appPath, 'build', 'sqlite-amalgamation');
|
|
const resp = await npm(
|
|
`install better-sqlite3@${appDependencies['better-sqlite3']} ` +
|
|
`--no-save --no-audit --build-from-source --sqlite3="${amalgamationPath}"`,
|
|
{ cwd: './app', env: 'electron' }
|
|
);
|
|
console.log(`better-sqlite stdout: ${resp}`);
|
|
|
|
// remove the build symlinks so that we can build an installer for the app without
|
|
// symlinks out to the sqlite-amalgamation directory.
|
|
rimraf.sync(path.join(appModulesPath, 'better-sqlite3', 'build', 'Release', 'obj'));
|
|
|
|
if (!fs.existsSync(path.join(appModulesPath, 'better-sqlite3', 'build', 'Release'))) {
|
|
console.error(`better-sqlite did not recompile successfully!`);
|
|
process.exit(1001);
|
|
} else {
|
|
console.error(`better-sqlite recompiled successfully!`);
|
|
}
|
|
}
|
|
|
|
// if SQlite was STILL not built with HAVE_USLEEP, do not ship this build! We need usleep
|
|
// support so that multiple processes can connect to the sqlite file at the same time.
|
|
// Without it, transactions only retry every 1 sec instead of every 10ms, leading to
|
|
// awful db lock contention.
|
|
if (['linux', 'darwin'].includes(process.platform) && (await sqliteMissingUsleep())) {
|
|
console.error(`better-sqlite compiled without -HAVE_USLEEP, do not ship this build!`);
|
|
process.exit(1001);
|
|
}
|
|
|
|
// write the marker with the electron version
|
|
fs.writeFileSync(cacheVersionPath, npmElectronTarget);
|
|
|
|
// if the user hasn't cloned the mailsync module, download
|
|
// the binary for their operating system that was shipped to S3.
|
|
if (!fs.existsSync('./mailsync/build.sh')) {
|
|
console.log(`\n-- Downloading the last released version of Mailspring mailsync --`);
|
|
downloadMailsync();
|
|
} else {
|
|
console.log(
|
|
`\n-- You have the Mailspring mailsync submodule. If you'd prefer ` +
|
|
`to develop with a pre-built binary, remove the submodule and re-run ` +
|
|
`'npm run postinstall' to download the latest binary for your machine. --`
|
|
);
|
|
}
|
|
}
|
|
|
|
run();
|