aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorVojta Jina2012-09-01 19:31:06 -0700
committerMisko Hevery2012-09-11 16:12:40 -0700
commit5dbd942bac8b8d045162d29467640107db14263b (patch)
tree49ef5ead9636f44dae6c737064ae5396e531827d
parent84c13d96ff6e993b2ee9ff6bf49614fc1d514b04 (diff)
downloadangular.js-5dbd942bac8b8d045162d29467640107db14263b.tar.bz2
chore(scripts): add commit-msg hook (validation)
-rwxr-xr-xvalidate-commit-msg.js104
-rw-r--r--validate-commit-msg.spec.js73
2 files changed, 177 insertions, 0 deletions
diff --git a/validate-commit-msg.js b/validate-commit-msg.js
new file mode 100755
index 00000000..453c56c7
--- /dev/null
+++ b/validate-commit-msg.js
@@ -0,0 +1,104 @@
+#!/usr/bin/env node
+
+/**
+ * Git COMMIT-MSG hook for validating commit message
+ * See https://docs.google.com/document/d/1rk04jEuGfk9kYzfqCuOlPTSJw3hEDZJTBN5E5f1SALo/edit
+ *
+ * Installation:
+ * >> cd <angular-repo>
+ * >> ln -s ../../validate-commit-msg.js .git/hooks/commit-msg
+ */
+var fs = require('fs');
+var util = require('util');
+
+
+var MAX_LENGTH = 70;
+var PATTERN = /^(\w*)(\(([\w\$\.]*)\))?\: (.*)$/;
+var IGNORED = /^WIP\:/;
+var TYPES = {
+ feat: true,
+ fix: true,
+ docs: true,
+ style: true,
+ refactor: true,
+ test: true,
+ chore: true
+};
+
+
+var error = function() {
+ // gitx does not display it
+ // http://gitx.lighthouseapp.com/projects/17830/tickets/294-feature-display-hook-error-message-when-hook-fails
+ // https://groups.google.com/group/gitx/browse_thread/thread/a03bcab60844b812
+ console.error('INVALID COMMIT MSG: ' + util.format.apply(null, arguments));
+};
+
+
+var validateMessage = function(message) {
+ var isValid = true;
+
+ if (IGNORED.test(message)) {
+ console.log('Commit message validation ignored.');
+ return true;
+ }
+
+ if (message.length > MAX_LENGTH) {
+ error('is longer than %d characters !', MAX_LENGTH);
+ isValid = false;
+ }
+
+ var match = PATTERN.exec(message);
+
+ if (!match) {
+ error('does not match "<type>(<scope>): <subject>" !');
+ return false;
+ }
+
+ var type = match[1];
+ var scope = match[3];
+ var subject = match[4];
+
+ if (!TYPES.hasOwnProperty(type)) {
+ error('"%s" is not allowed type !', type);
+ return false;
+ }
+
+ // Some more ideas, do want anything like this ?
+ // - allow only specific scopes (eg. fix(docs) should not be allowed ?
+ // - auto correct the type to lower case ?
+ // - auto correct first letter of the subject to lower case ?
+ // - auto add empty line after subject ?
+ // - auto remove empty () ?
+ // - auto correct typos in type ?
+ // - store incorrect messages, so that we can learn
+
+ return isValid;
+};
+
+
+var firstLineFromBuffer = function(buffer) {
+ return buffer.toString().split('\n').shift();
+};
+
+
+
+// publish for testing
+exports.validateMessage = validateMessage;
+
+// hacky start if not run by jasmine :-D
+if (process.argv.join('').indexOf('jasmine-node') === -1) {
+ var commitMsgFile = process.argv[2];
+ var incorrectLogFile = commitMsgFile.replace('COMMIT_EDITMSG', 'logs/incorrect-commit-msgs');
+
+ fs.readFile(commitMsgFile, function(err, buffer) {
+ var msg = firstLineFromBuffer(buffer);
+
+ if (!validateMessage(msg)) {
+ fs.appendFile(incorrectLogFile, msg + '\n', function() {
+ process.exit(1);
+ });
+ } else {
+ process.exit(0);
+ }
+ });
+}
diff --git a/validate-commit-msg.spec.js b/validate-commit-msg.spec.js
new file mode 100644
index 00000000..8eb4f3bd
--- /dev/null
+++ b/validate-commit-msg.spec.js
@@ -0,0 +1,73 @@
+describe('validate-commit-msg.js', function() {
+ var m = require('./validate-commit-msg');
+ var errors = [];
+ var logs = [];
+
+ var VALID = true;
+ var INVALID = false;
+
+ beforeEach(function() {
+ errors.length = 0;
+ logs.length = 0;
+
+ spyOn(console, 'error').andCallFake(function(msg) {
+ errors.push(msg.replace(/\x1B\[\d+m/g, '')); // uncolor
+ });
+
+ spyOn(console, 'log').andCallFake(function(msg) {
+ logs.push(msg.replace(/\x1B\[\d+m/g, '')); // uncolor
+ });
+ });
+
+ describe('validateMessage', function() {
+
+ it('should be valid', function() {
+ expect(m.validateMessage('fix($compile): something')).toBe(VALID);
+ expect(m.validateMessage('feat($location): something')).toBe(VALID);
+ expect(m.validateMessage('docs($filter): something')).toBe(VALID);
+ expect(m.validateMessage('style($http): something')).toBe(VALID);
+ expect(m.validateMessage('refactor($httpBackend): something')).toBe(VALID);
+ expect(m.validateMessage('test($resource): something')).toBe(VALID);
+ expect(m.validateMessage('chore($controller): something')).toBe(VALID);
+ expect(errors).toEqual([]);
+ });
+
+
+ it('should validate 70 characters length', function() {
+ var msg = 'fix($compile): something super mega extra giga tera long, maybe even longer... ' +
+ 'way over 80 characters';
+
+ expect(m.validateMessage(msg)).toBe(INVALID);
+ expect(errors).toEqual(['INVALID COMMIT MSG: is longer than 70 characters !']);
+ });
+
+
+ it('should validate "<type>(<scope>): <subject>" format', function() {
+ var msg = 'not correct format';
+
+ expect(m.validateMessage(msg)).toBe(INVALID);
+ expect(errors).toEqual(['INVALID COMMIT MSG: does not match "<type>(<scope>): <subject>" !']);
+ });
+
+
+ it('should validate type', function() {
+ expect(m.validateMessage('weird($filter): something')).toBe(INVALID);
+ expect(errors).toEqual(['INVALID COMMIT MSG: "weird" is not allowed type !']);
+ });
+
+
+ it('should allow empty scope', function() {
+ expect(m.validateMessage('fix: blablabla')).toBe(VALID);
+ });
+
+
+ it('should allow dot in scope', function() {
+ expect(m.validateMessage('chore(mocks.$httpBackend): something')).toBe(VALID);
+ });
+
+
+ it('should ignore msg prefixed with "WIP: "', function() {
+ expect(m.validateMessage('WIP: bullshit')).toBe(VALID);
+ });
+ });
+});