Skip to content

Commit 4d3dd5f

Browse files
committed
fix(query): encode space as +
Close #561
1 parent 0f00915 commit 4d3dd5f

File tree

4 files changed

+42
-12
lines changed

4 files changed

+42
-12
lines changed

__tests__/encoding.spec.ts

+13-3
Original file line numberDiff line numberDiff line change
@@ -58,9 +58,9 @@ describe('Encoding', () => {
5858
})
5959

6060
describe('query params', () => {
61-
const safePerSpec = "!$'*+,:;@[]_|?/{}^()`"
62-
const toEncodeForKey = ' "<>#&='
63-
const toEncodeForValue = ' "<>#&'
61+
const safePerSpec = "!$'*,:;@[]_|?/{}^()`"
62+
const toEncodeForKey = '"<>#&='
63+
const toEncodeForValue = '"<>#&'
6464
const encodedToEncodeForKey = toEncodeForKey
6565
.split('')
6666
.map(c => {
@@ -100,6 +100,16 @@ describe('Encoding', () => {
100100
expect(encodeQueryKey(toEncodeForKey)).toBe(encodedToEncodeForKey)
101101
expect(encodeQueryValue(toEncodeForValue)).toBe(encodedToEncodeForValue)
102102
})
103+
104+
it('encodes space as +', () => {
105+
expect(encodeQueryKey(' ')).toBe('+')
106+
expect(encodeQueryValue(' ')).toBe('+')
107+
})
108+
109+
it('encodes +', () => {
110+
expect(encodeQueryKey('+')).toBe('%2B')
111+
expect(encodeQueryValue('+')).toBe('%2B')
112+
})
103113
})
104114

105115
describe('hash', () => {

__tests__/parseQuery.spec.ts

+12
Original file line numberDiff line numberDiff line change
@@ -65,6 +65,18 @@ describe('parseQuery', () => {
6565
})
6666
})
6767

68+
it('decodes the + as space', () => {
69+
expect(parseQuery('a+b=c+d')).toEqual({
70+
'a b': 'c d',
71+
})
72+
})
73+
74+
it('decodes the encoded + as +', () => {
75+
expect(parseQuery('a%2Bb=c%2Bd')).toEqual({
76+
'a+b': 'c+d',
77+
})
78+
})
79+
6880
// this is for browsers like IE that allow invalid characters
6981
it('keep invalid values as is', () => {
7082
expect(parseQuery('e=%&e=%25')).toEqual({

src/encoding.ts

+14-8
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@ const AMPERSAND_RE = /&/g // %26
2323
const SLASH_RE = /\//g // %2F
2424
const EQUAL_RE = /=/g // %3D
2525
const IM_RE = /\?/g // %3F
26+
const PLUS_RE = /\+/g // %2B
2627
/**
2728
* NOTE: It's not clear to me if we should encode the + symbol in queries, it
2829
* seems to be less flexible than not doing so and I can't find out the legacy
@@ -37,7 +38,6 @@ const IM_RE = /\?/g // %3F
3738
* - https://url.spec.whatwg.org/#urlencoded-parsing
3839
* - https://stackoverflow.com/questions/1634271/url-encoding-the-space-character-or-20
3940
*/
40-
// const PLUS_RE = /\+/g // %3F
4141

4242
const ENC_BRACKET_OPEN_RE = /%5B/g // [
4343
const ENC_BRACKET_CLOSE_RE = /%5D/g // ]
@@ -46,6 +46,7 @@ const ENC_BACKTICK_RE = /%60/g // `
4646
const ENC_CURLY_OPEN_RE = /%7B/g // {
4747
const ENC_PIPE_RE = /%7C/g // |
4848
const ENC_CURLY_CLOSE_RE = /%7D/g // }
49+
const ENC_SPACE_RE = /%20/g // }
4950

5051
/**
5152
* Encode characters that need to be encoded on the path, search and hash
@@ -83,13 +84,18 @@ export function encodeHash(text: string): string {
8384
* @returns encoded string
8485
*/
8586
export function encodeQueryValue(text: string | number): string {
86-
return commonEncode(text)
87-
.replace(HASH_RE, '%23')
88-
.replace(AMPERSAND_RE, '%26')
89-
.replace(ENC_BACKTICK_RE, '`')
90-
.replace(ENC_CURLY_OPEN_RE, '{')
91-
.replace(ENC_CURLY_CLOSE_RE, '}')
92-
.replace(ENC_CARET_RE, '^')
87+
return (
88+
commonEncode(text)
89+
// Encode the space as +, encode the + to differentiate it from the space
90+
.replace(PLUS_RE, '%2B')
91+
.replace(ENC_SPACE_RE, '+')
92+
.replace(HASH_RE, '%23')
93+
.replace(AMPERSAND_RE, '%26')
94+
.replace(ENC_BACKTICK_RE, '`')
95+
.replace(ENC_CURLY_OPEN_RE, '{')
96+
.replace(ENC_CURLY_CLOSE_RE, '}')
97+
.replace(ENC_CARET_RE, '^')
98+
)
9399
}
94100

95101
/**

src/query.ts

+3-1
Original file line numberDiff line numberDiff line change
@@ -50,7 +50,9 @@ 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-
const searchParam = searchParams[i]
53+
// pre decode the + into space
54+
// FIXME: can't import PLUS_RE because it becomes a different regex ???
55+
const searchParam = searchParams[i].replace(/\+/g, ' ')
5456
// allow the = character
5557
let eqPos = searchParam.indexOf('=')
5658
let key = decode(eqPos < 0 ? searchParam : searchParam.slice(0, eqPos))

0 commit comments

Comments
 (0)