diff --git a/hugr-core/src/hugr/rewrite/replace.rs b/hugr-core/src/hugr/rewrite/replace.rs index 9c8ce6731..979b0d928 100644 --- a/hugr-core/src/hugr/rewrite/replace.rs +++ b/hugr-core/src/hugr/rewrite/replace.rs @@ -458,30 +458,27 @@ mod test { use crate::ops::dataflow::DataflowOpTrait; use crate::ops::handle::{BasicBlockID, ConstID, NodeHandle}; use crate::ops::{self, Case, DataflowBlock, OpTag, OpType, DFG}; - use crate::std_extensions::collections; - use crate::types::{Signature, Type, TypeArg, TypeRow}; + use crate::std_extensions::collections::{self, list_type, ListOp}; + use crate::types::{Signature, Type, TypeRow}; use crate::utils::depth; use crate::{type_row, Direction, Hugr, HugrView, OutgoingPort}; use super::{NewEdgeKind, NewEdgeSpec, ReplaceError, Replacement}; #[test] + #[ignore] // FIXME: This needs a rewrite now that `pop` returns an optional value -.-' fn cfg() -> Result<(), Box> { let reg = ExtensionRegistry::try_new([PRELUDE.to_owned(), collections::EXTENSION.to_owned()]) .unwrap(); - let listy = Type::new_extension( - collections::EXTENSION - .get_type(&collections::LIST_TYPENAME) - .unwrap() - .instantiate([TypeArg::Type { ty: USIZE_T }]) - .unwrap(), - ); - let pop: ExtensionOp = collections::EXTENSION - .instantiate_extension_op("pop", [TypeArg::Type { ty: USIZE_T }], ®) + let listy = list_type(USIZE_T); + let pop: ExtensionOp = ListOp::pop + .with_type(USIZE_T) + .to_extension_op(®) .unwrap(); - let push: ExtensionOp = collections::EXTENSION - .instantiate_extension_op("push", [TypeArg::Type { ty: USIZE_T }], ®) + let push: ExtensionOp = ListOp::push + .with_type(USIZE_T) + .to_extension_op(®) .unwrap(); let just_list = TypeRow::from(vec![listy.clone()]); let intermed = TypeRow::from(vec![listy.clone(), USIZE_T]); diff --git a/hugr-core/src/std_extensions/collections.rs b/hugr-core/src/std_extensions/collections.rs index 67178c076..d56fb9f07 100644 --- a/hugr-core/src/std_extensions/collections.rs +++ b/hugr-core/src/std_extensions/collections.rs @@ -9,6 +9,7 @@ use lazy_static::lazy_static; use serde::{Deserialize, Serialize}; use strum_macros::{EnumIter, EnumString, IntoStaticStr}; +use crate::extension::prelude::{option_type, result_type, USIZE_T}; use crate::extension::simple_op::{MakeOpDef, MakeRegisteredOp}; use crate::extension::{ExtensionBuildError, OpDef, SignatureFunc, PRELUDE}; use crate::ops::constant::ValueName; @@ -110,17 +111,30 @@ impl CustomConst for ListValue { #[allow(non_camel_case_types)] #[non_exhaustive] pub enum ListOp { - /// Pop from end of list + /// Pop from the end of list. Return an optional value. pop, - /// Push to end of list + /// Push to end of list. Return the new list. push, + /// Lookup an element in a list by index. + get, + /// Replace the element at index `i` with value `v`, and return the old value. + /// + /// If the index is out of bounds, returns the input value as an error. + set, + /// Insert an element at index `i`. + /// + /// Elements at higher indices are shifted one position to the right. + /// Returns an Err with the element if the index is out of bounds. + insert, + /// Get the length of a list. + length, } impl ListOp { /// Type parameter used in the list types. const TP: TypeParam = TypeParam::Type { b: TypeBound::Any }; - /// Instantiate a list operation with an `element_type` + /// Instantiate a list operation with an `element_type`. pub fn with_type(self, element_type: Type) -> ListOpInst { ListOpInst { elem_type: element_type, @@ -135,9 +149,25 @@ impl ListOp { let l = self.list_type(list_type_def, 0); match self { pop => self - .list_polytype(vec![l.clone()], vec![l.clone(), e.clone()]) + .list_polytype(vec![l.clone()], vec![l, Type::from(option_type(e))]) .into(), push => self.list_polytype(vec![l.clone(), e], vec![l]).into(), + get => self + .list_polytype(vec![l, USIZE_T], vec![Type::from(option_type(e))]) + .into(), + set => self + .list_polytype( + vec![l.clone(), USIZE_T, e.clone()], + vec![l, Type::from(result_type(e.clone(), e))], + ) + .into(), + insert => self + .list_polytype( + vec![l.clone(), USIZE_T, e.clone()], + vec![l, result_type(Type::UNIT, e).into()], + ) + .into(), + length => self.list_polytype(vec![l], vec![USIZE_T]).into(), } } @@ -191,8 +221,12 @@ impl MakeOpDef for ListOp { use ListOp::*; match self { - pop => "Pop from back of list", - push => "Push to back of list", + pop => "Pop from the back of list. Returns an optional value.", + push => "Push to the back of list", + get => "Lookup an element in a list by index. Panics if the index is out of bounds.", + set => "Replace the element at index `i` with value `v`.", + insert => "Insert an element at index `i`. Elements at higher indices are shifted one position to the right. Panics if the index is out of bounds.", + length => "Get the length of a list", } .into() } @@ -205,7 +239,6 @@ impl MakeOpDef for ListOp { lazy_static! { /// Extension for list operations. pub static ref EXTENSION: Extension = { - println!("creating collections extension"); let mut extension = Extension::new(EXTENSION_ID, VERSION); // The list type must be defined before the operations are added. @@ -323,7 +356,13 @@ impl ListOpInst { #[cfg(test)] mod test { + use rstest::rstest; + + use crate::extension::prelude::{ + const_err_tuple, const_none, const_ok_tuple, const_some_tuple, + }; use crate::ops::OpTrait; + use crate::PortIndex; use crate::{ extension::{ prelude::{ConstUsize, QB_T, USIZE_T}, @@ -378,7 +417,7 @@ mod test { let list_t = list_type(QB_T); - let both_row: TypeRow = vec![list_t.clone(), QB_T].into(); + let both_row: TypeRow = vec![list_t.clone(), option_type(QB_T).into()].into(); let just_list_row: TypeRow = vec![list_t].into(); assert_eq!(pop_sig.input(), &just_list_row); assert_eq!(pop_sig.output(), &both_row); @@ -396,4 +435,81 @@ mod test { assert_eq!(push_sig.input(), &both_row); assert_eq!(push_sig.output(), &just_list_row); } + + /// Values used in the `list_fold` test cases. + #[derive(Debug, Clone, PartialEq, Eq)] + enum TestVal { + Idx(usize), + List(Vec), + Elem(usize), + Some(Vec), + None(TypeRow), + Ok(Vec, TypeRow), + Err(TypeRow, Vec), + } + + impl TestVal { + fn to_value(&self) -> Value { + match self { + TestVal::Idx(i) => Value::extension(ConstUsize::new(*i as u64)), + TestVal::Elem(e) => Value::extension(ConstUsize::new(*e as u64)), + TestVal::List(l) => { + let elems = l + .iter() + .map(|&i| Value::extension(ConstUsize::new(i as u64))) + .collect(); + Value::extension(ListValue(elems, USIZE_T)) + } + TestVal::Some(l) => { + let elems = l.iter().map(TestVal::to_value); + const_some_tuple(elems) + } + TestVal::None(tr) => const_none(tr.clone()), + TestVal::Ok(l, tr) => { + let elems = l.iter().map(TestVal::to_value); + const_ok_tuple(elems, tr.clone()) + } + TestVal::Err(tr, l) => { + let elems = l.iter().map(TestVal::to_value); + const_err_tuple(tr.clone(), elems) + } + } + } + } + + #[rstest] + #[case::pop(ListOp::pop, &[TestVal::List(vec![77,88, 42])], &[TestVal::List(vec![77,88]), TestVal::Some(vec![TestVal::Elem(42)])])] + #[case::pop_empty(ListOp::pop, &[TestVal::List(vec![])], &[TestVal::List(vec![]), TestVal::None(vec![USIZE_T].into())])] + #[case::push(ListOp::push, &[TestVal::List(vec![77,88]), TestVal::Elem(42)], &[TestVal::List(vec![77,88,42])])] + #[case::get(ListOp::get, &[TestVal::List(vec![77,88,42]), TestVal::Idx(1)], &[TestVal::Some(vec![TestVal::Elem(88)])])] + #[case::get_invalid(ListOp::get, &[TestVal::List(vec![77,88,42]), TestVal::Idx(99)], &[TestVal::None(vec![USIZE_T].into())])] + #[case::insert(ListOp::insert, &[TestVal::List(vec![77,88,42]), TestVal::Idx(1), TestVal::Elem(99)], &[TestVal::List(vec![77,99,88,42]), TestVal::Ok(vec![], vec![USIZE_T].into())])] + #[case::insert_invalid(ListOp::insert, &[TestVal::List(vec![77,88,42]), TestVal::Idx(52), TestVal::Elem(99)], &[TestVal::List(vec![77,88,42]), TestVal::Err(Type::UNIT.into(), vec![TestVal::Elem(99)])])] + #[case::length(ListOp::length, &[TestVal::List(vec![77,88,42])], &[TestVal::Elem(3)])] + fn list_fold(#[case] op: ListOp, #[case] inputs: &[TestVal], #[case] outputs: &[TestVal]) { + let consts: Vec<_> = inputs + .iter() + .enumerate() + .map(|(i, x)| (i.into(), x.to_value())) + .collect(); + + let res = op + .with_type(USIZE_T) + .to_extension_op(&COLLECTIONS_REGISTRY) + .unwrap() + .constant_fold(&consts) + .unwrap(); + + for (i, expected) in outputs.iter().enumerate() { + let expected = expected.to_value(); + let res_val = res + .iter() + .find(|(port, _)| port.index() == i) + .unwrap() + .1 + .clone(); + + assert_eq!(res_val, expected); + } + } } diff --git a/hugr-core/src/std_extensions/collections/list_fold.rs b/hugr-core/src/std_extensions/collections/list_fold.rs index ca9b78c77..09820887a 100644 --- a/hugr-core/src/std_extensions/collections/list_fold.rs +++ b/hugr-core/src/std_extensions/collections/list_fold.rs @@ -1,9 +1,14 @@ //! Folding definitions for list operations. -use crate::extension::{ConstFold, OpDef}; -use crate::ops; +use crate::extension::prelude::{ + const_err, const_none, const_ok, const_ok_tuple, const_some, ConstUsize, +}; +use crate::extension::{ConstFold, ConstFoldResult, OpDef}; +use crate::ops::Value; use crate::types::type_param::TypeArg; +use crate::types::Type; use crate::utils::sorted_consts; +use crate::IncomingPort; use super::{ListOp, ListValue}; @@ -11,6 +16,10 @@ pub(super) fn set_fold(op: &ListOp, def: &mut OpDef) { match op { ListOp::pop => def.set_constant_folder(PopFold), ListOp::push => def.set_constant_folder(PushFold), + ListOp::get => def.set_constant_folder(GetFold), + ListOp::set => def.set_constant_folder(SetFold), + ListOp::insert => def.set_constant_folder(InsertFold), + ListOp::length => def.set_constant_folder(LengthFold), } } @@ -20,14 +29,22 @@ impl ConstFold for PopFold { fn fold( &self, _type_args: &[TypeArg], - consts: &[(crate::IncomingPort, ops::Value)], - ) -> crate::extension::ConstFoldResult { - let [list]: [&ops::Value; 1] = sorted_consts(consts).try_into().ok()?; + consts: &[(crate::IncomingPort, Value)], + ) -> ConstFoldResult { + let [list]: [&Value; 1] = sorted_consts(consts).try_into().ok()?; let list: &ListValue = list.get_custom_value().expect("Should be list value."); let mut list = list.clone(); - let elem = list.0.pop()?; // empty list fails to evaluate "pop" - Some(vec![(0.into(), list.into()), (1.into(), elem)]) + match list.0.pop() { + Some(elem) => Some(vec![(0.into(), list.into()), (1.into(), const_some(elem))]), + None => { + let elem_type = list.1.clone(); + Some(vec![ + (0.into(), list.into()), + (1.into(), const_none(elem_type)), + ]) + } + } } } @@ -37,9 +54,9 @@ impl ConstFold for PushFold { fn fold( &self, _type_args: &[TypeArg], - consts: &[(crate::IncomingPort, ops::Value)], - ) -> crate::extension::ConstFoldResult { - let [list, elem]: [&ops::Value; 2] = sorted_consts(consts).try_into().ok()?; + consts: &[(crate::IncomingPort, Value)], + ) -> ConstFoldResult { + let [list, elem]: [&Value; 2] = sorted_consts(consts).try_into().ok()?; let list: &ListValue = list.get_custom_value().expect("Should be list value."); let mut list = list.clone(); list.0.push(elem.clone()); @@ -47,3 +64,76 @@ impl ConstFold for PushFold { Some(vec![(0.into(), list.into())]) } } + +pub struct GetFold; + +impl ConstFold for GetFold { + fn fold(&self, _type_args: &[TypeArg], consts: &[(IncomingPort, Value)]) -> ConstFoldResult { + let [list, index]: [&Value; 2] = sorted_consts(consts).try_into().ok()?; + let list: &ListValue = list.get_custom_value().expect("Should be list value."); + let index: &ConstUsize = index.get_custom_value().expect("Should be int value."); + let idx = index.value() as usize; + + match list.0.get(idx) { + Some(elem) => Some(vec![(0.into(), const_some(elem.clone()))]), + None => Some(vec![(0.into(), const_none(list.1.clone()))]), + } + } +} + +pub struct SetFold; + +impl ConstFold for SetFold { + fn fold(&self, _type_args: &[TypeArg], consts: &[(IncomingPort, Value)]) -> ConstFoldResult { + let [list, idx, elem]: [&Value; 3] = sorted_consts(consts).try_into().ok()?; + let list: &ListValue = list.get_custom_value().expect("Should be list value."); + + let idx: &ConstUsize = idx.get_custom_value().expect("Should be int value."); + let idx = idx.value() as usize; + + let mut list = list.clone(); + let mut elem = elem.clone(); + let res_elem: Value = match list.0.get_mut(idx) { + Some(old_elem) => { + std::mem::swap(old_elem, &mut elem); + const_ok(elem, list.1.clone()) + } + None => const_err(list.1.clone(), elem), + }; + Some(vec![(0.into(), list.into()), (1.into(), res_elem)]) + } +} + +pub struct InsertFold; + +impl ConstFold for InsertFold { + fn fold(&self, _type_args: &[TypeArg], consts: &[(IncomingPort, Value)]) -> ConstFoldResult { + let [list, idx, elem]: [&Value; 3] = sorted_consts(consts).try_into().ok()?; + let list: &ListValue = list.get_custom_value().expect("Should be list value."); + + let idx: &ConstUsize = idx.get_custom_value().expect("Should be int value."); + let idx = idx.value() as usize; + + let mut list = list.clone(); + let elem = elem.clone(); + let res_elem: Value = if list.0.len() > idx { + list.0.insert(idx, elem); + const_ok_tuple([], list.1.clone()) + } else { + const_err(Type::UNIT, elem) + }; + Some(vec![(0.into(), list.into()), (1.into(), res_elem)]) + } +} + +pub struct LengthFold; + +impl ConstFold for LengthFold { + fn fold(&self, _type_args: &[TypeArg], consts: &[(IncomingPort, Value)]) -> ConstFoldResult { + let [list]: [&Value; 1] = sorted_consts(consts).try_into().ok()?; + let list: &ListValue = list.get_custom_value().expect("Should be list value."); + let len = list.0.len(); + + Some(vec![(0.into(), ConstUsize::new(len as u64).into())]) + } +} diff --git a/hugr-py/src/hugr/std/_json_defs/collections.json b/hugr-py/src/hugr/std/_json_defs/collections.json index 6263c2074..906cec92e 100644 --- a/hugr-py/src/hugr/std/_json_defs/collections.json +++ b/hugr-py/src/hugr/std/_json_defs/collections.json @@ -23,10 +23,10 @@ }, "values": {}, "operations": { - "pop": { + "get": { "extension": "collections", - "name": "pop", - "description": "Pop from back of list", + "name": "get", + "description": "Lookup an element in a list by index. Panics if the index is out of bounds.", "signature": { "params": [ { @@ -51,9 +51,51 @@ } ], "bound": "A" + }, + { + "t": "I" } ], "output": [ + { + "t": "Sum", + "s": "General", + "rows": [ + [ + { + "t": "V", + "i": 0, + "b": "A" + } + ], + [ + { + "t": "Sum", + "s": "Unit", + "size": 1 + } + ] + ] + } + ], + "extension_reqs": [] + } + }, + "binary": false + }, + "insert": { + "extension": "collections", + "name": "insert", + "description": "Insert an element at index `i`. Elements at higher indices are shifted one position to the right. Panics if the index is out of bounds.", + "signature": { + "params": [ + { + "tp": "Type", + "b": "A" + } + ], + "body": { + "input": [ { "t": "Opaque", "extension": "collections", @@ -70,12 +112,166 @@ ], "bound": "A" }, + { + "t": "I" + }, { "t": "V", "i": 0, "b": "A" } ], + "output": [ + { + "t": "Opaque", + "extension": "collections", + "id": "List", + "args": [ + { + "tya": "Type", + "ty": { + "t": "V", + "i": 0, + "b": "A" + } + } + ], + "bound": "A" + }, + { + "t": "Sum", + "s": "General", + "rows": [ + [ + { + "t": "Sum", + "s": "Unit", + "size": 1 + } + ], + [ + { + "t": "V", + "i": 0, + "b": "A" + } + ] + ] + } + ], + "extension_reqs": [] + } + }, + "binary": false + }, + "length": { + "extension": "collections", + "name": "length", + "description": "Get the length of a list", + "signature": { + "params": [ + { + "tp": "Type", + "b": "A" + } + ], + "body": { + "input": [ + { + "t": "Opaque", + "extension": "collections", + "id": "List", + "args": [ + { + "tya": "Type", + "ty": { + "t": "V", + "i": 0, + "b": "A" + } + } + ], + "bound": "A" + } + ], + "output": [ + { + "t": "I" + } + ], + "extension_reqs": [] + } + }, + "binary": false + }, + "pop": { + "extension": "collections", + "name": "pop", + "description": "Pop from the back of list. Returns an optional value.", + "signature": { + "params": [ + { + "tp": "Type", + "b": "A" + } + ], + "body": { + "input": [ + { + "t": "Opaque", + "extension": "collections", + "id": "List", + "args": [ + { + "tya": "Type", + "ty": { + "t": "V", + "i": 0, + "b": "A" + } + } + ], + "bound": "A" + } + ], + "output": [ + { + "t": "Opaque", + "extension": "collections", + "id": "List", + "args": [ + { + "tya": "Type", + "ty": { + "t": "V", + "i": 0, + "b": "A" + } + } + ], + "bound": "A" + }, + { + "t": "Sum", + "s": "General", + "rows": [ + [ + { + "t": "V", + "i": 0, + "b": "A" + } + ], + [ + { + "t": "Sum", + "s": "Unit", + "size": 1 + } + ] + ] + } + ], "extension_reqs": [] } }, @@ -84,7 +280,65 @@ "push": { "extension": "collections", "name": "push", - "description": "Push to back of list", + "description": "Push to the back of list", + "signature": { + "params": [ + { + "tp": "Type", + "b": "A" + } + ], + "body": { + "input": [ + { + "t": "Opaque", + "extension": "collections", + "id": "List", + "args": [ + { + "tya": "Type", + "ty": { + "t": "V", + "i": 0, + "b": "A" + } + } + ], + "bound": "A" + }, + { + "t": "V", + "i": 0, + "b": "A" + } + ], + "output": [ + { + "t": "Opaque", + "extension": "collections", + "id": "List", + "args": [ + { + "tya": "Type", + "ty": { + "t": "V", + "i": 0, + "b": "A" + } + } + ], + "bound": "A" + } + ], + "extension_reqs": [] + } + }, + "binary": false + }, + "set": { + "extension": "collections", + "name": "set", + "description": "Replace the element at index `i` with value `v`.", "signature": { "params": [ { @@ -110,6 +364,9 @@ ], "bound": "A" }, + { + "t": "I" + }, { "t": "V", "i": 0, @@ -132,6 +389,26 @@ } ], "bound": "A" + }, + { + "t": "Sum", + "s": "General", + "rows": [ + [ + { + "t": "V", + "i": 0, + "b": "A" + } + ], + [ + { + "t": "V", + "i": 0, + "b": "A" + } + ] + ] } ], "extension_reqs": [] diff --git a/hugr-py/src/hugr/tys.py b/hugr-py/src/hugr/tys.py index a779212c8..c6544b5bf 100644 --- a/hugr-py/src/hugr/tys.py +++ b/hugr-py/src/hugr/tys.py @@ -9,8 +9,6 @@ from hugr.utils import ser_it if TYPE_CHECKING: - from collections.abc import Iterable - from hugr import ext @@ -305,41 +303,6 @@ def __repr__(self) -> str: return f"Tuple{tuple(self.variant_rows[0])}" -@dataclass(eq=False) -class Option(Sum): - """Optional tuple of elements. - - Instances of this type correspond to :class:`Sum` with two variants. - The first variant is the tuple of elements, the second is empty. - """ - - def __init__(self, *tys: Type): - self.variant_rows = [list(tys), []] - - def __repr__(self) -> str: - return f"Option({', '.join(map(repr, self.variant_rows[0]))})" - - -@dataclass(eq=False) -class Result(Sum): - """Fallible tuple of elements. - - Instances of this type correspond to :class:`Sum` with two variants. The - first variant is a tuple of elements representing the successful state, the - second is a tuple of elements representing failure. - """ - - def __init__(self, ok: Iterable[Type], err: Iterable[Type]): - self.variant_rows = [list(ok), list(err)] - - def __repr__(self) -> str: - ok = self.variant_rows[0] - err = self.variant_rows[1] - ok_str = ok[0] if len(ok) == 1 else tuple(ok) - err_str = err[0] if len(err) == 1 else tuple(err) - return f"Result({ok_str}, {err_str})" - - @dataclass(frozen=True) class Variable(Type): """A type variable with a given bound, identified by index.""" diff --git a/hugr-py/src/hugr/val.py b/hugr-py/src/hugr/val.py index acc42e84c..210b19dff 100644 --- a/hugr-py/src/hugr/val.py +++ b/hugr-py/src/hugr/val.py @@ -11,8 +11,6 @@ from hugr.utils import ser_it if TYPE_CHECKING: - from collections.abc import Iterable - from hugr.hugr import Hugr @@ -151,114 +149,6 @@ def __repr__(self) -> str: return f"Tuple({', '.join(map(repr, self.vals))})" -@dataclass -class Some(Sum): - """Optional tuple of value, containing a list of values. - Internally a :class:`Sum` with two variant rows. - - Example: - >>> some = Some(TRUE, FALSE) - >>> some - Some(TRUE, FALSE) - >>> some.type_() - Option(Bool, Bool) - - """ - - #: The values of this tuple. - vals: list[Value] - - def __init__(self, *vals: Value): - val_list = list(vals) - super().__init__( - tag=0, typ=tys.Option(*(v.type_() for v in val_list)), vals=val_list - ) - - def __repr__(self) -> str: - return f"Some({', '.join(map(repr, self.vals))})" - - -@dataclass -class None_(Sum): - """Optional tuple of value, containing no values. - Internally a :class:`Sum` with two variant rows. - - Example: - >>> none = None_(tys.Bool) - >>> none - None(Bool) - >>> none.type_() - Option(Bool) - - """ - - def __init__(self, *types: tys.Type): - super().__init__(tag=1, typ=tys.Option(*types), vals=[]) - - def __repr__(self) -> str: - return f"None({', '.join(map(repr, self.typ.variant_rows[0]))})" - - -@dataclass -class Ok(Sum): - """Success variant of a :class:`tys.Result` type, containing a list of values. - - Internally a :class:`Sum` with two variant rows. - - Example: - >>> ok = Ok([TRUE, FALSE], [tys.Bool]) - >>> ok - Ok((TRUE, FALSE), Bool) - >>> ok.type_() - Result((Bool, Bool), Bool) - """ - - #: The values of this tuple. - vals: list[Value] - - def __init__(self, vals: Iterable[Value], err_typ: Iterable[tys.Type]): - val_list = list(vals) - super().__init__( - tag=0, typ=tys.Result([v.type_() for v in val_list], err_typ), vals=val_list - ) - - def __repr__(self) -> str: - vals_str = self.vals[0] if len(self.vals) == 1 else tuple(self.vals) - err = self.typ.variant_rows[1] - err_str = err[0] if len(err) == 1 else tuple(err) - return f"Ok({vals_str}, {err_str})" - - -@dataclass -class Err(Sum): - """Error variant of a :class:`tys.Result` type, containing a list of values. - - Internally a :class:`Sum` with two variant rows. - - Example: - >>> err = Err([tys.Bool, tys.Bool], [TRUE, FALSE]) - >>> err - Err((Bool, Bool), (TRUE, FALSE)) - >>> err.type_() - Result((Bool, Bool), (Bool, Bool)) - """ - - #: The values of this tuple. - vals: list[Value] - - def __init__(self, ok_typ: Iterable[tys.Type], vals: Iterable[Value]): - val_list = list(vals) - super().__init__( - tag=1, typ=tys.Result(ok_typ, [v.type_() for v in val_list]), vals=val_list - ) - - def __repr__(self) -> str: - ok = self.typ.variant_rows[1] - ok_str = ok[0] if len(ok) == 1 else tuple(ok) - vals_str = self.vals[0] if len(self.vals) == 1 else tuple(self.vals) - return f"Err({ok_str}, {vals_str})" - - @dataclass class Function(Value): """Higher order function value, defined by a :class:`Hugr `.""" diff --git a/specification/std_extensions/collections.json b/specification/std_extensions/collections.json index 6263c2074..906cec92e 100644 --- a/specification/std_extensions/collections.json +++ b/specification/std_extensions/collections.json @@ -23,10 +23,10 @@ }, "values": {}, "operations": { - "pop": { + "get": { "extension": "collections", - "name": "pop", - "description": "Pop from back of list", + "name": "get", + "description": "Lookup an element in a list by index. Panics if the index is out of bounds.", "signature": { "params": [ { @@ -51,9 +51,51 @@ } ], "bound": "A" + }, + { + "t": "I" } ], "output": [ + { + "t": "Sum", + "s": "General", + "rows": [ + [ + { + "t": "V", + "i": 0, + "b": "A" + } + ], + [ + { + "t": "Sum", + "s": "Unit", + "size": 1 + } + ] + ] + } + ], + "extension_reqs": [] + } + }, + "binary": false + }, + "insert": { + "extension": "collections", + "name": "insert", + "description": "Insert an element at index `i`. Elements at higher indices are shifted one position to the right. Panics if the index is out of bounds.", + "signature": { + "params": [ + { + "tp": "Type", + "b": "A" + } + ], + "body": { + "input": [ { "t": "Opaque", "extension": "collections", @@ -70,12 +112,166 @@ ], "bound": "A" }, + { + "t": "I" + }, { "t": "V", "i": 0, "b": "A" } ], + "output": [ + { + "t": "Opaque", + "extension": "collections", + "id": "List", + "args": [ + { + "tya": "Type", + "ty": { + "t": "V", + "i": 0, + "b": "A" + } + } + ], + "bound": "A" + }, + { + "t": "Sum", + "s": "General", + "rows": [ + [ + { + "t": "Sum", + "s": "Unit", + "size": 1 + } + ], + [ + { + "t": "V", + "i": 0, + "b": "A" + } + ] + ] + } + ], + "extension_reqs": [] + } + }, + "binary": false + }, + "length": { + "extension": "collections", + "name": "length", + "description": "Get the length of a list", + "signature": { + "params": [ + { + "tp": "Type", + "b": "A" + } + ], + "body": { + "input": [ + { + "t": "Opaque", + "extension": "collections", + "id": "List", + "args": [ + { + "tya": "Type", + "ty": { + "t": "V", + "i": 0, + "b": "A" + } + } + ], + "bound": "A" + } + ], + "output": [ + { + "t": "I" + } + ], + "extension_reqs": [] + } + }, + "binary": false + }, + "pop": { + "extension": "collections", + "name": "pop", + "description": "Pop from the back of list. Returns an optional value.", + "signature": { + "params": [ + { + "tp": "Type", + "b": "A" + } + ], + "body": { + "input": [ + { + "t": "Opaque", + "extension": "collections", + "id": "List", + "args": [ + { + "tya": "Type", + "ty": { + "t": "V", + "i": 0, + "b": "A" + } + } + ], + "bound": "A" + } + ], + "output": [ + { + "t": "Opaque", + "extension": "collections", + "id": "List", + "args": [ + { + "tya": "Type", + "ty": { + "t": "V", + "i": 0, + "b": "A" + } + } + ], + "bound": "A" + }, + { + "t": "Sum", + "s": "General", + "rows": [ + [ + { + "t": "V", + "i": 0, + "b": "A" + } + ], + [ + { + "t": "Sum", + "s": "Unit", + "size": 1 + } + ] + ] + } + ], "extension_reqs": [] } }, @@ -84,7 +280,65 @@ "push": { "extension": "collections", "name": "push", - "description": "Push to back of list", + "description": "Push to the back of list", + "signature": { + "params": [ + { + "tp": "Type", + "b": "A" + } + ], + "body": { + "input": [ + { + "t": "Opaque", + "extension": "collections", + "id": "List", + "args": [ + { + "tya": "Type", + "ty": { + "t": "V", + "i": 0, + "b": "A" + } + } + ], + "bound": "A" + }, + { + "t": "V", + "i": 0, + "b": "A" + } + ], + "output": [ + { + "t": "Opaque", + "extension": "collections", + "id": "List", + "args": [ + { + "tya": "Type", + "ty": { + "t": "V", + "i": 0, + "b": "A" + } + } + ], + "bound": "A" + } + ], + "extension_reqs": [] + } + }, + "binary": false + }, + "set": { + "extension": "collections", + "name": "set", + "description": "Replace the element at index `i` with value `v`.", "signature": { "params": [ { @@ -110,6 +364,9 @@ ], "bound": "A" }, + { + "t": "I" + }, { "t": "V", "i": 0, @@ -132,6 +389,26 @@ } ], "bound": "A" + }, + { + "t": "Sum", + "s": "General", + "rows": [ + [ + { + "t": "V", + "i": 0, + "b": "A" + } + ], + [ + { + "t": "V", + "i": 0, + "b": "A" + } + ] + ] } ], "extension_reqs": []