use crate::{Expression, IntegerType, Literal, LiteralVariant, ProgramId, Type};
use snarkvm::console::program::ArrayType as ConsoleArrayType;
use leo_span::Span;
use serde::{Deserialize, Serialize};
use snarkvm::prelude::Network;
use std::fmt;
#[derive(Clone, Debug, PartialEq, Eq, Serialize, Deserialize)]
pub struct ArrayType {
pub element_type: Box<Type>,
pub length: Box<Expression>,
}
impl ArrayType {
pub fn new(element: Type, length: Expression) -> Self {
Self { element_type: Box::new(element), length: Box::new(length) }
}
pub fn bit_array(length: u32) -> Self {
Self {
element_type: Box::new(Type::Boolean),
length: Box::new(Expression::Literal(Literal {
variant: LiteralVariant::Integer(IntegerType::U32, length.to_string()),
id: Default::default(),
span: Span::default(),
})),
}
}
pub fn element_type(&self) -> &Type {
&self.element_type
}
pub fn base_element_type(&self) -> &Type {
match self.element_type.as_ref() {
Type::Array(array_type) => array_type.base_element_type(),
type_ => type_,
}
}
pub fn from_snarkvm<N: Network>(array_type: &ConsoleArrayType<N>, program_id: ProgramId) -> Self {
Self {
element_type: Box::new(Type::from_snarkvm(array_type.next_element_type(), program_id)),
length: Box::new(Expression::Literal(Literal {
variant: LiteralVariant::Integer(IntegerType::U32, array_type.length().to_string().replace("u32", "")),
id: Default::default(),
span: Span::default(),
})),
}
}
pub fn to_snarkvm<N: Network>(&self) -> anyhow::Result<ConsoleArrayType<N>> {
let length = if let Expression::Literal(literal) = &*self.length {
match &literal.variant {
LiteralVariant::Integer(_, s) => {
s.parse::<u32>().map_err(|e| anyhow::anyhow!("Array length is not a valid u32: {e}"))?
}
LiteralVariant::Unsuffixed(s) => {
s.parse::<u32>().map_err(|e| anyhow::anyhow!("Array length is not a valid u32: {e}"))?
}
_ => anyhow::bail!("Array length must be an integer literal"),
}
} else {
anyhow::bail!("Array length must be an integer literal")
};
ConsoleArrayType::new(self.element_type.to_snarkvm()?, vec![snarkvm::console::types::U32::new(length)])
}
}
impl fmt::Display for ArrayType {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
if let Expression::Literal(literal) = &*self.length
&& let LiteralVariant::Integer(_, s) = &literal.variant
{
return write!(f, "[{}; {s}]", self.element_type);
}
write!(f, "[{}; {}]", self.element_type, self.length)
}
}
impl From<ArrayType> for Type {
fn from(value: ArrayType) -> Self {
Type::Array(value)
}
}