use std::{rc::Rc, sync::Arc};
use clvmr::Atom;
use num_bigint::BigInt;
use crate::{encode_number, ClvmEncoder, ToClvmError};
pub trait ToClvm<E>
where
E: ClvmEncoder,
{
fn to_clvm(&self, encoder: &mut E) -> Result<E::Node, ToClvmError>;
}
macro_rules! clvm_primitive {
($primitive:ty) => {
impl<N, E: ClvmEncoder<Node = N>> ToClvm<E> for $primitive {
fn to_clvm(&self, encoder: &mut E) -> Result<N, ToClvmError> {
let bytes = self.to_be_bytes();
#[allow(unused_comparisons)]
encoder.encode_atom(Atom::Borrowed(&encode_number(&bytes, *self < 0)))
}
}
};
}
clvm_primitive!(u8);
clvm_primitive!(i8);
clvm_primitive!(u16);
clvm_primitive!(i16);
clvm_primitive!(u32);
clvm_primitive!(i32);
clvm_primitive!(u64);
clvm_primitive!(i64);
clvm_primitive!(u128);
clvm_primitive!(i128);
clvm_primitive!(usize);
clvm_primitive!(isize);
impl<N, E: ClvmEncoder<Node = N>> ToClvm<E> for BigInt {
fn to_clvm(&self, encoder: &mut E) -> Result<<E as ClvmEncoder>::Node, ToClvmError> {
encoder.encode_bigint(self.clone())
}
}
impl<N, E: ClvmEncoder<Node = N>> ToClvm<E> for bool {
fn to_clvm(&self, encoder: &mut E) -> Result<N, ToClvmError> {
i32::from(*self).to_clvm(encoder)
}
}
impl<N, E: ClvmEncoder<Node = N>, T> ToClvm<E> for &T
where
T: ToClvm<E>,
{
fn to_clvm(&self, encoder: &mut E) -> Result<N, ToClvmError> {
T::to_clvm(*self, encoder)
}
}
impl<N, E: ClvmEncoder<Node = N>, T> ToClvm<E> for Box<T>
where
T: ToClvm<E>,
{
fn to_clvm(&self, encoder: &mut E) -> Result<N, ToClvmError> {
T::to_clvm(self, encoder)
}
}
impl<N, E: ClvmEncoder<Node = N>, T> ToClvm<E> for Rc<T>
where
T: ToClvm<E>,
{
fn to_clvm(&self, encoder: &mut E) -> Result<N, ToClvmError> {
T::to_clvm(self, encoder)
}
}
impl<N, E: ClvmEncoder<Node = N>, T> ToClvm<E> for Arc<T>
where
T: ToClvm<E>,
{
fn to_clvm(&self, encoder: &mut E) -> Result<N, ToClvmError> {
T::to_clvm(self, encoder)
}
}
impl<N, E: ClvmEncoder<Node = N>, A, B> ToClvm<E> for (A, B)
where
A: ToClvm<E>,
B: ToClvm<E>,
{
fn to_clvm(&self, encoder: &mut E) -> Result<N, ToClvmError> {
let first = self.0.to_clvm(encoder)?;
let rest = self.1.to_clvm(encoder)?;
encoder.encode_pair(first, rest)
}
}
impl<N, E: ClvmEncoder<Node = N>> ToClvm<E> for () {
fn to_clvm(&self, encoder: &mut E) -> Result<N, ToClvmError> {
encoder.encode_atom(Atom::Borrowed(&[]))
}
}
impl<N, E: ClvmEncoder<Node = N>, T> ToClvm<E> for &[T]
where
T: ToClvm<E>,
{
fn to_clvm(&self, encoder: &mut E) -> Result<N, ToClvmError> {
let mut result = encoder.encode_atom(Atom::Borrowed(&[]))?;
for item in self.iter().rev() {
let value = item.to_clvm(encoder)?;
result = encoder.encode_pair(value, result)?;
}
Ok(result)
}
}
impl<N, E: ClvmEncoder<Node = N>, T, const LEN: usize> ToClvm<E> for [T; LEN]
where
T: ToClvm<E>,
{
fn to_clvm(&self, encoder: &mut E) -> Result<N, ToClvmError> {
self.as_slice().to_clvm(encoder)
}
}
impl<N, E: ClvmEncoder<Node = N>, T> ToClvm<E> for Vec<T>
where
T: ToClvm<E>,
{
fn to_clvm(&self, encoder: &mut E) -> Result<N, ToClvmError> {
self.as_slice().to_clvm(encoder)
}
}
impl<N, E: ClvmEncoder<Node = N>, T> ToClvm<E> for Option<T>
where
T: ToClvm<E>,
{
fn to_clvm(&self, encoder: &mut E) -> Result<N, ToClvmError> {
match self {
Some(value) => value.to_clvm(encoder),
None => encoder.encode_atom(Atom::Borrowed(&[])),
}
}
}
impl<N, E: ClvmEncoder<Node = N>> ToClvm<E> for &str {
fn to_clvm(&self, encoder: &mut E) -> Result<N, ToClvmError> {
encoder.encode_atom(Atom::Borrowed(self.as_bytes()))
}
}
impl<N, E: ClvmEncoder<Node = N>> ToClvm<E> for String {
fn to_clvm(&self, encoder: &mut E) -> Result<N, ToClvmError> {
self.as_str().to_clvm(encoder)
}
}
#[cfg(feature = "chia-bls")]
impl<N, E: ClvmEncoder<Node = N>> ToClvm<E> for chia_bls::PublicKey {
fn to_clvm(&self, encoder: &mut E) -> Result<N, ToClvmError> {
encoder.encode_atom(Atom::Borrowed(&self.to_bytes()))
}
}
#[cfg(feature = "chia-bls")]
impl<N, E: ClvmEncoder<Node = N>> ToClvm<E> for chia_bls::Signature {
fn to_clvm(&self, encoder: &mut E) -> Result<N, ToClvmError> {
encoder.encode_atom(Atom::Borrowed(&self.to_bytes()))
}
}
#[cfg(feature = "chia-secp")]
impl<E> ToClvm<E> for chia_secp::K1PublicKey
where
E: ClvmEncoder,
{
fn to_clvm(&self, encoder: &mut E) -> Result<E::Node, ToClvmError> {
encoder.encode_atom(Atom::Borrowed(&self.to_bytes()))
}
}
#[cfg(feature = "chia-secp")]
impl<E> ToClvm<E> for chia_secp::K1Signature
where
E: ClvmEncoder,
{
fn to_clvm(&self, encoder: &mut E) -> Result<E::Node, ToClvmError> {
encoder.encode_atom(Atom::Borrowed(&self.to_bytes()))
}
}
#[cfg(feature = "chia-secp")]
impl<E> ToClvm<E> for chia_secp::R1PublicKey
where
E: ClvmEncoder,
{
fn to_clvm(&self, encoder: &mut E) -> Result<E::Node, ToClvmError> {
encoder.encode_atom(Atom::Borrowed(&self.to_bytes()))
}
}
#[cfg(feature = "chia-secp")]
impl<E> ToClvm<E> for chia_secp::R1Signature
where
E: ClvmEncoder,
{
fn to_clvm(&self, encoder: &mut E) -> Result<E::Node, ToClvmError> {
encoder.encode_atom(Atom::Borrowed(&self.to_bytes()))
}
}
#[cfg(test)]
mod tests {
use clvmr::{serde::node_to_bytes, Allocator};
use hex::ToHex;
use super::*;
fn encode<T>(a: &mut Allocator, value: T) -> Result<String, ToClvmError>
where
T: ToClvm<Allocator>,
{
let actual = value.to_clvm(a)?;
let actual_bytes = node_to_bytes(a, actual).unwrap();
Ok(actual_bytes.encode_hex())
}
#[test]
fn test_nodeptr() {
let a = &mut Allocator::new();
let ptr = a.one();
assert_eq!(ptr.to_clvm(a).unwrap(), ptr);
}
#[test]
fn test_primitives() {
let a = &mut Allocator::new();
assert_eq!(encode(a, 0u8), Ok("80".to_owned()));
assert_eq!(encode(a, 0i8), Ok("80".to_owned()));
assert_eq!(encode(a, 5u8), Ok("05".to_owned()));
assert_eq!(encode(a, 5u32), Ok("05".to_owned()));
assert_eq!(encode(a, 5i32), Ok("05".to_owned()));
assert_eq!(encode(a, -27i32), Ok("81e5".to_owned()));
assert_eq!(encode(a, -0), Ok("80".to_owned()));
assert_eq!(encode(a, -128i8), Ok("8180".to_owned()));
}
#[test]
fn test_bool() {
let a = &mut Allocator::new();
assert_eq!(encode(a, true), Ok("01".to_owned()));
assert_eq!(encode(a, false), Ok("80".to_owned()));
}
#[test]
fn test_reference() {
let a = &mut Allocator::new();
assert_eq!(encode(a, [1, 2, 3]), encode(a, [1, 2, 3]));
assert_eq!(encode(a, Some(42)), encode(a, Some(42)));
assert_eq!(encode(a, Some(&42)), encode(a, Some(42)));
assert_eq!(encode(a, Some(&42)), encode(a, Some(42)));
}
#[test]
fn test_smart_pointers() {
let a = &mut Allocator::new();
assert_eq!(encode(a, Box::new(42)), encode(a, 42));
assert_eq!(encode(a, Rc::new(42)), encode(a, 42));
assert_eq!(encode(a, Arc::new(42)), encode(a, 42));
}
#[test]
fn test_pair() {
let a = &mut Allocator::new();
assert_eq!(encode(a, (5, 2)), Ok("ff0502".to_owned()));
assert_eq!(
encode(a, (-72, (90121, ()))),
Ok("ff81b8ff8301600980".to_owned())
);
assert_eq!(
encode(a, (((), ((), ((), (((), ((), ((), ()))), ())))), ())),
Ok("ffff80ff80ff80ffff80ff80ff80808080".to_owned())
);
}
#[test]
fn test_nil() {
let a = &mut Allocator::new();
assert_eq!(encode(a, ()), Ok("80".to_owned()));
}
#[test]
fn test_slice() {
let a = &mut Allocator::new();
assert_eq!(
encode(a, [1, 2, 3, 4].as_slice()),
Ok("ff01ff02ff03ff0480".to_owned())
);
assert_eq!(encode(a, [0; 0].as_slice()), Ok("80".to_owned()));
}
#[test]
fn test_array() {
let a = &mut Allocator::new();
assert_eq!(encode(a, [1, 2, 3, 4]), Ok("ff01ff02ff03ff0480".to_owned()));
assert_eq!(encode(a, [0; 0]), Ok("80".to_owned()));
}
#[test]
fn test_vec() {
let a = &mut Allocator::new();
assert_eq!(
encode(a, vec![1, 2, 3, 4]),
Ok("ff01ff02ff03ff0480".to_owned())
);
assert_eq!(encode(a, vec![0; 0]), Ok("80".to_owned()));
}
#[test]
fn test_option() {
let a = &mut Allocator::new();
assert_eq!(encode(a, Some("hello")), Ok("8568656c6c6f".to_owned()));
assert_eq!(encode(a, None::<&str>), Ok("80".to_owned()));
assert_eq!(encode(a, Some("")), Ok("80".to_owned()));
}
#[test]
fn test_str() {
let a = &mut Allocator::new();
assert_eq!(encode(a, "hello"), Ok("8568656c6c6f".to_owned()));
assert_eq!(encode(a, ""), Ok("80".to_owned()));
}
#[test]
fn test_string() {
let a = &mut Allocator::new();
assert_eq!(
encode(a, "hello".to_string()),
Ok("8568656c6c6f".to_owned())
);
assert_eq!(encode(a, String::new()), Ok("80".to_owned()));
}
#[cfg(feature = "chia-bls")]
#[test]
fn test_public_key() {
use chia_bls::PublicKey;
use hex_literal::hex;
let a = &mut Allocator::new();
let bytes = hex!(
"
b8f7dd239557ff8c49d338f89ac1a258a863fa52cd0a502e
3aaae4b6738ba39ac8d982215aa3fa16bc5f8cb7e44b954d
"
);
assert_eq!(
encode(a, PublicKey::from_bytes(&bytes).unwrap()),
Ok("b0b8f7dd239557ff8c49d338f89ac1a258a863fa52cd0a502e3aaae4b6738ba39ac8d982215aa3fa16bc5f8cb7e44b954d".to_string())
);
}
#[cfg(feature = "chia-bls")]
#[test]
fn test_signature() {
use chia_bls::Signature;
use hex_literal::hex;
let a = &mut Allocator::new();
let bytes = hex!(
"
a3994dc9c0ef41a903d3335f0afe42ba16c88e7881706798492da4a1653cd10c
69c841eeb56f44ae005e2bad27fb7ebb16ce8bbfbd708ea91dd4ff24f030497b
50e694a8270eccd07dbc206b8ffe0c34a9ea81291785299fae8206a1e1bbc1d1
"
);
assert_eq!(
encode(a, Signature::from_bytes(&bytes).unwrap()),
Ok("c060a3994dc9c0ef41a903d3335f0afe42ba16c88e7881706798492da4a1653cd10c69c841eeb56f44ae005e2bad27fb7ebb16ce8bbfbd708ea91dd4ff24f030497b50e694a8270eccd07dbc206b8ffe0c34a9ea81291785299fae8206a1e1bbc1d1".to_string())
);
}
#[cfg(feature = "chia-secp")]
#[test]
fn test_secp_public_key() {
use chia_secp::{K1PublicKey, R1PublicKey};
use hex_literal::hex;
let a = &mut Allocator::new();
let k1_pk = K1PublicKey::from_bytes(&hex!(
"02827cdbbed87e45683d448be2ea15fb72ba3732247bda18474868cf5456123fb4"
))
.unwrap();
assert_eq!(
encode(a, k1_pk),
Ok("a102827cdbbed87e45683d448be2ea15fb72ba3732247bda18474868cf5456123fb4".to_string())
);
let r1_pk = R1PublicKey::from_bytes(&hex!(
"037dc85102f5eb7867b9580fea8b242c774173e1a47db320c798242d3a7a7579e4"
))
.unwrap();
assert_eq!(
encode(a, r1_pk),
Ok("a1037dc85102f5eb7867b9580fea8b242c774173e1a47db320c798242d3a7a7579e4".to_string())
);
}
#[cfg(feature = "chia-secp")]
#[test]
fn test_secp_signature() {
use chia_secp::{K1Signature, R1Signature};
use hex_literal::hex;
let a = &mut Allocator::new();
let k1_sig = K1Signature::from_bytes(&hex!(
"6f07897d1d28b8698af5dec5ca06907b1304b227dc9f740b8c4065cf04d5e8653ae66aa17063e7120ee7f22fae54373b35230e259244b90400b65cf00d86c591"
))
.unwrap();
assert_eq!(
encode(a, k1_sig),
Ok("c0406f07897d1d28b8698af5dec5ca06907b1304b227dc9f740b8c4065cf04d5e8653ae66aa17063e7120ee7f22fae54373b35230e259244b90400b65cf00d86c591".to_string())
);
let r1_sig = R1Signature::from_bytes(&hex!(
"550e83da8cf9b2d407ed093ae213869ebd7ceaea603920f87d535690e52b40537915d8fe3d5a96c87e700c56dc638c32f7a2954f2ba409367d1a132000cc2228"
))
.unwrap();
assert_eq!(
encode(a, r1_sig),
Ok("c040550e83da8cf9b2d407ed093ae213869ebd7ceaea603920f87d535690e52b40537915d8fe3d5a96c87e700c56dc638c32f7a2954f2ba409367d1a132000cc2228".to_string())
);
}
}