cli.ts
5.86 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
#!/usr/bin/env node
import 'colors';
import * as fs from 'fs-extra';
import * as path from 'path';
import ora from 'ora';
import * as argParser from 'yargs';
import { rebuild, ModuleType } from './rebuild';
import { getProjectRootPath } from './search-module';
import { locateElectronModule } from './electron-locator';
const yargs = argParser
.usage('Usage: electron-rebuild --version [version] --module-dir [path]')
.help('h')
.alias('h', 'help')
.version(false)
.describe('v', 'The version of Electron to build against')
.alias('v', 'version')
.describe('f', 'Force rebuilding modules, even if we would skip it otherwise')
.alias('f', 'force')
.describe('a', "Override the target architecture to something other than your system's")
.alias('a', 'arch')
.describe('m', 'The path to the node_modules directory to rebuild')
.alias('m', 'module-dir')
.describe('w', 'A specific module to build, or comma separated list of modules')
.alias('w', 'which-module')
.describe('o', 'Only build specified module, or comma separated list of modules. All others are ignored.')
.alias('o', 'only')
.describe('e', 'The path to prebuilt electron module')
.alias('e', 'electron-prebuilt-dir')
.describe('d', 'Custom header tarball URL')
.alias('d', 'dist-url')
.describe('t', 'The types of dependencies to rebuild. Comma seperated list of "prod", "dev" and "optional". Default is "prod,optional"')
.alias('t', 'types')
.describe('p', 'Rebuild in parallel, this is enabled by default on macOS and Linux')
.alias('p', 'parallel')
.describe('s', 'Rebuild modules sequentially, this is enabled by default on Windows')
.alias('s', 'sequential')
.describe('b', 'Build debug version of modules')
.alias('b','debug')
.describe('prebuild-tag-prefix', 'GitHub tag prefix passed to prebuild-install. Default is "v"')
.describe('force-abi', 'Override the ABI version for the version of Electron you are targeting. Only use when targeting Nightly releases.')
.epilog('Copyright 2016');
const argv = yargs.argv;
if (argv.h) {
yargs.showHelp();
process.exit(0);
}
if (process.argv.length === 3 && process.argv[2] === '--version') {
try {
console.log('Electron Rebuild Version:', require(path.resolve(__dirname, '../../package.json')).version);
} catch (err) {
console.log('Electron Rebuild Version:', require(path.resolve(__dirname, '../package.json')).version);
}
process.exit(0);
}
const handler = (err: Error): void => {
console.error('\nAn unhandled error occurred inside electron-rebuild'.red);
console.error(`${err.message}\n\n${err.stack}`.red);
process.exit(-1);
};
process.on('uncaughtException', handler);
process.on('unhandledRejection', handler);
(async (): Promise<void> => {
const projectRootPath = await getProjectRootPath(process.cwd());
const electronModulePath = argv.e ? path.resolve(process.cwd(), (argv.e as string)) : await locateElectronModule(projectRootPath);
let electronModuleVersion = argv.v as string;
if (!electronModuleVersion) {
try {
if (!electronModulePath) throw new Error('Prebuilt electron module not found');
// eslint-disable-next-line @typescript-eslint/no-var-requires
const pkgJson = require(path.join(electronModulePath, 'package.json'));
electronModuleVersion = pkgJson.version;
} catch (e) {
throw new Error(`Unable to find electron's version number, either install it or specify an explicit version`);
}
}
let rootDirectory = argv.m as string;
if (!rootDirectory) {
// NB: We assume here that we're going to rebuild the immediate parent's
// node modules, which might not always be the case but it's at least a
// good guess
rootDirectory = path.resolve(__dirname, '../../..');
if (!await fs.pathExists(rootDirectory) || !await fs.pathExists(path.resolve(rootDirectory, 'package.json'))) {
// Then we try the CWD
rootDirectory = process.cwd();
if (!await fs.pathExists(rootDirectory) || !await fs.pathExists(path.resolve(rootDirectory, 'package.json'))) {
throw new Error('Unable to find parent node_modules directory, specify it via --module-dir, E.g. "--module-dir ." for the current directory');
}
}
} else {
rootDirectory = path.resolve(process.cwd(), rootDirectory);
}
if (argv.forceAbi && typeof argv.forceAbi !== 'number') {
throw new Error('force-abi must be a number');
}
let modulesDone = 0;
let moduleTotal = 0;
const rebuildSpinner = ora('Searching dependency tree').start();
let lastModuleName: string;
const redraw = (moduleName?: string): void => {
if (moduleName) lastModuleName = moduleName;
if (argv.p) {
rebuildSpinner.text = `Building modules: ${modulesDone}/${moduleTotal}`;
} else {
rebuildSpinner.text = `Building module: ${lastModuleName}, Completed: ${modulesDone}`;
}
};
const rebuilder = rebuild({
buildPath: rootDirectory,
electronVersion: electronModuleVersion,
arch: (argv.a as string) || process.arch,
extraModules: argv.w ? (argv.w as string).split(',') : [],
onlyModules: argv.o ? (argv.o as string).split(',') : null,
force: argv.f as boolean,
headerURL: argv.d as string,
types: argv.t ? (argv.t as string).split(',') as ModuleType[] : ['prod', 'optional'],
mode: argv.p ? 'parallel' : (argv.s ? 'sequential' : undefined),
debug: argv.b as boolean,
prebuildTagPrefix: (argv.prebuildTagPrefix as string) || 'v',
forceABI: argv.forceAbi as number,
projectRootPath,
});
const lifecycle = rebuilder.lifecycle;
lifecycle.on('module-found', (moduleName: string) => {
moduleTotal += 1;
redraw(moduleName);
});
lifecycle.on('module-done', () => {
modulesDone += 1;
redraw();
});
try {
await rebuilder;
} catch (err) {
rebuildSpinner.text = 'Rebuild Failed';
rebuildSpinner.fail();
throw err;
}
rebuildSpinner.text = 'Rebuild Complete';
rebuildSpinner.succeed();
})();