use super::*;
impl<N: Network> Parser for Value<N> {
#[inline]
fn parse(string: &str) -> ParserResult<Self> {
alt((
map(Future::parse, Value::Future),
map(DynamicFuture::parse, Value::DynamicFuture),
map(Plaintext::parse, Value::Plaintext),
map(Record::parse, Value::Record),
map(DynamicRecord::parse, Value::DynamicRecord),
))(string)
}
}
impl<N: Network> FromStr for Value<N> {
type Err = Error;
#[inline]
fn from_str(string: &str) -> Result<Self> {
match Self::parse(string) {
Ok((remainder, object)) => {
ensure!(remainder.is_empty(), "Failed to parse string. Found invalid character in: \"{remainder}\"");
Ok(object)
}
Err(error) => bail!("Failed to parse string. {error}"),
}
}
}
impl<N: Network> Debug for Value<N> {
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
Display::fmt(self, f)
}
}
impl<N: Network> Display for Value<N> {
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
match self {
Value::Plaintext(plaintext) => Display::fmt(plaintext, f),
Value::Record(record) => Display::fmt(record, f),
Value::Future(future) => Display::fmt(future, f),
Value::DynamicRecord(dynamic_record) => Display::fmt(dynamic_record, f),
Value::DynamicFuture(dynamic_future) => Display::fmt(dynamic_future, f),
}
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::{Argument, Entry, Identifier, Literal, Owner, ProgramID};
use snarkvm_console_network::MainnetV0;
use snarkvm_console_types::{Group, U8, U64};
use snarkvm_utilities::{TestRng, Uniform};
type CurrentNetwork = MainnetV0;
#[test]
fn test_value_plaintext_parse() {
let string = r"{
owner: aleo1d5hg2z3ma00382pngntdp68e74zv54jdxy249qhaujhks9c72yrs33ddah,
token_amount: 100u64
}";
let expected = Value::<CurrentNetwork>::from_str(string).unwrap();
assert!(matches!(expected, Value::Plaintext(..)));
assert_eq!(string, format!("{expected}"));
}
#[test]
fn test_value_record_parse() {
let string = r"{
owner: aleo1d5hg2z3ma00382pngntdp68e74zv54jdxy249qhaujhks9c72yrs33ddah.private,
token_amount: 100u64.private,
_nonce: 6122363155094913586073041054293642159180066699840940609722305038224296461351group.public,
_version: 0u8.public
}";
let expected = Value::<CurrentNetwork>::from_str(string).unwrap();
assert!(matches!(expected, Value::Record(..)));
assert_eq!(string, format!("{expected}"));
}
#[test]
fn test_value_future_parse() {
let string = r"{
program_id: credits.aleo,
function_name: transfer_public_to_private,
arguments: [
aleo1g8qul5a44vk22u9uuvaewdcjw4v6xg8wx0llru39nnjn7eu08yrscxe4e2,
100000000u64
]
}";
let expected = Value::<CurrentNetwork>::from_str(string).unwrap();
assert!(matches!(expected, Value::Future(..)));
assert_eq!(string, format!("{expected}"));
}
#[test]
fn test_value_dynamic_record_parse() {
let rng = &mut TestRng::default();
let data = indexmap::indexmap! {
Identifier::from_str("amount").unwrap() => Entry::Private(Plaintext::from(Literal::U64(U64::rand(rng)))),
};
let owner = Owner::Public(Address::rand(rng));
let record = Record::<CurrentNetwork, Plaintext<CurrentNetwork>>::from_plaintext(
owner,
data,
Group::rand(rng),
U8::new(0),
)
.unwrap();
let expected = DynamicRecord::from_record(&record).unwrap();
let string = Value::DynamicRecord(expected.clone()).to_string();
let Value::DynamicRecord(candidate) = Value::<CurrentNetwork>::from_str(&string).unwrap() else {
panic!("Expected DynamicRecord value");
};
assert_eq!(expected.owner(), candidate.owner());
assert_eq!(expected.root(), candidate.root());
assert_eq!(expected.nonce(), candidate.nonce());
assert_eq!(expected.version(), candidate.version());
}
#[test]
fn test_value_dynamic_future_parse() {
let future = Future::<CurrentNetwork>::new(
ProgramID::from_str("test.aleo").unwrap(),
Identifier::from_str("foo").unwrap(),
vec![Argument::Plaintext(Plaintext::from_str("100u64").unwrap())],
);
let expected = DynamicFuture::from_future(&future).unwrap();
let string = Value::DynamicFuture(expected.clone()).to_string();
let Value::DynamicFuture(candidate) = Value::<CurrentNetwork>::from_str(&string).unwrap() else {
panic!("Expected DynamicFuture value");
};
assert_eq!(expected.program_name(), candidate.program_name());
assert_eq!(expected.program_network(), candidate.program_network());
assert_eq!(expected.function_name(), candidate.function_name());
assert_eq!(expected.checksum(), candidate.checksum());
}
}