Skip to content

Commit c5eb354

Browse files
authored
Use Base64Url in WebEncoders (#56959)
1 parent 62ece1b commit c5eb354

File tree

1 file changed

+51
-1
lines changed

1 file changed

+51
-1
lines changed

src/Shared/WebEncoders/WebEncoders.cs

+51-1
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@
55
using System;
66
#if NETCOREAPP
77
using System.Buffers;
8+
using System.Buffers.Text;
89
#endif
910
using System.Diagnostics;
1011
using System.Globalization;
@@ -66,6 +67,23 @@ public static byte[] Base64UrlDecode(string input, int offset, int count)
6667
return Array.Empty<byte>();
6768
}
6869

70+
#if NET9_0_OR_GREATER
71+
// Legacy behavior of Base64UrlDecode supports either Base64 or Base64Url input.
72+
// If it doesn't have + or /, it can be treated as Base64Url.
73+
ReadOnlySpan<char> inputSpan = input.AsSpan(offset, count);
74+
if (!inputSpan.ContainsAny('+', '/'))
75+
{
76+
return Base64Url.DecodeFromChars(inputSpan);
77+
}
78+
79+
// Otherwise, maintain the legacy behavior of accepting Base64 input. Input that
80+
// contained both +/ and -_ is neither Base64 nor Base64Url and is considered invalid.
81+
if (offset == 0 && count == input.Length)
82+
{
83+
return Convert.FromBase64String(input);
84+
}
85+
#endif
86+
6987
// Create array large enough for the Base64 characters, not just shorter Base64-URL-encoded form.
7088
var buffer = new char[GetArraySizeRequiredToDecode(count)];
7189

@@ -104,6 +122,23 @@ public static byte[] Base64UrlDecode(string input, int offset, char[] buffer, in
104122
return Array.Empty<byte>();
105123
}
106124

125+
#if NET9_0_OR_GREATER
126+
// Legacy behavior of Base64UrlDecode supports either Base64 or Base64Url input.
127+
// If it doesn't have + or /, it can be treated as Base64Url.
128+
ReadOnlySpan<char> inputSpan = input.AsSpan(offset, count);
129+
if (!inputSpan.ContainsAny('+', '/'))
130+
{
131+
return Base64Url.DecodeFromChars(inputSpan);
132+
}
133+
134+
// Otherwise, maintain the legacy behavior of accepting Base64 input. Input that
135+
// contained both +/ and -_ is neither Base64 nor Base64Url and is considered invalid.
136+
if (offset == 0 && count == input.Length)
137+
{
138+
return Convert.FromBase64String(input);
139+
}
140+
#endif
141+
107142
// Assumption: input is base64url encoded without padding and contains no whitespace.
108143

109144
var paddingCharsToAdd = GetNumBase64PaddingCharsToAddForDecode(count);
@@ -124,6 +159,13 @@ public static byte[] Base64UrlDecode(string input, int offset, char[] buffer, in
124159

125160
// Copy input into buffer, fixing up '-' -> '+' and '_' -> '/'.
126161
var i = bufferOffset;
162+
#if NET8_0_OR_GREATER
163+
Span<char> bufferSpan = buffer.AsSpan(i, count);
164+
inputSpan.CopyTo(bufferSpan);
165+
bufferSpan.Replace('-', '+');
166+
bufferSpan.Replace('_', '/');
167+
i += count;
168+
#else
127169
for (var j = offset; i - bufferOffset < count; i++, j++)
128170
{
129171
var ch = input[j];
@@ -140,6 +182,7 @@ public static byte[] Base64UrlDecode(string input, int offset, char[] buffer, in
140182
buffer[i] = ch;
141183
}
142184
}
185+
#endif
143186

144187
// Add the padding characters back.
145188
for (; paddingCharsToAdd > 0; i++, paddingCharsToAdd--)
@@ -314,6 +357,9 @@ public static int GetArraySizeRequiredToEncode(int count)
314357
[SkipLocalsInit]
315358
public static string Base64UrlEncode(ReadOnlySpan<byte> input)
316359
{
360+
#if NET9_0_OR_GREATER
361+
return Base64Url.EncodeToString(input);
362+
#else
317363
const int StackAllocThreshold = 128;
318364

319365
if (input.IsEmpty)
@@ -337,6 +383,7 @@ public static string Base64UrlEncode(ReadOnlySpan<byte> input)
337383
}
338384

339385
return base64Url;
386+
#endif
340387
}
341388

342389
#if NET9_0_OR_GREATER
@@ -347,9 +394,11 @@ public static string Base64UrlEncode(ReadOnlySpan<byte> input)
347394
/// <param name="output">The buffer to place the result in.</param>
348395
/// <returns></returns>
349396
public static int Base64UrlEncode(ReadOnlySpan<byte> input, Span<char> output)
397+
{
398+
return Base64Url.EncodeToChars(input, output);
399+
}
350400
#else
351401
private static int Base64UrlEncode(ReadOnlySpan<byte> input, Span<char> output)
352-
#endif
353402
{
354403
Debug.Assert(output.Length >= GetArraySizeRequiredToEncode(input.Length));
355404

@@ -383,6 +432,7 @@ private static int Base64UrlEncode(ReadOnlySpan<byte> input, Span<char> output)
383432

384433
return charsWritten;
385434
}
435+
#endif
386436
#endif
387437

388438
private static int GetNumBase64PaddingCharsToAddForDecode(int inputLength)

0 commit comments

Comments
 (0)