Skip to content

Commit f73925d

Browse files
committedNov 10, 2022
fix(sfc): ensure <script setup> binding behavior consistency on this between prod and dev
close #6248
1 parent 15e889a commit f73925d

File tree

2 files changed

+37
-24
lines changed

2 files changed

+37
-24
lines changed
 

‎packages/runtime-core/__tests__/componentPublicInstance.spec.ts

+22-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,8 @@ import {
44
getCurrentInstance,
55
nodeOps,
66
createApp,
7-
shallowReadonly
7+
shallowReadonly,
8+
defineComponent
89
} from '@vue/runtime-test'
910
import { ComponentInternalInstance, ComponentOptions } from '../src/component'
1011

@@ -458,4 +459,24 @@ describe('component: proxy', () => {
458459
)} was accessed during render ` + `but is not defined on instance.`
459460
).toHaveBeenWarned()
460461
})
462+
463+
test('should prevent mutating script setup bindings', () => {
464+
const Comp = defineComponent({
465+
render() {},
466+
setup() {
467+
return {
468+
__isScriptSetup: true,
469+
foo: 1
470+
}
471+
},
472+
mounted() {
473+
expect('foo' in this).toBe(false)
474+
try {
475+
this.foo = 123
476+
} catch (e) {}
477+
}
478+
})
479+
render(h(Comp), nodeOps.createElement('div'))
480+
expect(`Cannot mutate <script setup> binding "foo"`).toHaveBeenWarned()
481+
})
461482
})

‎packages/runtime-core/src/componentPublicInstance.ts

+15-23
Original file line numberDiff line numberDiff line change
@@ -270,6 +270,9 @@ export interface ComponentRenderContext {
270270

271271
export const isReservedPrefix = (key: string) => key === '_' || key === '$'
272272

273+
const hasSetupBinding = (state: Data, key: string) =>
274+
state !== EMPTY_OBJ && !state.__isScriptSetup && hasOwn(state, key)
275+
273276
export const PublicInstanceProxyHandlers: ProxyHandler<any> = {
274277
get({ _: instance }: ComponentRenderContext, key: string) {
275278
const { ctx, setupState, data, props, accessCache, type, appContext } =
@@ -280,19 +283,6 @@ export const PublicInstanceProxyHandlers: ProxyHandler<any> = {
280283
return true
281284
}
282285

283-
// prioritize <script setup> bindings during dev.
284-
// this allows even properties that start with _ or $ to be used - so that
285-
// it aligns with the production behavior where the render fn is inlined and
286-
// indeed has access to all declared variables.
287-
if (
288-
__DEV__ &&
289-
setupState !== EMPTY_OBJ &&
290-
setupState.__isScriptSetup &&
291-
hasOwn(setupState, key)
292-
) {
293-
return setupState[key]
294-
}
295-
296286
// data / props / ctx
297287
// This getter gets called for every property access on the render context
298288
// during render and is a major hotspot. The most expensive part of this
@@ -314,7 +304,7 @@ export const PublicInstanceProxyHandlers: ProxyHandler<any> = {
314304
return props![key]
315305
// default: just fallthrough
316306
}
317-
} else if (setupState !== EMPTY_OBJ && hasOwn(setupState, key)) {
307+
} else if (hasSetupBinding(setupState, key)) {
318308
accessCache![key] = AccessTypes.SETUP
319309
return setupState[key]
320310
} else if (data !== EMPTY_OBJ && hasOwn(data, key)) {
@@ -403,26 +393,28 @@ export const PublicInstanceProxyHandlers: ProxyHandler<any> = {
403393
value: any
404394
): boolean {
405395
const { data, setupState, ctx } = instance
406-
if (setupState !== EMPTY_OBJ && hasOwn(setupState, key)) {
396+
if (hasSetupBinding(setupState, key)) {
407397
setupState[key] = value
408398
return true
399+
} else if (
400+
__DEV__ &&
401+
setupState.__isScriptSetup &&
402+
hasOwn(setupState, key)
403+
) {
404+
warn(`Cannot mutate <script setup> binding "${key}" from Options API.`)
405+
return false
409406
} else if (data !== EMPTY_OBJ && hasOwn(data, key)) {
410407
data[key] = value
411408
return true
412409
} else if (hasOwn(instance.props, key)) {
413-
__DEV__ &&
414-
warn(
415-
`Attempting to mutate prop "${key}". Props are readonly.`,
416-
instance
417-
)
410+
__DEV__ && warn(`Attempting to mutate prop "${key}". Props are readonly.`)
418411
return false
419412
}
420413
if (key[0] === '$' && key.slice(1) in instance) {
421414
__DEV__ &&
422415
warn(
423416
`Attempting to mutate public property "${key}". ` +
424-
`Properties starting with $ are reserved and readonly.`,
425-
instance
417+
`Properties starting with $ are reserved and readonly.`
426418
)
427419
return false
428420
} else {
@@ -449,7 +441,7 @@ export const PublicInstanceProxyHandlers: ProxyHandler<any> = {
449441
return (
450442
!!accessCache![key] ||
451443
(data !== EMPTY_OBJ && hasOwn(data, key)) ||
452-
(setupState !== EMPTY_OBJ && hasOwn(setupState, key)) ||
444+
hasSetupBinding(setupState, key) ||
453445
((normalizedProps = propsOptions[0]) && hasOwn(normalizedProps, key)) ||
454446
hasOwn(ctx, key) ||
455447
hasOwn(publicPropertiesMap, key) ||

0 commit comments

Comments
 (0)
Please sign in to comment.