-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathspellcheck-commit.js
127 lines (111 loc) · 3.79 KB
/
spellcheck-commit.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
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
// @ts-check
/* eslint-disable unicorn/prefer-top-level-await */
/* eslint-disable no-console */
const spellcheck = require('spellchecker');
const read = require('node:fs/promises').readFile;
const execa = require('execa');
const debug = require('debug')('commit-spell');
const tryToRead = async (
/** @type {import("fs").PathLike | import("fs/promises").FileHandle} */ path
) => {
try {
debug(`attempting to read ${path}`);
const data = await read(path, 'utf8');
debug(`successfully read ${path}`);
return data;
} catch {
debug(`failed to read ${path}`);
return '';
}
};
const asJson = (/** @type {string} */ str) => {
try {
const json = JSON.parse(str);
debug('json parse successful!');
return [
...(json?.['cSpell.words'] || []),
...(json?.['cSpell.userWords'] || []),
...(json?.['cSpell.ignoreWords'] || [])
];
} catch {
debug('json parse failed!');
return [];
}
};
const asText = (/** @type {string} */ str) => str.split('\n');
const isCamelCase = (/** @type {string} */ w) => /^[a-z]+[A-Z]+.*$/.test(w);
const isAllCaps = (/** @type {string} */ w) => /^[^a-z]+$/.test(w);
const isPascalCase = (/** @type {string} */ w) =>
/^([A-Z]{2,}.+|[A-Z][a-z]+[A-Z].*)$/.test(w);
const splitOutWords = (/** @type {string} */ phrase) =>
[...phrase.split(/[^A-Za-z]+/g), phrase].filter(Boolean);
const keys = (/** @type {{}} */ obj = {}) =>
Object.keys(obj).map((k) => splitOutWords(k));
void (async () => {
const pkgFile = await tryToRead('./package.json');
const pkg = pkgFile ? JSON.parse(pkgFile) : {};
const lastCommitMessage = (await read('./.git/COMMIT_EDITMSG'), 'utf8');
const homeDir = require('node:os').homedir();
debug(`lastCommitMsg: ${lastCommitMessage}`);
debug(`homeDir: ${homeDir}`);
const ignoreWords = new Set(
Array.from(
new Set(
[
...(await Promise.all([
tryToRead('./.spellcheckignore').then(asText),
tryToRead(`${homeDir}/.config/_spellcheckignore`).then(asText),
tryToRead('./.vscode/settings.json').then(asJson),
tryToRead(`${homeDir}/.config/Code/User/settings.json`).then(asJson)
])),
...(await import('text-extensions')).default,
// ? Popular contractions
've',
're',
's',
'll',
't',
'd',
'o',
'ol',
...keys(pkg.dependencies),
...keys(pkg.devDependencies),
...keys(pkg.scripts),
...splitOutWords(
(await execa('git', ['log', '--format="%B"', 'HEAD~1'])).stdout
).slice(0, -1)
]
.flat()
.filter(Boolean)
.flatMap((word) => splitOutWords(word.trim().toLowerCase()))
)
)
);
const typos = Array.from(
new Set(
spellcheck
.checkSpelling(lastCommitMessage)
.flatMap((typoLocation) =>
lastCommitMessage.slice(typoLocation.start, typoLocation.end).trim().split("'")
)
.filter((w) => !isAllCaps(w) && !isCamelCase(w) && !isPascalCase(w))
.map((w) => w.toLowerCase())
.filter((typo) => !ignoreWords.has(typo))
)
);
debug('typos: %O', typos);
if (typos.length) {
console.warn('WARNING: there may be misspelled words in your commit message!');
console.warn('Commit messages can be fixed before push with `git commit -S --amend`');
console.warn('---');
for (const typo of typos.slice(0, 5)) {
const corrections = spellcheck.getCorrectionsForMisspelling(typo);
const suggestion = corrections.length
? ` (did you mean ${corrections.slice(0, 5).join(', ')}?)`
: '';
console.warn(`${typo}${suggestion}`);
}
typos.length > 5 && console.warn(`${typos.length - 5} more...`);
typos.length && console.warn('---');
}
})();