Protocol Buffers are a way of encoding structured data in an efficient yet extensible format. It might be used in file formats and RPC protocols.
PHP Protobuf is Google's Protocol Buffers implementation for PHP with a goal to provide high performance, including a protoc
plugin to generate PHP classes from .proto files. The heavy-lifting (a parsing and a serialization) is done by a PHP extension.
- PHP 7.0 or above (for PHP 5 support refer to php5 branch)
- Pear's Console_CommandLine (for the protoc plugin)
- Google's protoc compiler version 2.6 or above
-
Clone the source code
git clone https://github.com/allegro/php-protobuf
-
Go to the source code directory
cd php-protobuf
-
Build and install the PHP extension (follow instructions at php.net)
-
Install protoc plugin dependencies
composer install
-
Locate your global composer directory (usually ~/.composer)
-
Add the git repository to the composer.json
"repositories": [ { "type": "vcs", "url": "https://github.com/JackNoordhuis/php-protobuf" } ]
-
Add the library as a global dependency
"require": [ "jacknoordhuis/php-protobuf": "dev-master" ]
-
Finally, run
composer global update
to install the required dependencies -
You can now run the
protoc-gen-php
command from anywhere
-
Assume you have a file
foo.proto
message Foo { required int32 bar = 1; optional string baz = 2; repeated float spam = 3; }
-
Compile
foo.proto
php protoc-gen-php.php foo.proto
-
Create
Foo
message and populate it with some datarequire_once 'Foo.php'; $foo = new Foo(); $foo->setBar(1); $foo->setBaz('two'); $foo->appendSpam(3.0); $foo->appendSpam(4.0);
-
Serialize a message to a string
$packed = $foo->serializeToString();
-
Parse a message from a string
$parsedFoo = new Foo(); try { $parsedFoo->parseFromString($packed); } catch (Exception $ex) { die('Oops.. there is a bug in this example, ' . $ex->getMessage()); }
-
Let's see what we parsed out
$parsedFoo->dump();
It should produce output similar to the following:
Foo { 1: bar => 1 2: baz => 'two' 3: spam(2) => [0] => 3 [1] => 4 }
-
If you would like you can reset an object to its initial state
$parsedFoo->reset();
PHP Protobuf comes with Google's protoc compiler plugin. You can run in directly:
php protoc-gen-php.php -o output_dir foo.proto
or pass it to the protoc:
protoc --plugin=protoc-gen-allegrophp=protoc-gen-php.php --allegrophp_out=output_dir foo.proto
On Windows use protoc-gen-php.bat
instead.
- -o out, --out=out - the destination directory for generated files (defaults to the current directory).
- -I proto_path, --proto_path=proto_path - the directory in which to search for imports.
- --protoc=protoc - the protoc compiler executable path.
- -D define, --define=define - define a generator option (i.e. -Dnamespace='Foo\Bar\Baz').
- namespace - the namespace to be used by the generated PHP classes.
The classes generated during the compilation are PSR-0 compliant (each class is put into it's own file). If namespace
generator option is not defined then a package name (if present) is used to create a namespace. If the package name is not set then a class is put into global space.
PHP Protobuf module implements ProtobufMessage
class which encapsulates the protocol logic. A message compiled from a proto file extends this class providing message field descriptors. Based on these descriptors ProtobufMessage knows how to parse and serialize a message of a given type.
For each field a set of accessors is generated. The set of methods is different for single value fields (required
/ optional
) and multi-value fields (repeated
).
-
required
/optional
get{FIELD}() // return a field value has{FIELD}() // check whether a field is set set{FIELD}($value) // set a field value to $value
-
repeated
append{FIELD}($value) // append $value to a field clear{FIELD}() // empty field get{FIELD}() // return an array of field values getAt{FIELD}($index) // return a field value at $index index getCount{FIELD}() // return a number of field values has{FIELD}() // check whether a field is set getIterator{FIELD}() // return an ArrayIterator
{FIELD} is a camel cased field name.
PHP does not natively support enum type. Hence enum is represented by the PHP integer type. For convenience enum is compiled to a class with set of constants corresponding to its possible values.
The range of available build-in PHP types poses some limitations. PHP does not support 64-bit positive integer type. Note that parsing big integer values might result in getting unexpected results.
Protocol Buffers types map to PHP types as follows (x86_64):
| Protocol Buffers | PHP |
| ---------------- | ------ |
| double | float |
| float | |
| ---------------- | ------ |
| int32 | int |
| int64 | |
| uint32 | |
| uint64 | |
| sint32 | |
| sint64 | |
| fixed32 | |
| fixed64 | |
| sfixed32 | |
| sfixed64 | |
| ---------------- | ------ |
| bool | bool |
| ---------------- | ------ |
| string | string |
| bytes | |
Protocol Buffers types map to PHP types as follows (x86):
| Protocol Buffers | PHP |
| ---------------- | --------------------------- |
| double | float |
| float | |
| ---------------- | --------------------------- |
| int32 | int |
| uint32 | |
| sint32 | |
| fixed32 | |
| sfixed32 | |
| ---------------- | --------------------------- |
| int64 | if val <= PHP_INT_MAX |
| uint64 | then value is stored as int |
| sint64 | otherwise as double |
| fixed64 | |
| sfixed64 | |
| ---------------- | --------------------------- |
| bool | bool |
| ---------------- | --------------------------- |
| string | string |
| bytes | |
Not set value is represented by null
type. To unset value just set its value to null
.
To parse message create a message class instance and call its parseFromString
method passing it a serialized message. The errors encountered are signaled by throwing Exception
. Exception message provides detailed explanation. Required fields not set are silently ignored.
$packed = /* serialized FooMessage */;
$foo = new FooMessage();
try {
$foo->parseFromString($packed);
} catch (Exception $ex) {
die('Parse error: ' . $e->getMessage());
}
$foo->dump(); // see what you got
To serialize a message call serializeToString
method. It returns a string containing protobuf-encoded message. The errors encountered are signaled by throwing Exception
. Exception message provides detailed explanation. A required field not set triggers an error.
$foo = new FooMessage()
$foo->setBar(1);
try {
$packed = $foo->serializeToString();
} catch (Exception $ex) {
die 'Serialize error: ' . $e->getMessage();
}
/* do some cool stuff with protobuf-encoded $packed */
There might be situations you need to investigate what an actual content of a given message is. What var_dump
gives on a message instance is somewhat obscure.
The ProtobufMessage
class comes with dump
method which prints out a message content to the standard output. It takes one optional argument specifying whether you want to dump only set fields (by default it dumps only set fields). Pass false
as an argument to dump all fields. Format it produces is similar to var_dump
.
Alternatively you can use printDebugString()
method which produces output in protocol buffers text format.
To integrate this extension with your IDE (PhpStorm, Eclipse etc.) and get auto-complete support, simply include stubs\ProtobufMessage.php
anywhere under your project root.
- PHP7 support (Sergey)