libnetkeeper 0.1.0

The netkeeper toolkits write in rust.
Documentation
use std::str;
use std::slice;

use byteorder::{NetworkEndian, NativeEndian, ByteOrder};

use common::dialer::Dialer;
use common::utils::current_timestamp;

#[derive(Debug)]
pub enum Configuration {
    Hainan,
}

#[derive(Debug)]
pub struct SingleNetDialer {
    share_key: String,
    secret_key: String,
    key_table: String,
}

impl SingleNetDialer {
    pub fn new(share_key: &str, secret_key: &str, key_table: &str) -> Self {
        SingleNetDialer {
            share_key: share_key.to_string(),
            secret_key: secret_key.to_string(),
            key_table: key_table.to_string(),
        }
    }

    pub fn encrypt_account(&self, username: &str, timestamp: Option<u32>) -> String {
        let username = username.to_uppercase();
        let time_now = timestamp.unwrap_or_else(current_timestamp);

        let first_hash: u16;
        {
            let mut time_now_bytes = [0u8; 4];
            NetworkEndian::write_u32(&mut time_now_bytes, time_now);
            let mut hash_data: Vec<u8> = Vec::new();
            hash_data.extend_from_slice(&time_now_bytes);
            hash_data.extend(self.share_key.as_bytes());
            hash_data.extend(username.split('@').nth(0).unwrap().as_bytes());
            first_hash = Self::calc_hash(&hash_data)
        }

        let second_hash: u16;
        {
            let mut first_hash_bytes = [0u8; 2];
            NetworkEndian::write_u16(&mut first_hash_bytes, first_hash);
            let mut hash_data: Vec<u8> = Vec::new();
            hash_data.extend_from_slice(&first_hash_bytes);
            hash_data.extend(self.secret_key.as_bytes());
            second_hash = Self::calc_hash(&hash_data);
        }

        let mut scheduled_table: Vec<u8> = Vec::with_capacity(8);
        {
            let time_now_high = (time_now >> 16) as u16;
            let time_now_low = (time_now & 0xFFFF) as u16;

            let mut time_now_high_bytes = [0u8; 2];
            let mut time_now_low_bytes = [0u8; 2];
            let mut first_hash_bytes = [0u8; 2];
            let mut second_hash_bytes = [0u8; 2];

            NetworkEndian::write_u16(&mut time_now_high_bytes, time_now_high);
            NetworkEndian::write_u16(&mut time_now_low_bytes, time_now_low);
            NativeEndian::write_u16(&mut first_hash_bytes, first_hash);
            NativeEndian::write_u16(&mut second_hash_bytes, second_hash);

            scheduled_table.extend_from_slice(&time_now_high_bytes);
            scheduled_table.extend_from_slice(&first_hash_bytes);
            scheduled_table.extend_from_slice(&time_now_low_bytes);
            scheduled_table.extend_from_slice(&second_hash_bytes);
        }

        let mut vectors: [u8; 12] = [0; 12];
        for i in 0..4 {
            let j = 2 * i + 1;
            let k = 3 * i + 1;
            vectors[k - 1] = scheduled_table[j - 1] >> 0x3 & 0x1F;
            vectors[k] = ((scheduled_table[j - 1] & 0x7) << 0x2) |
                (scheduled_table[j] >> 0x6 & 0x3);
            vectors[k + 1] = scheduled_table[j] & 0x3F;
        }

        let key_table_bytes = self.key_table_bytes();
        let pin: Vec<u8> = vectors.iter().map(|c| key_table_bytes[*c as usize]).collect();

        let pin_str = str::from_utf8(&pin).unwrap();
        format!("~LL_{}_{}", pin_str, username)
    }

    fn calc_hash(data: &[u8]) -> u16 {
        let length = data.len();
        let mut summary: u32 = 0;
        let mut data = data;

        if length % 2 != 0 {
            summary = u32::from(data[length - 1]);
            data = &data[0..length - 2];
        }

        let data_shorts: &[u16];
        unsafe {
            data_shorts = slice::from_raw_parts::<u16>((data as *const [u8]) as *const u16,
                                                       length / 2);
        }

        summary = data_shorts.iter().fold(summary, |sum, x| sum + u32::from(*x));
        if summary & 0xFFFF_0000 != 0 {
            summary = ((summary >> 0x10) + summary) & 0xFFFF;
        }

        !summary as u16
    }

    fn key_table_bytes(&self) -> &[u8] {
        self.key_table.as_bytes()
    }
}

impl Configuration {
    pub fn share_key(&self) -> &'static str {
        match *self {
            Configuration::Hainan => "hngx01",
        }
    }

    pub fn secret_key(&self) -> &'static str {
        match *self {
            Configuration::Hainan => "000c29270712",
        }
    }

    pub fn key_table(&self) -> &'static str {
        match *self {
            Configuration::Hainan => {
                "abcdefghijklmnopqrstuvwxyz1234567890ZYXWVUTSRQPONMLKJIHGFEDCBA:_"
            }
        }
    }
}

impl Dialer for SingleNetDialer {
    type C = Configuration;

    fn load_from_config(config: Self::C) -> Self {
        SingleNetDialer::new(config.share_key(), config.secret_key(), config.key_table())
    }
}

#[test]
fn test_hash_key() {
    let str1 = "123456".to_string();
    let str2 = "1234567".to_string();
    let hash1 = SingleNetDialer::calc_hash(str1.as_bytes());
    let hash2 = SingleNetDialer::calc_hash(str2.as_bytes());
    assert_eq!(hash1, 25446u16);
    assert_eq!(hash2, 25391u16);
}