Skip to content

Commit 5cb209f

Browse files
authored
feat(guards): allow guards to return a value instead of calling next (#343)
1 parent 3475486 commit 5cb209f

File tree

2 files changed

+127
-2
lines changed

2 files changed

+127
-2
lines changed

__tests__/guards/guardToPromiseFn.spec.ts

+122
Original file line numberDiff line numberDiff line change
@@ -110,4 +110,126 @@ describe('guardToPromiseFn', () => {
110110
expect(err).toBe(error)
111111
}
112112
})
113+
114+
describe('no next argument', () => {
115+
it('rejects if returns false', async () => {
116+
expect.assertions(1)
117+
try {
118+
await guardToPromiseFn((to, from) => false, to, from)()
119+
} catch (err) {
120+
expect(err).toMatchObject({
121+
from,
122+
to,
123+
type: ErrorTypes.NAVIGATION_ABORTED,
124+
})
125+
}
126+
})
127+
128+
it('resolves no value is returned', async () => {
129+
await expect(
130+
guardToPromiseFn((to, from) => {}, to, from)()
131+
).resolves.toEqual(undefined)
132+
})
133+
134+
it('resolves if true is returned', async () => {
135+
await expect(
136+
guardToPromiseFn((to, from) => true, to, from)()
137+
).resolves.toEqual(undefined)
138+
})
139+
140+
it('rejects if false is returned', async () => {
141+
expect.assertions(1)
142+
try {
143+
await guardToPromiseFn((to, from) => false, to, from)()
144+
} catch (err) {
145+
expect(err).toMatchObject({
146+
from,
147+
to,
148+
type: ErrorTypes.NAVIGATION_ABORTED,
149+
})
150+
}
151+
})
152+
153+
it('rejects if async false is returned', async () => {
154+
expect.assertions(1)
155+
try {
156+
await guardToPromiseFn(async (to, from) => false, to, from)()
157+
} catch (err) {
158+
expect(err).toMatchObject({
159+
from,
160+
to,
161+
type: ErrorTypes.NAVIGATION_ABORTED,
162+
})
163+
}
164+
})
165+
166+
it('rejects if a string location is returned', async () => {
167+
expect.assertions(1)
168+
try {
169+
await guardToPromiseFn((to, from) => '/new', to, from)()
170+
} catch (err) {
171+
expect(err).toMatchObject({
172+
from: to,
173+
to: '/new',
174+
type: ErrorTypes.NAVIGATION_GUARD_REDIRECT,
175+
})
176+
}
177+
})
178+
179+
it('rejects if an object location is returned', async () => {
180+
expect.assertions(1)
181+
let redirectTo = { path: '/new' }
182+
try {
183+
await guardToPromiseFn((to, from) => redirectTo, to, from)()
184+
} catch (err) {
185+
expect(err).toMatchObject({
186+
from: to,
187+
to: redirectTo,
188+
type: ErrorTypes.NAVIGATION_GUARD_REDIRECT,
189+
})
190+
}
191+
})
192+
193+
it('rejects if an error is returned', async () => {
194+
expect.assertions(1)
195+
let error = new Error('nope')
196+
try {
197+
await guardToPromiseFn((to, from) => error, to, from)()
198+
} catch (err) {
199+
expect(err).toBe(error)
200+
}
201+
})
202+
203+
it('rejects if guard rejects a Promise', async () => {
204+
expect.assertions(1)
205+
let error = new Error('nope')
206+
try {
207+
await guardToPromiseFn(
208+
async (to, from) => {
209+
throw error
210+
},
211+
to,
212+
from
213+
)()
214+
} catch (err) {
215+
expect(err).toBe(error)
216+
}
217+
})
218+
219+
it('rejects if guard throws an error', async () => {
220+
expect.assertions(1)
221+
let error = new Error('nope')
222+
try {
223+
await guardToPromiseFn(
224+
(to, from) => {
225+
throw error
226+
},
227+
to,
228+
from
229+
)()
230+
} catch (err) {
231+
expect(err).toBe(error)
232+
}
233+
})
234+
})
113235
})

src/navigationGuards.ts

+5-2
Original file line numberDiff line numberDiff line change
@@ -152,14 +152,17 @@ export function guardToPromiseFn(
152152
}
153153

154154
// wrapping with Promise.resolve allows it to work with both async and sync guards
155-
Promise.resolve(
155+
let guardCall = Promise.resolve(
156156
guard.call(
157157
record && record.instances[name!],
158158
to,
159159
from,
160160
__DEV__ ? canOnlyBeCalledOnce(next, to, from) : next
161161
)
162-
).catch(err => reject(err))
162+
)
163+
164+
if (guard.length < 3) guardCall.then(next)
165+
guardCall.catch(err => reject(err))
163166
})
164167
}
165168

0 commit comments

Comments
 (0)