use super::*;
impl<N: Network> Parser for Output<N> {
#[inline]
fn parse(string: &str) -> ParserResult<Self> {
let (string, _) = Sanitizer::parse(string)?;
let (string, _) = tag(Self::type_name())(string)?;
let (string, _) = Sanitizer::parse_whitespaces(string)?;
let (string, operand) = Operand::parse(string)?;
let (string, _) = Sanitizer::parse_whitespaces(string)?;
let (string, _) = tag("as")(string)?;
let (string, _) = Sanitizer::parse_whitespaces(string)?;
let (string, value_type) = ValueType::parse(string)?;
let (string, _) = Sanitizer::parse_whitespaces(string)?;
let (string, _) = tag(";")(string)?;
Ok((string, Self { operand, value_type }))
}
}
impl<N: Network> FromStr for Output<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 Output<N> {
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
Display::fmt(self, f)
}
}
impl<N: Network> Display for Output<N> {
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
write!(
f,
"{type_} {operand} as {value_type};",
type_ = Self::type_name(),
operand = self.operand,
value_type = self.value_type
)
}
}
#[cfg(test)]
mod tests {
use super::*;
use console::{
network::Testnet3,
program::{Literal, Register, U8},
};
type CurrentNetwork = Testnet3;
#[test]
fn test_output_parse() -> Result<()> {
let output = Output::<CurrentNetwork>::parse("output r0 as field.private;").unwrap().1;
assert_eq!(output.operand(), &Operand::Register(Register::<CurrentNetwork>::Locator(0)));
assert_eq!(output.value_type(), &ValueType::<CurrentNetwork>::from_str("field.private")?);
let output = Output::<CurrentNetwork>::parse("output 0u8 as u8.public;").unwrap().1;
assert_eq!(output.operand(), &Operand::Literal(Literal::<CurrentNetwork>::U8(U8::new(0))));
assert_eq!(output.value_type(), &ValueType::<CurrentNetwork>::from_str("u8.public")?);
let output = Output::<CurrentNetwork>::parse("output r1 as signature.private;").unwrap().1;
assert_eq!(output.operand(), &Operand::Register(Register::<CurrentNetwork>::Locator(1)));
assert_eq!(output.value_type(), &ValueType::<CurrentNetwork>::from_str("signature.private")?);
let output = Output::<CurrentNetwork>::parse("output r2 as token.record;").unwrap().1;
assert_eq!(output.operand(), &Operand::Register(Register::<CurrentNetwork>::Locator(2)));
assert_eq!(output.value_type(), &ValueType::<CurrentNetwork>::from_str("token.record")?);
Ok(())
}
#[test]
fn test_output_display() {
let output = Output::<CurrentNetwork>::parse("output r0 as field.private;").unwrap().1;
assert_eq!(format!("{output}"), "output r0 as field.private;");
let output = Output::<CurrentNetwork>::parse("output 0u8 as u8.public;").unwrap().1;
assert_eq!(format!("{output}"), "output 0u8 as u8.public;");
let output = Output::<CurrentNetwork>::parse("output r1 as signature.private;").unwrap().1;
assert_eq!(format!("{output}"), "output r1 as signature.private;");
let output = Output::<CurrentNetwork>::parse("output r2 as token.record;").unwrap().1;
assert_eq!(format!("{output}"), "output r2 as token.record;");
}
}