use alloc::boxed::Box;
use alloc::string::String;
use core::convert::Infallible;
use core::fmt::Debug;
use core::future::Future;
use core::pin::Pin;
#[cfg(feature = "std")]
use std::sync::LazyLock;
#[cfg(feature = "rand")]
use rand::RngCore;
#[cfg(all(feature = "std", feature = "os-rng"))]
use rand::TryRngCore;
#[cfg(feature = "os-rng")]
use rand::rngs::OsRng;
#[cfg(feature = "std")]
use secp256k1::{All, Secp256k1};
use secp256k1::{Parity, PublicKey as NormalizedPublicKey, XOnlyPublicKey, ecdh};
use serde::Serialize;
use serde::de::DeserializeOwned;
#[cfg(feature = "nip44")]
pub mod hkdf;
use crate::{PublicKey, SecretKey, key};
#[cfg(not(all(target_arch = "wasm32", target_os = "unknown")))]
pub type BoxedFuture<'a, T> = Pin<Box<dyn Future<Output = T> + Send + 'a>>;
#[cfg(all(target_arch = "wasm32", target_os = "unknown"))]
pub type BoxedFuture<'a, T> = Pin<Box<dyn Future<Output = T> + 'a>>;
#[cfg(feature = "rand")]
fn random_bytes<R, const N: usize>(rng: &mut R) -> [u8; N]
where
R: RngCore,
{
let mut ret: [u8; N] = [0u8; N];
rng.fill_bytes(&mut ret);
ret
}
#[inline]
#[cfg(feature = "rand")]
pub(crate) fn random_32_bytes<R>(rng: &mut R) -> [u8; 32]
where
R: RngCore,
{
random_bytes(rng)
}
#[cfg(feature = "rand")]
pub(crate) fn random_hex_string<R, const N: usize>(rng: &mut R) -> String
where
R: RngCore,
{
let bytes: [u8; N] = random_bytes(rng);
faster_hex::hex_string(&bytes)
}
pub fn generate_shared_key(
secret_key: &SecretKey,
public_key: &PublicKey,
) -> Result<[u8; 32], key::Error> {
let pk: XOnlyPublicKey = public_key.xonly()?;
let public_key_normalized: NormalizedPublicKey =
NormalizedPublicKey::from_x_only_public_key(pk, Parity::Even);
let ssp: [u8; 64] = ecdh::shared_secret_point(&public_key_normalized, secret_key);
let mut shared_key: [u8; 32] = [0u8; 32];
shared_key.copy_from_slice(&ssp[..32]);
Ok(shared_key)
}
#[cfg(feature = "std")]
pub static SECP256K1: LazyLock<Secp256k1<All>> = LazyLock::new(|| {
#[cfg(feature = "os-rng")]
let mut ctx: Secp256k1<All> = Secp256k1::new();
#[cfg(not(feature = "os-rng"))]
let ctx: Secp256k1<All> = Secp256k1::new();
#[cfg(feature = "os-rng")]
{
let seed: [u8; 32] = random_32_bytes(&mut OsRng.unwrap_err());
ctx.seeded_randomize(&seed);
}
ctx
});
pub trait JsonUtil: Sized + Serialize + DeserializeOwned
where
<Self as JsonUtil>::Err: From<serde_json::Error>,
{
type Err: Debug;
#[inline]
fn from_json<T>(json: T) -> Result<Self, Self::Err>
where
T: AsRef<[u8]>,
{
Ok(serde_json::from_slice(json.as_ref())?)
}
#[inline]
fn as_json(&self) -> String {
self.try_as_json().unwrap()
}
#[inline]
fn try_as_json(&self) -> Result<String, Self::Err> {
Ok(serde_json::to_string(self)?)
}
#[inline]
fn as_pretty_json(&self) -> String {
self.try_as_pretty_json().unwrap()
}
#[inline]
fn try_as_pretty_json(&self) -> Result<String, Self::Err> {
Ok(serde_json::to_string_pretty(self)?)
}
}
pub(crate) trait UnwrapInfallible<T>: Sized {
fn unwrap_infallible(self) -> T;
}
impl<T> UnwrapInfallible<T> for Result<T, Infallible> {
#[inline]
fn unwrap_infallible(self) -> T {
match self {
Ok(value) => value,
Err(e) => match e {},
}
}
}