use ffi;
#[cfg(not(feature = "std"))]
use prelude::*;
use libc::c_ulonglong;
use super::super::box_::curve25519xsalsa20poly1305 as box_;
pub const SEALBYTES: usize = ffi::crypto_box_SEALBYTES as usize;
pub fn seal(m: &[u8], pk: &box_::PublicKey) -> Vec<u8> {
let mut c = vec![0u8; m.len() + SEALBYTES];
unsafe {
ffi::crypto_box_seal(
c.as_mut_ptr(),
m.as_ptr(),
m.len() as c_ulonglong,
pk.0.as_ptr(),
);
}
c
}
pub fn open(c: &[u8], pk: &box_::PublicKey, sk: &box_::SecretKey) -> Result<Vec<u8>, ()> {
if c.len() < SEALBYTES {
return Err(());
}
let mut m = vec![0u8; c.len() - SEALBYTES];
let ret = unsafe {
ffi::crypto_box_seal_open(
m.as_mut_ptr(),
c.as_ptr(),
c.len() as c_ulonglong,
pk.0.as_ptr(),
sk.0.as_ptr(),
)
};
if ret == 0 {
Ok(m)
} else {
Err(())
}
}
#[cfg(test)]
mod test {
use super::super::super::box_::curve25519xsalsa20poly1305 as box_;
use super::*;
#[test]
fn test_seal_open() {
use randombytes::randombytes;
for i in 0..256usize {
let (pk, sk) = box_::gen_keypair();
let m = randombytes(i);
let c = seal(&m, &pk);
let opened = open(&c, &pk, &sk);
assert!(Ok(m) == opened);
}
}
#[test]
fn test_seal_open_tamper() {
use randombytes::randombytes;
for i in 0..32usize {
let (pk, sk) = box_::gen_keypair();
let m = randombytes(i);
let mut c = seal(&m, &pk);
for j in 0..c.len() {
c[j] ^= 0x20;
assert!(Err(()) == open(&c, &pk, &sk));
c[j] ^= 0x20;
}
}
}
}