如何描述EBNF中的范围?

时间:2016-10-03 14:22:31

标签: python ebnf cisco-ios grako ciscoconfparse

我正在尝试使用Grako和Python为Cisco IOS和ASA配置编写解析器。我正在尝试弄清楚如何在EBNF中表示“范围”关键字 - 例如,'description'关键字必须出现在interface范围内,但interface有多个选项,它们是所有可选的(我相信订单可以在设备之间更改):

interface Vlan1435
 nameif untrust
 description the outside interface for customer X
 bridge-group 1
 security-level 0

我发现的最近的例子是名为Farly的Perl应用程序,该应用程序使用perl Parse :: Recdescent模块,它似乎与Grako类似。

从那里我有这种类型的递归定义:

@@eol_comments :: /!([^\n]*?)\n/
@@whitespace :: /[\t ]+/

start
    =
    file_input $
    ;


file_input
    =
    {NEWLINE | asa_line}
    ;


asa_line
    =
      'names' NEWLINE
    | interface NEWLINE
    ;

interface
    =
    'interface' string NEWLINE interface_options
    ;


interface_options
    =
    if_name | sec_level | if_addr | if_bridgegroup | NEWLINE
    ;


if_bridgegroup
    =
    'bridge-group' digit NEWLINE interface_options
    ;


if_name
    =
    'nameif' string NEWLINE interface_options
    ;


sec_level
    =
    'security-level' digit NEWLINE interface_options
    ;

但它会产生一个奇怪的嵌套AST,并且它不会“重置”以检测第二个接口或后续配置中的任何其他内容。

这些范围通常如何在EBNF中定义? (这个类型的东西是否有一个有用的教程?我的google-fu一般没有为Grako或解析器提供任何东西)

1 个答案:

答案 0 :(得分:1)

我在这些情况下使用的技巧是使用重复,即使选项只出现一次:

interface_options
    =
    { @+:(if_name | sec_level | if_addr | if_bridgegroup)  NEWLINE}*
    ;

如有必要,您可以使用语义操作来验证不会重复选项。