use alloc::vec::Vec;
use miden_core::serde::{
ByteReader, ByteWriter, Deserializable, DeserializationError, Serializable,
};
use miden_debug_types::{SourceSpan, Spanned};
#[cfg(feature = "serde")]
use serde::{Deserialize, Serialize};
use super::MetaExpr;
use crate::ast::Ident;
#[derive(Debug, Clone)]
#[cfg_attr(feature = "serde", derive(Serialize, Deserialize))]
#[cfg_attr(
all(feature = "arbitrary", test),
miden_test_serde_macros::serde_test(binary_serde(true))
)]
pub struct MetaList {
#[cfg_attr(feature = "serde", serde(skip, default))]
pub span: SourceSpan,
pub name: Ident,
pub items: Vec<MetaExpr>,
}
impl Spanned for MetaList {
#[inline(always)]
fn span(&self) -> SourceSpan {
self.span
}
}
impl MetaList {
pub fn new<I>(name: Ident, items: I) -> Self
where
I: IntoIterator<Item = MetaExpr>,
{
Self {
span: SourceSpan::default(),
name,
items: items.into_iter().collect(),
}
}
pub fn with_span(mut self, span: SourceSpan) -> Self {
self.span = span;
self
}
pub fn name(&self) -> &str {
self.name.as_str()
}
pub fn id(&self) -> Ident {
self.name.clone()
}
#[inline]
pub fn is_empty(&self) -> bool {
self.items.is_empty()
}
#[inline]
pub fn len(&self) -> usize {
self.items.len()
}
#[inline]
pub fn as_slice(&self) -> &[MetaExpr] {
self.items.as_slice()
}
#[inline]
pub fn as_mut_slice(&mut self) -> &mut [MetaExpr] {
self.items.as_mut_slice()
}
}
impl Eq for MetaList {}
impl PartialEq for MetaList {
fn eq(&self, other: &Self) -> bool {
self.name == other.name && self.items == other.items
}
}
impl PartialOrd for MetaList {
#[inline]
fn partial_cmp(&self, other: &Self) -> Option<core::cmp::Ordering> {
Some(self.cmp(other))
}
}
impl Ord for MetaList {
fn cmp(&self, other: &Self) -> core::cmp::Ordering {
self.name.cmp(&other.name).then_with(|| self.items.cmp(&other.items))
}
}
impl core::hash::Hash for MetaList {
fn hash<H: core::hash::Hasher>(&self, state: &mut H) {
self.name.hash(state);
self.items.hash(state);
}
}
#[cfg(feature = "arbitrary")]
impl proptest::arbitrary::Arbitrary for MetaList {
type Parameters = ();
fn arbitrary_with(_args: Self::Parameters) -> Self::Strategy {
use proptest::{arbitrary::any, strategy::Strategy};
let name = any::<Ident>();
let items = proptest::collection::vec(any::<MetaExpr>(), 1..3);
(name, items)
.prop_map(|(name, items)| Self { span: SourceSpan::UNKNOWN, name, items })
.boxed()
}
type Strategy = proptest::prelude::BoxedStrategy<Self>;
}
impl Serializable for MetaList {
fn write_into<W: ByteWriter>(&self, target: &mut W) {
self.name.write_into(target);
self.items.write_into(target);
}
}
impl Deserializable for MetaList {
fn read_from<R: ByteReader>(source: &mut R) -> Result<Self, DeserializationError> {
let name = Ident::read_from(source)?;
let items = Vec::read_from(source)?;
Ok(Self { span: SourceSpan::UNKNOWN, name, items })
}
}