#[macro_use]
use crate::*;
use crate::base::*;
use crate::label;
use crate::labelr;
use nom::{
multi::separated_list0,
sequence::tuple as nom_tuple,
};
use crate::nodes::Matrix;
pub fn max_err<'a>(x: Option<ParseError<'a>>, y: ParseError<'a>) -> ParseError<'a> {
match (x,&y) {
(None, y) => y.clone(),
_ => y.clone(),
}
}
pub fn structure(input: ParseString) -> ParseResult<Structure> {
match empty_set(input.clone()) {
Ok((input, set)) => {return Ok((input, Structure::Set(set)));},
_ => (),
}
match empty_map(input.clone()) {
Ok((input, map)) => {return Ok((input, Structure::Map(map)));},
_ => (),
}
match table(input.clone()) {
Ok((input, tbl)) => {return Ok((input, Structure::Table(tbl)));},
_ => (),
}
match matrix(input.clone()) {
Ok((input, mtrx)) => {return Ok((input, Structure::Matrix(mtrx)));},
_ => (),
}
match tuple(input.clone()) {
Ok((input, tpl)) => {return Ok((input, Structure::Tuple(tpl)));},
_ => (),
}
match tuple_struct(input.clone()) {
Ok((input, tpl)) => {return Ok((input, Structure::TupleStruct(tpl)));},
_ => (),
}
match record(input.clone()) {
Ok((input, table)) => {return Ok((input, Structure::Record(table)));},
_ => (),
}
match map(input.clone()) {
Ok((input, map)) => {return Ok((input, Structure::Map(map)));},
_ => (),
}
match set(input.clone()) {
Ok((input, set)) => {return Ok((input, Structure::Set(set)));},
Err(err) => {return Err(err);}
}
}
pub fn tuple(input: ParseString) -> ParseResult<Tuple> {
let (input, _) = left_parenthesis(input)?;
let (input, _) = whitespace0(input)?;
let (input, exprs) = separated_list0(list_separator, expression)(input)?;
let (input, _) = whitespace0(input)?;
let (input, _) = right_parenthesis(input)?;
Ok((input, Tuple{elements: exprs}))
}
pub fn atom(input: ParseString) -> ParseResult<Atom> {
let (input, _) = grave(input)?;
let (input, name) = identifier(input)?;
Ok((input, Atom{name}))
}
pub fn tuple_struct(input: ParseString) -> ParseResult<TupleStruct> {
let (input, _) = grave(input)?;
let (input, name) = identifier(input)?;
let (input, _) = left_parenthesis(input)?;
let (input, _) = whitespace0(input)?;
let (input, value) = expression(input)?;
let (input, _) = whitespace0(input)?;
let (input, _) = right_parenthesis(input)?;
Ok((input, TupleStruct{name, value: Box::new(value)}))
}
pub fn binding(input: ParseString) -> ParseResult<Binding> {
let msg1 = "Unexpected space before colon ':'";
let msg2 = "Expects a value";
let msg3 = "Expects whitespace or comma followed by whitespace";
let msg4 = "Expects whitespace";
let (input, _) = whitespace0(input)?;
let (input, name) = identifier(input)?;
let (input, kind) = opt(kind_annotation)(input)?;
let (input, _) = label!(is_not(nom_tuple((many1(space), colon))), msg1)(input)?;
let (input, _) = colon(input)?;
let (input, _) = whitespace1(input)?;
let (input, value) = label!(expression, msg2)(input)?;
let (input, _) = whitespace0(input)?;
let (input, _) = opt(comma)(input)?;
let (input, _) = whitespace0(input)?;
Ok((input, Binding{name, kind, value}))
}
pub fn table_column(input: ParseString) -> ParseResult<TableColumn> {
let (input, _) = many0(space_tab)(input)?;
let (input, element) = match expression(input) {
Ok(result) => result,
Err(err) => {
return Err(err);
}
};
let (input, _) = nom_tuple((many0(space_tab),opt(alt((comma,table_separator))), many0(space_tab)))(input)?;
Ok((input, TableColumn{element}))
}
pub fn matrix_column(input: ParseString) -> ParseResult<MatrixColumn> {
let (input, _) = many0(space_tab)(input)?;
let (input, element) = match expression(input) {
Ok(result) => result,
Err(err) => {
return Err(err);
}
};
let (input, _) = nom_tuple((many0(space_tab),opt(alt((comma,table_separator))), many0(space_tab)))(input)?;
Ok((input, MatrixColumn{element}))
}
pub fn table_row(input: ParseString) -> ParseResult<TableRow> {
let (input, _) = opt(table_separator)(input)?;
let (input, _) = many0(space_tab)(input)?;
let (input, columns) = match many1(table_column)(input) {
Ok(result) => result,
Err(error) => {
return Err(error);
}
};
let (input, _) = nom_tuple((opt(semicolon), opt(new_line)))(input)?;
let (input, _) = opt(nom_tuple((many1(box_drawing_char),new_line)))(input)?;
Ok((input, TableRow{columns}))
}
pub fn matrix_row(input: ParseString) -> ParseResult<MatrixRow> {
let (input, _) = opt(table_separator)(input)?;
let (input, _) = many0(space_tab)(input)?;
let (input, columns) = match many1(matrix_column)(input) {
Ok(result) => result,
Err(error) => {
return Err(error);
}
};
let (input, _) = nom_tuple((opt(semicolon), opt(new_line)))(input)?;
let (input, _) = opt(nom_tuple((many1(box_drawing_char),new_line)))(input)?;
Ok((input, MatrixRow{columns}))
}
pub fn table_header(input: ParseString) -> ParseResult<Vec<Field>> {
let (input, fields) = separated_list1(many1(space_tab),field)(input)?;
let (input, _) = many0(space_tab)(input)?;
let (input, _) = alt((bar,box_vert))(input)?;
let (input, _) = whitespace0(input)?;
Ok((input, fields))
}
pub fn field(input: ParseString) -> ParseResult<Field> {
let (input, name) = identifier(input)?;
let (input, kind) = kind_annotation(input)?;
Ok((input, Field{name, kind}))
}
pub fn box_drawing_char(input: ParseString) -> ParseResult<Token> {
alt((box_tr_round, box_bl_round, box_vert, box_cross, box_horz, box_t_left, box_t_right, box_t_top, box_t_bottom))(input)
}
pub fn box_drawing_emoji(input: ParseString) -> ParseResult<Token> {
alt((box_tl_round, box_br_round, box_tr_round, box_bl_round, box_vert, box_cross, box_horz, box_t_left, box_t_right, box_t_top, box_t_bottom))(input)
}
pub fn matrix_start(input: ParseString) -> ParseResult<Token> {
alt((box_tl_round, left_bracket))(input)
}
pub fn matrix_end(input: ParseString) -> ParseResult<Token> {
let result = alt((box_br_round, right_bracket))(input);
result
}
pub fn table_start(input: ParseString) -> ParseResult<Token> {
alt((box_tl_round, left_brace))(input)
}
pub fn table_end(input: ParseString) -> ParseResult<Token> {
let result = alt((box_br_round, right_brace))(input);
result
}
pub fn table_separator(input: ParseString) -> ParseResult<Token> {
let (input, token) = box_vert(input)?;
Ok((input, token))
}
pub fn matrix(input: ParseString) -> ParseResult<Matrix> {
let msg = "Expects right bracket ']' to finish the matrix";
let (input, (_, r)) = range(matrix_start)(input)?;
let (input, _) = many0(alt((box_drawing_char,whitespace)))(input)?;
let (input, rows) = many0(matrix_row)(input)?;
let (input, _) = many0(box_drawing_char)(input)?;
let (input, _) = whitespace0(input)?;
let (input, _) = match label!(matrix_end, msg, r)(input) {
Ok(k) => k,
Err(err) => {
return Err(err);
}
};
Ok((input, Matrix{rows}))
}
pub fn table(input: ParseString) -> ParseResult<Table> {
let msg = "Expects right bracket '}' to finish the table";
let (input, (_, r)) = range(table_start)(input)?;
let (input, _) = many0(alt((box_drawing_char,whitespace)))(input)?;
let (input, header) = table_header(input)?;
let (input, _) = many0(alt((box_drawing_char,whitespace)))(input)?;
let (input, rows) = many1(table_row)(input)?;
let (input, _) = many0(box_drawing_char)(input)?;
let (input, _) = whitespace0(input)?;
let (input, _) = match label!(table_end, msg, r)(input) {
Ok(k) => k,
Err(err) => {
return Err(err);
}
};
Ok((input, Table{header,rows}))
}
pub fn empty_map(input: ParseString) -> ParseResult<Map> {
let (input, _) = table_start(input)?;
let (input, _) = whitespace0(input)?;
let (input, _) = table_end(input)?;
Ok((input, Map{elements: vec![]}))
}
pub fn empty_set(input: ParseString) -> ParseResult<Set> {
let (input, _) = table_start(input)?;
let (input, _) = whitespace0(input)?;
let (input, _) = empty(input)?;
let (input, _) = whitespace0(input)?;
let (input, _) = table_end(input)?;
Ok((input, Set{elements: vec![]}))
}
pub fn record(input: ParseString) -> ParseResult<Record> {
let msg = "Expects right bracket ']' to terminate inline table";
let (input, (_, r)) = range(table_start)(input)?;
let (input, _) = whitespace0(input)?;
let (input, bindings) = many1(binding)(input)?;
let (input, _) = whitespace0(input)?;
let (input, _) = label!(table_end, msg, r)(input)?;
Ok((input, Record{bindings}))
}
pub fn map(input: ParseString) -> ParseResult<Map> {
let msg = "Expects right bracket '}' to terminate inline table";
let (input, (_, r)) = range(left_brace)(input)?;
let (input, _) = whitespace0(input)?;
let (input, elements) = many0(mapping)(input)?;
let (input, _) = whitespace0(input)?;
let (input, _) = label!(right_brace, msg, r)(input)?;
Ok((input, Map{elements}))
}
pub fn mapping(input: ParseString) -> ParseResult<Mapping> {
let msg1 = "Unexpected space before colon ':'";
let msg2 = "Expects a value";
let msg3 = "Expects whitespace or comma followed by whitespace";
let msg4 = "Expects whitespace";
let (input, _) = whitespace0(input)?;
let (input, key) = expression(input)?;
let (input, _) = whitespace0(input)?;
let (input, _) = colon(input)?;
let (input, _) = whitespace0(input)?;
let (input, value) = label!(expression, msg2)(input)?;
let (input, _) = whitespace0(input)?;
let (input, _) = opt(comma)(input)?;
let (input, _) = whitespace0(input)?;
Ok((input, Mapping{key, value}))
}
pub fn set(input: ParseString) -> ParseResult<Set> {
let msg = "Expects right bracket '}' to terminate inline table";
let (input, (_, r)) = range(left_brace)(input)?;
let (input, _) = whitespace0(input)?;
let (input, elements) = separated_list0(list_separator, expression)(input)?;
let (input, _) = whitespace0(input)?;
let (input, _) = label!(right_brace, msg, r)(input)?;
Ok((input, Set{elements}))
}
pub fn define_operator(input: ParseString) -> ParseResult<()> {
let (input, _) = whitespace0(input)?;
let (input, _) = tag(":=")(input)?;
let (input, _) = whitespace0(input)?;
Ok((input, ()))
}
pub fn output_operator(input: ParseString) -> ParseResult<()> {
let (input, _) = whitespace0(input)?;
let (input, _) = tag("=>")(input)?;
let (input, _) = whitespace0(input)?;
Ok((input, ()))
}
pub fn async_transition_operator(input: ParseString) -> ParseResult<()> {
let (input, _) = whitespace0(input)?;
let (input, _) = tag("~>")(input)?;
let (input, _) = whitespace0(input)?;
Ok((input, ()))
}
pub fn transition_operator(input: ParseString) -> ParseResult<()> {
let (input, _) = whitespace0(input)?;
let (input, _) = tag("->")(input)?;
let (input, _) = whitespace0(input)?;
Ok((input, ()))
}
pub fn guard_operator(input: ParseString) -> ParseResult<()> {
let (input, _) = whitespace0(input)?;
let (input, _) = alt((tag("|"),tag("│"),tag("├"),tag("└")))(input)?;
let (input, _) = whitespace0(input)?;
Ok((input, ()))
}
pub fn fsm_implementation(input: ParseString) -> ParseResult<FsmImplementation> {
let ((input, _)) = hashtag(input)?;
let ((input, name)) = identifier(input)?;
let ((input, _)) = left_parenthesis(input)?;
let ((input, input_vars)) = separated_list0(list_separator, identifier)(input)?;
let ((input, _)) = right_parenthesis(input)?;
let ((input, _)) = transition_operator(input)?;
let ((input, start)) = fsm_pattern(input)?;
let ((input, _)) = whitespace0(input)?;
let ((input, arms)) = many1(fsm_arm)(input)?;
let ((input, _)) = period(input)?;
Ok((input, FsmImplementation{name,input: input_vars,start,arms}))
}
pub fn fsm_arm(input: ParseString) -> ParseResult<FsmArm> {
let ((input, _)) = many0(comment)(input)?;
let ((input, start)) = fsm_pattern(input)?;
let ((input, trns)) = many1(alt((fsm_state_transition,fsm_output,fsm_guard)))(input)?;
let ((input, _)) = whitespace0(input)?;
Ok((input, FsmArm{start, transitions: trns}))
}
pub fn fsm_guard(input: ParseString) -> ParseResult<Transition> {
let (input, _) = alt((transition_operator,guard_operator))(input)?;
let (input, expr) = match wildcard(input.clone()) {
Ok((input, _)) => (input, Guard::Wildcard),
_ => match expression(input.clone()) {
Ok((input, expr)) => (input, Guard::Expression(expr)),
Err(err) => {return Err(err);}
}
};
Ok((input, Transition::Guard(expr)))
}
pub fn wildcard(input: ParseString) -> ParseResult<Pattern> {
let ((input, _)) = asterisk(input)?;
Ok((input, Pattern::Wildcard))
}
pub fn fsm_state_transition(input: ParseString) -> ParseResult<Transition> {
let (input, _) = transition_operator(input)?;
let ((input, ptrn)) = fsm_pattern(input)?;
Ok((input, Transition::Next(ptrn)))
}
pub fn fsm_async_transition(input: ParseString) -> ParseResult<Transition> {
let (input, _) = async_transition_operator(input)?;
let ((input, ptrn)) = fsm_pattern(input)?;
Ok((input, Transition::Async(ptrn)))
}
pub fn fsm_output(input: ParseString) -> ParseResult<Transition> {
let (input, _) = output_operator(input)?;
let ((input, ptrn)) = fsm_pattern(input)?;
Ok((input, Transition::Output(ptrn)))
}
pub fn fsm_specification(input: ParseString) -> ParseResult<FsmSpecification> {
let ((input, _)) = hashtag(input)?;
let ((input, name)) = identifier(input)?;
let ((input, _)) = left_parenthesis(input)?;
let ((input, input_vars)) = separated_list0(list_separator, identifier)(input)?;
let ((input, _)) = right_parenthesis(input)?;
let ((input, _)) = output_operator(input)?;
let ((input, output)) = identifier(input)?;
let ((input, _)) = define_operator(input)?;
let ((input, states)) = many1(fsm_state_definition)(input)?;
let ((input, _)) = period(input)?;
Ok((input, FsmSpecification{name,input: input_vars,output,states}))
}
pub fn fsm_pattern(input: ParseString) -> ParseResult<Pattern> {
match fsm_tuple_struct(input.clone()) {
Ok((input, tpl)) => {return Ok((input, Pattern::TupleStruct(tpl)))},
_ => ()
}
match wildcard(input.clone()) {
Ok((input, _)) => {return Ok((input, Pattern::Wildcard))},
_ => ()
}
match formula(input.clone()) {
Ok((input, Factor::Expression(expr))) => {return Ok((input, Pattern::Expression(*expr)))},
Ok((input, frmla)) => {return Ok((input, Pattern::Formula(frmla)))},
Err(err) => {return Err(err)},
}
}
pub fn fsm_tuple_struct(input: ParseString) -> ParseResult<PatternTupleStruct> {
let (input, id) = identifier(input)?;
let ((input, _)) = left_parenthesis(input)?;
let ((input, patterns)) = separated_list1(list_separator, fsm_pattern)(input)?;
let ((input, _)) = right_parenthesis(input)?;
Ok((input, PatternTupleStruct{name: id, patterns}))
}
pub fn fsm_state_definition(input: ParseString) -> ParseResult<StateDefinition> {
let ((input, _)) = guard_operator(input)?;
let ((input, name)) = identifier(input)?;
let ((input, vars)) = opt(fsm_state_definition_variables)(input)?;
Ok((input, StateDefinition{name,state_variables: vars}))
}
pub fn fsm_state_definition_variables(input: ParseString) -> ParseResult<Vec<Identifier>> {
let ((input, _)) = left_parenthesis(input)?;
let ((input, names)) = separated_list1(list_separator, identifier)(input)?;
let ((input, _)) = right_parenthesis(input)?;
Ok((input, names))
}
pub fn fsm_pipe(input: ParseString) -> ParseResult<FsmPipe> {
let ((input, start)) = fsm_instance(input)?;
let ((input, trns)) = many0(alt((fsm_state_transition,fsm_async_transition,fsm_output,fsm_guard)))(input)?;
Ok((input, FsmPipe{start, transitions: trns}))
}
pub fn fsm_declare(input: ParseString) -> ParseResult<FsmDeclare> {
let (input, fsm) = fsm(input)?;
let (input, _) = define_operator(input)?;
let (input, pipe) = fsm_pipe(input)?;
Ok((input, FsmDeclare{fsm,pipe}))
}
pub fn fsm(input: ParseString) -> ParseResult<Fsm> {
let ((input, _)) = hashtag(input)?;
let ((input, name)) = identifier(input)?;
let ((input, args)) = opt(argument_list)(input)?;
let ((input, kind)) = opt(kind_annotation)(input)?;
Ok((input, Fsm{ name, args, kind }))
}
pub fn fsm_instance(input: ParseString) -> ParseResult<FsmInstance> {
let ((input, _)) = hashtag(input)?;
let (input, name) = identifier(input)?;
let (input, args) = opt(fsm_args)(input)?;
Ok((input, FsmInstance{name,args} ))
}
pub fn fsm_args(input: ParseString) -> ParseResult<Vec<(Option<Identifier>,Expression)>> {
let (input, _) = left_parenthesis(input)?;
let (input, args) = separated_list0(list_separator, alt((call_arg_with_binding,call_arg)))(input)?;
let (input, _) = right_parenthesis(input)?;
Ok((input, args))
}