Skip to content

Commit eae833e

Browse files
committed
feat: resolve relative paths
1 parent fea6382 commit eae833e

File tree

5 files changed

+139
-12
lines changed

5 files changed

+139
-12
lines changed

__tests__/location.spec.ts

+72
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,78 @@ describe('parseURL', () => {
1919
})
2020
})
2121

22+
it('works with partial path with no query', () => {
23+
expect(parseURL('foo#hash')).toEqual({
24+
fullPath: '/foo#hash',
25+
path: '/foo',
26+
hash: '#hash',
27+
query: {},
28+
})
29+
})
30+
31+
it('works with partial path', () => {
32+
expect(parseURL('foo?f=foo#hash')).toEqual({
33+
fullPath: '/foo?f=foo#hash',
34+
path: '/foo',
35+
hash: '#hash',
36+
query: { f: 'foo' },
37+
})
38+
})
39+
40+
it('works with only query', () => {
41+
expect(parseURL('?f=foo')).toEqual({
42+
fullPath: '/?f=foo',
43+
path: '/',
44+
hash: '',
45+
query: { f: 'foo' },
46+
})
47+
})
48+
49+
it('works with only hash', () => {
50+
expect(parseURL('#foo')).toEqual({
51+
fullPath: '/#foo',
52+
path: '/',
53+
hash: '#foo',
54+
query: {},
55+
})
56+
})
57+
58+
it('works with partial path and current location', () => {
59+
expect(parseURL('foo', '/parent/bar')).toEqual({
60+
fullPath: '/parent/foo',
61+
path: '/parent/foo',
62+
hash: '',
63+
query: {},
64+
})
65+
})
66+
67+
it('works with partial path with query and hash and current location', () => {
68+
expect(parseURL('foo?f=foo#hash', '/parent/bar')).toEqual({
69+
fullPath: '/parent/foo?f=foo#hash',
70+
path: '/parent/foo',
71+
hash: '#hash',
72+
query: { f: 'foo' },
73+
})
74+
})
75+
76+
it('works with relative query and current location', () => {
77+
expect(parseURL('?f=foo', '/parent/bar')).toEqual({
78+
fullPath: '/parent/bar?f=foo',
79+
path: '/parent/bar',
80+
hash: '',
81+
query: { f: 'foo' },
82+
})
83+
})
84+
85+
it('works with relative hash and current location', () => {
86+
expect(parseURL('#hash', '/parent/bar')).toEqual({
87+
fullPath: '/parent/bar#hash',
88+
path: '/parent/bar',
89+
hash: '#hash',
90+
query: {},
91+
})
92+
})
93+
2294
it('extracts the query', () => {
2395
expect(parseURL('/foo?a=one&b=two')).toEqual({
2496
fullPath: '/foo?a=one&b=two',

__tests__/matcher/resolve.spec.ts

+21
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@ import {
77
MatcherLocation,
88
} from '../../src/types'
99
import { MatcherLocationNormalizedLoose } from '../utils'
10+
import { mockWarn } from 'jest-mock-warn'
1011

1112
// @ts-ignore
1213
const component: RouteComponent = null
@@ -756,6 +757,26 @@ describe('RouterMatcher.resolve', () => {
756757
})
757758

758759
describe('LocationAsRelative', () => {
760+
mockWarn()
761+
it('warns if a path isn not absolute', () => {
762+
const record = {
763+
path: '/parent',
764+
components,
765+
}
766+
const matcher = createRouterMatcher([record], {})
767+
matcher.resolve(
768+
{ path: 'two' },
769+
{
770+
path: '/parent/one',
771+
name: undefined,
772+
params: {},
773+
matched: [] as any,
774+
meta: {},
775+
}
776+
)
777+
expect('received "two"').toHaveBeenWarned()
778+
})
779+
759780
it('matches with nothing', () => {
760781
const record = { path: '/home', name: 'Home', components }
761782
assertRecordMatch(

src/location.ts

+16-4
Original file line numberDiff line numberDiff line change
@@ -36,13 +36,16 @@ export const removeTrailingSlash = (path: string) =>
3636
*
3737
* @param parseQuery
3838
* @param location - URI to normalize
39+
* @param currentLocation - current absolute location. Allows resolving relative
40+
* paths. Must start with `/`. Defaults to `/`
3941
* @returns a normalized history location
4042
*/
4143
export function parseURL(
4244
parseQuery: (search: string) => LocationQuery,
43-
location: string
45+
location: string,
46+
currentLocation: string = '/'
4447
): LocationNormalized {
45-
let path = '',
48+
let path: string | undefined,
4649
query: LocationQuery = {},
4750
searchString = '',
4851
hash = ''
@@ -68,10 +71,19 @@ export function parseURL(
6871
}
6972

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

7385
return {
74-
fullPath: location,
86+
fullPath: path + (searchString && '?') + searchString + hash,
7587
path,
7688
query,
7789
hash,

src/matcher/index.ts

+12-4
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ import {
1313
comparePathParserScore,
1414
PathParserOptions,
1515
} from './pathParserRanker'
16+
import { warn } from 'vue'
1617

1718
let noop = () => {}
1819

@@ -208,15 +209,22 @@ export function createRouterMatcher(
208209
// throws if cannot be stringified
209210
path = matcher.stringify(params)
210211
} else if ('path' in location) {
211-
matcher = matchers.find(m => m.re.test(location.path))
212-
// matcher should have a value after the loop
213-
214212
// no need to resolve the path with the matcher as it was provided
215213
// this also allows the user to control the encoding
216214
path = location.path
215+
216+
if (__DEV__ && path[0] !== '/') {
217+
warn(
218+
`The Matcher cannot resolve relative paths but received "${path}". Unless you directly called \`matcher.resolve("${path}")\`, this is probably a bug in vue-router. Please open an issue at https://new-issue.vuejs.org/?repo=vuejs/vue-router-next.`
219+
)
220+
}
221+
222+
matcher = matchers.find(m => m.re.test(path))
223+
// matcher should have a value after the loop
224+
217225
if (matcher) {
218226
// TODO: dev warning of unused params if provided
219-
params = matcher.parse(location.path)!
227+
params = matcher.parse(path)!
220228
name = matcher.record.name
221229
}
222230
// location is a relative path

src/router.ts

+18-4
Original file line numberDiff line numberDiff line change
@@ -207,13 +207,17 @@ export function createRouter({
207207
}
208208

209209
function resolve(
210-
location: RouteLocationRaw,
211-
currentLocation?: RouteLocationNormalizedLoaded
210+
location: Readonly<RouteLocationRaw>,
211+
currentLocation?: Readonly<RouteLocationNormalizedLoaded>
212212
): RouteLocation & { href: string } {
213213
// const objectLocation = routerLocationAsObject(location)
214214
currentLocation = currentLocation || currentRoute.value
215215
if (typeof location === 'string') {
216-
let locationNormalized = parseURL(parseQuery, location)
216+
let locationNormalized = parseURL(
217+
parseQuery,
218+
location,
219+
currentLocation.path
220+
)
217221
let matchedRoute = matcher.resolve(
218222
{ path: locationNormalized.path },
219223
currentLocation
@@ -235,6 +239,16 @@ export function createRouter({
235239
}
236240
}
237241

242+
// TODO: dev warning if params and path at the same time
243+
244+
// path could be relative in object as well
245+
if ('path' in location) {
246+
location = {
247+
...location,
248+
path: parseURL(parseQuery, location.path, currentLocation.path).path,
249+
}
250+
}
251+
238252
let matchedRoute: MatcherLocation = // relative or named location, path is ignored
239253
// for same reason TS thinks location.params can be undefined
240254
matcher.resolve(
@@ -473,7 +487,7 @@ export function createRouter({
473487
return runGuardQueue(guards)
474488
})
475489
.then(() => {
476-
// check global guards beforeEach
490+
// check global guards beforeResolve
477491
guards = []
478492
for (const guard of beforeResolveGuards.list()) {
479493
guards.push(guardToPromiseFn(guard, to, from))

0 commit comments

Comments
 (0)