2016-11-04 17:37:42 -04:00
|
|
|
// Copyright 2016 The Rust Project Developers. See the COPYRIGHT
|
|
|
|
// file at the top-level directory of this distribution and at
|
|
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
|
|
|
|
|
|
|
/// Convert unsigned integers into a string representation with some base.
|
|
|
|
/// Bases up to and including 36 can be used for case-insensitive things.
|
|
|
|
|
|
|
|
use std::str;
|
|
|
|
|
2018-01-08 12:30:52 +01:00
|
|
|
pub const MAX_BASE: usize = 64;
|
|
|
|
pub const ALPHANUMERIC_ONLY: usize = 62;
|
|
|
|
pub const CASE_INSENSITIVE: usize = 36;
|
2016-12-12 21:56:52 +03:00
|
|
|
|
2016-11-04 17:37:42 -04:00
|
|
|
const BASE_64: &'static [u8; MAX_BASE as usize] =
|
|
|
|
b"0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ@$";
|
|
|
|
|
|
|
|
#[inline]
|
2018-01-08 12:30:52 +01:00
|
|
|
pub fn push_str(mut n: u128, base: usize, output: &mut String) {
|
2016-11-04 17:37:42 -04:00
|
|
|
debug_assert!(base >= 2 && base <= MAX_BASE);
|
2018-01-08 12:30:52 +01:00
|
|
|
let mut s = [0u8; 128];
|
2016-11-04 17:37:42 -04:00
|
|
|
let mut index = 0;
|
|
|
|
|
2018-01-08 12:30:52 +01:00
|
|
|
let base = base as u128;
|
|
|
|
|
2016-11-04 17:37:42 -04:00
|
|
|
loop {
|
|
|
|
s[index] = BASE_64[(n % base) as usize];
|
|
|
|
index += 1;
|
|
|
|
n /= base;
|
|
|
|
|
|
|
|
if n == 0 {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
&mut s[0..index].reverse();
|
|
|
|
output.push_str(str::from_utf8(&s[0..index]).unwrap());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2018-01-08 12:30:52 +01:00
|
|
|
pub fn encode(n: u128, base: usize) -> String {
|
|
|
|
let mut s = String::new();
|
2016-11-04 17:37:42 -04:00
|
|
|
push_str(n, base, &mut s);
|
|
|
|
s
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_encode() {
|
2018-01-08 12:30:52 +01:00
|
|
|
fn test(n: u128, base: usize) {
|
|
|
|
assert_eq!(Ok(n), u128::from_str_radix(&encode(n, base), base as u32));
|
2016-11-04 17:37:42 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
for base in 2..37 {
|
|
|
|
test(0, base);
|
|
|
|
test(1, base);
|
|
|
|
test(35, base);
|
|
|
|
test(36, base);
|
|
|
|
test(37, base);
|
2018-01-08 12:30:52 +01:00
|
|
|
test(u64::max_value() as u128, base);
|
|
|
|
test(u128::max_value(), base);
|
2016-11-04 17:37:42 -04:00
|
|
|
|
|
|
|
for i in 0 .. 1_000 {
|
|
|
|
test(i * 983, base);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|