如何使用spirit解析器获取错误位置

2024-02-18

我用spirit编写了一个简单的解析器,类似于json(但更简单、更专业)。按照以下建议here http://boost-spirit.com/home/articles/qi-example/tracking-the-input-position-while-parsing/,我尝试通过跟踪错误位置来实现错误处理。具体来说,我的解析函数如下

bool parse_properties(std::istream& is, const std::string &filename, PropertyList &pset)
{
    namespace qi = boost::spirit::qi;
    namespace ascii = boost::spirit::ascii;
    namespace classic = boost::spirit::classic;

    typedef std::istreambuf_iterator<char> base_iterator_type;
    base_iterator_type in_begin(is);

    // convert input iterator to forward iterator, usable by spirit parser
    typedef boost::spirit::multi_pass<base_iterator_type> forward_iterator_type;
    forward_iterator_type fwd_begin = boost::spirit::make_default_multi_pass(in_begin);
    forward_iterator_type fwd_end;

    // wrap forward iterator with position iterator, to record the position
    typedef classic::position_iterator2<forward_iterator_type> pos_iterator_type;
    pos_iterator_type position_begin(fwd_begin, fwd_end, filename);
    pos_iterator_type position_end;

    qi::rule<pos_iterator_type> skipper = ascii::space | 
    '#' >> *(ascii::char_ - qi::eol) >> qi::eol; 

    property_set_grammar<pos_iterator_type, qi::rule<pos_iterator_type> > g;
    bool r = false;
    try {
        r = phrase_parse(position_begin, 
                         position_end, 
             g, skipper, pset);
    }
    catch(const qi::expectation_failure<pos_iterator_type>& e) {
        const classic::file_position_base<std::string>& pos = e.first.get_position();
        std::stringstream msg;
        msg <<
            "parse error at file " << pos.file <<
            " line " << pos.line << " column " << pos.column << std::endl <<
            "'" << e.first.get_currentline() << "'" << std::endl <<
            std::setw(pos.column) << " " << "^- here";
        throw std::runtime_error(msg.str());
    }

    return r;
}

不幸的是,它不起作用。功能phrase_parse对于正确和不正确的文件,总是立即返回 false,并且从不引发任何异常。

但是,当我修改上面的代码以使用简单的forward_iterator而不是classic::position_iterator2它工作正常,但当然它不会跟踪错误位置。非常奇怪的是,原来的例子here http://boost-spirit.com/home/articles/qi-example/tracking-the-input-position-while-parsing/工作正常。所以也许问题与我的语法有关。如下:

template <typename Iterator, typename Skipper>
struct property_set_grammar : qi::grammar<Iterator, PropertyList(),
                                          Skipper>
{
    qi::rule<Iterator, Property(), Skipper> prop;
    qi::rule<Iterator, std::string(), Skipper> name;
    qi::rule<Iterator, std::string(), Skipper> type;
    qi::rule<Iterator, std::string(), Skipper> value;
    qi::rule<Iterator, std::string(), Skipper> value_simple;
    qi::rule<Iterator, std::string(), Skipper> value_quoted;
    qi::rule<Iterator, PropertyList(), Skipper> plist;

    property_set_grammar() : 
        property_set_grammar::base_type(plist, "Set of Properties") {
        using qi::lit;
        using qi::alpha;
        using qi::alnum;
        using qi::lexeme;
        using qi::char_;

        name = lexeme[alpha >> *alnum];
        type = lexeme[alpha >> *alnum];
        value_simple = lexeme[*(alnum - lit('"'))];
        value_quoted = lit('"') > lexeme[*(char_ - lit('"'))] > lit('"');
        value = (value_quoted | value_simple);

        prop = name >> '=' > value > ';';
        plist = type >> '(' > name > ')' > '{' >> *(prop | plist) > '}' > ';';
    }
};   

我正在使用 g++ (Ubuntu/Linaro 4.7.2-2ubuntu1) 4.7.2 和 boost 库版本 1.50。

我忽略了什么愚蠢的事情吗?如果需要,我可以提供完整的代码(只是几个文件)。


看来 boost::spirit::position_iterator 有问题。

您可以编辑标题position_iterator正如答案中所建议的here https://stackoverflow.com/questions/14005273/strange-error-with-boostspiritposition-iterator2.

或者你可以实现你自己的position_iterator,我基本上是通过复制并粘贴原始代码来实现的boost::spirit::position_iterator,然后删除一些不需要的东西。

Also, when parsing from istream, make sure to set the noskipws manip:

//is >> std::noskipws;

EDIT:istreambuf_iterator 不需要

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

如何使用spirit解析器获取错误位置 的相关文章

随机推荐