-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathsemi.ts
103 lines (100 loc) · 2.66 KB
/
semi.ts
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
import type { Rule } from "eslint";
import { makeListener } from "../utils";
import type { visitor } from "@peggyjs/eslint-parser";
function check(
context: Rule.RuleContext,
required: boolean,
before: visitor.AST.Node,
semi?: visitor.AST.Punctuation[]
): void {
if (required) {
if (!semi) {
context.report({
loc: {
start: before.loc.end,
end: before.loc.end,
},
messageId: "required",
fix(fixer: Rule.RuleFixer): Rule.Fix {
return fixer.replaceTextRange([
before.range[1],
before.range[1],
], ";");
},
});
} else if (semi.length > 1) {
const lastSemi = semi[semi.length - 1];
context.report({
loc: {
start: before.loc.end,
end: lastSemi.loc.end,
},
messageId: "prohibited",
fix(fixer: Rule.RuleFixer): Rule.Fix {
return fixer.replaceTextRange([
before.range[1],
lastSemi.range[1],
], ";");
},
});
}
} else {
if (semi) {
const lastSemi = semi[semi.length - 1];
context.report({
loc: {
start: before.loc.end,
end: lastSemi.loc.end,
},
messageId: "prohibited",
fix(fixer: Rule.RuleFixer): Rule.Fix {
return fixer.removeRange([
before.range[1],
lastSemi.range[1],
]);
},
});
}
}
}
const rule: Rule.RuleModule = {
meta: {
type: "suggestion",
docs: {
description: "enforce consistent use of semicolons",
recommended: true,
url: "https://github.com/peggyjs/peggyjs-eslint-plugin/blob/main/docs/rules/semi.md",
},
messages: {
required: "Missing semicolon.",
prohibited: "Extra semicolon.",
},
fixable: "code",
schema: [
{
type: "string",
enum: ["always", "never"],
},
],
},
create(context: Rule.RuleContext): Rule.RuleListener {
const required = context.options[0] === "always";
return makeListener({
top_level_initializer(node: visitor.AST.TopLevelInitializer): void {
check(context, required, node.close, node.semi);
},
initializer(node: visitor.AST.Initializer): void {
check(context, required, node.code.close, node.semi);
},
rule(node: visitor.AST.Rule): void {
// Named expressions include trailing whitespace, which we don't want
// before the semi.
const expr = node.expression.type === "named"
? node.expression.expression
: node.expression;
check(context, required, expr, node.semi);
},
});
},
};
export = rule;