use super::super::box_::curve25519xsalsa20poly1305 as box_;
use crate::ffi;
use libc::c_ulonglong;
pub const SEALBYTES: usize = ffi::crypto_box_SEALBYTES as usize;
pub fn seal(m: &[u8], &box_::PublicKey(ref pk): &box_::PublicKey) -> Vec<u8> {
let mut c = vec![0u8; m.len() + SEALBYTES];
unsafe {
let _todo_use_result = ffi::crypto_box_seal(
c.as_mut_ptr(),
m.as_ptr(),
m.len() as c_ulonglong,
pk.as_ptr(),
);
}
c
}
pub fn open(
c: &[u8],
&box_::PublicKey(ref pk): &box_::PublicKey,
&box_::SecretKey(ref 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.as_ptr(),
sk.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 crate::randombytes::randombytes;
unwrap!(crate::init());
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 crate::randombytes::randombytes;
unwrap!(crate::init());
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;
}
}
}
}