-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathunicode-escape.ts
112 lines (103 loc) · 3.69 KB
/
unicode-escape.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
104
105
106
107
108
109
110
111
112
import type { Character } from "@eslint-community/regexpp/ast"
import type { RegExpVisitor } from "@eslint-community/regexpp/visitor"
import type { RegExpContext } from "../utils"
import { createRule, defineRegexpVisitor } from "../utils"
import {
EscapeSequenceKind,
getEscapeSequenceKind,
} from "../utils/regex-syntax"
export default createRule("unicode-escape", {
meta: {
docs: {
description:
"enforce consistent usage of unicode escape or unicode codepoint escape",
category: "Stylistic Issues",
recommended: false,
},
fixable: "code",
schema: [
{
enum: ["unicodeCodePointEscape", "unicodeEscape"], // default unicodeCodePointEscape
},
],
messages: {
expectedUnicodeCodePointEscape:
"Expected unicode code point escape ('{{unicodeCodePointEscape}}'), but unicode escape ('{{unicodeEscape}}') is used.",
expectedUnicodeEscape:
"Expected unicode escape ('{{unicodeEscape}}'), but unicode code point escape ('{{unicodeCodePointEscape}}') is used.",
},
type: "suggestion", // "problem",
},
create(context) {
const preferUnicodeCodePointEscape =
context.options[0] !== "unicodeEscape"
function verifyForUnicodeCodePointEscape(
{ node, getRegexpLocation, fixReplaceNode }: RegExpContext,
kind: EscapeSequenceKind,
cNode: Character,
) {
if (kind !== EscapeSequenceKind.unicode) {
return
}
const unicodeCodePointEscape = `\\u{${cNode.value.toString(16)}}`
context.report({
node,
loc: getRegexpLocation(cNode),
messageId: "expectedUnicodeCodePointEscape",
data: {
unicodeCodePointEscape,
unicodeEscape: cNode.raw,
},
fix: fixReplaceNode(cNode, unicodeCodePointEscape),
})
}
function verifyForUnicodeEscape(
{ node, getRegexpLocation, fixReplaceNode }: RegExpContext,
kind: EscapeSequenceKind,
cNode: Character,
) {
if (kind !== EscapeSequenceKind.unicodeCodePoint) {
return
}
const unicodeEscape = `\\u${cNode.value
.toString(16)
.padStart(4, "0")}`
context.report({
node,
loc: getRegexpLocation(cNode),
messageId: "expectedUnicodeEscape",
data: {
unicodeEscape,
unicodeCodePointEscape: cNode.raw,
},
fix: fixReplaceNode(cNode, unicodeEscape),
})
}
const verify = preferUnicodeCodePointEscape
? verifyForUnicodeCodePointEscape
: verifyForUnicodeEscape
function createVisitor(
regexpContext: RegExpContext,
): RegExpVisitor.Handlers {
const { flags } = regexpContext
if (!flags.unicode && !flags.unicodeSets) {
return {}
}
return {
onCharacterEnter(cNode) {
if (cNode.value >= 0x10000) {
return
}
const kind = getEscapeSequenceKind(cNode.raw)
if (!kind) {
return
}
verify(regexpContext, kind, cNode)
},
}
}
return defineRegexpVisitor(context, {
createVisitor,
})
},
})