Skip to content

Commit a967e42

Browse files
committed
fix(encoding): differentiate keys and values in query
1 parent 11acb3d commit a967e42

6 files changed

+71
-34
lines changed

__tests__/encoding.spec.ts

+22-10
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,8 @@
11
import {
22
encodeHash,
33
encodeParam,
4-
encodeQueryProperty,
4+
encodeQueryKey,
5+
encodeQueryValue,
56
// decode,
67
} from '../src/encoding'
78

@@ -58,8 +59,16 @@ describe('Encoding', () => {
5859

5960
describe('query params', () => {
6061
const safePerSpec = "!$'*+,:;@[]_|?/{}^()`"
61-
const toEncode = ' "<>#&='
62-
const encodedToEncode = toEncode
62+
const toEncodeForKey = ' "<>#&='
63+
const toEncodeForValue = ' "<>#&'
64+
const encodedToEncodeForKey = toEncodeForKey
65+
.split('')
66+
.map(c => {
67+
const hex = c.charCodeAt(0).toString(16).toUpperCase()
68+
return '%' + (hex.length > 1 ? hex : '0' + hex)
69+
})
70+
.join('')
71+
const encodedToEncodeForValue = toEncodeForValue
6372
.split('')
6473
.map(c => {
6574
const hex = c.charCodeAt(0).toString(16).toUpperCase()
@@ -68,25 +77,28 @@ describe('Encoding', () => {
6877
.join('')
6978

7079
it('does not encode safe chars', () => {
71-
expect(encodeQueryProperty(unreservedSet)).toBe(unreservedSet)
80+
expect(encodeQueryValue(unreservedSet)).toBe(unreservedSet)
81+
expect(encodeQueryKey(unreservedSet)).toBe(unreservedSet)
7282
})
7383

7484
it('encodes non-ascii', () => {
75-
expect(encodeQueryProperty('é')).toBe('%C3%A9')
85+
expect(encodeQueryValue('é')).toBe('%C3%A9')
86+
expect(encodeQueryKey('é')).toBe('%C3%A9')
7687
})
7788

7889
it('encodes non-printable ascii', () => {
79-
expect(encodeQueryProperty(nonPrintableASCII)).toBe(
80-
encodedNonPrintableASCII
81-
)
90+
expect(encodeQueryValue(nonPrintableASCII)).toBe(encodedNonPrintableASCII)
91+
expect(encodeQueryKey(nonPrintableASCII)).toBe(encodedNonPrintableASCII)
8292
})
8393

8494
it('does not encode a safe set', () => {
85-
expect(encodeQueryProperty(safePerSpec)).toBe(safePerSpec)
95+
expect(encodeQueryValue(safePerSpec)).toBe(safePerSpec)
96+
expect(encodeQueryKey(safePerSpec)).toBe(safePerSpec)
8697
})
8798

8899
it('encodes a specific charset', () => {
89-
expect(encodeQueryProperty(toEncode)).toBe(encodedToEncode)
100+
expect(encodeQueryKey(toEncodeForKey)).toBe(encodedToEncodeForKey)
101+
expect(encodeQueryValue(toEncodeForValue)).toBe(encodedToEncodeForValue)
90102
})
91103
})
92104

__tests__/parseQuery.spec.ts

+7-1
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,13 @@ describe('parseQuery', () => {
3636
})
3737
})
3838

39-
it('decodes empty values as null', () => {
39+
it('allows = inside values', () => {
40+
expect(parseQuery('e=c=a')).toEqual({
41+
e: 'c=a',
42+
})
43+
})
44+
45+
it('parses empty values as null', () => {
4046
expect(parseQuery('e&b&c=a')).toEqual({
4147
e: null,
4248
b: null,

__tests__/stringifyQuery.spec.ts

+8
Original file line numberDiff line numberDiff line change
@@ -39,4 +39,12 @@ describe('stringifyQuery', () => {
3939
it('encodes values in arrays', () => {
4040
expect(stringifyQuery({ e: ['%', 'a'], b: 'c' })).toEqual('e=%25&e=a&b=c')
4141
})
42+
43+
it('encodes = in key', () => {
44+
expect(stringifyQuery({ '=': 'a' })).toEqual('%3D=a')
45+
})
46+
47+
it('keeps = in value', () => {
48+
expect(stringifyQuery({ a: '=' })).toEqual('a==')
49+
})
4250
})

__tests__/urlEncoding.spec.ts

+13-9
Original file line numberDiff line numberDiff line change
@@ -96,9 +96,10 @@ describe('URL Encoding', () => {
9696
it('calls encodeQueryProperty with query', async () => {
9797
const router = createRouter()
9898
await router.push({ name: 'home', query: { p: 'foo' } })
99-
expect(encoding.encodeQueryProperty).toHaveBeenCalledTimes(2)
100-
expect(encoding.encodeQueryProperty).toHaveBeenNthCalledWith(1, 'p')
101-
expect(encoding.encodeQueryProperty).toHaveBeenNthCalledWith(2, 'foo')
99+
expect(encoding.encodeQueryValue).toHaveBeenCalledTimes(1)
100+
expect(encoding.encodeQueryKey).toHaveBeenCalledTimes(1)
101+
expect(encoding.encodeQueryKey).toHaveBeenNthCalledWith(1, 'p')
102+
expect(encoding.encodeQueryValue).toHaveBeenNthCalledWith(1, 'foo')
102103
})
103104

104105
it('calls decode with query', async () => {
@@ -113,21 +114,24 @@ describe('URL Encoding', () => {
113114
it('calls encodeQueryProperty with arrays in query', async () => {
114115
const router = createRouter()
115116
await router.push({ name: 'home', query: { p: ['foo', 'bar'] } })
116-
expect(encoding.encodeQueryProperty).toHaveBeenCalledTimes(3)
117-
expect(encoding.encodeQueryProperty).toHaveBeenNthCalledWith(1, 'p')
118-
expect(encoding.encodeQueryProperty).toHaveBeenNthCalledWith(2, 'foo')
119-
expect(encoding.encodeQueryProperty).toHaveBeenNthCalledWith(3, 'bar')
117+
expect(encoding.encodeQueryValue).toHaveBeenCalledTimes(2)
118+
expect(encoding.encodeQueryKey).toHaveBeenCalledTimes(1)
119+
expect(encoding.encodeQueryKey).toHaveBeenNthCalledWith(1, 'p')
120+
expect(encoding.encodeQueryValue).toHaveBeenNthCalledWith(1, 'foo')
121+
expect(encoding.encodeQueryValue).toHaveBeenNthCalledWith(2, 'bar')
120122
})
121123

122124
it('keeps decoded values in query', async () => {
123125
// @ts-ignore: override to make the difference
124126
encoding.decode = () => 'd'
125127
// @ts-ignore
126-
encoding.encodeQueryProperty = () => 'e'
128+
encoding.encodeQueryValue = () => 'ev'
129+
// @ts-ignore
130+
encoding.encodeQueryKey = () => 'ek'
127131
const router = createRouter()
128132
await router.push({ name: 'home', query: { p: '%' } })
129133
expect(router.currentRoute.value).toMatchObject({
130-
fullPath: '/?e=e',
134+
fullPath: '/?ek=ev',
131135
query: { p: '%' },
132136
})
133137
})

src/encoding.ts

+11-3
Original file line numberDiff line numberDiff line change
@@ -61,23 +61,31 @@ export function encodeHash(text: string): string {
6161
}
6262

6363
/**
64-
* Encode characters that need to be encoded query keys and values on the query
64+
* Encode characters that need to be encoded query values on the query
6565
* section of the URL.
6666
*
6767
* @param text - string to encode
6868
* @returns encoded string
6969
*/
70-
export function encodeQueryProperty(text: string | number): string {
70+
export function encodeQueryValue(text: string | number): string {
7171
return commonEncode(text)
7272
.replace(HASH_RE, '%23')
7373
.replace(AMPERSAND_RE, '%26')
74-
.replace(EQUAL_RE, '%3D')
7574
.replace(ENC_BACKTICK_RE, '`')
7675
.replace(ENC_CURLY_OPEN_RE, '{')
7776
.replace(ENC_CURLY_CLOSE_RE, '}')
7877
.replace(ENC_CARET_RE, '^')
7978
}
8079

80+
/**
81+
* Like `encodeQueryValue` but also encodes the `=` character.
82+
*
83+
* @param text - string to encode
84+
*/
85+
export function encodeQueryKey(text: string | number): string {
86+
return encodeQueryValue(text).replace(EQUAL_RE, '%3D')
87+
}
88+
8189
/**
8290
* Encode characters that need to be encoded on the path section of the URL.
8391
*

src/query.ts

+10-11
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import { decode, encodeQueryProperty } from './encoding'
1+
import { decode, encodeQueryKey, encodeQueryValue } from './encoding'
22

33
/**
44
* Possible values in normalized {@link LocationQuery}
@@ -50,13 +50,12 @@ export function parseQuery(search: string): LocationQuery {
5050
const hasLeadingIM = search[0] === '?'
5151
const searchParams = (hasLeadingIM ? search.slice(1) : search).split('&')
5252
for (let i = 0; i < searchParams.length; ++i) {
53-
let [key, rawValue] = searchParams[i].split('=') as [
54-
string,
55-
string | undefined
56-
]
57-
key = decode(key)
58-
// avoid decoding null
59-
let value = rawValue == null ? null : decode(rawValue)
53+
const searchParam = searchParams[i]
54+
// allow the = character
55+
let eqPos = searchParam.indexOf('=')
56+
let key = decode(eqPos < 0 ? searchParam : searchParam.slice(0, eqPos))
57+
let value = eqPos < 0 ? null : decode(searchParam.slice(eqPos + 1))
58+
6059
if (key in query) {
6160
// an extra variable for ts types
6261
let currentValue = query[key]
@@ -85,16 +84,16 @@ export function stringifyQuery(query: LocationQueryRaw): string {
8584
for (let key in query) {
8685
if (search.length) search += '&'
8786
const value = query[key]
88-
key = encodeQueryProperty(key)
87+
key = encodeQueryKey(key)
8988
if (value == null) {
9089
// only null adds the value
9190
if (value !== undefined) search += key
9291
continue
9392
}
9493
// keep null values
9594
let values: LocationQueryValueRaw[] = Array.isArray(value)
96-
? value.map(v => v && encodeQueryProperty(v))
97-
: [value && encodeQueryProperty(value)]
95+
? value.map(v => v && encodeQueryValue(v))
96+
: [value && encodeQueryValue(value)]
9897

9998
for (let i = 0; i < values.length; i++) {
10099
// only append & with i > 0

0 commit comments

Comments
 (0)