use super::*;
impl<N: Network> Parser for Struct<N> {
#[inline]
fn parse(string: &str) -> ParserResult<Self> {
fn parse_tuple<N: Network>(string: &str) -> ParserResult<(Identifier<N>, PlaintextType<N>)> {
let (string, _) = Sanitizer::parse(string)?;
let (string, identifier) = Identifier::parse(string)?;
let (string, _) = Sanitizer::parse_whitespaces(string)?;
let (string, _) = tag("as")(string)?;
let (string, _) = Sanitizer::parse_whitespaces(string)?;
let (string, plaintext_type) = PlaintextType::parse(string)?;
let (string, _) = Sanitizer::parse_whitespaces(string)?;
let (string, _) = tag(";")(string)?;
Ok((string, (identifier, plaintext_type)))
}
let (string, _) = Sanitizer::parse(string)?;
let (string, _) = tag(Self::type_name())(string)?;
let (string, _) = Sanitizer::parse_whitespaces(string)?;
let (string, name) = Identifier::parse(string)?;
let (string, _) = Sanitizer::parse_whitespaces(string)?;
let (string, _) = tag(":")(string)?;
let (string, members) = map_res(many1(parse_tuple), |members| {
if has_duplicates(members.iter().map(|(identifier, _)| identifier)) {
return Err(error(format!("Duplicate identifier found in struct '{}'", name)));
}
if members.len() > N::MAX_DATA_ENTRIES {
return Err(error("Failed to parse struct: too many members"));
}
Ok(members)
})(string)?;
Ok((string, Self { name, members: IndexMap::from_iter(members.into_iter()) }))
}
}
impl<N: Network> FromStr for Struct<N> {
type Err = Error;
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 Struct<N> {
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
Display::fmt(self, f)
}
}
#[allow(clippy::format_push_string)]
impl<N: Network> Display for Struct<N> {
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
let mut output = format!("{} {}:\n", Self::type_name(), self.name);
for (identifier, plaintext_type) in &self.members {
output += &format!(" {identifier} as {plaintext_type};\n");
}
output.pop(); write!(f, "{}", output)
}
}
#[cfg(test)]
mod tests {
use super::*;
use snarkvm_console_network::Testnet3;
type CurrentNetwork = Testnet3;
#[test]
fn test_parse() -> Result<()> {
let expected = Struct::<CurrentNetwork> {
name: Identifier::from_str("message")?,
members: IndexMap::from_iter(
vec![
(Identifier::from_str("sender")?, PlaintextType::from_str("address")?),
(Identifier::from_str("amount")?, PlaintextType::from_str("u64")?),
]
.into_iter(),
),
};
let (remainder, candidate) = Struct::<CurrentNetwork>::parse(
r"
struct message:
sender as address;
amount as u64;
",
)?;
assert_eq!("\n", remainder);
assert_eq!(expected, candidate);
Ok(())
}
#[test]
fn test_parse_fails() {
assert!(Struct::<CurrentNetwork>::parse("").is_err());
assert!(Struct::<CurrentNetwork>::parse("struct message:").is_err());
assert!(Struct::<CurrentNetwork>::parse("{}").is_err());
assert!(Struct::<CurrentNetwork>::parse("_").is_err());
assert!(Struct::<CurrentNetwork>::parse("__").is_err());
assert!(Struct::<CurrentNetwork>::parse("___").is_err());
assert!(Struct::<CurrentNetwork>::parse("-").is_err());
assert!(Struct::<CurrentNetwork>::parse("--").is_err());
assert!(Struct::<CurrentNetwork>::parse("---").is_err());
assert!(Struct::<CurrentNetwork>::parse("*").is_err());
assert!(Struct::<CurrentNetwork>::parse("**").is_err());
assert!(Struct::<CurrentNetwork>::parse("***").is_err());
assert!(Struct::<CurrentNetwork>::parse("1").is_err());
assert!(Struct::<CurrentNetwork>::parse("2").is_err());
assert!(Struct::<CurrentNetwork>::parse("3").is_err());
assert!(Struct::<CurrentNetwork>::parse("1foo").is_err());
assert!(Struct::<CurrentNetwork>::parse("12").is_err());
assert!(Struct::<CurrentNetwork>::parse("111").is_err());
let struct_ =
Struct::<CurrentNetwork>::parse("foo_bar_baz_qux_quux_quuz_corge_grault_garply_waldo_fred_plugh_xyzzy");
assert!(struct_.is_err());
}
#[test]
fn test_display() {
let expected = "struct message:\n first as field;\n second as field;";
let message = Struct::<CurrentNetwork>::parse(expected).unwrap().1;
assert_eq!(expected, format!("{}", message));
}
#[test]
fn test_display_fails() {
let candidate = Struct::<CurrentNetwork>::parse("struct message:\n first as field;\n first as field;");
assert!(candidate.is_err());
let candidate =
Struct::<CurrentNetwork>::parse("struct message:\n first as field.public;\n first as field.private;");
assert!(candidate.is_err());
}
#[test]
fn test_max_members() {
let mut string = "struct message:\n".to_string();
for i in 0..CurrentNetwork::MAX_DATA_ENTRIES {
string += &format!(" member_{} as field;\n", i);
}
assert!(Struct::<CurrentNetwork>::parse(&string).is_ok());
}
#[test]
fn test_too_many_members() {
let mut string = "struct message:\n".to_string();
for i in 0..=CurrentNetwork::MAX_DATA_ENTRIES {
string += &format!(" member_{} as field;\n", i);
}
assert!(Struct::<CurrentNetwork>::parse(&string).is_err());
}
}