Skip to content

Commit af1deaa

Browse files
committed
feat: resolve simple relative links
Works with links like ../../url but not with ../url/./path
1 parent 1367867 commit af1deaa

File tree

3 files changed

+30
-9
lines changed

3 files changed

+30
-9
lines changed

__tests__/location.spec.ts

+6
Original file line numberDiff line numberDiff line change
@@ -335,6 +335,12 @@ describe('resolveRelativePath', () => {
335335
expect(resolveRelativePath('/add', '/users/posva')).toBe('/add')
336336
})
337337

338+
it('resolves empty path', () => {
339+
expect(resolveRelativePath('', '/users/posva')).toBe('/users/posva')
340+
expect(resolveRelativePath('', '/users')).toBe('/users')
341+
expect(resolveRelativePath('', '/')).toBe('/')
342+
})
343+
338344
it('warns if from path is not absolute', () => {
339345
resolveRelativePath('path', 'other')
340346
resolveRelativePath('path', './other')

__tests__/router.spec.ts

+21
Original file line numberDiff line numberDiff line change
@@ -72,6 +72,7 @@ const routes: RouteRecordRaw[] = [
7272
},
7373
],
7474
},
75+
{ path: '/:pathMatch(.*)', component: components.Home, name: 'catch-all' },
7576
]
7677

7778
async function newRouter(
@@ -331,6 +332,26 @@ describe('Router', () => {
331332
})
332333
})
333334

335+
it('resolves relative locations', async () => {
336+
const { router } = await newRouter()
337+
await router.push('/users/posva')
338+
await router.push('add')
339+
expect(router.currentRoute.value.path).toBe('/users/add')
340+
await router.push('/users/posva')
341+
await router.push('./add')
342+
expect(router.currentRoute.value.path).toBe('/users/add')
343+
})
344+
345+
it('resolves parent relative locations', async () => {
346+
const { router } = await newRouter()
347+
await router.push('/users/posva')
348+
await router.push('../add')
349+
expect(router.currentRoute.value.path).toBe('/add')
350+
await router.push('/users/posva')
351+
await router.push('../../../add')
352+
expect(router.currentRoute.value.path).toBe('/add')
353+
})
354+
334355
describe('Warnings', () => {
335356
it.skip('avoid infinite redirection loops', async () => {
336357
const history = createMemoryHistory()

src/location.ts

+3-9
Original file line numberDiff line numberDiff line change
@@ -72,16 +72,8 @@ export function parseURL(
7272
}
7373

7474
// no search and no query
75-
path = path != null ? path : location
75+
path = resolveRelativePath(path != null ? path : location, currentLocation)
7676
// empty path means a relative query or hash `?foo=f`, `#thing`
77-
if (!path) {
78-
path = currentLocation + path
79-
} else if (path[0] !== '/') {
80-
// relative to current location. Currently we only support simple relative
81-
// but no `..`, `.`, or complex like `../.././..`. We will always leave the
82-
// leading slash so we can safely append path
83-
path = currentLocation.replace(/[^\/]*$/, '') + path
84-
}
8577

8678
return {
8779
fullPath: path + (searchString && '?') + searchString + hash,
@@ -211,6 +203,8 @@ export function resolveRelativePath(to: string, from: string): string {
211203
return to
212204
}
213205

206+
if (!to) return from
207+
214208
const fromSegments = from.split('/')
215209
const toSegments = to.split('/')
216210

0 commit comments

Comments
 (0)