mirror of
https://github.com/Foundry376/Mailspring.git
synced 2025-11-08 07:21:18 +08:00
Summary:
This diff removes significant cruft from N1's compilation and build tooling:
- Electron-Packager replaces most of the code in build/tasks/* used to copy things,
bundle things, download electron, etc.
- script/bootstrap has been replaced with a much simpler script that does not use
APM, does not download Electron (we just use electron as an NPM dep) and does
not manully compile sqlite. It requires NPMv3, but I think that's safe.
- babel and eslint are now devDependencies of the main project. The main project
also supports optionalDependencies now.
- npm test and npm start replace ./N1.sh
- APM is still around, and is only put into N1 so it can install plugins at runtime.
It should be removed as soon as we notify package maintainers and have them provide zips.
- N1 no longer has it's own compile-cache or babel/typescript/coffeescript compilers.
It delegates to electron-compile and electron-compilers. Both of these packages had
to be forked and modified slightly, but I'm hopeful the modifications will make it back
in to the projects and you can still consult their documentation for more info.
+ In the near future, I think we should stop shipping electron-compilers with N1. This
would mean that all plugins would need to be compiled on pre-publish, just like NPM
packages, and would complicate the local development story a bit, but would make the
app smaller. electron-compile is not supposed to compile at runtime in the prod app,
just pull from the compile cache.
- I've re-organized Grunt according to Grunt best practices, where each tasks/* file
specifies it's own config and imports grunt tasks.
- Unfortunately, I was not able to use any open source projects for the deb and rpm builds,
because we have things like postinst hooks and start menu items which are not supported
by the electron installer-generators.
WIP
Turn off all LESS compilation, because themes. Doh.
Use Grunt for new build process too, just remove tasks
More changes
Add babel-eslint
Remove unused react-devtools
WIP
Add name
Ignore nonexistent
Switch to more modern approach to config for grunt
Move zipping to mac installer task
Restructure publish task so it aggregates first, can log useful info if publishing is disabled
Fix build dirs
Fix win installer
Fix linux installer
Fix linux installer
Try making linux
A few more
Updates
Upadtes
fixes
fixes
Get rid of non-meaningful variables
Resolve assets path
Insert nylas.sh
Clean up args more
Actually use description
Fix display name ugh
More tweaks
Expliclty write /usr/bin/nylas
Improve vars
Use old nylas.sh
Reinstate APM to better scope this diff
Test Plan: Test on Mac, Windows, Linux
Reviewers: evan, jackie, juan
Reviewed By: jackie, juan
Differential Revision: https://phab.nylas.com/D3411
189 lines
7 KiB
JavaScript
189 lines
7 KiB
JavaScript
/* eslint no-cond-assign: 0 */
|
|
const path = require('path');
|
|
const fs = require('fs-plus');
|
|
|
|
function normalizeRequirePath(requirePath, fPath) {
|
|
if (requirePath[0] === ".") {
|
|
return path.normalize(path.join(path.dirname(fPath), requirePath));
|
|
}
|
|
return requirePath;
|
|
}
|
|
|
|
|
|
module.exports = (grunt) => {
|
|
grunt.config.merge({
|
|
nylaslint: {
|
|
src: grunt.config('source:coffeescript').concat(grunt.config('source:es6')),
|
|
},
|
|
});
|
|
|
|
grunt.registerMultiTask('nylaslint', 'Check requires for file extensions compiled away', function nylaslint() {
|
|
const done = this.async();
|
|
|
|
// Enable once path errors are fixed.
|
|
if (process.platform === 'win32') {
|
|
done();
|
|
return;
|
|
}
|
|
|
|
const extensionRegex = /require ['"].*\.(coffee|cjsx|jsx|es6|es)['"]/i;
|
|
|
|
for (const fileset of this.files) {
|
|
grunt.log.writeln(`Nylinting ${fileset.src.length} files.`);
|
|
|
|
const esExtensions = {
|
|
".es6": true,
|
|
".es": true,
|
|
".jsx": true,
|
|
};
|
|
|
|
const errors = [];
|
|
|
|
const esExport = {};
|
|
const esNoExport = {};
|
|
const esExportDefault = {};
|
|
|
|
// Temp TODO. Fix spec files
|
|
for (const f of fileset.src) {
|
|
if (!esExtensions[path.extname(f)]) { continue; }
|
|
if (!/-spec/.test(f)) { continue; }
|
|
|
|
const content = fs.readFileSync(f, {encoding: 'utf8'});
|
|
|
|
// https://regex101.com/r/rQ3eD0/1
|
|
// Matches only the first describe block
|
|
const describeRe = /[\n]describe\(['"](.*?)['"], ?\(\) ?=> ?/m;
|
|
if (describeRe.test(content)) {
|
|
errors.push(`${f}: Spec has to start with function`);
|
|
}
|
|
}
|
|
|
|
// NOTE: Comment me in if you want to fix these files.
|
|
// _str = require('underscore.string')
|
|
// replacer = (match, describeName) ->
|
|
// fnName = _str.camelize(describeName, true)
|
|
// return "\ndescribe('#{describeName}', function #{fnName}() "
|
|
// newContent = content.replace(describeRe, replacer)
|
|
// fs.writeFileSync(f, newContent, encoding:'utf8')
|
|
|
|
// Build the list of ES6 files that export things and categorize
|
|
for (const f of fileset.src) {
|
|
if (!esExtensions[path.extname(f)]) { continue; }
|
|
const lookupPath = `${path.dirname(f)}/${path.basename(f, path.extname(f))}`;
|
|
const content = fs.readFileSync(f, {encoding: 'utf8'});
|
|
|
|
if (/module.exports\s?=\s?.+/gmi.test(content)) {
|
|
if (!f.endsWith('nylas-exports.es6')) {
|
|
errors.push(`${f}: Don't use module.exports in ES6`);
|
|
}
|
|
}
|
|
|
|
if (/^export/gmi.test(content)) {
|
|
if (/^export default/gmi.test(content)) {
|
|
esExportDefault[lookupPath] = true;
|
|
} else {
|
|
esExport[lookupPath] = true;
|
|
}
|
|
} else {
|
|
esNoExport[lookupPath] = true;
|
|
}
|
|
}
|
|
|
|
// Now look again through all ES6 files, this time to check imports
|
|
// instead of exports.
|
|
for (const f of fileset.src) {
|
|
let result = null;
|
|
if (!esExtensions[path.extname(f)]) {
|
|
continue;
|
|
}
|
|
const content = fs.readFileSync(f, {encoding: 'utf8'});
|
|
const importRe = /import \{.*\} from ['"](.*?)['"]/gmi;
|
|
|
|
while (result = importRe.exec(content)) {
|
|
for (const requirePath of result.slice(1)) {
|
|
const lookupPath = normalizeRequirePath(requirePath, f);
|
|
if (esExportDefault[lookupPath] || esNoExport[lookupPath]) {
|
|
errors.push(`${f}: Don't destructure default export ${requirePath}`);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Now look through all coffeescript files
|
|
// If they require things from ES6 files, ensure they're using the
|
|
// proper syntax.
|
|
for (const f of fileset.src) {
|
|
let result = null;
|
|
if (esExtensions[path.extname(f)]) {
|
|
continue;
|
|
}
|
|
const content = fs.readFileSync(f, {encoding: 'utf8'});
|
|
if (extensionRegex.test(content)) {
|
|
errors.push(`${f}: Remove extensions when requiring files`);
|
|
}
|
|
|
|
const requireRe = /require[ (]['"]([\w_./-]*?)['"]/gmi;
|
|
|
|
while (result = requireRe.exec(content)) {
|
|
for (const requirePath of result.slice(1)) {
|
|
const lookupPath = normalizeRequirePath(requirePath, f);
|
|
|
|
const baseRequirePath = path.basename(requirePath);
|
|
|
|
const plainRequireRe = new RegExp(`require[ (]['"].*${baseRequirePath}['"]\\)?$`, "gm");
|
|
const defaultRequireRe = new RegExp(`require\\(['"].*${baseRequirePath}['"]\\)\\.default`, "gm");
|
|
|
|
if (esExport[lookupPath]) {
|
|
if (!plainRequireRe.test(content)) {
|
|
errors.push(`${f}: No \`default\` exported ${requirePath}`);
|
|
}
|
|
} else if (esNoExport[lookupPath]) {
|
|
errors.push(`${f}: Nothing exported from ${requirePath}`);
|
|
} else if (esExportDefault[lookupPath]) {
|
|
if (!defaultRequireRe.test(content)) {
|
|
errors.push(`${f}: Add \`default\` to require ${requirePath}`);
|
|
}
|
|
} else {
|
|
// must be a coffeescript or core file
|
|
if (defaultRequireRe.test(content)) {
|
|
errors.push(`${f}: Don't ask for \`default\` from ${requirePath}`);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (errors.length > 0) {
|
|
for (const err of errors) { grunt.log.error(err); }
|
|
const error = `
|
|
Please fix the #{errors.length} linter errors above. These are the issues we're looking for:
|
|
|
|
ISSUES WITH COFFEESCRIPT FILES:
|
|
|
|
1. Remove extensions when requiring files:
|
|
Since we compile files in production to plain ".js" files it's very important you do NOT include the file extension when "require"ing a file.
|
|
|
|
2. Add "default" to require:
|
|
As of Babel 6, "require" no longer returns whatever the "default" value is. If you are "require"ing an es6 file from a coffeescript file, you must explicitly request the "default" property. For example: do "require('./my-es6-file').default"
|
|
|
|
3. Don't ask for "default":
|
|
If you're requiring a coffeescript file from a coffeescript file, you will almost never need to load a "default" object. This is likely an indication you incorrectly thought you were importing an ES6 file.
|
|
|
|
ISSUES WITH ES6 FILES:
|
|
|
|
4. Don't use module.exports in ES6:
|
|
You sholudn't manually assign module.exports anymore. Use proper ES6 module syntax like "export default" or "export const FOO".
|
|
|
|
5. Don't destructure default export:
|
|
If you're using "import {FOO} from './bar'" in ES6 files, it's important that "./bar" does NOT export a "default". Instead, in './bar', do "export const FOO = 'foo'"
|
|
|
|
6. Spec has to start with function
|
|
Top-level "describe" blocks can no longer use the "() => {}" function syntax. This will incorrectly bind "this" to the "window" object instead of the jasmine object. The top-level "describe" block must use the "function describeName() {}" syntax
|
|
`;
|
|
done(new Error(error));
|
|
}
|
|
}
|
|
|
|
done(null);
|
|
});
|
|
}
|