schala/schala-lang/src/parsing/mod.rs

115 lines
3.2 KiB
Rust
Raw Normal View History

2021-10-19 22:24:27 -07:00
#![allow(clippy::upper_case_acronyms)]
2021-11-17 01:54:35 -08:00
pub mod combinator;
2021-11-14 04:25:24 -08:00
mod peg_parser;
2021-11-04 21:11:19 -07:00
mod test;
2021-11-18 01:37:05 -08:00
use std::{cell::RefCell, fmt, rc::Rc};
use combinator::Span;
2018-06-04 19:25:40 -07:00
#[cfg(test)]
use crate::ast::{Block, Expression};
2021-11-14 03:55:35 -08:00
use crate::{
ast::{ASTItem, AST},
identifier::{Id, IdStore},
};
pub struct Parser {
id_store: IdStore<ASTItem>,
}
impl Parser {
pub(crate) fn new() -> Self {
Self { id_store: IdStore::new() }
}
pub(crate) fn parse(&mut self, input: &str) -> Result<AST, ParseError> {
peg_parser::schala_parser::program(input, self).map_err(ParseError::from_peg)
}
2021-11-18 15:17:47 -08:00
pub(crate) fn parse_comb(&mut self, input: &str) -> Result<AST, ParseError> {
let id_store: IdStore<ASTItem> = IdStore::new();
let span = Span::new_extra(input, Rc::new(RefCell::new(id_store)));
combinator::program(span).map_err(|err| convert_err(input, err)).map(|(_, output)| output)
}
#[cfg(test)]
fn expression(&mut self, input: &str) -> Result<Expression, ParseError> {
peg_parser::schala_parser::expression(input, self).map_err(ParseError::from_peg)
}
2021-11-14 03:55:35 -08:00
2021-11-17 16:53:03 -08:00
#[cfg(test)]
fn expression_comb(&mut self, input: &str) -> Result<Expression, ParseError> {
let id_store: IdStore<ASTItem> = IdStore::new();
let span = Span::new_extra(input, Rc::new(RefCell::new(id_store)));
2021-11-18 01:37:05 -08:00
2021-11-18 17:16:39 -08:00
combinator::expression(span).map_err(|err| convert_err(input, err)).map(|(rest, output)| output)
2021-11-17 16:53:03 -08:00
}
#[cfg(test)]
fn block(&mut self, input: &str) -> Result<Block, ParseError> {
peg_parser::schala_parser::block(input, self).map_err(ParseError::from_peg)
2021-11-14 03:55:35 -08:00
}
2021-11-18 01:37:05 -08:00
#[cfg(test)]
fn block_comb(&mut self, input: &str) -> Result<Block, ParseError> {
let id_store: IdStore<ASTItem> = IdStore::new();
let span = Span::new_extra(input, Rc::new(RefCell::new(id_store)));
combinator::block(span).map_err(|err| convert_err(input, err)).map(|(_, output)| output)
}
2021-11-14 03:55:35 -08:00
fn fresh(&mut self) -> Id<ASTItem> {
self.id_store.fresh()
}
}
2021-11-18 01:37:05 -08:00
fn convert_err<'a>(
input: &'a str,
err: nom::Err<nom::error::VerboseError<combinator::Span<'a>>>,
) -> ParseError {
use nom::{error::VerboseError, Err};
match err {
Err::Error(err) | Err::Failure(err) => {
let err = VerboseError {
errors: err.errors.into_iter().map(|(sp, kind)| (*sp.fragment(), kind)).collect(),
};
let msg = nom::error::convert_error(input, err);
ParseError { msg, location: (0).into() }
}
_ => panic!(),
}
}
/// Represents a parsing error
#[derive(Debug)]
2017-09-11 02:07:17 -07:00
pub struct ParseError {
pub msg: String,
2021-11-14 02:15:08 -08:00
pub location: Location,
}
impl ParseError {
fn from_peg(err: peg::error::ParseError<peg::str::LineCol>) -> Self {
let msg = err.to_string();
Self { msg, location: err.location.offset.into() }
}
}
#[derive(Debug, Clone, Copy, PartialEq, Default)]
pub struct Location {
pub(crate) offset: usize,
2017-09-11 02:07:17 -07:00
}
2017-09-09 00:31:15 -07:00
impl From<usize> for Location {
fn from(offset: usize) -> Self {
Self { offset }
}
2017-09-11 23:27:15 -07:00
}
impl fmt::Display for Location {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "{}", self.offset)
}
2018-01-08 05:57:36 -08:00
}