conlang: Introduce ..rest Patterns, refactor Ranges
This commit is contained in:
@@ -8,7 +8,10 @@ use crate::{
|
||||
error::{Error, IResult},
|
||||
};
|
||||
use cl_ast::{Literal, Pattern, Sym};
|
||||
use std::{collections::HashMap, rc::Rc};
|
||||
use std::{
|
||||
collections::{HashMap, VecDeque},
|
||||
rc::Rc,
|
||||
};
|
||||
|
||||
/// Gets the path variables in the given Pattern
|
||||
pub fn variables(pat: &Pattern) -> Vec<&Sym> {
|
||||
@@ -17,6 +20,8 @@ pub fn variables(pat: &Pattern) -> Vec<&Sym> {
|
||||
Pattern::Name(name) if &**name == "_" => {}
|
||||
Pattern::Name(name) => set.push(name),
|
||||
Pattern::Literal(_) => {}
|
||||
Pattern::Rest(Some(pattern)) => patvars(set, pattern),
|
||||
Pattern::Rest(None) => {}
|
||||
Pattern::Ref(_, pattern) => patvars(set, pattern),
|
||||
Pattern::Tuple(patterns) | Pattern::Array(patterns) => {
|
||||
patterns.iter().for_each(|pat| patvars(set, pat))
|
||||
@@ -37,6 +42,41 @@ pub fn variables(pat: &Pattern) -> Vec<&Sym> {
|
||||
set
|
||||
}
|
||||
|
||||
fn rest_binding<'pat>(
|
||||
sub: &mut HashMap<&'pat Sym, ConValue>,
|
||||
mut patterns: &'pat [Pattern],
|
||||
mut values: VecDeque<ConValue>,
|
||||
) -> IResult<Option<(&'pat Pattern, VecDeque<ConValue>)>> {
|
||||
// Bind the head of the list
|
||||
while let [pattern, tail @ ..] = patterns {
|
||||
if matches!(pattern, Pattern::Rest(_)) {
|
||||
break;
|
||||
}
|
||||
let value = values
|
||||
.pop_front()
|
||||
.ok_or_else(|| Error::PatFailed(Box::new(pattern.clone())))?;
|
||||
append_sub(sub, pattern, value)?;
|
||||
patterns = tail;
|
||||
}
|
||||
// Bind the tail of the list
|
||||
while let [head @ .., pattern] = patterns {
|
||||
if matches!(pattern, Pattern::Rest(_)) {
|
||||
break;
|
||||
}
|
||||
let value = values
|
||||
.pop_back()
|
||||
.ok_or_else(|| Error::PatFailed(Box::new(pattern.clone())))?;
|
||||
append_sub(sub, pattern, value)?;
|
||||
patterns = head;
|
||||
}
|
||||
// Bind the ..rest of the list
|
||||
match patterns {
|
||||
[] | [Pattern::Rest(None)] => Ok(None),
|
||||
[Pattern::Rest(Some(pattern))] => Ok(Some((pattern.as_ref(), values))),
|
||||
_ => Err(Error::PatFailed(Box::new(Pattern::Array(patterns.into())))),
|
||||
}
|
||||
}
|
||||
|
||||
/// Appends a substitution to the provided table
|
||||
pub fn append_sub<'pat>(
|
||||
sub: &mut HashMap<&'pat Sym, ConValue>,
|
||||
@@ -44,18 +84,6 @@ pub fn append_sub<'pat>(
|
||||
value: ConValue,
|
||||
) -> IResult<()> {
|
||||
match (pat, value) {
|
||||
(Pattern::Array(patterns), ConValue::Array(values))
|
||||
| (Pattern::Tuple(patterns), ConValue::Tuple(values)) => {
|
||||
if patterns.len() != values.len() {
|
||||
Err(Error::ArgNumber { want: patterns.len(), got: values.len() })?
|
||||
}
|
||||
for (pat, value) in patterns.iter().zip(Vec::from(values).into_iter()) {
|
||||
append_sub(sub, pat, value)?;
|
||||
}
|
||||
Ok(())
|
||||
}
|
||||
(Pattern::Tuple(patterns), ConValue::Empty) if patterns.is_empty() => Ok(()),
|
||||
|
||||
(Pattern::Literal(Literal::Bool(a)), ConValue::Bool(b)) => {
|
||||
(*a == b).then_some(()).ok_or(Error::NotAssignable)
|
||||
}
|
||||
@@ -73,6 +101,25 @@ pub fn append_sub<'pat>(
|
||||
}
|
||||
(Pattern::Literal(_), _) => Err(Error::NotAssignable),
|
||||
|
||||
(Pattern::Rest(Some(pat)), value) => match (pat.as_ref(), value) {
|
||||
(Pattern::Literal(Literal::Int(a)), ConValue::Int(b)) => {
|
||||
(b < *a as _).then_some(()).ok_or(Error::NotAssignable)
|
||||
}
|
||||
(Pattern::Literal(Literal::Char(a)), ConValue::Char(b)) => {
|
||||
(b < *a as _).then_some(()).ok_or(Error::NotAssignable)
|
||||
}
|
||||
(Pattern::Literal(Literal::Bool(a)), ConValue::Bool(b)) => {
|
||||
(!b & *a).then_some(()).ok_or(Error::NotAssignable)
|
||||
}
|
||||
(Pattern::Literal(Literal::Float(a)), ConValue::Float(b)) => {
|
||||
(b < *a as _).then_some(()).ok_or(Error::NotAssignable)
|
||||
}
|
||||
(Pattern::Literal(Literal::String(a)), ConValue::String(b)) => {
|
||||
(&*b < a).then_some(()).ok_or(Error::NotAssignable)
|
||||
}
|
||||
_ => Err(Error::NotAssignable)
|
||||
},
|
||||
|
||||
(Pattern::Name(name), _) if "_".eq(&**name) => Ok(()),
|
||||
(Pattern::Name(name), value) => {
|
||||
sub.insert(name, value);
|
||||
@@ -81,14 +128,43 @@ pub fn append_sub<'pat>(
|
||||
|
||||
(Pattern::Ref(_, pat), ConValue::Ref(r)) => append_sub(sub, pat, Rc::unwrap_or_clone(r)),
|
||||
|
||||
(Pattern::Array(patterns), ConValue::Array(values)) => {
|
||||
match rest_binding(sub, patterns, values.into_vec().into())? {
|
||||
Some((pattern, values)) => {
|
||||
append_sub(sub, pattern, ConValue::Array(Vec::from(values).into()))
|
||||
}
|
||||
_ => Ok(()),
|
||||
}
|
||||
}
|
||||
|
||||
(Pattern::Tuple(patterns), ConValue::Empty) if patterns.is_empty() => Ok(()),
|
||||
(Pattern::Tuple(patterns), ConValue::Tuple(values)) => {
|
||||
match rest_binding(sub, patterns, values.into_vec().into())? {
|
||||
Some((pattern, values)) => {
|
||||
append_sub(sub, pattern, ConValue::Tuple(Vec::from(values).into()))
|
||||
}
|
||||
_ => Ok(()),
|
||||
}
|
||||
}
|
||||
|
||||
(Pattern::TupleStruct(path, patterns), ConValue::TupleStruct(parts)) => {
|
||||
let (name, values) = *parts;
|
||||
if !path.ends_with(name) {
|
||||
Err(Error::TypeError)?
|
||||
}
|
||||
match rest_binding(sub, patterns, values.into_vec().into())? {
|
||||
Some((pattern, values)) => {
|
||||
append_sub(sub, pattern, ConValue::Tuple(Vec::from(values).into()))
|
||||
}
|
||||
_ => Ok(()),
|
||||
}
|
||||
}
|
||||
|
||||
(Pattern::Struct(path, patterns), ConValue::Struct(parts)) => {
|
||||
let (name, mut values) = *parts;
|
||||
if !path.ends_with(&name) {
|
||||
Err(Error::TypeError)?
|
||||
}
|
||||
if patterns.len() != values.len() {
|
||||
return Err(Error::ArgNumber { want: patterns.len(), got: values.len() });
|
||||
}
|
||||
for (name, pat) in patterns {
|
||||
let value = values.remove(name).ok_or(Error::TypeError)?;
|
||||
match pat {
|
||||
@@ -101,23 +177,9 @@ pub fn append_sub<'pat>(
|
||||
Ok(())
|
||||
}
|
||||
|
||||
(Pattern::TupleStruct(path, patterns), ConValue::TupleStruct(parts)) => {
|
||||
let (name, values) = *parts;
|
||||
if !path.ends_with(&name) {
|
||||
Err(Error::TypeError)?
|
||||
}
|
||||
if patterns.len() != values.len() {
|
||||
Err(Error::ArgNumber { want: patterns.len(), got: values.len() })?
|
||||
}
|
||||
for (pat, value) in patterns.iter().zip(Vec::from(values).into_iter()) {
|
||||
append_sub(sub, pat, value)?;
|
||||
}
|
||||
Ok(())
|
||||
}
|
||||
|
||||
(pat, value) => {
|
||||
eprintln!("Could not match pattern `{pat}` with value `{value}`!");
|
||||
Err(Error::NotAssignable)
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user