#lexer #parser-combinator #tokenizer #interpreter #tokenize #parser

nightly langbox

A simple framework to build compilers and interpreters

9 releases (5 breaking)

0.6.0 Jul 14, 2024
0.5.0 Oct 11, 2023
0.4.0 Aug 31, 2023
0.3.0 Apr 12, 2023
0.1.1 Jul 29, 2022

#507 in Programming languages

Download history 110/week @ 2024-07-13 11/week @ 2024-07-20 22/week @ 2024-07-27 3/week @ 2024-08-31 14/week @ 2024-09-14 44/week @ 2024-09-21 7/week @ 2024-09-28

65 downloads per month

MIT license

60KB
1.5K SLoC

A simple framework to build compilers and interpreters

This crate requires a nightly compiler because of the try_trait_v2 feature.

Usage

use langbox::*;

enum JsonTokenKind {
    // ...
}

// Make the reader an uninhabited type because we don't construct any objects of it
enum JsonTokenReader {}

impl TokenReader for JsonTokenReader {
    type Token = JsonTokenKind;
    
    fn read_token(text: &str) -> ReadTokenResult<Self::Token> {
        // ...
    }
}

struct JsonValue {
    // ...
}

fn jvalue() -> impl Parser<JsonTokenKind, JsonValue, String> {
    // ...
}

fn main() {
    // FileServer manages loading files for us.
    // It ensures the same physical file is never loaded twice.
    let mut file_server = FileServer::new();
    let file_id = file_server.register_file( /* ... */ );

    // After we have loaded a file we can tokenize it using a Lexer.
    type JLexer<'a> = Lexer<'a, JsonTokenReader, whitespace_mode::Remove>;
    let lexer = JLexer::new(file_id, &file_server);
    let tokens = lexer.collect::<Vec<_>>();
    let stream = TokenStream::new(&tokens);

    // Finally after all files have been tokenized we can parse the token stream.
    match jvalue().run(stream).expect("malformed JSON input") {
        InfallibleParseResult::Match { value, .. } => {
            // `value` contains the parsed JSON value
        }
        InfallibleParseResult::NoMatch => { /* empty input */ }
    }
}

Dependencies

~4.5MB
~86K SLoC