Skip to content

Commit 5983f36

Browse files
fix(html): escape html attribute (#18067)
Co-authored-by: 翠 / green <green@sapphi.red>
1 parent d81dc59 commit 5983f36

File tree

3 files changed

+24
-1
lines changed

3 files changed

+24
-1
lines changed

packages/vite/src/node/plugins/html.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ import MagicString from 'magic-string'
1010
import colors from 'picocolors'
1111
import type { DefaultTreeAdapterMap, ParserError, Token } from 'parse5'
1212
import { stripLiteral } from 'strip-literal'
13+
import escapeHtml from 'escape-html'
1314
import type { Plugin } from '../plugin'
1415
import type { ViteDevServer } from '../server'
1516
import {
@@ -1510,7 +1511,7 @@ function serializeAttrs(attrs: HtmlTagDescriptor['attrs']): string {
15101511
if (typeof attrs[key] === 'boolean') {
15111512
res += attrs[key] ? ` ${key}` : ``
15121513
} else {
1513-
res += ` ${key}=${JSON.stringify(attrs[key])}`
1514+
res += ` ${key}="${escapeHtml(attrs[key])}"`
15141515
}
15151516
}
15161517
return res

playground/html/__tests__/html.spec.ts

+5
Original file line numberDiff line numberDiff line change
@@ -471,3 +471,8 @@ test('html fallback works non browser accept header', async () => {
471471
).status,
472472
).toBe(200)
473473
})
474+
475+
test('escape html attribute', async () => {
476+
const el = await page.$('.unescape-div')
477+
expect(el).toBeNull()
478+
})

playground/html/vite.config.js

+17
Original file line numberDiff line numberDiff line change
@@ -214,5 +214,22 @@ ${
214214
]
215215
},
216216
},
217+
{
218+
name: 'escape-html-attribute',
219+
transformIndexHtml: {
220+
order: 'post',
221+
handler() {
222+
return [
223+
{
224+
tag: 'link',
225+
attrs: {
226+
href: `"><div class=unescape-div>extra content</div>`,
227+
},
228+
injectTo: 'body',
229+
},
230+
]
231+
},
232+
},
233+
},
217234
],
218235
})

0 commit comments

Comments
 (0)