mirror of
https://github.com/angular/angular-cli.git
synced 2025-05-15 01:54:04 +08:00
154 lines
4.2 KiB
TypeScript
154 lines
4.2 KiB
TypeScript
/**
|
|
* @license
|
|
* Copyright Google Inc. All Rights Reserved.
|
|
*
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
* found in the LICENSE file at https://angular.io/license
|
|
*/
|
|
// tslint:disable:no-implicit-dependencies
|
|
import { logging } from '@angular-devkit/core';
|
|
import { execSync } from 'child_process';
|
|
import { packages } from '../lib/packages';
|
|
|
|
|
|
const blacklist = [
|
|
'9ce1aed331ad0742463b587f1f5555486ccc202f',
|
|
'de7a44f23514594274394322adaf40ac87c38d8b',
|
|
];
|
|
|
|
|
|
export interface ValidateCommitsOptions {
|
|
ci?: boolean;
|
|
base?: string;
|
|
head?: string;
|
|
}
|
|
|
|
|
|
export default function (argv: ValidateCommitsOptions, logger: logging.Logger) {
|
|
logger.info('Getting merge base...');
|
|
|
|
const prNumber = process.env['CIRCLE_PR_NUMBER'] || '';
|
|
let baseSha = '';
|
|
let sha = '';
|
|
|
|
if (prNumber) {
|
|
const url = `https://api.github.com/repos/angular/angular-cli/pulls/${prNumber}`;
|
|
const prJson = JSON.parse(execSync(`curl "${url}"`, {
|
|
stdio: ['ignore', 'pipe', 'ignore'],
|
|
encoding: 'utf8',
|
|
}).toString());
|
|
baseSha = prJson['base']['sha'];
|
|
sha = prJson['head']['sha'];
|
|
} else if (argv.base) {
|
|
baseSha = argv.base;
|
|
sha = argv.head || 'HEAD';
|
|
} else {
|
|
const parentRemote = process.env['GIT_REMOTE'] ? process.env['GIT_REMOTE'] + '/' : '';
|
|
const parentBranch = process.env['GIT_BRANCH'] || 'master';
|
|
baseSha = execSync(`git merge-base --fork-point "${parentRemote}${parentBranch}"`)
|
|
.toString().trim();
|
|
sha = 'HEAD';
|
|
}
|
|
|
|
logger.createChild('sha').info(`Base: ${baseSha}\nHEAD: ${sha}`);
|
|
|
|
const log = execSync(`git log --oneline "${baseSha}..${sha}"`).toString().trim();
|
|
logger.debug('Commits:');
|
|
logger.createChild('commits').debug(log);
|
|
logger.debug('');
|
|
|
|
const commits = log.split(/\n/)
|
|
.map(i => i.match(/(^[0-9a-f]+) (.+)$/))
|
|
.map(x => x ? Array.from(x).slice(1) : null)
|
|
.filter(x => !!x) as [string, string][];
|
|
logger.info(`Found ${commits.length} commits...`);
|
|
|
|
const output = logger.createChild('check');
|
|
let invalidCount = 0;
|
|
|
|
function _invalid(sha: string, message: string, error: string) {
|
|
invalidCount++;
|
|
output.error(`The following commit ${error}:`);
|
|
output.error(` ${sha} ${message}`);
|
|
}
|
|
|
|
for (const [sha, message] of commits) {
|
|
if (blacklist.find(i => i.startsWith(sha))) {
|
|
// Some commits are better ignored.
|
|
continue;
|
|
}
|
|
|
|
const subject = message.match(/^([^:(]+)(?:\((.*?)\))?:/);
|
|
if (!subject) {
|
|
_invalid(sha, message, 'does not have a subject');
|
|
continue;
|
|
}
|
|
|
|
const [type, scope] = subject.slice(1);
|
|
switch (type) {
|
|
// Types that can contain both a scope or no scope.
|
|
case 'docs':
|
|
case 'refactor':
|
|
case 'style':
|
|
case 'test':
|
|
if (scope && !packages[scope]) {
|
|
_invalid(sha, message, 'has a scope that does not exist');
|
|
continue;
|
|
}
|
|
break;
|
|
|
|
// Types that MUST contain a scope.
|
|
case 'feat':
|
|
case 'fix':
|
|
if (!scope) {
|
|
_invalid(sha, message, 'should always have a scope');
|
|
continue;
|
|
}
|
|
if (!packages[scope]) {
|
|
_invalid(sha, message, 'has a scope that does not exist');
|
|
continue;
|
|
}
|
|
break;
|
|
|
|
// Types that MUST NOT contain a scope.
|
|
case 'build':
|
|
case 'revert':
|
|
case 'ci':
|
|
if (scope) {
|
|
_invalid(sha, message, 'should not have a scope');
|
|
continue;
|
|
}
|
|
break;
|
|
|
|
case 'release':
|
|
if (scope) {
|
|
_invalid(sha, message, 'should not have a scope');
|
|
continue;
|
|
}
|
|
if (argv.ci && commits.length > 1) {
|
|
_invalid(sha, message, 'release should always be alone in a PR');
|
|
continue;
|
|
}
|
|
break;
|
|
|
|
case 'wip':
|
|
if (argv.ci) {
|
|
_invalid(sha, message, 'wip are not allowed in a PR');
|
|
}
|
|
break;
|
|
|
|
// Unknown types.
|
|
default:
|
|
_invalid(sha, message, 'has an unknown type. You can use wip: to avoid this.');
|
|
}
|
|
}
|
|
|
|
if (invalidCount > 0) {
|
|
logger.fatal(`${invalidCount} commits were found invalid...`);
|
|
} else {
|
|
logger.info('All green. Thank you, come again.');
|
|
}
|
|
|
|
return invalidCount;
|
|
}
|