vscode/build/hygiene.js

419 lines
10 KiB
JavaScript
Raw Normal View History

2020-09-22 17:12:35 +08:00
/*---------------------------------------------------------------------------------------------
* Copyright (c) Microsoft Corporation. All rights reserved.
* Licensed under the MIT License. See License.txt in the project root for license information.
*--------------------------------------------------------------------------------------------*/
2020-09-22 20:37:04 +08:00
const filter = require('gulp-filter');
const es = require('event-stream');
const VinylFile = require('vinyl');
const vfs = require('vinyl-fs');
const path = require('path');
const fs = require('fs');
const pall = require('p-all');
2020-09-22 17:12:35 +08:00
/**
* Hygiene works by creating cascading subsets of all our files and
* passing them through a sequence of checks. Here are the current subsets,
* named according to the checks performed on them. Each subset contains
* the following one, as described in mathematical notation:
*
* all eol indentation copyright typescript
*/
const all = [
2020-09-22 20:37:04 +08:00
'*',
'build/**/*',
'extensions/**/*',
'scripts/**/*',
'src/**/*',
'test/**/*',
2021-01-07 03:36:37 +08:00
'!out*/**',
2020-09-22 20:37:04 +08:00
'!test/**/out/**',
'!**/node_modules/**',
2020-09-22 17:12:35 +08:00
];
module.exports.all = all;
const indentationFilter = [
2020-09-22 20:37:04 +08:00
'**',
2020-09-22 17:12:35 +08:00
// except specific files
2020-09-22 20:37:04 +08:00
'!**/ThirdPartyNotices.txt',
'!**/LICENSE.{txt,rtf}',
'!LICENSES.chromium.html',
'!**/LICENSE',
'!src/vs/nls.js',
'!src/vs/nls.build.js',
'!src/vs/css.js',
'!src/vs/css.build.js',
'!src/vs/loader.js',
'!src/vs/base/common/insane/insane.js',
'!src/vs/base/common/marked/marked.js',
2020-10-26 19:11:55 +08:00
'!src/vs/base/common/semver/semver.js',
2020-09-22 20:37:04 +08:00
'!src/vs/base/node/terminateProcess.sh',
'!src/vs/base/node/cpuUsage.sh',
'!test/unit/assert.js',
'!resources/linux/snap/electron-launch',
2020-09-22 17:12:35 +08:00
// except specific folders
2020-09-22 20:37:04 +08:00
'!test/automation/out/**',
'!test/monaco/out/**',
2020-09-22 20:37:04 +08:00
'!test/smoke/out/**',
'!extensions/typescript-language-features/test-workspace/**',
'!extensions/vscode-api-tests/testWorkspace/**',
'!extensions/vscode-api-tests/testWorkspace2/**',
'!build/monaco/**',
'!build/win32/**',
2020-09-22 17:12:35 +08:00
// except multiple specific files
2020-09-22 20:37:04 +08:00
'!**/package.json',
'!**/yarn.lock',
'!**/yarn-error.log',
2020-09-22 17:12:35 +08:00
// except multiple specific folders
2020-09-22 20:37:04 +08:00
'!**/codicon/**',
'!**/fixtures/**',
'!**/lib/**',
'!extensions/**/dist/**',
2020-09-22 20:37:04 +08:00
'!extensions/**/out/**',
'!extensions/**/snippets/**',
'!extensions/**/syntaxes/**',
'!extensions/**/themes/**',
'!extensions/**/colorize-fixtures/**',
2020-09-22 17:12:35 +08:00
// except specific file types
2020-09-22 20:37:04 +08:00
'!src/vs/*/**/*.d.ts',
'!src/typings/**/*.d.ts',
'!extensions/**/*.d.ts',
2020-12-01 12:56:50 +08:00
'!**/*.{svg,exe,png,bmp,jpg,scpt,bat,cmd,cur,ttf,woff,eot,md,ps1,template,yaml,yml,d.ts.recipe,ico,icns,plist}',
2020-09-22 20:37:04 +08:00
'!build/{lib,download,darwin}/**/*.js',
'!build/**/*.sh',
'!build/azure-pipelines/**/*.js',
'!build/azure-pipelines/**/*.config',
'!**/Dockerfile',
'!**/Dockerfile.*',
'!**/*.Dockerfile',
'!**/*.dockerfile',
'!extensions/markdown-language-features/media/*.js',
'!extensions/simple-browser/media/*.js',
2020-09-22 17:12:35 +08:00
];
const copyrightFilter = [
2020-09-22 20:37:04 +08:00
'**',
'!**/*.desktop',
'!**/*.json',
'!**/*.html',
'!**/*.template',
'!**/*.md',
'!**/*.bat',
'!**/*.cmd',
'!**/*.ico',
'!**/*.icns',
'!**/*.xml',
'!**/*.sh',
'!**/*.txt',
'!**/*.xpm',
'!**/*.opts',
'!**/*.disabled',
'!**/*.code-workspace',
'!**/*.js.map',
'!build/**/*.init',
'!resources/linux/snap/snapcraft.yaml',
'!resources/win32/bin/code.js',
'!resources/web/code-web.js',
'!resources/completions/**',
'!extensions/configuration-editing/build/inline-allOf.ts',
2020-09-22 20:37:04 +08:00
'!extensions/markdown-language-features/media/highlight.css',
'!extensions/html-language-features/server/src/modes/typescript/*',
'!extensions/*/server/bin/*',
'!src/vs/editor/test/node/classification/typescript-test.ts',
2020-09-22 17:12:35 +08:00
];
const jsHygieneFilter = [
2020-09-22 20:37:04 +08:00
'src/**/*.js',
'build/gulpfile.*.js',
'!src/vs/loader.js',
'!src/vs/css.js',
'!src/vs/nls.js',
'!src/vs/css.build.js',
'!src/vs/nls.build.js',
'!src/**/insane.js',
'!src/**/marked.js',
2020-10-26 19:11:55 +08:00
'!src/**/semver.js',
2020-09-22 20:37:04 +08:00
'!**/test/**',
2020-09-22 17:12:35 +08:00
];
module.exports.jsHygieneFilter = jsHygieneFilter;
const tsHygieneFilter = [
2020-09-22 20:37:04 +08:00
'src/**/*.ts',
'test/**/*.ts',
'extensions/**/*.ts',
'!**/fixtures/**',
'!**/typings/**',
'!**/node_modules/**',
'!extensions/typescript-basics/test/colorize-fixtures/**',
'!extensions/vscode-api-tests/testWorkspace/**',
'!extensions/vscode-api-tests/testWorkspace2/**',
'!extensions/**/*.test.ts',
'!extensions/html-language-features/server/lib/jquery.d.ts',
2020-09-22 17:12:35 +08:00
];
module.exports.tsHygieneFilter = tsHygieneFilter;
const copyrightHeaderLines = [
2020-09-22 20:37:04 +08:00
'/*---------------------------------------------------------------------------------------------',
' * Copyright (c) Microsoft Corporation. All rights reserved.',
' * Licensed under the MIT License. See License.txt in the project root for license information.',
' *--------------------------------------------------------------------------------------------*/',
2020-09-22 17:12:35 +08:00
];
function hygiene(some) {
2020-12-23 02:55:56 +08:00
const gulpeslint = require('gulp-eslint');
const tsfmt = require('typescript-formatter');
2020-09-22 17:12:35 +08:00
let errorCount = 0;
const productJson = es.through(function (file) {
2020-09-22 20:37:04 +08:00
const product = JSON.parse(file.contents.toString('utf8'));
2020-09-22 17:12:35 +08:00
if (product.extensionsGallery) {
2020-09-22 20:37:04 +08:00
console.error(`product.json: Contains 'extensionsGallery'`);
2020-09-22 17:12:35 +08:00
errorCount++;
}
2020-09-22 20:37:04 +08:00
this.emit('data', file);
2020-09-22 17:12:35 +08:00
});
const indentation = es.through(function (file) {
2020-09-22 20:37:04 +08:00
const lines = file.contents.toString('utf8').split(/\r\n|\r|\n/);
2020-09-22 17:12:35 +08:00
file.__lines = lines;
lines.forEach((line, i) => {
if (/^\s*$/.test(line)) {
// empty or whitespace lines are OK
} else if (/^[\t]*[^\s]/.test(line)) {
// good indent
} else if (/^[\t]* \*/.test(line)) {
// block comment using an extra space
} else {
console.error(
2020-09-22 20:37:04 +08:00
file.relative + '(' + (i + 1) + ',1): Bad whitespace indentation'
2020-09-22 17:12:35 +08:00
);
errorCount++;
}
});
2020-09-22 20:37:04 +08:00
this.emit('data', file);
2020-09-22 17:12:35 +08:00
});
const copyrights = es.through(function (file) {
const lines = file.__lines;
for (let i = 0; i < copyrightHeaderLines.length; i++) {
if (lines[i] !== copyrightHeaderLines[i]) {
2020-09-22 20:37:04 +08:00
console.error(file.relative + ': Missing or bad copyright statement');
2020-09-22 17:12:35 +08:00
errorCount++;
break;
}
}
2020-09-22 20:37:04 +08:00
this.emit('data', file);
2020-09-22 17:12:35 +08:00
});
const formatting = es.map(function (file, cb) {
tsfmt
2020-09-22 20:37:04 +08:00
.processString(file.path, file.contents.toString('utf8'), {
2020-09-22 17:12:35 +08:00
verify: false,
tsfmt: true,
// verbose: true,
// keep checkJS happy
editorconfig: undefined,
replace: undefined,
tsconfig: undefined,
tsconfigFile: undefined,
tsfmtFile: undefined,
vscode: undefined,
vscodeFile: undefined,
})
.then(
(result) => {
2020-09-22 20:37:04 +08:00
let original = result.src.replace(/\r\n/gm, '\n');
let formatted = result.dest.replace(/\r\n/gm, '\n');
2020-09-22 17:12:35 +08:00
if (original !== formatted) {
console.error(
2020-09-22 20:37:04 +08:00
`File not formatted. Run the 'Format Document' command to fix it:`,
2020-09-22 17:12:35 +08:00
file.relative
);
errorCount++;
}
cb(null, file);
},
(err) => {
cb(err);
}
);
});
let input;
2020-09-22 20:37:04 +08:00
if (Array.isArray(some) || typeof some === 'string' || !some) {
const options = { base: '.', follow: true, allowEmpty: true };
2020-09-22 17:12:35 +08:00
if (some) {
input = vfs.src(some, options).pipe(filter(all)); // split this up to not unnecessarily filter all a second time
} else {
input = vfs.src(all, options);
}
} else {
input = some;
}
2020-09-22 20:37:04 +08:00
const productJsonFilter = filter('product.json', { restore: true });
2020-09-22 17:12:35 +08:00
const result = input
.pipe(filter((f) => !f.stat.isDirectory()))
.pipe(productJsonFilter)
2020-09-22 20:37:04 +08:00
.pipe(process.env['BUILD_SOURCEVERSION'] ? es.through() : productJson)
2020-09-22 17:12:35 +08:00
.pipe(productJsonFilter.restore)
.pipe(filter(indentationFilter))
.pipe(indentation)
.pipe(filter(copyrightFilter))
.pipe(copyrights);
const typescript = result.pipe(filter(tsHygieneFilter)).pipe(formatting);
const javascript = result
.pipe(filter(jsHygieneFilter.concat(tsHygieneFilter)))
.pipe(
gulpeslint({
2020-09-22 20:37:04 +08:00
configFile: '.eslintrc.json',
rulePaths: ['./build/lib/eslint'],
2020-09-22 17:12:35 +08:00
})
)
2020-09-22 20:37:04 +08:00
.pipe(gulpeslint.formatEach('compact'))
2020-09-22 17:12:35 +08:00
.pipe(
gulpeslint.results((results) => {
errorCount += results.warningCount;
errorCount += results.errorCount;
})
);
let count = 0;
return es.merge(typescript, javascript).pipe(
es.through(
function (data) {
count++;
2020-09-22 20:37:04 +08:00
if (process.env['TRAVIS'] && count % 10 === 0) {
process.stdout.write('.');
2020-09-22 17:12:35 +08:00
}
2020-09-22 20:37:04 +08:00
this.emit('data', data);
2020-09-22 17:12:35 +08:00
},
function () {
2020-09-22 20:37:04 +08:00
process.stdout.write('\n');
2020-09-22 17:12:35 +08:00
if (errorCount > 0) {
this.emit(
2020-09-22 20:37:04 +08:00
'error',
'Hygiene failed with ' +
errorCount +
` errors. Check 'build / gulpfile.hygiene.js'.`
2020-09-22 17:12:35 +08:00
);
} else {
2020-09-22 20:37:04 +08:00
this.emit('end');
2020-09-22 17:12:35 +08:00
}
}
)
);
}
module.exports.hygiene = hygiene;
function createGitIndexVinyls(paths) {
2020-09-22 20:37:04 +08:00
const cp = require('child_process');
2020-09-22 17:12:35 +08:00
const repositoryPath = process.cwd();
const fns = paths.map((relativePath) => () =>
new Promise((c, e) => {
const fullPath = path.join(repositoryPath, relativePath);
fs.stat(fullPath, (err, stat) => {
2020-09-22 20:37:04 +08:00
if (err && err.code === 'ENOENT') {
2020-09-22 17:12:35 +08:00
// ignore deletions
return c(null);
} else if (err) {
return e(err);
}
cp.exec(
`git show :${relativePath}`,
2020-09-22 20:37:04 +08:00
{ maxBuffer: 2000 * 1024, encoding: 'buffer' },
2020-09-22 17:12:35 +08:00
(err, out) => {
if (err) {
return e(err);
}
c(
new VinylFile({
path: fullPath,
base: repositoryPath,
contents: out,
stat,
})
);
}
);
});
})
);
return pall(fns, { concurrency: 4 }).then((r) => r.filter((p) => !!p));
}
// this allows us to run hygiene as a git pre-commit hook
if (require.main === module) {
2020-09-22 20:37:04 +08:00
const cp = require('child_process');
2020-09-22 17:12:35 +08:00
2020-09-22 20:37:04 +08:00
process.on('unhandledRejection', (reason, p) => {
console.log('Unhandled Rejection at: Promise', p, 'reason:', reason);
2020-09-22 17:12:35 +08:00
process.exit(1);
});
if (process.argv.length > 2) {
2020-09-22 20:37:04 +08:00
hygiene(process.argv.slice(2)).on('error', (err) => {
2020-09-22 17:12:35 +08:00
console.error();
console.error(err);
process.exit(1);
});
} else {
cp.exec(
2020-09-22 20:37:04 +08:00
'git diff --cached --name-only',
2020-09-22 17:12:35 +08:00
{ maxBuffer: 2000 * 1024 },
(err, out) => {
if (err) {
console.error();
console.error(err);
process.exit(1);
}
const some = out.split(/\r?\n/).filter((l) => !!l);
if (some.length > 0) {
2020-09-22 20:37:04 +08:00
console.log('Reading git index versions...');
2020-09-22 17:12:35 +08:00
createGitIndexVinyls(some)
.then(
(vinyls) =>
new Promise((c, e) =>
hygiene(es.readArray(vinyls))
2020-09-22 20:37:04 +08:00
.on('end', () => c())
.on('error', e)
2020-09-22 17:12:35 +08:00
)
)
.catch((err) => {
console.error();
console.error(err);
process.exit(1);
});
}
}
);
}
}