plutus_ledger_api/
aux.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
use std::{
    collections::BTreeMap,
    iter::{empty, once},
    str::FromStr,
};

use nom::{
    branch::alt,
    character::complete::{char, digit1},
    combinator::{map_res, opt, recognize},
    error::VerboseError,
    multi::many1,
    sequence::tuple,
    IResult,
};
use num_bigint::BigInt;

use crate::error::ConversionError;

/// Create a container C from one element.
pub fn singleton<T, C>(value: T) -> C
where
    C: FromIterator<T>,
{
    once(value).collect()
}

/// Create an empty container.
pub fn none<T, C>() -> C
where
    C: FromIterator<T>,
{
    empty::<T>().collect()
}

/// Union two BTreeMaps, call f to resolve conflicts if duplicate keys are encountered.
pub fn union_btree_maps_with<K: Clone + Ord, V: Clone, F: Fn(V, V) -> V>(
    f: F,
    l: BTreeMap<K, V>,
    r: BTreeMap<K, V>,
) -> BTreeMap<K, V> {
    r.into_iter().fold(l.clone(), |mut acc, (k, vr)| {
        let v = if let Some((_, vl)) = acc.remove_entry(&k) {
            f(vl, vr)
        } else {
            vr
        };
        acc.insert(k, v);
        acc
    })
}

pub fn union_b_tree_maps_with<const N: usize, K: Clone + Ord, V: Clone, F: Fn(&V, &V) -> V>(
    f: F,
    maps: [&BTreeMap<K, V>; N],
) -> BTreeMap<K, V> {
    maps.into_iter().fold(BTreeMap::new(), |acc, m| {
        m.iter().fold(acc, |mut acc, (k, v)| {
            acc.entry(k.clone())
                .and_modify(|va: &mut V| *va = f(va, v))
                .or_insert(v.clone());

            acc
        })
    })
}

/// Verify that a given bytestring has the expected length
pub(crate) fn guard_bytes(
    ctx: &str,
    bytes: Vec<u8>,
    expected: usize,
) -> Result<Vec<u8>, ConversionError> {
    if bytes.len() == expected {
        Ok(bytes)
    } else {
        Err(ConversionError::invalid_bytestring_length(
            ctx, expected, "equal to", &bytes,
        ))
    }
}

/// Nom parser for BigInt
/// Expects an arbitrary length decimal integer, optionally signed
pub(crate) fn big_int(i: &str) -> IResult<&str, BigInt, VerboseError<&str>> {
    map_res(
        recognize(tuple((opt(alt((char('-'), char('+')))), many1(digit1)))),
        |s: &str| BigInt::from_str(s),
    )(i)
}