forked from l8on/affiance
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathCoffeeLint.js
93 lines (83 loc) · 2.57 KB
/
CoffeeLint.js
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
'use strict';
const PreCommitBase = require('./Base');
/**
* @class CoffeeLint
* @extends PreCommitBase
* @classdesc Run coffeelint on changed files
*/
class CoffeeLint extends PreCommitBase {
/**
* Run `coffeelint` against files that apply.
* Uses spawnPromiseOnApplicableFiles to parallelize
*
* @returns {Promise}
* @resolves {HookMessage[]} An array of hook messages produced by the hook
* @rejects {Error} An Error thrown or emitted while running the hook
*/
run() {
return new Promise((resolve, reject) => {
this.spawnPromiseOnApplicableFiles().then((result) => {
if(result.error) {
return resolve(['fail', 'coffeelint failed to start:\n' + result.error.message]);
}
if(result.stderr.toString()) {
return resolve(['fail', 'coffeelint has error output:\n' + result.stderr.toString()]);
}
return resolve(this.extractMessages(
this.parseCoffeelintOutput(result.stdout),
CoffeeLint.MESSAGE_REGEX,
CoffeeLint.MESSAGE_CAPTURE_MAP,
CoffeeLint.MESSAGE_TYPE_CATEGORIZER
));
}, reject);
});
}
/**
* Parses the output stream of coffeelint to produce an array of
* output messages. Ensures we only send valid lines to the
* standard `extractMessages` function.
* For example, coffeelint will print the headers of the csv like:
* path,lineNumber,lineNumberEnd,level,message
* which is useful for a csv file, but not for our purposes.
*
* @param {Buffer|string} stdout - the stdout content of a `coffeelint` run
* @returns {string[]}
*/
parseCoffeelintOutput(stdout) {
let outputLines = stdout.toString().trim().split('\n');
return outputLines.filter((outputLine) => {
return CoffeeLint.MESSAGE_REGEX.test(outputLine);
});
}
static MESSAGE_TYPE_CATEGORIZER(capturedType) {
if (capturedType && capturedType.indexOf('w') > -1) {
return 'warning';
} else {
return 'error';
}
}
}
/**
* Regex to capture various parts of coffeelint csv output.
* The output lines look like this:
* `dir/path.coffee,10,20,warn,Some Issue Described`
* @type {RegExp}
*/
CoffeeLint.MESSAGE_REGEX = new RegExp(
'^(.+),' + //1: File name
'(\\d*),\\d*,' + //2: Line number
'(\\w+),' + //3: Type
'(.+)$' //4: Content
);
/**
* Maps the types of captured data to the index in the
* matches array produced when executing `MESSAGE_REGEX`
*
* @type {{file: number, line: number, type: number}}
*/
CoffeeLint.MESSAGE_CAPTURE_MAP = {
'file': 1,
'line': 2,
'type': 3
};
module.exports = CoffeeLint;