rust - 使用 nom 匹配模板过滤器表达式

标签 rust parser-combinators nom

我正在开发一个模板引擎,其中一些语法可能是这样的:

{{ somevar|filter }}

代替 somevar 可以是任意“表达式”,也就是说,可以是像 somevar 这样的变量名,也可以是嵌套的过滤器表达式(如 {{ somevar|filter|anotherfilter }}).我正在尝试使用 Rust 的 nom 解析器组合器库来解析它,但到目前为止未能让它工作。

这是我到目前为止提出的解析器:

#[macro_use]
extern crate nom;

use std::str;

#[derive(Debug)]
pub enum Expr<'a> {
    Var(&'a [u8]),
    Filter(&'a str, Box<Expr<'a>>),
}

#[derive(Debug)]
pub enum Node<'a> {
    Lit(&'a [u8]),
    Expr(Expr<'a>),
}

named!(expr_var<Expr>, dbg_dmp!(map!(nom::alphanumeric, Expr::Var)));

named!(expr_filter<Expr>,
    dbg_dmp!(do_parse!(
         val: any_expr >>
         tag_s!("|") >>
         name: map_res!(nom::alphanumeric, str::from_utf8) >>
         (Expr::Filter(name, Box::new(val)))
    ))
);

named!(any_expr<Expr>, dbg_dmp!(ws!(
    alt_complete!(
        expr_filter |
        expr_var  
    ))));

named!(expr_node<Node>, dbg_dmp!(map!(
    delimited!(tag_s!("{{"), any_expr, tag_s!("}}")),
    Node::Expr)));

named!(parse_template< Vec<Node> >, many1!(expr_node));

playground .当前版本因堆栈溢出而 panic 。我可以通过反转 expr_var | 来解决这个问题expr_filter 命令在 any_expr 中,但我又回到了与以前基本相同的错误。

最佳答案

我不能说我理解你的问题:没有应该解析的文本示例,你也没有描述你在构建解析器时遇到的问题。

不过,也许下面的例子会有帮助。一个有效的递归解析器:

#[macro_use]
extern crate nom;

use nom::alphanumeric;

type Variable = String;
type Filter = String;

named! (plain_expression (&str) -> (Variable, Filter), do_parse! (
    tag_s! ("{{") >>
    variable: alphanumeric >>
    tag_s! ("|") >>
    filter: alphanumeric >>
    tag_s! ("}}") >>
    ((variable.into(), filter.into()))));

#[derive(Debug)]
enum Expression {
    Plain(Variable, Filter),
    Recursive(Box<Expression>, Filter),
}

named! (recursive_expression (&str) -> Expression,
  alt_complete! (
    map! (plain_expression, |(v, f)| Expression::Plain (v, f)) |
    do_parse! (
      tag_s! ("{{") >>
      sub: recursive_expression >>
      tag_s! ("|") >>
      filter: alphanumeric >>
      tag_s! ("}}") >>
      (Expression::Recursive (Box::new (sub), filter.into())))));

fn main() {
    let plain = "{{var|fil}}";
    let recursive = "{{{{{{var1|fil1}}|fil2}}|fil3}}";
    // Prints: Done("", ("var", "fil")).
    println!("{:?}", plain_expression(plain));
    // Prints: Done("", Recursive(Recursive(Plain("var1", "fil1"), "fil2"), "fil3")).
    println!("{:?}", recursive_expression(recursive));
}

(playground)。

关于rust - 使用 nom 匹配模板过滤器表达式,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41449860/

相关文章:

rust - 使用 nom 捕获整个连续的匹配输入

parsing - 如何使用 nom 精确匹配一个字节?

rust - 如何从函数返回 Any?

rust - 如何从构建脚本 (build.rs) 访问当前的 cargo profile (build, test, bench, doc, ....)

parsing - Scala 解析器组合器与 Haskell 的 Parsec 相比如何?

Scala解析器组合器: how to invert matches?

rust - Rust函数语法问题,示例出现在nom中

parsing - 如何将任何字符串向量与 nom 匹配?

utf-8 - 使用rust 的println!在某些情况下打印奇怪的字符

python - 在 Rust 中使用 python 的多处理库