From 7cd10bc08c72cbce630bbbc569be7ab19703ae74 Mon Sep 17 00:00:00 2001 From: auvred Date: Mon, 26 Jun 2023 20:48:40 +0000 Subject: [PATCH 1/4] fix(compiler-sfc): consistently escape type-only prop names --- .../__snapshots__/defineProps.spec.ts.snap | 45 +++++++++ .../compileScript/defineProps.spec.ts | 99 +++++++++++++++++++ packages/compiler-sfc/src/script/utils.ts | 4 +- 3 files changed, 146 insertions(+), 2 deletions(-) diff --git a/packages/compiler-sfc/__tests__/compileScript/__snapshots__/defineProps.spec.ts.snap b/packages/compiler-sfc/__tests__/compileScript/__snapshots__/defineProps.spec.ts.snap index 62d9bef5d27..9f36e1fadc5 100644 --- a/packages/compiler-sfc/__tests__/compileScript/__snapshots__/defineProps.spec.ts.snap +++ b/packages/compiler-sfc/__tests__/compileScript/__snapshots__/defineProps.spec.ts.snap @@ -52,6 +52,51 @@ const { foo } = __props; +return { } +} + +})" +`; + +exports[`defineProps > should escape names w/ special symbols 1`] = ` +"import { defineComponent as _defineComponent } from 'vue' + +export default /*#__PURE__*/_defineComponent({ + props: { + \\"spa ce\\": { type: null, required: true }, + \\"exclamation!mark\\": { type: null, required: true }, + \\"double\\\\\\"quote\\": { type: null, required: true }, + \\"hash#tag\\": { type: null, required: true }, + \\"dollar$sign\\": { type: null, required: true }, + \\"percentage%sign\\": { type: null, required: true }, + \\"amper&sand\\": { type: null, required: true }, + \\"single'quote\\": { type: null, required: true }, + \\"round(brack)ets\\": { type: null, required: true }, + \\"aste*risk\\": { type: null, required: true }, + \\"pl+us\\": { type: null, required: true }, + \\"com,ma\\": { type: null, required: true }, + \\"do.t\\": { type: null, required: true }, + \\"sla/sh\\": { type: null, required: true }, + \\"co:lon\\": { type: null, required: true }, + \\"semi;colon\\": { type: null, required: true }, + \\"angleets\\": { type: null, required: true }, + \\"equal=sign\\": { type: null, required: true }, + \\"question?mark\\": { type: null, required: true }, + \\"at@sign\\": { type: null, required: true }, + \\"square[brack]ets\\": { type: null, required: true }, + \\"back\\\\\\\\slash\\": { type: null, required: true }, + \\"ca^ret\\": { type: null, required: true }, + \\"back\`tick\\": { type: null, required: true }, + \\"curly{bra}ces\\": { type: null, required: true }, + \\"pi|pe\\": { type: null, required: true }, + \\"til~de\\": { type: null, required: true }, + \\"da-sh\\": { type: null, required: true } + }, + setup(__props: any, { expose: __expose }) { + __expose(); + + + return { } } diff --git a/packages/compiler-sfc/__tests__/compileScript/defineProps.spec.ts b/packages/compiler-sfc/__tests__/compileScript/defineProps.spec.ts index 43f54b0aa1e..a68b7adbd35 100644 --- a/packages/compiler-sfc/__tests__/compileScript/defineProps.spec.ts +++ b/packages/compiler-sfc/__tests__/compileScript/defineProps.spec.ts @@ -608,4 +608,103 @@ const props = defineProps({ foo: String }) }).toThrow(`cannot accept both type and non-type arguments`) }) }) + + test('should escape names w/ special symbols', () => { + const { content, bindings } = compile(` + `) + assertCode(content) + expect(content).toMatch(`"spa ce": { type: null, required: true }`) + expect(content).toMatch( + `"exclamation!mark": { type: null, required: true }` + ) + expect(content).toMatch(`"double\\"quote": { type: null, required: true }`) + expect(content).toMatch(`"hash#tag": { type: null, required: true }`) + expect(content).toMatch(`"dollar$sign": { type: null, required: true }`) + expect(content).toMatch(`"percentage%sign": { type: null, required: true }`) + expect(content).toMatch(`"amper&sand": { type: null, required: true }`) + expect(content).toMatch(`"single'quote": { type: null, required: true }`) + expect(content).toMatch(`"round(brack)ets": { type: null, required: true }`) + expect(content).toMatch(`"aste*risk": { type: null, required: true }`) + expect(content).toMatch(`"pl+us": { type: null, required: true }`) + expect(content).toMatch(`"com,ma": { type: null, required: true }`) + expect(content).toMatch(`"do.t": { type: null, required: true }`) + expect(content).toMatch(`"sla/sh": { type: null, required: true }`) + expect(content).toMatch(`"co:lon": { type: null, required: true }`) + expect(content).toMatch(`"semi;colon": { type: null, required: true }`) + expect(content).toMatch(`"angleets": { type: null, required: true }`) + expect(content).toMatch(`"equal=sign": { type: null, required: true }`) + expect(content).toMatch(`"question?mark": { type: null, required: true }`) + expect(content).toMatch(`"at@sign": { type: null, required: true }`) + expect(content).toMatch( + `"square[brack]ets": { type: null, required: true }` + ) + expect(content).toMatch(`"back\\\\slash": { type: null, required: true }`) + expect(content).toMatch(`"ca^ret": { type: null, required: true }`) + expect(content).toMatch(`"back\`tick": { type: null, required: true }`) + expect(content).toMatch(`"curly{bra}ces": { type: null, required: true }`) + expect(content).toMatch(`"pi|pe": { type: null, required: true }`) + expect(content).toMatch(`"til~de": { type: null, required: true }`) + expect(content).toMatch(`"da-sh": { type: null, required: true }`) + expect(bindings).toStrictEqual({ + 'spa ce': BindingTypes.PROPS, + 'exclamation!mark': BindingTypes.PROPS, + 'double"quote': BindingTypes.PROPS, + 'hash#tag': BindingTypes.PROPS, + dollar$sign: BindingTypes.PROPS, + 'percentage%sign': BindingTypes.PROPS, + 'amper&sand': BindingTypes.PROPS, + "single'quote": BindingTypes.PROPS, + 'round(brack)ets': BindingTypes.PROPS, + 'aste*risk': BindingTypes.PROPS, + 'pl+us': BindingTypes.PROPS, + 'com,ma': BindingTypes.PROPS, + 'do.t': BindingTypes.PROPS, + 'sla/sh': BindingTypes.PROPS, + 'co:lon': BindingTypes.PROPS, + 'semi;colon': BindingTypes.PROPS, + 'angleets': BindingTypes.PROPS, + 'equal=sign': BindingTypes.PROPS, + 'question?mark': BindingTypes.PROPS, + 'at@sign': BindingTypes.PROPS, + 'square[brack]ets': BindingTypes.PROPS, + 'back\\slash': BindingTypes.PROPS, + 'ca^ret': BindingTypes.PROPS, + 'back`tick': BindingTypes.PROPS, + 'curly{bra}ces': BindingTypes.PROPS, + 'pi|pe': BindingTypes.PROPS, + 'til~de': BindingTypes.PROPS, + 'da-sh': BindingTypes.PROPS + }) + }) }) diff --git a/packages/compiler-sfc/src/script/utils.ts b/packages/compiler-sfc/src/script/utils.ts index 42c4718e3a8..26f1d56f7d9 100644 --- a/packages/compiler-sfc/src/script/utils.ts +++ b/packages/compiler-sfc/src/script/utils.ts @@ -113,8 +113,8 @@ export const joinPaths = (path.posix || path).join * key may contain symbols * e.g. onUpdate:modelValue -> "onUpdate:modelValue" */ -export const escapeSymbolsRE = /[ !"#$%&'()*+,./:;<=>?@[\\\]^`{|}~]/g +export const escapeSymbolsRE = /[ !"#$%&'()*+,./:;<=>?@[\\\]^`{|}~\-]/g export function getEscapedKey(key: string) { - return escapeSymbolsRE.test(key) ? JSON.stringify(key) : key + return key.search(escapeSymbolsRE) !== -1 ? JSON.stringify(key) : key } From 83d99a42110bf36c5a81458bbd446631e00e4fc6 Mon Sep 17 00:00:00 2001 From: auvred Date: Tue, 27 Jun 2023 10:18:21 +0000 Subject: [PATCH 2/4] fix: don't escape dash for css vars names --- packages/compiler-sfc/src/style/cssVars.ts | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/packages/compiler-sfc/src/style/cssVars.ts b/packages/compiler-sfc/src/style/cssVars.ts index c86dbf2a5b9..e7b889ebac8 100644 --- a/packages/compiler-sfc/src/style/cssVars.ts +++ b/packages/compiler-sfc/src/style/cssVars.ts @@ -32,7 +32,10 @@ function genVarName(id: string, raw: string, isProd: boolean): string { return hash(id + raw) } else { // escape ASCII Punctuation & Symbols - return `${id}-${raw.replace(escapeSymbolsRE, s => `\\${s}`)}` + return `${id}-${raw.replace(escapeSymbolsRE, s => + // dash is allowed symbol in variable name + s === '-' ? s : `\\${s}` + )}` } } From d24da587b06321b46d54c9759e91d721fadd8a14 Mon Sep 17 00:00:00 2001 From: auvred Date: Sun, 13 Aug 2023 13:55:28 +0000 Subject: [PATCH 3/4] chore: separate regexps for css var names and prop names --- packages/compiler-sfc/src/script/utils.ts | 4 ++-- packages/compiler-sfc/src/style/cssVars.ts | 8 +++----- 2 files changed, 5 insertions(+), 7 deletions(-) diff --git a/packages/compiler-sfc/src/script/utils.ts b/packages/compiler-sfc/src/script/utils.ts index 26f1d56f7d9..89424daf350 100644 --- a/packages/compiler-sfc/src/script/utils.ts +++ b/packages/compiler-sfc/src/script/utils.ts @@ -113,8 +113,8 @@ export const joinPaths = (path.posix || path).join * key may contain symbols * e.g. onUpdate:modelValue -> "onUpdate:modelValue" */ -export const escapeSymbolsRE = /[ !"#$%&'()*+,./:;<=>?@[\\\]^`{|}~\-]/g +export const escapeSymbolsRE = /[ !"#$%&'()*+,./:;<=>?@[\\\]^`{|}~\-]/ export function getEscapedKey(key: string) { - return key.search(escapeSymbolsRE) !== -1 ? JSON.stringify(key) : key + return escapeSymbolsRE.test(key) ? JSON.stringify(key) : key } diff --git a/packages/compiler-sfc/src/style/cssVars.ts b/packages/compiler-sfc/src/style/cssVars.ts index e7b889ebac8..abb5114bca3 100644 --- a/packages/compiler-sfc/src/style/cssVars.ts +++ b/packages/compiler-sfc/src/style/cssVars.ts @@ -8,7 +8,6 @@ import { BindingMetadata } from '@vue/compiler-dom' import { SFCDescriptor } from '../parse' -import { escapeSymbolsRE } from '../script/utils' import { PluginCreator } from 'postcss' import hash from 'hash-sum' @@ -27,15 +26,14 @@ export function genCssVarsFromList( .join(',\n ')}\n}` } +const cssVarNameEscapeSymbolsRE = /[ !"#$%&'()*+,./:;<=>?@[\\\]^`{|}~]/g + function genVarName(id: string, raw: string, isProd: boolean): string { if (isProd) { return hash(id + raw) } else { // escape ASCII Punctuation & Symbols - return `${id}-${raw.replace(escapeSymbolsRE, s => - // dash is allowed symbol in variable name - s === '-' ? s : `\\${s}` - )}` + return `${id}-${raw.replace(cssVarNameEscapeSymbolsRE, s => `\\${s}`)}` } } From ab30a8dc4c3bdfd721e87a56920e39f36172892b Mon Sep 17 00:00:00 2001 From: auvred Date: Mon, 14 Aug 2023 11:38:07 +0000 Subject: [PATCH 4/4] chore: move regexps to a single file --- packages/compiler-sfc/src/script/defineProps.ts | 6 +++--- packages/compiler-sfc/src/script/utils.ts | 12 +++++++++--- packages/compiler-sfc/src/style/cssVars.ts | 5 ++--- 3 files changed, 14 insertions(+), 9 deletions(-) diff --git a/packages/compiler-sfc/src/script/defineProps.ts b/packages/compiler-sfc/src/script/defineProps.ts index 5004e314da1..9de15b92b76 100644 --- a/packages/compiler-sfc/src/script/defineProps.ts +++ b/packages/compiler-sfc/src/script/defineProps.ts @@ -17,7 +17,7 @@ import { isCallOf, unwrapTSNode, toRuntimeTypeString, - getEscapedKey + getEscapedPropName } from './utils' import { genModelProps } from './defineModel' import { getObjectOrArrayExpressionKeys } from './analyzeScriptBindings' @@ -135,7 +135,7 @@ export function genRuntimeProps(ctx: ScriptCompileContext): string | undefined { const defaults: string[] = [] for (const key in ctx.propsDestructuredBindings) { const d = genDestructuredDefaultValue(ctx, key) - const finalKey = getEscapedKey(key) + const finalKey = getEscapedPropName(key) if (d) defaults.push( `${finalKey}: ${d.valueString}${ @@ -251,7 +251,7 @@ function genRuntimePropFromType( } } - const finalKey = getEscapedKey(key) + const finalKey = getEscapedPropName(key) if (!ctx.options.isProd) { return `${finalKey}: { ${concatStrings([ `type: ${toRuntimeTypeString(type)}`, diff --git a/packages/compiler-sfc/src/script/utils.ts b/packages/compiler-sfc/src/script/utils.ts index 89424daf350..8afadf63ee4 100644 --- a/packages/compiler-sfc/src/script/utils.ts +++ b/packages/compiler-sfc/src/script/utils.ts @@ -113,8 +113,14 @@ export const joinPaths = (path.posix || path).join * key may contain symbols * e.g. onUpdate:modelValue -> "onUpdate:modelValue" */ -export const escapeSymbolsRE = /[ !"#$%&'()*+,./:;<=>?@[\\\]^`{|}~\-]/ +export const propNameEscapeSymbolsRE = /[ !"#$%&'()*+,./:;<=>?@[\\\]^`{|}~\-]/ -export function getEscapedKey(key: string) { - return escapeSymbolsRE.test(key) ? JSON.stringify(key) : key +export function getEscapedPropName(key: string) { + return propNameEscapeSymbolsRE.test(key) ? JSON.stringify(key) : key +} + +export const cssVarNameEscapeSymbolsRE = /[ !"#$%&'()*+,./:;<=>?@[\\\]^`{|}~]/g + +export function getEscapedCssVarName(key: string) { + return key.replace(cssVarNameEscapeSymbolsRE, s => `\\${s}`) } diff --git a/packages/compiler-sfc/src/style/cssVars.ts b/packages/compiler-sfc/src/style/cssVars.ts index 0d20f2df535..9fe727bc5dc 100644 --- a/packages/compiler-sfc/src/style/cssVars.ts +++ b/packages/compiler-sfc/src/style/cssVars.ts @@ -8,6 +8,7 @@ import { BindingMetadata } from '@vue/compiler-dom' import { SFCDescriptor } from '../parse' +import { getEscapedCssVarName } from '../script/utils' import { PluginCreator } from 'postcss' import hash from 'hash-sum' @@ -26,14 +27,12 @@ export function genCssVarsFromList( .join(',\n ')}\n}` } -const cssVarNameEscapeSymbolsRE = /[ !"#$%&'()*+,./:;<=>?@[\\\]^`{|}~]/g - function genVarName(id: string, raw: string, isProd: boolean): string { if (isProd) { return hash(id + raw) } else { // escape ASCII Punctuation & Symbols - return `${id}-${raw.replace(cssVarNameEscapeSymbolsRE, s => `\\${s}`)}` + return `${id}-${getEscapedCssVarName(raw)}` } }