Skip to content

Commit 360272b

Browse files
committed
fix(ssr/reactivity): fix composition api behavior in SSR
fix #12615
1 parent 94ccca2 commit 360272b

File tree

4 files changed

+265
-59
lines changed

4 files changed

+265
-59
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,192 @@
1+
// @vitest-environment node
2+
3+
import Vue from 'vue'
4+
import {
5+
reactive,
6+
ref,
7+
isReactive,
8+
shallowRef,
9+
isRef,
10+
set,
11+
nextTick,
12+
getCurrentInstance
13+
} from 'v3'
14+
import { createRenderer } from '../src'
15+
16+
describe('SSR Reactive', () => {
17+
beforeEach(() => {
18+
// force SSR env
19+
global.process.env.VUE_ENV = 'server'
20+
})
21+
22+
it('should not affect non reactive APIs', () => {
23+
expect(typeof window).toBe('undefined')
24+
expect((Vue.observable({}) as any).__ob__).toBeUndefined()
25+
})
26+
27+
it('reactive behavior should be consistent in SSR', () => {
28+
const obj = reactive({
29+
foo: ref(1),
30+
bar: {
31+
baz: ref(2)
32+
},
33+
arr: [{ foo: ref(3) }]
34+
})
35+
expect(isReactive(obj)).toBe(true)
36+
expect(obj.foo).toBe(1)
37+
38+
expect(isReactive(obj.bar)).toBe(true)
39+
expect(obj.bar.baz).toBe(2)
40+
41+
expect(isReactive(obj.arr)).toBe(true)
42+
expect(isReactive(obj.arr[0])).toBe(true)
43+
expect(obj.arr[0].foo).toBe(3)
44+
})
45+
46+
it('ref value', () => {
47+
const r = ref({})
48+
expect(isReactive(r.value)).toBe(true)
49+
})
50+
51+
it('should render', async () => {
52+
const app = new Vue({
53+
setup() {
54+
return {
55+
count: ref(42)
56+
}
57+
},
58+
render(this: any, h) {
59+
return h('div', this.count)
60+
}
61+
})
62+
63+
const serverRenderer = createRenderer()
64+
const html = await serverRenderer.renderToString(app)
65+
expect(html).toBe('<div data-server-rendered="true">42</div>')
66+
})
67+
68+
it('reactive + isReactive', () => {
69+
const state = reactive({})
70+
expect(isReactive(state)).toBe(true)
71+
})
72+
73+
it('shallowRef + isRef', () => {
74+
const state = shallowRef({})
75+
expect(isRef(state)).toBe(true)
76+
})
77+
78+
it('should work on objects sets with set()', () => {
79+
const state = ref<any>({})
80+
81+
set(state.value, 'a', {})
82+
expect(isReactive(state.value.a)).toBe(true)
83+
84+
set(state.value, 'a', {})
85+
expect(isReactive(state.value.a)).toBe(true)
86+
})
87+
88+
it('should work on arrays sets with set()', () => {
89+
const state = ref<any>([])
90+
91+
set(state.value, 1, {})
92+
expect(isReactive(state.value[1])).toBe(true)
93+
94+
set(state.value, 1, {})
95+
expect(isReactive(state.value[1])).toBe(true)
96+
})
97+
98+
// #550
99+
it('props should work with set', async done => {
100+
let props: any
101+
102+
const app = new Vue({
103+
render(this: any, h) {
104+
return h('child', { attrs: { msg: this.msg } })
105+
},
106+
setup() {
107+
return { msg: ref('hello') }
108+
},
109+
components: {
110+
child: {
111+
render(this: any, h: any) {
112+
return h('span', this.data.msg)
113+
},
114+
props: ['msg'],
115+
setup(_props) {
116+
props = _props
117+
118+
return { data: _props }
119+
}
120+
}
121+
}
122+
})
123+
124+
const serverRenderer = createRenderer()
125+
const html = await serverRenderer.renderToString(app)
126+
127+
expect(html).toBe('<span data-server-rendered="true">hello</span>')
128+
129+
expect(props.bar).toBeUndefined()
130+
set(props, 'bar', 'bar')
131+
expect(props.bar).toBe('bar')
132+
133+
done()
134+
})
135+
136+
// #721
137+
it('should behave correctly', () => {
138+
const state = ref({ old: ref(false) })
139+
set(state.value, 'new', ref(true))
140+
// console.log(process.server, 'state.value', JSON.stringify(state.value))
141+
142+
expect(state.value).toMatchObject({
143+
old: false,
144+
new: true
145+
})
146+
})
147+
148+
// #721
149+
it('should behave correctly for the nested ref in the object', () => {
150+
const state = { old: ref(false) }
151+
set(state, 'new', ref(true))
152+
expect(JSON.stringify(state)).toBe(
153+
'{"old":{"value":false},"new":{"value":true}}'
154+
)
155+
})
156+
157+
// #721
158+
it('should behave correctly for ref of object', () => {
159+
const state = ref({ old: ref(false) })
160+
set(state.value, 'new', ref(true))
161+
expect(JSON.stringify(state.value)).toBe('{"old":false,"new":true}')
162+
})
163+
164+
it('ssr should not RangeError: Maximum call stack size exceeded', async () => {
165+
new Vue({
166+
setup() {
167+
// @ts-expect-error
168+
const app = getCurrentInstance().proxy
169+
let mockNt: any = []
170+
mockNt.__ob__ = {}
171+
const test = reactive({
172+
app,
173+
mockNt
174+
})
175+
return {
176+
test
177+
}
178+
}
179+
})
180+
await nextTick()
181+
expect(
182+
`"RangeError: Maximum call stack size exceeded"`
183+
).not.toHaveBeenWarned()
184+
})
185+
186+
it('should work on objects sets with set()', () => {
187+
const state = ref<any>({})
188+
set(state.value, 'a', {})
189+
190+
expect(isReactive(state.value.a)).toBe(true)
191+
})
192+
})

0 commit comments

Comments
 (0)