Techioz Blog

Ruby でコンパイルされた protobuf メッセージから enum 値を取得する

概要

次のようなコンパイル済み Ruby protobuf メッセージがあります。

  require 'google/protobuf'

  Google::Protobuf::DescriptorPool.generated_pool.build do
    add_message "PingPacket" do
      optional :message_counter, :int32, 1
      optional :message_type, :enum, 2, "PingPacket.MessageType"
    end
    add_enum "PingPacket.MessageType" do
      value :REPORT, 0
      value :LOW_BATTERY, 1
      value :LOCATE_REQUEST, 2
      value :CHECK_IN, 3
      value :SOS, 4
      value :RESTING, 5
      value :MOVING, 6
      value :EVENT, 7
      value :SYSTEM_TEST, 8
    end
  end

  PingPacket = Google::Protobuf::DescriptorPool.generated_pool.lookup("PingPacket").msgclass
  PingPacket::MessageType = Google::Protobuf::DescriptorPool.generated_pool.lookup("PingPacket.MessageType").enummodule

すべての MessageType 値を含む配列を取得しようとしています。私は明白なことを試してみました:

PingPacket::MessageType.enums
PingPacket::MessageType.values
PingPacket::MessageType.to_s

しかし、何も機能しません。どうすればそれらの値を取得できますか?

解決策

私は Pry で物事を検査するのが好きです。pry コンソールにコードをロードすると、次の結果が得られます。

  1. あなたのクラスはモジュールです
[2] pry(main)> PingPacket::MessageType.class
=> Module

クラス内に入ると、次のようになります。

[4] pry(main)> cd PingPacket::MessageType
[5] pry(PingPacket::MessageType):1> ls
constants: 
  CHECK_IN  LOCATE_REQUEST  MOVING  RESTING  SYSTEM_TEST
  EVENT     LOW_BATTERY     REPORT  SOS    
PingPacket::MessageType.methods: descriptor  lookup  resolve
locals: _  __  _dir_  _ex_  _file_  _in_  _out_  _pry_

次に、すべての定数を検査できます。

[6] pry(PingPacket::MessageType):1> constants
=> [:CHECK_IN,
 :SOS,
 :RESTING,
 :MOVING,
 :EVENT,
 :SYSTEM_TEST,
 :REPORT,
 :LOW_BATTERY,
 :LOCATE_REQUEST]

最後に、このトリックを使用してモジュールを形成する定数値を取得できます。

[9] pry(PingPacket::MessageType):1> constants(false).map &method(:const_get)
=> [3, 4, 5, 6, 7, 8, 0, 1, 2]

これでうまくいきます

[12] pry(main)> PingPacket::MessageType.constants(false).map &PingPacket::MessageType.method(:const_get)
=> [3, 4, 5, 6, 7, 8, 0, 1, 2]

また、次のように機能する 3 つのメソッドがあることもわかります。

[31] pry(PingPacket::MessageType):1> resolve :CHECK_IN
=> 3
[33] pry(PingPacket::MessageType):1> lookup 3
=> :CHECK_IN
[37] pry(PingPacket::MessageType):1> descriptor.each do |i|
[37] pry(PingPacket::MessageType):1* puts i
[37] pry(PingPacket::MessageType):1* end
LOCATE_REQUEST
SOS
SYSTEM_TEST
LOW_BATTERY
EVENT
CHECK_IN
RESTING
MOVING
REPORT
=> nil

たとえばこれを確認してください:

[42] pry(PingPacket::MessageType):1> descriptor.each do |i|
[42] pry(PingPacket::MessageType):1* puts resolve i
[42] pry(PingPacket::MessageType):1* end
2
4
8
1
7
3
5
6
0
=> nil

最後にすべてを結合して、すべてのキーと値をハッシュに入れましょう

[54] pry(main)> Hash[PingPacket::MessageType.descriptor.collect do |i| [i, PingPacket::MessageType.resolve(i)] end]
=> {:LOCATE_REQUEST=>2,
 :SOS=>4,
 :SYSTEM_TEST=>8,
 :LOW_BATTERY=>1,
 :EVENT=>7,
 :CHECK_IN=>3,
 :RESTING=>5,
 :MOVING=>6,
 :REPORT=>0}