openssh-keys 0.6.5

read and write OpenSSH public keys
Documentation
//! This module provides a struct for writing bytes in the OpenSSH public key format.

use byteorder::{BigEndian, WriteBytesExt};

pub struct Writer {
    data: Vec<u8>,
}

impl Writer {
    pub fn new() -> Writer {
        Writer { data: vec![] }
    }

    pub fn into_vec(self) -> Vec<u8> {
        self.data
    }

    pub fn write_int(&mut self, val: u32) {
        if self.data.write_u32::<BigEndian>(val).is_err() {
            unreachable!()
        };
    }

    pub fn write_bytes(&mut self, mut buf: Vec<u8>) {
        if buf.is_empty() {
            return;
        }
        // The first four bytes represent the length of the encoded data.
        self.write_int(buf.len() as u32);
        // the rest of the bytes are the data itself
        self.data.append(&mut buf);
    }

    // according to RFC 4251, the mpint datatype representation is a big-endian
    // arbitrary-precision integer encoded in two's complement and stored as a
    // string with the minimum possible number of characters.
    // see mpint definition in https://tools.ietf.org/html/rfc4251#section-5
    pub fn write_mpint(&mut self, mut num: Vec<u8>) {
        // If the number is positive then we are required to guarentee that the
        // most significant bit is set to zero if the first bit in the first
        // byte is going to be one.
        if num.first().unwrap_or(&0) & 0x80 != 0 {
            num.insert(0, 0);
        }
        // other than that it's just normal ssh encoding
        self.write_bytes(num)
    }

    pub fn write_string(&mut self, val: &str) {
        self.write_bytes(val.as_bytes().to_vec())
    }
}

#[cfg(test)]
mod tests {
    use super::*;

    #[test]
    fn writer_empty() {
        let w = Writer::new();
        assert_eq!(w.into_vec().len(), 0);
        let mut w = Writer::new();
        w.write_bytes(vec![]);
        assert_eq!(w.into_vec().len(), 0);
        let mut w = Writer::new();
        w.write_mpint(vec![]);
        assert_eq!(w.into_vec().len(), 0);
    }
}