signet-types 0.16.2

A collection of types used in Signet.
Documentation
use crate::SignedOrder;
use alloy::primitives::{Address, U256};
use serde::{Deserialize, Serialize};
use signet_zenith::RollupOrders;
use std::borrow::Cow;
use std::collections::{HashMap, HashSet};

/// Aggregated orders for a transaction or set of transactions.
#[derive(Debug, Default, Clone, Eq, PartialEq, Deserialize, Serialize)]
pub struct AggregateOrders {
    /// Outputs to be transferred to the user. These may be on the rollup or
    /// the host or potentially elsewhere in the future.
    /// (chain_id, token) -> recipient -> amount
    pub outputs: HashMap<(u64, Address), HashMap<Address, U256>>,
    /// Inputs to be transferred to the filler. These are always on the
    /// rollup.
    pub inputs: HashMap<Address, U256>,
}

impl AggregateOrders {
    /// Instantiate a new [`AggregateOrders`].
    pub fn new() -> Self {
        Default::default()
    }

    /// Instantiate a new [`AggregateOrders`] with a custom capacity. The
    /// capcity is for the number of assets in inputs or outputs.
    pub fn with_capacity(capacity: usize) -> Self {
        Self { outputs: HashMap::with_capacity(capacity), inputs: HashMap::with_capacity(capacity) }
    }

    /// Ingest an output into the aggregate orders.
    pub(crate) fn ingest_output(&mut self, output: &RollupOrders::Output) {
        self.ingest_raw_output(
            output.chainId as u64,
            output.token,
            output.recipient,
            output.amount,
        );
    }

    /// Ingest raw output values into the aggregate orders.
    pub(crate) fn ingest_raw_output(
        &mut self,
        chain_id: u64,
        token: Address,
        recipient: Address,
        amount: U256,
    ) {
        let entry =
            self.outputs.entry((chain_id, token)).or_default().entry(recipient).or_default();
        *entry = entry.saturating_add(amount);
    }

    /// Ingest an input into the aggregate orders.
    pub(crate) fn ingest_input(&mut self, input: &RollupOrders::Input) {
        self.ingest_raw_input(input.token, input.amount);
    }

    /// Ingest raw input values into the aggregate orders.
    pub(crate) fn ingest_raw_input(&mut self, token: Address, amount: U256) {
        let entry = self.inputs.entry(token).or_default();
        *entry = entry.saturating_add(amount);
    }

    /// Ingest a new order into the aggregate orders.
    pub fn ingest(&mut self, order: &RollupOrders::Order) {
        order.outputs.iter().for_each(|o| self.ingest_output(o));
        order.inputs.iter().for_each(|i| self.ingest_input(i));
    }

    /// Ingest a signed order into the aggregate orders.
    pub fn ingest_signed(&mut self, order: &SignedOrder) {
        order
            .outputs()
            .iter()
            .for_each(|o| self.ingest_raw_output(o.chainId as u64, o.token, o.recipient, o.amount));
        order
            .permit()
            .permit
            .permitted
            .iter()
            .for_each(|tp| self.ingest_raw_input(tp.token, tp.amount));
    }

    /// Extend the orders with a new set of orders.
    pub fn extend<'a>(&mut self, orders: impl IntoIterator<Item = &'a RollupOrders::Order>) {
        for order in orders {
            self.ingest(order);
        }
    }

    /// Extend the orders with a new set of signed orders.
    pub fn extend_signed<'a>(&mut self, orders: impl IntoIterator<Item = &'a SignedOrder>) {
        for order in orders {
            self.ingest_signed(order);
        }
    }

    /// Get the unique target chain ids for the aggregated outputs.
    pub fn target_chain_ids(&self) -> Vec<u64> {
        HashSet::<u64>::from_iter(self.outputs.keys().map(|(chain_id, _)| *chain_id))
            .into_iter()
            .collect()
    }

    /// Get the aggregated Outputs for a given target chain id.
    ///
    /// # Warning
    ///
    /// All Orders in the AggregateOrders MUST have originated on the same rollup.
    /// Otherwise, the aggregated Outputs will be incorrectly credited.
    ///
    /// This function truncates `ru_chain_id` to u32. Chain IDs exceeding
    /// `u32::MAX` will be silently truncated.
    pub fn outputs_for(&self, target_chain_id: u64, ru_chain_id: u64) -> Vec<RollupOrders::Output> {
        let mut o = Vec::new();
        for ((chain_id, token), recipient_map) in &self.outputs {
            if *chain_id == target_chain_id {
                for (recipient, amount) in recipient_map {
                    o.push(RollupOrders::Output {
                        token: *token,
                        amount: U256::from(*amount),
                        recipient: *recipient,
                        chainId: ru_chain_id as u32,
                    });
                }
            }
        }
        o
    }

    /// Absorb the orders from another context.
    pub fn absorb(&mut self, other: &Self) {
        for (address, amount) in other.inputs.iter() {
            self.ingest_raw_input(*address, *amount);
        }

        for ((chain_id, output_asset), recipients) in other.outputs.iter() {
            for (recipient, value) in recipients {
                self.ingest_raw_output(*chain_id, *output_asset, *recipient, *value);
            }
        }
    }
}

impl<'a> FromIterator<&'a RollupOrders::Order> for AggregateOrders {
    fn from_iter<T: IntoIterator<Item = &'a RollupOrders::Order>>(iter: T) -> Self {
        let mut orders = AggregateOrders::new();
        orders.extend(iter);
        orders
    }
}

impl<'a> FromIterator<&'a SignedOrder> for AggregateOrders {
    fn from_iter<T: IntoIterator<Item = &'a SignedOrder>>(iter: T) -> Self {
        let mut orders = AggregateOrders::new();
        orders.extend_signed(iter);
        orders
    }
}

impl<'a> From<&'a RollupOrders::Order> for AggregateOrders {
    fn from(order: &'a RollupOrders::Order) -> Self {
        let mut orders = AggregateOrders::new();
        orders.ingest(order);
        orders
    }
}

impl<'a> From<&'a SignedOrder> for AggregateOrders {
    fn from(order: &'a SignedOrder) -> Self {
        let mut orders = AggregateOrders::new();
        orders.ingest_signed(order);
        orders
    }
}

impl<'a> From<&'a AggregateOrders> for Cow<'a, AggregateOrders> {
    fn from(orders: &'a AggregateOrders) -> Self {
        Cow::Borrowed(orders)
    }
}

#[cfg(test)]
mod test {
    use super::*;
    use alloy::primitives::{Address, U256};

    const ASSET_A: Address = Address::repeat_byte(1);
    const ASSET_B: Address = Address::repeat_byte(2);
    const ASSET_C: Address = Address::repeat_byte(3);

    const USER_A: Address = Address::repeat_byte(4);
    const USER_B: Address = Address::repeat_byte(5);
    const USER_C: Address = Address::repeat_byte(6);

    fn input(asset: Address, amount: u64) -> RollupOrders::Input {
        RollupOrders::Input { token: asset, amount: U256::from(amount) }
    }

    fn output(asset: Address, recipient: Address, amount: u64) -> RollupOrders::Output {
        RollupOrders::Output { chainId: 1, token: asset, recipient, amount: U256::from(amount) }
    }

    #[test]
    fn test_single_order() {
        let order = RollupOrders::Order {
            inputs: vec![input(ASSET_A, 100), input(ASSET_B, 200)],
            outputs: vec![
                output(ASSET_A, USER_A, 50),
                output(ASSET_A, USER_B, 50),
                output(ASSET_B, USER_B, 100),
                output(ASSET_C, USER_C, 200),
                output(ASSET_C, USER_C, 200),
            ],
            deadline: U256::ZERO,
        };

        let agg: AggregateOrders = [&order].into_iter().collect();
        assert_eq!(agg.inputs.get(&ASSET_A), Some(&U256::from(100)), "ASSET_A input");
        assert_eq!(agg.inputs.get(&ASSET_B), Some(&U256::from(200)), "ASSET_B input");

        assert_eq!(
            agg.outputs.get(&(1, ASSET_A)).map(|m| m.get(&USER_A)),
            Some(Some(&U256::from(50))),
            "ASSET_A USER_A output"
        );
        assert_eq!(
            agg.outputs.get(&(1, ASSET_A)).map(|m| m.get(&USER_B)),
            Some(Some(&U256::from(50))),
            "ASSET_A USER_B output"
        );
        assert_eq!(
            agg.outputs.get(&(1, ASSET_B)).map(|m| m.get(&USER_B)),
            Some(Some(&U256::from(100))),
            "ASSET_B USER_B output"
        );
        assert_eq!(
            agg.outputs.get(&(1, ASSET_C)).map(|m| m.get(&USER_C)),
            Some(Some(&U256::from(400))),
            "ASSET_C USER_C output"
        );
    }

    #[test]
    fn test_two_orders() {
        let order_1 = RollupOrders::Order {
            inputs: vec![input(ASSET_A, 100), input(ASSET_B, 200)],
            outputs: vec![
                output(ASSET_A, USER_A, 50),
                output(ASSET_A, USER_B, 50),
                output(ASSET_B, USER_B, 100),
                output(ASSET_C, USER_C, 200),
                output(ASSET_C, USER_C, 200),
            ],
            deadline: U256::ZERO,
        };
        let order_2 = RollupOrders::Order {
            inputs: vec![input(ASSET_A, 50), input(ASSET_C, 100)],
            outputs: vec![
                output(ASSET_A, USER_A, 50),
                output(ASSET_B, USER_B, 100),
                output(ASSET_C, USER_C, 100),
            ],
            deadline: U256::ZERO,
        };

        let agg: AggregateOrders = [&order_1, &order_2].into_iter().collect();

        assert_eq!(agg.inputs.get(&ASSET_A), Some(&U256::from(150)), "ASSET_A input");
        assert_eq!(agg.inputs.get(&ASSET_B), Some(&U256::from(200)), "ASSET_B input");
        assert_eq!(agg.inputs.get(&ASSET_C), Some(&U256::from(100)), "ASSET_C input");

        assert_eq!(
            agg.outputs.get(&(1, ASSET_A)).map(|m| m.get(&USER_A)),
            Some(Some(&U256::from(100))),
            "ASSET_A USER_A output"
        );
        assert_eq!(
            agg.outputs.get(&(1, ASSET_A)).map(|m| m.get(&USER_B)),
            Some(Some(&U256::from(50))),
            "ASSET_A USER_B output"
        );
        assert_eq!(
            agg.outputs.get(&(1, ASSET_B)).map(|m| m.get(&USER_B)),
            Some(Some(&U256::from(200))),
            "ASSET_B USER_B output"
        );
        assert_eq!(
            agg.outputs.get(&(1, ASSET_C)).map(|m| m.get(&USER_C)),
            Some(Some(&U256::from(500))),
            "ASSET_C USER_C output"
        );
    }

    // === Chain ID Truncation Tests ===

    #[test]
    fn chain_id_truncation_at_u32_max() {
        let mut orders = AggregateOrders::new();
        orders.ingest_raw_output(u64::from(u32::MAX), ASSET_A, USER_A, U256::from(100));

        let outputs = orders.outputs_for(u64::from(u32::MAX), u64::from(u32::MAX));
        assert_eq!(outputs.len(), 1);
        assert_eq!(outputs[0].chainId, u32::MAX);
    }

    #[test]
    fn chain_id_truncation_exceeds_u32() {
        let mut orders = AggregateOrders::new();
        let large_chain_id: u64 = u64::from(u32::MAX) + 1000;

        orders.ingest_raw_output(large_chain_id, ASSET_A, USER_A, U256::from(100));

        let outputs = orders.outputs_for(large_chain_id, large_chain_id);
        assert_eq!(outputs.len(), 1);
        // Documents truncation behavior - chainId wraps
        assert_eq!(outputs[0].chainId, 999); // (u32::MAX + 1000) as u32 = 999
    }

    // === Multi-Chain Tests ===

    #[test]
    fn target_chain_ids_multiple_chains() {
        let mut orders = AggregateOrders::new();
        orders.ingest_raw_output(1, ASSET_A, USER_A, U256::from(100));
        orders.ingest_raw_output(10, ASSET_A, USER_A, U256::from(200));
        orders.ingest_raw_output(42161, ASSET_A, USER_A, U256::from(300));

        let ids = orders.target_chain_ids();
        assert_eq!(ids.len(), 3);
        assert!(ids.contains(&1));
        assert!(ids.contains(&10));
        assert!(ids.contains(&42161));
    }

    // === Clone/Equality Tests ===

    #[test]
    fn clone_equality() {
        let mut orders = AggregateOrders::new();
        orders.ingest_raw_output(1, ASSET_A, USER_A, U256::from(100));
        orders.ingest_raw_input(ASSET_B, U256::from(200));

        let cloned = orders.clone();
        assert_eq!(orders, cloned);
    }
}