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
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
use super::*;
impl<E: Environment> Parser for Group<E> {
#[inline]
fn parse(string: &str) -> ParserResult<Self> {
let (string, negation) = map(opt(tag("-")), |neg: Option<&str>| neg.is_some())(string)?;
let (string, primitive) = recognize(many1(terminated(one_of("0123456789"), many0(char('_')))))(string)?;
let (string, group): (&str, Self) = map_res(tag(Self::type_name()), |_| {
let x_coordinate = primitive.replace('_', "").parse()?;
match negation {
true => Ok(-Group::from_x_coordinate(Field::new(x_coordinate))?),
false => Group::from_x_coordinate(Field::new(x_coordinate)),
}
})(string)?;
Ok((string, group))
}
}
impl<E: Environment> FromStr for Group<E> {
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<E: Environment> Debug for Group<E> {
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
Display::fmt(self, f)
}
}
impl<E: Environment> Display for Group<E> {
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
write!(f, "{}{}", self.group.to_affine().to_x_coordinate(), Self::type_name())
}
}
#[cfg(test)]
mod tests {
use super::*;
use snarkvm_console_network_environment::Console;
type CurrentEnvironment = Console;
const ITERATIONS: u64 = 1_000;
#[test]
fn test_parse() -> Result<()> {
let rng = &mut TestRng::default();
assert!(Group::<CurrentEnvironment>::parse(Group::<CurrentEnvironment>::type_name()).is_err());
assert!(Group::<CurrentEnvironment>::parse("").is_err());
for _ in 0..ITERATIONS {
let group: <CurrentEnvironment as Environment>::Affine = Uniform::rand(rng);
let expected = format!("{}{}", group.to_x_coordinate(), Group::<CurrentEnvironment>::type_name());
let (remainder, candidate) = Group::<CurrentEnvironment>::parse(&expected).unwrap();
assert_eq!(format!("{expected}"), candidate.to_string());
assert_eq!("", remainder);
}
Ok(())
}
#[test]
fn test_display() {
fn check_display<E: Environment>(element: E::Affine) {
let candidate = Group::<E>::new(element);
assert_eq!(format!("{}{}", element.to_x_coordinate(), Group::<E>::type_name()), format!("{candidate}"));
let candidate_recovered = Group::<E>::from_str(&format!("{candidate}")).unwrap();
assert_eq!(candidate, candidate_recovered);
}
let mut rng = TestRng::default();
for _ in 0..ITERATIONS {
let element = Uniform::rand(&mut rng);
check_display::<CurrentEnvironment>(element);
}
}
#[test]
fn test_display_zero() {
let zero = <CurrentEnvironment as Environment>::Affine::zero();
let candidate = Group::<CurrentEnvironment>::new(zero);
assert_eq!("0group", &format!("{}", candidate));
}
}