Skip to content

Commit c9e3a5d

Browse files
committed
feat: detect and warn invalid dynamic argument expressions
1 parent 624c799 commit c9e3a5d

File tree

3 files changed

+41
-7
lines changed

3 files changed

+41
-7
lines changed

src/compiler/parser/html-parser.js

+2-1
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@ import { unicodeLetters } from 'core/util/lang'
1515

1616
// Regular Expressions for parsing tags and attributes
1717
const attribute = /^\s*([^\s"'<>\/=]+)(?:\s*(=)\s*(?:"([^"]*)"+|'([^']*)'+|([^\s"'=<>`]+)))?/
18+
const dynamicArgAttribute = /^\s*((?:v-[\w-]+:|@|:|#)\[[^=]+\][^\s"'<>\/=]*)(?:\s*(=)\s*(?:"([^"]*)"+|'([^']*)'+|([^\s"'=<>`]+)))?/
1819
const ncname = `[a-zA-Z_][\\-\\.0-9_a-zA-Z${unicodeLetters}]*`
1920
const qnameCapture = `((?:${ncname}\\:)?${ncname})`
2021
const startTagOpen = new RegExp(`^<${qnameCapture}`)
@@ -192,7 +193,7 @@ export function parseHTML (html, options) {
192193
}
193194
advance(start[0].length)
194195
let end, attr
195-
while (!(end = html.match(startTagClose)) && (attr = html.match(attribute))) {
196+
while (!(end = html.match(startTagClose)) && (attr = html.match(dynamicArgAttribute) || html.match(attribute))) {
196197
attr.start = index
197198
advance(attr[0].length)
198199
attr.end = index

src/compiler/parser/index.js

+22-6
Original file line numberDiff line numberDiff line change
@@ -38,6 +38,8 @@ const slotRE = /^v-slot(:|$)|^#/
3838
const lineBreakRE = /[\r\n]/
3939
const whitespaceRE = /\s+/g
4040

41+
const invalidAttributeRE = /[\s"'<>\/=]/
42+
4143
const decodeHTMLCached = cached(he.decode)
4244

4345
// configurable state
@@ -194,12 +196,26 @@ export function parse (
194196
element.ns = ns
195197
}
196198

197-
if (process.env.NODE_ENV !== 'production' && options.outputSourceRange) {
198-
element.start = start
199-
element.rawAttrsMap = element.attrsList.reduce((cumulated, attr) => {
200-
cumulated[attr.name] = attr
201-
return cumulated
202-
}, {})
199+
if (process.env.NODE_ENV !== 'production') {
200+
if (options.outputSourceRange) {
201+
element.start = start
202+
element.rawAttrsMap = element.attrsList.reduce((cumulated, attr) => {
203+
cumulated[attr.name] = attr
204+
return cumulated
205+
}, {})
206+
}
207+
attrs.forEach(attr => {
208+
if (invalidAttributeRE.test(attr.name)) {
209+
warn(
210+
`Invalid dynamic argument expression: attribute names cannot contain ` +
211+
`spaces, quotes, <, >, / or =.`,
212+
{
213+
start: attr.start + attr.name.indexOf(`[`),
214+
end: attr.start + attr.name.length
215+
}
216+
)
217+
}
218+
})
203219
}
204220

205221
if (isForbiddenTag(element) && !isServerRendering()) {

test/unit/modules/compiler/parser.spec.js

+17
Original file line numberDiff line numberDiff line change
@@ -550,6 +550,23 @@ describe('parser', () => {
550550
expect(ast.props).toEqual([{ name: 'id', value: 'foo', dynamic: true }])
551551
})
552552

553+
// This only works for string templates.
554+
// In-DOM templates will be malformed before Vue can parse it.
555+
describe('parse and warn invalid dynamic arguments', () => {
556+
[
557+
`<div v-bind:['foo' + bar]="baz"/>`,
558+
`<div :['foo' + bar]="baz"/>`,
559+
`<div @['foo' + bar]="baz"/>`,
560+
`<foo #['foo' + bar]="baz"/>`,
561+
`<div :['foo' + bar].some.mod="baz"/>`
562+
].forEach(template => {
563+
it(template, () => {
564+
const ast = parse(template, baseOptions)
565+
expect(`Invalid dynamic argument expression`).toHaveBeenWarned()
566+
})
567+
})
568+
})
569+
553570
// #6887
554571
it('special case static attribute that must be props', () => {
555572
const ast = parse('<video muted></video>', baseOptions)

0 commit comments

Comments
 (0)