Export to GitHub

ruby-protobuf - Tutorials.wiki


See also: http://code.google.com/apis/protocolbuffers/docs/tutorials.html

Table of Contents

Install

The gem is registered to a gem server, so that all you should do to install is to execute the following command:

$ sudo gem install ruby_protobuf

Compile Your .proto File

$ rprotoc PROTOFILE

When PROTOFILE is named 'person.proto,' the name of the generated ruby source is 'person.pb.rb.'

To know about other options, use -h option.

$ rprotoc -h rprotoc [OPTIONS] PROTO_FILE -p, --proto_path <PATH> Specify the directory in which to search for imports. The current directory is default. -o, --out <OUT_DIR> Specify the directory in which Ruby source file is generated. The current directory is default. -v, --version Show version. -h, --help Show this message.

Generated Ruby Code

A .proto file like this:

``` package tutorial;

message Person { required string name = 1; required int32 id = 2; optional string email = 3;

enum PhoneType { MOBILE = 0; HOME = 1; WORK = 2; }

message PhoneNumber { required string number = 1; optional PhoneType type = 2 [default = HOME]; }

repeated PhoneNumber phone = 4;

extensions 100 to 200; }

extend Person { optional int32 age = 100; }

message AddressBook { repeated Person person = 1; } ```

generates ruby code like this:

``` module Tutorial class Person < ::Protobuf::Message required :string, :name, 1 required :int32, :id, 2 optional :string, :email, 3

class PhoneType < ::Protobuf::Enum
  define :MOBILE, 0
  define :HOME, 1
  define :WORK, 2
end

class PhoneNumber < ::Protobuf::Message
  required :string, :number, 1
  optional :PhoneType, :type, 2, :default => :HOME
end

repeated :PhoneNumber, :phone, 4
extensions 100..200

end

class Person < ::Protobuf::Message optional :int32, :age, 100, :extension => true end

class AddressBook < ::Protobuf::Message repeated :Person, :person, 1 end end ```

The Protocol Buffer API

As you can see in the previous section, messages and fields in a proto file is respectively converted to ruby classes derived from Protobuf::Message class and their attributes. Using the class and its instance, you could write:

require 'addressbook.pb' include Tutorial person = Person.new person.id = 1234 person.name = 'John Doe' person.email = 'jdoe@example.com' phone = Person::PhoneNumber.new phone.number = '555-4321' phone.type = Person::PhoneType::HOME person.phone << phone

Note that the type of variable is checked in accordance with the proto file when assigned:

person.no_such_field = 1 # raises NoMethodError person.id = '1234' # raises TypeError

Enums

Enum field can accept integer value, symbol value or EnumValue object. On general case, you will use integers or symbols, and may not notice EnumValue object because EnumValue object is automatically converted into integer value.

For example, when .proto is: enum EnumType { FIRST = 1; SECOND = 2; } message Message { required EnumType enum = 1; }

generated code is:

class EnumType < ::Protobuf::Enum define :FIRST, 1 define :SECOND, 2 end class Message < ::Protobuf::Message required :EnumType, :enum, 1 end

and you can use like this: ``` EnumType::FIRST.class #=> Protobuf::EnumValue EnumType::FIRST == 1 #=> true 1 == EnumType::FIRST #=> true

message = Message.new message.enum = EnumType::FIRST # OK message.enum = 1 # OK message.enum = :FIRST # OK message.enum.class #=> Protobuf::EnumValue

puts "value is #{message.enum}" #=> "value is FIRST" ```

Standard Message Methods

  • initialized? : checks if all the required fields have been set.
  • dup : duplicate the message.
  • clear! : clears all the elements back to the empty state.
  • has_field?(field_name) : checks a field value is already set or not. It's useful for optional fields.
  • inspect : returns a human-readable representation of the message.

Parsing and Serialization

Finally, each protocol buffer class has methods for writing and reading messages of your chosen type using the protocol buffer binary format. These include:

  • serialize_to_string : serializes the message and returns it as a string. Note that the bytes are binary, not text; we only use the str type as a convenient container.
  • to_s : alias of serialize_to_string.
  • serialize_to_file(filename)
  • serialize_to(stream)
  • parse_from_string(data) : parses a message from the given string.
  • parse_from_file(filename)
  • parse_from(stream)

Sample (Mini) Applications

Writing A Message

examples/writing_a_message.rb

```

!/usr/local/bin/ruby

require 'addressbook.pb'

def prompt_for_address(person) print 'Enter person ID number: ' person.id = STDIN.gets.strip.to_i print 'Enter name: ' person.name = STDIN.gets.strip print 'Enter email address (blank for none): ' email = STDIN.gets.strip person.email = email unless email.empty?

loop do print 'Enter a phone number (or leave blank to finish): ' break if (number = STDIN.gets.strip).empty?

person.phone << Tutorial::Person::PhoneNumber.new
person.phone.last.number = number

print 'Is this a mobile, home, or work phone? '
person.phone.last.type =
  case type = STDIN.gets.strip
  when 'mobile'
    Tutorial::Person::PhoneType::MOBILE
  when 'home'
    Tutorial::Person::PhoneType::HOME
  when 'work'
    Tutorial::Person::PhoneType::WORK
  else
    puts 'Unknown phone type; leaving as default value.'
    nil
  end

end end

unless ARGV.size == 1 puts "Usage: #{$0} ADDRESS_BOOK_FILE" exit end

address_book = Tutorial::AddressBook.new address_book.parse_from_file ARGV[0] if File.exist? ARGV[0] address_book.person << Tutorial::Person.new prompt_for_address address_book.person.last address_book.serialize_to_file ARGV[0] ```

Reading A Message

examples/reading_a_message.rb

```

!/usr/local/bin/ruby

require 'addressbook.pb'

def list_people(address_book) address_book.person.each do |person| puts "Person ID: #{person.id}" puts " Name: #{person.name}" puts " E-mail: #{person.email}" unless person.email.empty? person.phone.each do |phone_number| print(case phone_number.type when Tutorial::Person::PhoneType::MOBILE ' Mobile phone #: ' when Tutorial::Person::PhoneType::HOME ' Home phone #: ' when Tutorial::Person::PhoneType::WORK ' Work phone #: ' end) puts phone_number.number end end end

unless ARGV.size == 1 puts "Usage: #{$0} ADDRESS_BOOK_FILE" exit end

address_book = Tutorial::AddressBook.new address_book.parse_from_file ARGV[0]

list_people address_book ```