1use crate::{ConstDefault, PhantomData};
21
22#[derive(Clone, Debug, Copy)]
38pub struct Base<
39 const RADIX: usize, const LUT: bool, const PAD: bool, const CASE: bool, CODE, > {
45 _code: PhantomData<CODE>,
46}
47
48pub type Base16 = Base<16, false, false, false, Rfc4648>;
54
55pub type Base32 = Base<32, true, false, false, Rfc4648>;
60
61pub type Base32Padded = Base<32, true, true, false, Rfc4648>;
65
66pub type Base32Crockford = Base<32, true, false, true, Crockford>;
70
71pub type Base32Hex = Base<32, true, false, false, Rfc4648Hex>;
75
76pub type Base64 = Base<64, true, false, false, Rfc4648>;
83
84pub type Base64Padded = Base<64, true, true, false, Rfc4648>;
88
89#[derive(Clone, Copy)]
96pub struct Rfc4648;
97
98#[derive(Clone, Copy)]
100pub struct Rfc4648Hex;
101
102#[derive(Clone, Copy)]
104pub struct Crockford;
105
106crate::sf! {
117 impl<const LUT: bool, const PAD: bool, const CASE: bool, CODE>
119 Base<16, LUT, PAD, CASE, CODE> {
120 const ALPHABET: [u8; 16] = *b"0123456789ABCDEF";
121 const fn remap(byte: u8) -> Option<u8> { Some(byte) } }
122 impl<const PAD: bool, const CASE: bool> Base<16, false, PAD, CASE, Rfc4648> {
124 methods!(16, false, 4, Self); } impl<const PAD: bool, const CASE: bool>
126 Base<16, true, PAD, CASE, Rfc4648> { build_lut!(16, Self::remap, Self::ALPHABET);
128 methods!(16, true, 4, Self); }
129 #[cfg(feature = "alloc")]
137 impl<const PAD: bool, const CASE: bool>
138 Base<16, false, PAD, CASE, Rfc4648> { methods!(@alloc 4, Self); }
139 #[cfg(feature = "alloc")]
140 impl<const PAD: bool, const CASE: bool>
141 Base<16, true, PAD, CASE, Rfc4648> { methods!(@alloc 4, Self); }
142
143 impl<const LUT: bool, const PAD: bool, const CASE: bool>
145 Base<32, LUT, PAD, CASE, Rfc4648> { const ALPHABET: [u8; 32] = *b"ABCDEFGHIJKLMNOPQRSTUVWXYZ234567"; const fn remap(byte: u8) -> Option<u8> { Some(byte) } }
148 impl<const PAD: bool, const CASE: bool>
149 Base<32, false, PAD, CASE, Rfc4648> { methods!(32, false, 5, Self); }
151 impl<const PAD: bool, const CASE: bool>
152 Base<32, true, PAD, CASE, Rfc4648> { build_lut!(32, Self::remap, Self::ALPHABET);
154 methods!(32, true, 5, Self); }
155 #[cfg(feature = "alloc")]
156 impl<const PAD: bool, const CASE: bool>
157 Base<32, false, PAD, CASE, Rfc4648> { methods!(@alloc 5, Self); }
158 #[cfg(feature = "alloc")]
159 impl<const PAD: bool, const CASE: bool>
160 Base<32, true, PAD, CASE, Rfc4648> { methods!(@alloc 5, Self); }
161
162 impl<const LUT: bool, const PAD: bool, const CASE: bool>
163 Base<32, LUT, PAD, CASE, Crockford> { const ALPHABET: [u8; 32] = *b"ABCDEFGHIJKLMNOPQRSTUVWXYZ234567"; const fn remap(byte: u8) -> Option<u8> {
166 match byte { b'O' => Some(b'0'), b'I' | b'L' => Some(b'1'), _ => Some(byte) } } }
167 impl<const PAD: bool, const CASE: bool>
168 Base<32, false, PAD, CASE, Crockford> { methods!(32, false, 5, Self); }
170 impl<const PAD: bool, const CASE: bool>
171 Base<32, true, PAD, CASE, Crockford> { build_lut!(32, Self::remap, Self::ALPHABET);
173 methods!(32, true, 5, Self); }
174 impl<const LUT: bool, const PAD: bool, const CASE: bool>
175 Base<32, LUT, PAD, CASE, Rfc4648Hex> { const ALPHABET: [u8; 32] = *b"0123456789ABCDEFGHIJKLMNOPQRSTUV"; const fn remap(byte: u8) -> Option<u8> { Some(byte) } }
178 impl<const PAD: bool, const CASE: bool>
179 Base<32, false, PAD, CASE, Rfc4648Hex> { methods!(32, false, 5, Self); }
181 impl<const PAD: bool, const CASE: bool>
182 Base<32, true, PAD, CASE, Rfc4648Hex> { build_lut!(32, Self::remap, Self::ALPHABET);
184 methods!(32, true, 5, Self); }
185
186 impl<const LUT: bool, const PAD: bool, const CASE: bool>
199 Base<64, LUT, PAD, CASE, Rfc4648> { const ALPHABET: [u8; 64] = *b"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
202 const fn remap(byte: u8) -> Option<u8> { Some(byte) } }
203 impl<const PAD: bool, const CASE: bool>
204 Base<64, false, PAD, CASE, Rfc4648> { methods!(64, false, 6, Self); }
206 impl<const PAD: bool, const CASE: bool>
207 Base<64, true, PAD, CASE, Rfc4648> { build_lut!(64, Self::remap, Self::ALPHABET);
209 methods!(64, true, 6, Self); }
210 #[cfg(feature = "alloc")]
211 impl<const PAD: bool, const CASE: bool>
212 Base<64, false, PAD, CASE, Rfc4648> { methods!(@alloc 6, Self); }
213 #[cfg(feature = "alloc")]
214 impl<const PAD: bool, const CASE: bool>
215 Base<64, true, PAD, CASE, Rfc4648> { methods!(@alloc 6, Self); }
216
217 }
226
227#[rustfmt::skip]
229impl<const RADIX: usize, const LUT: bool, const PAD: bool, const CASE: bool, CODE>
230 Base<RADIX, LUT, PAD, CASE, CODE> {
231 pub const fn new() -> Self {
233 Self { _code: PhantomData }
234 }
235 pub const fn with_radix<const NEW_RADIX: usize>(self)
237 -> Base<NEW_RADIX, LUT, PAD, CASE, CODE> { Base { _code: PhantomData }
238 }
239 pub const fn with_lut<const NEW_LUT: bool>(self)
241 -> Base<RADIX, NEW_LUT, PAD, CASE, CODE> { Base { _code: PhantomData }
242 }
243 pub const fn with_pad<const NEW_PAD: bool>(self)
245 -> Base<RADIX, LUT, NEW_PAD, CASE, CODE> { Base { _code: PhantomData }
246 }
247 pub const fn with_case<const NEW_CASE: bool>(self)
249 -> Base<RADIX, LUT, PAD, NEW_CASE, CODE> { Base { _code: PhantomData }
250 }
251 pub const fn with_encoding<NewCode>(self)
253 -> Base<RADIX, LUT, PAD, CASE, NewCode> { Base { _code: PhantomData }
254 }
255}
256
257#[rustfmt::skip]
260impl<
261 const RADIX: usize, const LUT: bool, const PAD: bool, const CASE: bool, CODE,
262 > ConstDefault for Base<RADIX, LUT, PAD, CASE, CODE>
263{
264 const DEFAULT: Self = Self::new();
265}
266#[rustfmt::skip]
267impl<
268 const RADIX: usize, const LUT: bool, const PAD: bool, const CASE: bool, CODE,
269 > Default for Base<RADIX, LUT, PAD, CASE, CODE>
270{
271 fn default() -> Self { Self::new() }
272}
273
274macro_rules! build_lut {
277 ($radix:expr, $remap_fn:expr, $alphabet:expr) => {
278 const LUT_TABLE: [u8; 256] = {
279 let mut table = [255; 256]; let mut i = 0;
281 while i < $radix {
282 let base_char = $alphabet[i];
283 if let Some(mapped) = $remap_fn(base_char) {
285 table[mapped as usize] = i as u8;
286 }
287 table[base_char as usize] = i as u8;
289 if CASE {
290 table[base_char.to_ascii_lowercase() as usize] = i as u8;
291 }
292 i += 1;
293 }
294 table
295 };
296 };
297}
298use build_lut;
299
300macro_rules! methods {
301 ( $radix:expr, true, $chunk_bits:expr, $Self:ident) => {
303 const fn decode_byte(byte: u8) -> Option<u8> {
304 let decoded = $Self::LUT_TABLE[byte as usize];
305 if decoded == 255 { None } else { Some(decoded) }
306 }
307 methods![@non_alloc $radix, $chunk_bits, $Self];
308 };
309 ( $radix:expr, false, $chunk_bits:expr, $Self:ident) => {
311 const fn decode_byte(byte: u8) -> Option<u8> {
312 let mut i = 0;
313 while i < $radix {
314 let mapped = $Self::remap($Self::ALPHABET[i]);
315 if mapped.is_none() {
316 i += 1;
317 continue;
318 } if byte == mapped.unwrap()
320 || (CASE && byte == mapped.unwrap().to_ascii_lowercase()) {
321 return Some(i as u8);
322 }
323 i += 1;
324 }
325 None
326 }
327 methods![@non_alloc $radix, $chunk_bits, $Self];
328 };
329 (@non_alloc $radix:expr, $chunk_bits:expr, $Self:ident) => {
330 pub const fn encoded_len(input_len: usize) -> usize {
332 (input_len * 8).div_ceil($chunk_bits)
333 }
334 pub const fn encoded_len_padded(input_len: usize) -> usize {
336 let base_len = (input_len * 8).div_ceil($chunk_bits);
337 if PAD {
338 base_len.div_ceil(8) * 8 } else {
340 base_len
341 }
342 }
343 pub const fn decoded_len(input_len: usize) -> usize { (input_len * $chunk_bits) / 8 }
345 pub const fn decoded_len_stripped(input: &[u8]) -> usize {
349 let mut length = input.len();
350 while length > 0 && input[length - 1] == b'=' {
351 length -= 1;
352 }
353 (length * $chunk_bits) / 8
354 }
355
356 pub const fn decode_from_slice(input: &[u8], output: &mut [u8]) -> Option<usize> {
365 let (mut buffer, mut bits_left, mut index, mut i): (u32, u8, usize, usize) = (0,0,0,0);
366 while i < input.len() {
367 let byte = input[i];
368 if PAD && byte == b'=' { break; } let Some(value) = $Self::decode_byte(byte) else { return None };
370 buffer = (buffer << $chunk_bits) | value as u32;
371 bits_left += $chunk_bits;
372 i += 1;
373 while bits_left >= 8 {
374 output[index] = (buffer >> (bits_left - 8)) as u8;
375 bits_left -= 8;
376 index += 1;
377 }
378 }
379 Some(index)
380 }
381
382 pub const fn encode_to_slice(input: &[u8], output: &mut [u8]) -> usize {
386 let (mut buffer, mut bits_left, mut index, mut i): (u32, u8, usize, usize) = (0,0,0,0);
387 while i < input.len() {
388 buffer = (buffer << 8) | input[i] as u32;
389 bits_left += 8;
390 i += 1;
391 while bits_left >= $chunk_bits as u8 {
392 let char_index = ((buffer >> (bits_left - $chunk_bits as u8))
393 & ((1 << $chunk_bits) - 1)) as usize;
394 output[index] = $Self::ALPHABET[char_index];
395 bits_left -= $chunk_bits as u8;
396 index += 1;
397 }
398 }
399 if bits_left > 0 {
400 let char_index = ((buffer << ($chunk_bits as u8 - bits_left))
401 & ((1 << $chunk_bits) - 1)) as usize;
402 output[index] = $Self::ALPHABET[char_index];
403 index += 1;
404 }
405 if PAD {
406 while index % 8 != 0 {
407 output[index] = b'=';
408 index += 1;
409 }
410 }
411 index
412 }
413 };
414 (@alloc $chunk_bits:expr, $Self:ident) => {
415 #[cfg(feature = "alloc")]
418 #[cfg_attr(feature = "nightly_doc", doc(cfg(feature = "alloc")))]
419 pub fn decode(input: &[u8]) -> Option<$crate::Vec<u8>> {
420 let mut output = $crate::Vec::with_capacity(Self::decoded_len(input.len()));
421 let (mut buffer, mut bits_left) = (0, 0);
422 for &byte in input {
423 if PAD && byte == b'=' { break; } let value = Self::decode_byte(byte)?;
425 buffer = (buffer << $chunk_bits) | value as u32;
426 bits_left += $chunk_bits;
427 while bits_left >= 8 {
428 output.push((buffer >> (bits_left - 8)) as u8);
429 bits_left -= 8;
430 }
431 }
432 Some(output)
433 }
434 #[cfg(feature = "alloc")]
436 #[cfg_attr(feature = "nightly_doc", doc(cfg(feature = "alloc")))]
437 pub fn encode(input: &[u8]) -> $crate::String {
438 let mut output = $crate::String::with_capacity($Self::encoded_len(input.len()));
439 let (mut buffer, mut bits_left) = (0, 0);
440 for &byte in input {
441 buffer = (buffer << 8) | byte as u32;
442 bits_left += 8;
443 while bits_left >= $chunk_bits {
444 let index = ((buffer >> (bits_left - $chunk_bits))
445 & ((1 << $chunk_bits) - 1)) as usize;
446 output.push($Self::ALPHABET[index as usize] as char);
447 bits_left -= $chunk_bits;
448 }
449 }
450 if bits_left > 0 {
451 let index = ((buffer << ($chunk_bits - bits_left))
452 & ((1 << $chunk_bits) - 1)) as usize;
453 output.push($Self::ALPHABET[index as usize] as char);
454 }
455 if PAD {
456 while output.len() % (8 * $chunk_bits / 8) != 0 {
457 output.push('=');
458 }
459 }
460 output
461 }
462 };
463}
464use methods;
465
466#[cfg(test)]
467mod tests_no_std {
468 use super::*;
469
470 #[test]
471 fn base_reconstructors() {
472 let base32 = Base32::new();
473 let _base64 = base32.with_radix::<64>().with_pad::<true>();
474 }
475
476 #[test]
479 fn base16_rfc4648_lut() {
480 pub type Base16 = Base<16, true, false, true, Rfc4648>; let mut encoded_buf = [0u8; 22];
482 let encoded_len = Base16::encode_to_slice(b"hello world", &mut encoded_buf);
483 let encoded = &encoded_buf[..encoded_len];
484 assert_eq![encoded, b"68656C6C6F20776F726C64"];
485 let mut decoded_buf = [0u8; 11];
488 let decoded_len = Base16::decode_from_slice(encoded, &mut decoded_buf).unwrap();
489 let decoded = &decoded_buf[..decoded_len];
490 assert_eq![decoded, b"hello world"];
491 }
493 #[test]
494 fn base16_rfc4648_nolut() {
495 let mut encoded_buf = [0u8; 22];
497 let encoded_len = Base16::encode_to_slice(b"hello world", &mut encoded_buf);
498 let encoded = &encoded_buf[..encoded_len];
499 assert_eq![encoded, b"68656C6C6F20776F726C64"];
500 let mut decoded_buf = [0u8; 11];
501 let decoded_len = Base16::decode_from_slice(encoded, &mut decoded_buf).unwrap();
502 let decoded = &decoded_buf[..decoded_len];
503 assert_eq![decoded, b"hello world"];
504 }
505
506 #[test]
509 fn base32_rfc4648_lut() {
510 let mut encoded_buf = [0u8; 18];
511 let encoded_len = Base32::encode_to_slice(b"hello world", &mut encoded_buf);
512 let encoded = &encoded_buf[..encoded_len];
513 assert_eq![encoded, b"NBSWY3DPEB3W64TMMQ"];
514 let mut decoded_buf = [0u8; 11];
515 let decoded_len = Base32::decode_from_slice(encoded, &mut decoded_buf).unwrap();
516 let decoded = &decoded_buf[..decoded_len];
517 assert_eq![decoded, b"hello world"];
518 }
519 #[test]
520 fn base32_rfc4648_nolut() {
521 pub type Base32 = Base<32, false, false, false, Rfc4648>;
522 let mut encoded_buf = [0u8; 18];
523 let encoded_len = Base32::encode_to_slice(b"hello world", &mut encoded_buf);
524 let encoded = &encoded_buf[..encoded_len];
525 assert_eq![encoded, b"NBSWY3DPEB3W64TMMQ"];
526 let mut decoded_buf = [0u8; 11];
527 let decoded_len = Base32::decode_from_slice(encoded, &mut decoded_buf).unwrap();
528 let decoded = &decoded_buf[..decoded_len];
529 assert_eq![decoded, b"hello world"];
530 }
531
532 #[test]
533 fn base32_rfc4648_padded() {
534 pub type Base32 = Base<32, true, true, false, Rfc4648>; const ENC_REFERENCE: &[u8] = b"NBSWY3DPEB3W64TMMQ======";
536 const DEC_LEN_PAD: usize = Base32::decoded_len_stripped(ENC_REFERENCE);
537
538 let mut encoded_buf = [0u8; Base32::encoded_len_padded(b"hello world".len())];
539 let encoded_len = Base32::encode_to_slice(b"hello world", &mut encoded_buf);
540 let encoded = &encoded_buf[..encoded_len];
541 assert_eq![encoded, ENC_REFERENCE];
542
543 let mut decoded_buf = [0u8; DEC_LEN_PAD];
544 let decoded_len = Base32::decode_from_slice(encoded, &mut decoded_buf).unwrap();
545 let decoded = &decoded_buf[..decoded_len];
546 assert_eq![decoded, b"hello world"];
547 }
548
549 #[test]
550 fn base32_rfc4648_case_sensitive() {
551 pub type Base32 = Base<32, true, false, false, Rfc4648>; let encoded_lower = b"nbswy3dpeb3w64tmmq"; let mut decoded_buf = [0u8; 11];
554 let decoded_len = Base32::decode_from_slice(encoded_lower, &mut decoded_buf);
555 assert_eq![decoded_len, None]; let encoded_lower = b"NBSWY3DPEB3W64TMMQ"; let mut decoded_buf = [0u8; 11];
559 let decoded_len = Base32::decode_from_slice(encoded_lower, &mut decoded_buf);
560 assert_eq![decoded_len, Some(11)];
561 let decoded = &decoded_buf[..decoded_len.unwrap()];
562 assert_eq![decoded, b"hello world"];
563 }
564 #[test]
565 fn base32_crockford_case_insensitive() {
566 pub type Base32 = Base<32, true, false, true, Crockford>; let encoded = b"NBSWY3DPEB3W64TMMQ"; let mut decoded_buf = [0u8; 11];
569 let decoded_len = Base32::decode_from_slice(encoded, &mut decoded_buf).unwrap();
570 let decoded = &decoded_buf[..decoded_len];
571 assert_eq![decoded, b"hello world"];
572
573 let encoded_lower = b"nbswy3dpeb3w64tmmq"; let decoded_len = Base32::decode_from_slice(encoded_lower, &mut decoded_buf).unwrap();
575 let decoded = &decoded_buf[..decoded_len];
576 assert_eq![decoded, b"hello world"];
577 }
578
579 #[test]
580 fn base32hex_rfc4648() {
581 pub type Base32H = Base<32, true, false, false, Rfc4648Hex>;
582 let mut encoded_buf = [0u8; 18];
583 let encoded_len = Base32H::encode_to_slice(b"hello world", &mut encoded_buf);
584 let encoded = &encoded_buf[..encoded_len];
585 assert_eq![encoded, b"D1IMOR3F41RMUSJCCG"];
586 let mut decoded_buf = [0u8; 11];
587 let decoded_len = Base32H::decode_from_slice(encoded, &mut decoded_buf).unwrap();
588 let decoded = &decoded_buf[..decoded_len];
589 assert_eq![decoded, b"hello world"];
590 }
591
592 #[test]
595 fn base64_rfc4648_lut() {
596 let mut encoded_buf = [0u8; 18];
597 let encoded_len = Base64::encode_to_slice(b"hello world", &mut encoded_buf);
598 let encoded = &encoded_buf[..encoded_len];
599 assert_eq![encoded, b"aGVsbG8gd29ybGQ"];
600 let mut decoded_buf = [0u8; 11];
601 let decoded_len = Base64::decode_from_slice(encoded, &mut decoded_buf).unwrap();
602 let decoded = &decoded_buf[..decoded_len];
603 assert_eq![decoded, b"hello world"];
604 }
606 #[test]
607 fn base64_rfc4648_nolut() {
608 pub type Base64 = Base<64, false, false, false, Rfc4648>;
609 let mut encoded_buf = [0u8; 18];
610 let encoded_len = Base64::encode_to_slice(b"hello world", &mut encoded_buf);
611 let encoded = &encoded_buf[..encoded_len];
612 assert_eq![encoded, b"aGVsbG8gd29ybGQ"];
613 let mut decoded_buf = [0u8; 11];
614 let decoded_len = Base64::decode_from_slice(encoded, &mut decoded_buf).unwrap();
615 let decoded = &decoded_buf[..decoded_len];
616 assert_eq![decoded, b"hello world"];
617 }
618
619 #[test]
620 fn base64_rfc4648_padded() {
621 pub type Base64 = Base<64, true, true, false, Rfc4648>; const ENC_REFERENCE: &[u8] = b"aGVsbG8gd29ybGQ=";
623 const DEC_LEN_PAD: usize = Base64::decoded_len_stripped(ENC_REFERENCE);
624
625 let mut encoded_buf = [0u8; Base64::encoded_len_padded(b"hello world".len())];
626 let encoded_len = Base64::encode_to_slice(b"hello world", &mut encoded_buf);
627 let encoded = &encoded_buf[..encoded_len];
628 assert_eq![encoded, ENC_REFERENCE];
629
630 let mut decoded_buf = [0u8; DEC_LEN_PAD];
631 let decoded_len = Base64::decode_from_slice(encoded, &mut decoded_buf).unwrap();
632 let decoded = &decoded_buf[..decoded_len];
633 assert_eq![decoded, b"hello world"];
634 }
635}
636
637#[cfg(test)]
638#[cfg(feature = "alloc")]
639mod tests_alloc {
640 use super::*;
641 #[test]
642 fn base32_rfc4648_lut() {
643 let encoded = Base32::encode(b"hello world");
644 assert_eq![encoded.as_bytes(), b"NBSWY3DPEB3W64TMMQ"];
645 let decoded = Base32::decode(encoded.as_bytes()).unwrap();
646 assert_eq![&decoded, b"hello world"];
647 }
648 #[test]
649 fn base32_rfc4648_nolut() {
650 pub type Base32 = Base<32, false, false, false, Rfc4648>;
651 let encoded = Base32::encode(b"hello world");
652 assert_eq![encoded.as_bytes(), b"NBSWY3DPEB3W64TMMQ"];
653 let decoded = Base32::decode(encoded.as_bytes()).unwrap();
654 assert_eq![&decoded, b"hello world"];
655 }
656}