Skip to content

Commit e3dcc8d

Browse files
committed
feat(warn): warn against infinite redirections
1 parent 3896f15 commit e3dcc8d

File tree

3 files changed

+47
-5
lines changed

3 files changed

+47
-5
lines changed

__tests__/warnings.spec.ts

+20
Original file line numberDiff line numberDiff line change
@@ -187,4 +187,24 @@ describe('warnings', () => {
187187
await expect(router.push({ path: '/foo2' })).resolves.toBe(undefined)
188188
expect(`No match found for location with path "/foo2"`).toHaveBeenWarned()
189189
})
190+
191+
it('warns if next is called with the same location too many times', async () => {
192+
let router = createRouter({
193+
history: createMemoryHistory(),
194+
routes: [
195+
{ path: '/', name: 'a', component },
196+
{ path: '/b', component },
197+
],
198+
})
199+
200+
router.beforeEach(to => {
201+
if (to.path === '/b') return '/b'
202+
return
203+
})
204+
205+
await router.push('/b').catch(() => {})
206+
expect(
207+
'Detected an infinite redirection in a navigation guard when going from "/" to "/b"'
208+
).toHaveBeenWarned()
209+
})
190210
})

src/navigationGuards.ts

-3
Original file line numberDiff line numberDiff line change
@@ -157,9 +157,6 @@ export function guardToPromiseFn(
157157
record && record.instances[name!],
158158
to,
159159
from,
160-
// TODO: could wrap in dev to check if the guard returns before
161-
// calling next with 3 or more arguments. This would help people
162-
// forgetting to remove the `next` argument
163160
__DEV__ ? canOnlyBeCalledOnce(next, to, from) : next
164161
)
165162
)

src/router.ts

+27-2
Original file line numberDiff line numberDiff line change
@@ -490,17 +490,42 @@ export function createRouter(options: RouterOptions): Router {
490490
if (failure) {
491491
if (
492492
isNavigationFailure(failure, ErrorTypes.NAVIGATION_GUARD_REDIRECT)
493-
)
494-
// preserve the original redirectedFrom if any
493+
) {
494+
if (
495+
__DEV__ &&
496+
// we are redirecting to the same location we were already at
497+
isSameRouteLocation(
498+
stringifyQuery,
499+
resolve(failure.to),
500+
toLocation
501+
) &&
502+
// and we have done it a couple of times
503+
redirectedFrom &&
504+
// @ts-ignore
505+
(redirectedFrom._count = redirectedFrom._count
506+
? // @ts-ignore
507+
redirectedFrom._count + 1
508+
: 1) > 10
509+
) {
510+
warn(
511+
`Detected an infinite redirection in a navigation guard when going from "${from.fullPath}" to "${toLocation.fullPath}". Aborting to avoid a Stack Overflow. This will break in production if not fixed.`
512+
)
513+
return Promise.reject(
514+
new Error('Infinite redirect in navigation guard')
515+
)
516+
}
517+
495518
return pushWithRedirect(
496519
// keep options
497520
assign(locationAsObject(failure.to), {
498521
state: data,
499522
force,
500523
replace,
501524
}),
525+
// preserve the original redirectedFrom if any
502526
redirectedFrom || toLocation
503527
)
528+
}
504529
} else {
505530
// if we fail we don't finalize the navigation
506531
failure = finalizeNavigation(

0 commit comments

Comments
 (0)