5
5
using System ;
6
6
#if NETCOREAPP
7
7
using System . Buffers ;
8
+ using System . Buffers . Text ;
8
9
#endif
9
10
using System . Diagnostics ;
10
11
using System . Globalization ;
@@ -66,6 +67,23 @@ public static byte[] Base64UrlDecode(string input, int offset, int count)
66
67
return Array . Empty < byte > ( ) ;
67
68
}
68
69
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
+
69
87
// Create array large enough for the Base64 characters, not just shorter Base64-URL-encoded form.
70
88
var buffer = new char [ GetArraySizeRequiredToDecode ( count ) ] ;
71
89
@@ -104,6 +122,23 @@ public static byte[] Base64UrlDecode(string input, int offset, char[] buffer, in
104
122
return Array . Empty < byte > ( ) ;
105
123
}
106
124
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
+
107
142
// Assumption: input is base64url encoded without padding and contains no whitespace.
108
143
109
144
var paddingCharsToAdd = GetNumBase64PaddingCharsToAddForDecode ( count ) ;
@@ -124,6 +159,13 @@ public static byte[] Base64UrlDecode(string input, int offset, char[] buffer, in
124
159
125
160
// Copy input into buffer, fixing up '-' -> '+' and '_' -> '/'.
126
161
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
127
169
for ( var j = offset ; i - bufferOffset < count ; i ++ , j ++ )
128
170
{
129
171
var ch = input [ j ] ;
@@ -140,6 +182,7 @@ public static byte[] Base64UrlDecode(string input, int offset, char[] buffer, in
140
182
buffer [ i ] = ch ;
141
183
}
142
184
}
185
+ #endif
143
186
144
187
// Add the padding characters back.
145
188
for ( ; paddingCharsToAdd > 0 ; i ++ , paddingCharsToAdd -- )
@@ -314,6 +357,9 @@ public static int GetArraySizeRequiredToEncode(int count)
314
357
[ SkipLocalsInit ]
315
358
public static string Base64UrlEncode ( ReadOnlySpan < byte > input )
316
359
{
360
+ #if NET9_0_OR_GREATER
361
+ return Base64Url . EncodeToString ( input ) ;
362
+ #else
317
363
const int StackAllocThreshold = 128 ;
318
364
319
365
if ( input . IsEmpty )
@@ -337,6 +383,7 @@ public static string Base64UrlEncode(ReadOnlySpan<byte> input)
337
383
}
338
384
339
385
return base64Url ;
386
+ #endif
340
387
}
341
388
342
389
#if NET9_0_OR_GREATER
@@ -347,9 +394,11 @@ public static string Base64UrlEncode(ReadOnlySpan<byte> input)
347
394
/// <param name="output">The buffer to place the result in.</param>
348
395
/// <returns></returns>
349
396
public static int Base64UrlEncode ( ReadOnlySpan < byte > input , Span < char > output )
397
+ {
398
+ return Base64Url . EncodeToChars ( input , output ) ;
399
+ }
350
400
#else
351
401
private static int Base64UrlEncode ( ReadOnlySpan < byte > input , Span < char > output )
352
- #endif
353
402
{
354
403
Debug . Assert ( output . Length >= GetArraySizeRequiredToEncode ( input . Length ) ) ;
355
404
@@ -383,6 +432,7 @@ private static int Base64UrlEncode(ReadOnlySpan<byte> input, Span<char> output)
383
432
384
433
return charsWritten ;
385
434
}
435
+ #endif
386
436
#endif
387
437
388
438
private static int GetNumBase64PaddingCharsToAddForDecode ( int inputLength )
0 commit comments