2

我尝试使用 protobuf 编译器解析 .proto 文件。但这是一个令人困惑的问题,我无法获得方法的选项。

似乎我的选项被视为“未知字段”,而不是选项。

有没有办法解决这个问题?谢谢。

(我讨厌在这里粘贴很多代码,但我认为完整描述问题很重要。对此感到抱歉。)

(环境:g++ 4.7,Ubuntu 16.04,Protobuf 3.0.0)

#include <google/protobuf/descriptor.h>
#include <google/protobuf/descriptor.pb.h>
#include <google/protobuf/dynamic_message.h>
#include <google/protobuf/compiler/importer.h>

using namespace std;
using namespace google::protobuf;
using namespace google::protobuf::compiler;

#define print(x) std::cout << x << std::endl
#define input(x) std::cin >> x

int main() {
    DiskSourceTree sourceTree;
    sourceTree.MapPath("", "./");
    sourceTree.MapPath("", "./protobuf/include/");
    Importer importer(&sourceTree, NULL);

    auto fd = importer.Import("example.proto");
    assert(fd);

    int service_count = fd->service_count();

    for (int i = 0; i < service_count; i++) {
        auto service_d = fd->service(i);
        int method_count = service_d->method_count();
        for (int j = 0; j < method_count; j++) {
            auto method_d = service_d->method(j);
            print(method_d->options().unknown_fields().field_count());$ 
            print(">> " << method_d->options().uninterpreted_option_size());
        }
    }
    return 0;
}

// lrpc.proto
syntax = "proto3";
package lrpc;
import "google/protobuf/descriptor.proto";


extend google.protobuf.MethodOptions {
    int32 CmdID      = 50000;
    string OptString = 50001;
    string Usage     = 50002;
}

// example.proto
syntax = "proto3";

package foobar;

import "google/protobuf/wrappers.proto";
import "google/protobuf/empty.proto";

import "lrpc.proto";

message SearchRequest {
    // ... 
}

message SearchResponse {
    // ...
}

service SearchService {
    rpc Search( SearchRequest ) returns( SearchResponse ) {
        option( lrpc.CmdID ) = 1;
    }
}

4

1 回答 1

2

这些选项不是未知字段,因为它们是扩展!扩展名应该在 proto3 中被删除,但是当您.proto使用 动态解析文件时Importer,无论您声明的语法版本如何,都会启用扩展名。

如果您在内部 for() 循环中添加一行,例如:

print(method_d->options().DebugString());

你会得到如下输出:

[lrpc.CmdID]: 1

您可以使用 protobuf 反射枚举扩展值——它们会在您调用时显示Reflection::ListFields()

于 2016-12-17T06:11:54.237 回答