mirror of
https://github.com/yhirose/cpp-peglib.git
synced 2025-01-22 13:25:30 +00:00
74 lines
1.5 KiB
C++
74 lines
1.5 KiB
C++
//
|
|
// indent.cc
|
|
//
|
|
// Copyright (c) 2022 Yuji Hirose. All rights reserved.
|
|
// MIT License
|
|
//
|
|
|
|
// Based on https://gist.github.com/dmajda/04002578dd41ae8190fc
|
|
|
|
#include <cstdlib>
|
|
#include <iostream>
|
|
#include <peglib.h>
|
|
|
|
using namespace peg;
|
|
|
|
int main(void) {
|
|
parser parser(R"(Start <- Statements {}
|
|
Statements <- Statement*
|
|
Statement <- Samedent (S / I)
|
|
|
|
S <- 'S' EOS { no_ast_opt }
|
|
I <- 'I' EOL Block / 'I' EOS { no_ast_opt }
|
|
|
|
Block <- Statements {}
|
|
|
|
~Samedent <- ' '* {}
|
|
|
|
~EOS <- EOL / EOF
|
|
~EOL <- '\n'
|
|
~EOF <- !.
|
|
)");
|
|
|
|
size_t indent = 0;
|
|
|
|
parser["Block"].enter = [&](const Context & /*c*/, const char * /*s*/,
|
|
size_t /*n*/, std::any & /*dt*/) { indent += 2; };
|
|
|
|
parser["Block"].leave = [&](const Context & /*c*/, const char * /*s*/,
|
|
size_t /*n*/, size_t /*matchlen*/,
|
|
std::any & /*value*/,
|
|
std::any & /*dt*/) { indent -= 2; };
|
|
|
|
parser["Samedent"].predicate =
|
|
[&](const SemanticValues &vs, const std::any & /*dt*/, std::string &msg) {
|
|
if (indent != vs.sv().size()) {
|
|
msg = "different indent...";
|
|
return false;
|
|
}
|
|
return true;
|
|
};
|
|
|
|
parser.enable_ast();
|
|
|
|
const auto source = R"(I
|
|
S
|
|
I
|
|
I
|
|
S
|
|
S
|
|
S
|
|
S
|
|
)";
|
|
|
|
std::shared_ptr<Ast> ast;
|
|
if (parser.parse(source, ast)) {
|
|
ast = parser.optimize_ast(ast);
|
|
std::cout << ast_to_s(ast);
|
|
return 0;
|
|
}
|
|
|
|
std::cout << "syntax error..." << std::endl;
|
|
return -1;
|
|
}
|