2
2
import {
3
3
createStore ,
4
4
combineReducers ,
5
- __DO_NOT_USE__ActionTypes as ActionTypes
6
- } from '../'
5
+ Reducer ,
6
+ AnyAction ,
7
+ __DO_NOT_USE__ActionTypes as ActionTypes ,
8
+ CombinedState
9
+ } from '..'
7
10
8
11
describe ( 'Utils' , ( ) => {
9
12
describe ( 'combineReducers' , ( ) => {
10
13
it ( 'returns a composite reducer that maps the state keys to given reducers' , ( ) => {
11
14
const reducer = combineReducers ( {
12
- counter : ( state = 0 , action ) =>
15
+ counter : ( state : number = 0 , action ) =>
13
16
action . type === 'increment' ? state + 1 : state ,
14
- stack : ( state = [ ] , action ) =>
17
+ stack : ( state : any [ ] = [ ] , action ) =>
15
18
action . type === 'push' ? [ ...state , action . value ] : state
16
19
} )
17
20
18
- const s1 = reducer ( { } , { type : 'increment' } )
21
+ const s1 = reducer ( undefined , { type : 'increment' } )
19
22
expect ( s1 ) . toEqual ( { counter : 1 , stack : [ ] } )
20
23
const s2 = reducer ( s1 , { type : 'push' , value : 'a' } )
21
24
expect ( s2 ) . toEqual ( { counter : 1 , stack : [ 'a' ] } )
22
25
} )
23
26
24
27
it ( 'ignores all props which are not a function' , ( ) => {
28
+ // we double-cast because these conditions can only happen in a javascript setting
25
29
const reducer = combineReducers ( {
26
- fake : true ,
27
- broken : 'string' ,
28
- another : { nested : 'object' } ,
30
+ fake : ( true as unknown ) as Reducer ,
31
+ broken : ( 'string' as unknown ) as Reducer ,
32
+ another : ( { nested : 'object' } as unknown ) as Reducer ,
29
33
stack : ( state = [ ] ) => state
30
34
} )
31
35
32
- expect ( Object . keys ( reducer ( { } , { type : 'push' } ) ) ) . toEqual ( [ 'stack' ] )
36
+ expect ( Object . keys ( reducer ( undefined , { type : 'push' } ) ) ) . toEqual ( [
37
+ 'stack'
38
+ ] )
33
39
} )
34
40
35
41
it ( 'warns if a reducer prop is undefined' , ( ) => {
@@ -55,7 +61,7 @@ describe('Utils', () => {
55
61
56
62
it ( 'throws an error if a reducer returns undefined handling an action' , ( ) => {
57
63
const reducer = combineReducers ( {
58
- counter ( state = 0 , action ) {
64
+ counter ( state : number = 0 , action ) {
59
65
switch ( action && action . type ) {
60
66
case 'increment' :
61
67
return state + 1
@@ -77,12 +83,14 @@ describe('Utils', () => {
77
83
expect ( ( ) => reducer ( { counter : 0 } , null ) ) . toThrow (
78
84
/ " c o u n t e r " .* a n a c t i o n /
79
85
)
80
- expect ( ( ) => reducer ( { counter : 0 } , { } ) ) . toThrow ( / " c o u n t e r " .* a n a c t i o n / )
86
+ expect ( ( ) =>
87
+ reducer ( { counter : 0 } , ( { } as unknown ) as AnyAction )
88
+ ) . toThrow ( / " c o u n t e r " .* a n a c t i o n / )
81
89
} )
82
90
83
91
it ( 'throws an error on first call if a reducer returns undefined initializing' , ( ) => {
84
92
const reducer = combineReducers ( {
85
- counter ( state , action ) {
93
+ counter ( state : number , action ) {
86
94
switch ( action . type ) {
87
95
case 'increment' :
88
96
return state + 1
@@ -93,7 +101,9 @@ describe('Utils', () => {
93
101
}
94
102
}
95
103
} )
96
- expect ( ( ) => reducer ( { } ) ) . toThrow ( / " c o u n t e r " .* i n i t i a l i z a t i o n / )
104
+ expect ( ( ) => reducer ( undefined , { type : '' } ) ) . toThrow (
105
+ / " c o u n t e r " .* i n i t i a l i z a t i o n /
106
+ )
97
107
} )
98
108
99
109
it ( 'catches error thrown in reducer when initializing and re-throw' , ( ) => {
@@ -102,14 +112,16 @@ describe('Utils', () => {
102
112
throw new Error ( 'Error thrown in reducer' )
103
113
}
104
114
} )
105
- expect ( ( ) => reducer ( { } ) ) . toThrow ( / E r r o r t h r o w n i n r e d u c e r / )
115
+ expect ( ( ) =>
116
+ reducer ( undefined , ( undefined as unknown ) as AnyAction )
117
+ ) . toThrow ( / E r r o r t h r o w n i n r e d u c e r / )
106
118
} )
107
119
108
120
it ( 'allows a symbol to be used as an action type' , ( ) => {
109
121
const increment = Symbol ( 'INCREMENT' )
110
122
111
123
const reducer = combineReducers ( {
112
- counter ( state = 0 , action ) {
124
+ counter ( state : number = 0 , action ) {
113
125
switch ( action . type ) {
114
126
case increment :
115
127
return state + 1
@@ -135,7 +147,7 @@ describe('Utils', () => {
135
147
}
136
148
} )
137
149
138
- const initialState = reducer ( undefined , '@@INIT' )
150
+ const initialState = reducer ( undefined , { type : '@@INIT' } )
139
151
expect ( reducer ( initialState , { type : 'FOO' } ) ) . toBe ( initialState )
140
152
} )
141
153
@@ -144,7 +156,7 @@ describe('Utils', () => {
144
156
child1 ( state = { } ) {
145
157
return state
146
158
} ,
147
- child2 ( state = { count : 0 } , action ) {
159
+ child2 ( state : { count : number } = { count : 0 } , action ) {
148
160
switch ( action . type ) {
149
161
case 'increment' :
150
162
return { count : state . count + 1 }
@@ -157,15 +169,15 @@ describe('Utils', () => {
157
169
}
158
170
} )
159
171
160
- const initialState = reducer ( undefined , '@@INIT' )
172
+ const initialState = reducer ( undefined , { type : '@@INIT' } )
161
173
expect ( reducer ( initialState , { type : 'increment' } ) ) . not . toBe (
162
174
initialState
163
175
)
164
176
} )
165
177
166
178
it ( 'throws an error on first call if a reducer attempts to handle a private action' , ( ) => {
167
179
const reducer = combineReducers ( {
168
- counter ( state , action ) {
180
+ counter ( state : number , action ) {
169
181
switch ( action . type ) {
170
182
case 'increment' :
171
183
return state + 1
@@ -179,7 +191,9 @@ describe('Utils', () => {
179
191
}
180
192
}
181
193
} )
182
- expect ( ( ) => reducer ( ) ) . toThrow ( / " c o u n t e r " .* p r i v a t e / )
194
+ expect ( ( ) =>
195
+ reducer ( undefined , ( undefined as unknown ) as AnyAction )
196
+ ) . toThrow ( / " c o u n t e r " .* p r i v a t e / )
183
197
} )
184
198
185
199
it ( 'warns if no reducers are passed to combineReducers' , ( ) => {
@@ -188,7 +202,7 @@ describe('Utils', () => {
188
202
console . error = spy
189
203
190
204
const reducer = combineReducers ( { } )
191
- reducer ( { } )
205
+ reducer ( undefined , { type : '' } )
192
206
expect ( spy . mock . calls [ 0 ] [ 0 ] ) . toMatch (
193
207
/ S t o r e d o e s n o t h a v e a v a l i d r e d u c e r /
194
208
)
@@ -200,9 +214,17 @@ describe('Utils', () => {
200
214
it ( 'warns if input state does not match reducer shape' , ( ) => {
201
215
const preSpy = console . error
202
216
const spy = jest . fn ( )
217
+ const nullAction = ( undefined as unknown ) as AnyAction
203
218
console . error = spy
204
219
205
- const reducer = combineReducers ( {
220
+ interface ShapeState {
221
+ foo : { bar : number }
222
+ baz : { qux : number }
223
+ }
224
+
225
+ type ShapeMismatchState = CombinedState < ShapeState >
226
+
227
+ const reducer = combineReducers < ShapeState > ( {
206
228
foo ( state = { bar : 1 } ) {
207
229
return state
208
230
} ,
@@ -211,44 +233,51 @@ describe('Utils', () => {
211
233
}
212
234
} )
213
235
214
- reducer ( )
236
+ reducer ( undefined , nullAction )
215
237
expect ( spy . mock . calls . length ) . toBe ( 0 )
216
238
217
- reducer ( { foo : { bar : 2 } } )
239
+ reducer ( ( { foo : { bar : 2 } } as unknown ) as ShapeState , nullAction )
218
240
expect ( spy . mock . calls . length ) . toBe ( 0 )
219
241
220
- reducer ( {
221
- foo : { bar : 2 } ,
222
- baz : { qux : 4 }
223
- } )
242
+ reducer (
243
+ {
244
+ foo : { bar : 2 } ,
245
+ baz : { qux : 4 }
246
+ } ,
247
+ nullAction
248
+ )
224
249
expect ( spy . mock . calls . length ) . toBe ( 0 )
225
250
226
- createStore ( reducer , { bar : 2 } )
251
+ createStore ( reducer , ( { bar : 2 } as unknown ) as ShapeState )
227
252
expect ( spy . mock . calls [ 0 ] [ 0 ] ) . toMatch (
228
253
/ U n e x p e c t e d k e y " b a r " .* c r e a t e S t o r e .* i n s t e a d : " f o o " , " b a z " /
229
254
)
230
255
231
- createStore ( reducer , { bar : 2 , qux : 4 , thud : 5 } )
256
+ createStore ( reducer , ( {
257
+ bar : 2 ,
258
+ qux : 4 ,
259
+ thud : 5
260
+ } as unknown ) as ShapeState )
232
261
expect ( spy . mock . calls [ 1 ] [ 0 ] ) . toMatch (
233
262
/ U n e x p e c t e d k e y s " q u x " , " t h u d " .* c r e a t e S t o r e .* i n s t e a d : " f o o " , " b a z " /
234
263
)
235
264
236
- createStore ( reducer , 1 )
265
+ createStore ( reducer , ( 1 as unknown ) as ShapeState )
237
266
expect ( spy . mock . calls [ 2 ] [ 0 ] ) . toMatch (
238
267
/ c r e a t e S t o r e h a s u n e x p e c t e d t y p e o f " N u m b e r " .* k e y s : " f o o " , " b a z " /
239
268
)
240
269
241
- reducer ( { corge : 2 } )
270
+ reducer ( ( { corge : 2 } as unknown ) as ShapeState , nullAction )
242
271
expect ( spy . mock . calls [ 3 ] [ 0 ] ) . toMatch (
243
272
/ U n e x p e c t e d k e y " c o r g e " .* r e d u c e r .* i n s t e a d : " f o o " , " b a z " /
244
273
)
245
274
246
- reducer ( { fred : 2 , grault : 4 } )
275
+ reducer ( ( { fred : 2 , grault : 4 } as unknown ) as ShapeState , nullAction )
247
276
expect ( spy . mock . calls [ 4 ] [ 0 ] ) . toMatch (
248
277
/ U n e x p e c t e d k e y s " f r e d " , " g r a u l t " .* r e d u c e r .* i n s t e a d : " f o o " , " b a z " /
249
278
)
250
279
251
- reducer ( 1 )
280
+ reducer ( ( 1 as unknown ) as ShapeState , nullAction )
252
281
expect ( spy . mock . calls [ 5 ] [ 0 ] ) . toMatch (
253
282
/ r e d u c e r h a s u n e x p e c t e d t y p e o f " N u m b e r " .* k e y s : " f o o " , " b a z " /
254
283
)
@@ -261,25 +290,32 @@ describe('Utils', () => {
261
290
const preSpy = console . error
262
291
const spy = jest . fn ( )
263
292
console . error = spy
293
+ const nullAction = { type : '' }
264
294
265
295
const foo = ( state = { foo : 1 } ) => state
266
296
const bar = ( state = { bar : 2 } ) => state
267
297
268
298
expect ( spy . mock . calls . length ) . toBe ( 0 )
269
299
300
+ interface WarnState {
301
+ foo : { foo : number }
302
+ bar : { bar : number }
303
+ }
304
+
270
305
const reducer = combineReducers ( { foo, bar } )
271
- const state = { foo : 1 , bar : 2 , qux : 3 }
306
+ const state = ( { foo : 1 , bar : 2 , qux : 3 } as unknown ) as WarnState
307
+ const bazState = ( { ...state , baz : 5 } as unknown ) as WarnState
272
308
273
- reducer ( state , { } )
274
- reducer ( state , { } )
275
- reducer ( state , { } )
276
- reducer ( state , { } )
309
+ reducer ( state , nullAction )
310
+ reducer ( state , nullAction )
311
+ reducer ( state , nullAction )
312
+ reducer ( state , nullAction )
277
313
expect ( spy . mock . calls . length ) . toBe ( 1 )
278
314
279
- reducer ( { ... state , baz : 5 } , { } )
280
- reducer ( { ...state , baz : 5 } , { } )
281
- reducer ( { ...state , baz : 5 } , { } )
282
- reducer ( { ...state , baz : 5 } , { } )
315
+ reducer ( bazState , nullAction )
316
+ reducer ( { ...bazState } , nullAction )
317
+ reducer ( { ...bazState } , nullAction )
318
+ reducer ( { ...bazState } , nullAction )
283
319
expect ( spy . mock . calls . length ) . toBe ( 2 )
284
320
285
321
spy . mockClear ( )
0 commit comments