2022-01-03 21:29:04 +00:00
|
|
|
#![allow(dead_code)]
|
2021-11-26 16:50:49 +00:00
|
|
|
#![allow(clippy::absurd_extreme_comparisons)]
|
2022-10-30 23:29:31 +00:00
|
|
|
use super::*;
|
2022-07-10 21:36:50 +00:00
|
|
|
use crate::*;
|
2021-11-22 16:28:30 +00:00
|
|
|
use core::convert::TryInto;
|
2022-05-26 00:56:13 +00:00
|
|
|
use data_encoding::BASE64URL_NOPAD;
|
2021-11-22 16:28:30 +00:00
|
|
|
|
2023-01-29 18:13:50 +00:00
|
|
|
/// Out-of-band receipts are versioned along with crypto versions
|
|
|
|
///
|
|
|
|
/// These are the formats for the on-the-wire serialization performed by this module
|
|
|
|
///
|
|
|
|
/// #[repr(C, packed)]
|
|
|
|
/// struct ReceiptHeader {
|
|
|
|
/// // Size is at least 8 bytes. Depending on the version specified, the size may vary and should be case to the appropriate struct
|
|
|
|
/// magic: [u8; 4], // 0x00: 0x52 0x43 0x50 0x54 ("RCPT")
|
|
|
|
/// version: u8, // 0x04: 0 = ReceiptV0
|
|
|
|
/// reserved: u8, // 0x05: Reserved for future use
|
|
|
|
/// }
|
|
|
|
///
|
|
|
|
/// #[repr(C, packed)]
|
|
|
|
/// struct ReceiptV0 {
|
|
|
|
/// // Size is 106 bytes.
|
|
|
|
/// magic: [u8; 4], // 0x00: 0x52 0x43 0x50 0x54 ("RCPT")
|
|
|
|
/// version: u8, // 0x04: 0 = ReceiptV0
|
|
|
|
/// reserved: u8, // 0x05: Reserved for future use
|
|
|
|
/// size: u16, // 0x06: Total size of the receipt including the extra data and the signature. Maximum size is 1152 bytes.
|
|
|
|
/// nonce: [u8; 24], // 0x08: Randomly chosen bytes that represent a unique receipt. Could be used to encrypt the extra data, but it's not required.
|
|
|
|
/// sender_id: [u8; 32], // 0x20: Node ID of the message source, which is the Ed25519 public key of the sender
|
|
|
|
/// extra_data: [u8; ??], // 0x40: Extra data is appended (arbitrary extra data, not encrypted by receipt itself, maximum size is 1024 bytes)
|
|
|
|
/// signature: [u8; 64], // 0x?? (end-0x40): Ed25519 signature of the entire receipt including header and extra data is appended to the packet
|
|
|
|
/// }
|
2021-11-22 16:28:30 +00:00
|
|
|
|
|
|
|
pub const MAX_RECEIPT_SIZE: usize = 1152;
|
|
|
|
pub const MAX_EXTRA_DATA_SIZE: usize = 1024;
|
|
|
|
pub const MIN_RECEIPT_SIZE: usize = 128;
|
|
|
|
pub const RECEIPT_MAGIC: &[u8; 4] = b"RCPT";
|
|
|
|
pub type ReceiptNonce = [u8; 24];
|
|
|
|
|
2022-05-26 00:56:13 +00:00
|
|
|
pub trait Encodable {
|
|
|
|
fn encode(&self) -> String;
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Encodable for ReceiptNonce {
|
|
|
|
fn encode(&self) -> String {
|
|
|
|
BASE64URL_NOPAD.encode(self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-22 16:28:30 +00:00
|
|
|
#[derive(Debug, Clone, PartialEq, Eq, Default)]
|
|
|
|
pub struct Receipt {
|
|
|
|
version: u8,
|
|
|
|
nonce: ReceiptNonce,
|
|
|
|
sender_id: DHTKey,
|
|
|
|
extra_data: Vec<u8>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Receipt {
|
|
|
|
pub fn try_new<D: AsRef<[u8]>>(
|
|
|
|
version: u8,
|
|
|
|
nonce: ReceiptNonce,
|
|
|
|
sender_id: DHTKey,
|
|
|
|
extra_data: D,
|
2022-07-10 21:36:50 +00:00
|
|
|
) -> Result<Self, VeilidAPIError> {
|
2021-11-22 16:28:30 +00:00
|
|
|
if extra_data.as_ref().len() > MAX_EXTRA_DATA_SIZE {
|
2022-11-26 19:16:02 +00:00
|
|
|
apibail_parse_error!(
|
2022-07-10 21:36:50 +00:00
|
|
|
"extra data too large for receipt",
|
2022-11-26 19:16:02 +00:00
|
|
|
extra_data.as_ref().len()
|
|
|
|
);
|
2021-11-22 16:28:30 +00:00
|
|
|
}
|
|
|
|
Ok(Self {
|
2021-11-26 16:50:49 +00:00
|
|
|
version,
|
|
|
|
nonce,
|
|
|
|
sender_id,
|
2021-11-22 16:28:30 +00:00
|
|
|
extra_data: Vec::from(extra_data.as_ref()),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2022-07-10 21:36:50 +00:00
|
|
|
pub fn from_signed_data(data: &[u8]) -> Result<Receipt, VeilidAPIError> {
|
2021-11-22 16:28:30 +00:00
|
|
|
// Ensure we are at least the length of the envelope
|
|
|
|
if data.len() < MIN_RECEIPT_SIZE {
|
2022-11-26 19:16:02 +00:00
|
|
|
apibail_parse_error!("receipt too small", data.len());
|
2021-11-22 16:28:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Verify magic number
|
2022-07-10 21:36:50 +00:00
|
|
|
let magic: [u8; 4] = data[0x00..0x04]
|
|
|
|
.try_into()
|
|
|
|
.map_err(VeilidAPIError::internal)?;
|
2021-11-22 16:28:30 +00:00
|
|
|
if magic != *RECEIPT_MAGIC {
|
2022-11-26 19:16:02 +00:00
|
|
|
apibail_generic!("bad magic number");
|
2021-11-22 16:28:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check version
|
|
|
|
let version = data[0x04];
|
2022-10-30 23:29:31 +00:00
|
|
|
if version > MAX_CRYPTO_VERSION || version < MIN_CRYPTO_VERSION {
|
2022-11-26 19:16:02 +00:00
|
|
|
apibail_parse_error!("unsupported cryptography version", version);
|
2021-11-22 16:28:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Get size and ensure it matches the size of the envelope and is less than the maximum message size
|
2022-07-10 21:36:50 +00:00
|
|
|
let size: u16 = u16::from_le_bytes(
|
|
|
|
data[0x06..0x08]
|
|
|
|
.try_into()
|
|
|
|
.map_err(VeilidAPIError::internal)?,
|
|
|
|
);
|
2021-11-22 16:28:30 +00:00
|
|
|
if (size as usize) > MAX_RECEIPT_SIZE {
|
2022-11-26 19:16:02 +00:00
|
|
|
apibail_parse_error!("receipt size is too large", size);
|
2021-11-22 16:28:30 +00:00
|
|
|
}
|
|
|
|
if (size as usize) != data.len() {
|
2022-11-26 19:16:02 +00:00
|
|
|
apibail_parse_error!(
|
2022-07-10 21:36:50 +00:00
|
|
|
"size doesn't match receipt size",
|
2022-11-26 19:16:02 +00:00
|
|
|
format!("size={} data.len()={}", size, data.len())
|
|
|
|
);
|
2021-11-22 16:28:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Get sender id
|
2022-07-10 21:36:50 +00:00
|
|
|
let sender_id = DHTKey::new(
|
|
|
|
data[0x20..0x40]
|
|
|
|
.try_into()
|
|
|
|
.map_err(VeilidAPIError::internal)?,
|
|
|
|
);
|
2021-11-26 16:50:49 +00:00
|
|
|
|
2021-11-22 16:28:30 +00:00
|
|
|
// Get signature
|
2022-07-10 21:36:50 +00:00
|
|
|
let signature = DHTSignature::new(
|
|
|
|
data[(data.len() - 64)..]
|
|
|
|
.try_into()
|
|
|
|
.map_err(VeilidAPIError::internal)?,
|
|
|
|
);
|
2021-11-22 16:28:30 +00:00
|
|
|
|
|
|
|
// Validate signature
|
2022-07-10 21:36:50 +00:00
|
|
|
verify(&sender_id, &data[0..(data.len() - 64)], &signature)
|
|
|
|
.map_err(VeilidAPIError::generic)?;
|
2021-11-22 16:28:30 +00:00
|
|
|
|
|
|
|
// Get nonce
|
2022-07-10 21:36:50 +00:00
|
|
|
let nonce: ReceiptNonce = data[0x08..0x20]
|
|
|
|
.try_into()
|
|
|
|
.map_err(VeilidAPIError::internal)?;
|
2021-11-22 16:28:30 +00:00
|
|
|
|
|
|
|
// Get extra data and signature
|
|
|
|
let extra_data: Vec<u8> = Vec::from(&data[0x40..(data.len() - 64)]);
|
|
|
|
|
|
|
|
// Return receipt
|
|
|
|
Ok(Self {
|
2021-11-26 16:50:49 +00:00
|
|
|
version,
|
|
|
|
nonce,
|
|
|
|
sender_id,
|
|
|
|
extra_data,
|
2021-11-22 16:28:30 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2022-07-10 21:36:50 +00:00
|
|
|
pub fn to_signed_data(&self, secret: &DHTKeySecret) -> Result<Vec<u8>, VeilidAPIError> {
|
2021-11-22 16:28:30 +00:00
|
|
|
// Ensure extra data isn't too long
|
|
|
|
let receipt_size: usize = self.extra_data.len() + MIN_RECEIPT_SIZE;
|
|
|
|
if receipt_size > MAX_RECEIPT_SIZE {
|
2022-11-26 19:16:02 +00:00
|
|
|
apibail_parse_error!("receipt too large", receipt_size);
|
2021-11-22 16:28:30 +00:00
|
|
|
}
|
2021-11-26 16:50:49 +00:00
|
|
|
let mut data: Vec<u8> = vec![0u8; receipt_size];
|
2021-11-22 16:28:30 +00:00
|
|
|
|
|
|
|
// Write magic
|
|
|
|
data[0x00..0x04].copy_from_slice(RECEIPT_MAGIC);
|
|
|
|
// Write version
|
|
|
|
data[0x04] = self.version;
|
|
|
|
// Write size
|
|
|
|
data[0x06..0x08].copy_from_slice(&(receipt_size as u16).to_le_bytes());
|
|
|
|
// Write nonce
|
|
|
|
data[0x08..0x20].copy_from_slice(&self.nonce);
|
|
|
|
// Write sender node id
|
|
|
|
data[0x20..0x40].copy_from_slice(&self.sender_id.bytes);
|
|
|
|
// Write extra data
|
2021-11-26 16:50:49 +00:00
|
|
|
if !self.extra_data.is_empty() {
|
2021-11-22 16:28:30 +00:00
|
|
|
data[0x40..(receipt_size - 64)].copy_from_slice(self.extra_data.as_slice());
|
|
|
|
}
|
|
|
|
// Sign the receipt
|
2022-07-10 21:36:50 +00:00
|
|
|
let signature = sign(&self.sender_id, secret, &data[0..(receipt_size - 64)])
|
|
|
|
.map_err(VeilidAPIError::generic)?;
|
2021-11-22 16:28:30 +00:00
|
|
|
// Append the signature
|
|
|
|
data[(receipt_size - 64)..].copy_from_slice(&signature.bytes);
|
|
|
|
|
|
|
|
Ok(data)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get_version(&self) -> u8 {
|
|
|
|
self.version
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get_nonce(&self) -> ReceiptNonce {
|
|
|
|
self.nonce
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get_sender_id(&self) -> DHTKey {
|
|
|
|
self.sender_id
|
|
|
|
}
|
|
|
|
pub fn get_extra_data(&self) -> &[u8] {
|
|
|
|
&self.extra_data
|
|
|
|
}
|
|
|
|
}
|