libsoliton/soliton_py/src/call.rs
Kamal Tufekcic 1d99048c95
Some checks failed
CI / lint (push) Successful in 1m37s
CI / test-python (push) Successful in 1m49s
CI / test-zig (push) Successful in 1m39s
CI / test-wasm (push) Successful in 1m54s
CI / test (push) Successful in 14m44s
CI / miri (push) Successful in 14m18s
CI / build (push) Successful in 1m9s
CI / fuzz-regression (push) Successful in 9m9s
CI / publish (push) Failing after 1m10s
CI / publish-python (push) Failing after 1m46s
CI / publish-wasm (push) Has been cancelled
initial commit
Signed-off-by: Kamal Tufekcic <kamal@lo.sh>
2026-04-02 23:48:10 +03:00

84 lines
2.3 KiB
Rust

//! Call key derivation for encrypted voice/video.
use pyo3::prelude::*;
use pyo3::types::PyBytes;
use crate::errors::to_py_err;
/// Call keys for encrypted voice/video media.
///
/// Derived from a ratchet session's root key and an ephemeral KEM shared secret.
/// Keys are role-assigned by fingerprint order — both parties get the same
/// send/recv assignment without coordination.
///
/// Use as a context manager for automatic zeroization::
///
/// with ratchet.derive_call_keys(kem_ss, call_id) as keys:
/// send = keys.send_key()
/// recv = keys.recv_key()
/// keys.advance() # derive next round
#[pyclass]
pub struct CallKeys {
inner: Option<soliton::call::CallKeys>,
}
#[pymethods]
impl CallKeys {
/// Current send key (32 bytes).
fn send_key<'py>(&self, py: Python<'py>) -> PyResult<Py<PyBytes>> {
let inner = self
.inner
.as_ref()
.ok_or_else(|| crate::errors::InvalidDataError::new_err("call keys consumed"))?;
Ok(PyBytes::new(py, inner.send_key()).into())
}
/// Current recv key (32 bytes).
fn recv_key<'py>(&self, py: Python<'py>) -> PyResult<Py<PyBytes>> {
let inner = self
.inner
.as_ref()
.ok_or_else(|| crate::errors::InvalidDataError::new_err("call keys consumed"))?;
Ok(PyBytes::new(py, inner.recv_key()).into())
}
/// Advance the call chain — derives fresh keys, zeroizes old ones.
///
/// Raises ``ChainExhaustedError`` after 2^24 advances.
fn advance(&mut self) -> PyResult<()> {
let inner = self
.inner
.as_mut()
.ok_or_else(|| crate::errors::InvalidDataError::new_err("call keys consumed"))?;
inner.advance().map_err(to_py_err)
}
/// Zeroize all key material.
fn close(&mut self) {
self.inner = None;
}
fn __enter__(slf: Py<Self>) -> Py<Self> {
slf
}
fn __exit__(
&mut self,
_exc_type: Option<&Bound<'_, PyAny>>,
_exc_val: Option<&Bound<'_, PyAny>>,
_exc_tb: Option<&Bound<'_, PyAny>>,
) {
self.close();
}
}
impl CallKeys {
pub fn from_inner(inner: soliton::call::CallKeys) -> Self {
Self { inner: Some(inner) }
}
}
pub fn register(m: &Bound<'_, PyModule>) -> PyResult<()> {
m.add_class::<CallKeys>()?;
Ok(())
}