Liking cljdoc? Tell your friends :D

rubberbuf

rubberbuf is a clojure(script) library to parse protobuf definition (.proto) into abstract syntax tree (AST).

It can be used to develop tools or libraries that transpile protobuf definitions into other langs like C/C++, Java, Python, etc, or to dynamically read/analyze protobuf definitions.

https://github.com/s-expresso/clojobuf uses this library to dynamically interprets protobuf files and use the resultant schemas to encode/decode plain clojure(script) map into/from protobuf binaries.

Usage

Add the following to deps.edn (or its equivalent for lein).

{:deps {com.github.s-expresso/rubberbuf {:mvn/version "0.3.0"}}}

then in code

(ns my-ns
  (:require [rubberbuf.core :refer [protoc]]))

; optionally add tap to be notified of compilation progress
;(add-tap println)

; :auto-import and :normalize both default to true if omitted
(protoc ["/path/to/protobuf1/", "/path/to/protobuf2/"] ; paths
        ["file1.proto" "file2.proto"]                  ; files
        ; :auto-import true, :normalize true
        ) 

produces a registry of AST, i.e. map of file name to AST

{"file1.proto" [[:syntax "proto3"]
                [:package "a.b.c"]
                [:import "file3.proto"] ...]
 "file2.proto" [[:syntax "proto2"]
                [:package "x.y.z"] ...]
 "file3.proto" [...]}

Note

  • :auto-import true instructs protoc to recursively load all imports
  • hence file3.proto is in registry because file1.proto imported it
  • if file3.proto imports another file, it will also appear in the registry
  • circular import is ok

AST Format

rubberbuf's AST format follows the original protocol buffer definition closely and only adds some keywords or substitute some protobuf syntaxes with keywords. Interpretation of the AST should be simple and easy.

Some less obvious features are shown using example below.

// content of example.proto
syntax = 'proto2';
package my.package.ns;

enum Enm1 {
  option allow_alias = true;
  ZERO = 0;
  ONE = 1;
  ANOTHER_ONE = 1 [deprecated = true];
}

message MsgA {
  optional Enm1 field_a1 = 1;
  optional sint32 field_a2 = 2 [deprecated = true, default = 5];
  message MsgB {
    optional uint32 field_b1 = 1;
    extensions 1, 2, 1000 to 2000;
  }
  repeated MsgB field_a3 = 3;
  extend MsgB {
    optional bool ext_1 = 1000;
  }
}

will yield

{"example.proto"
 [[:syntax "proto2"]
  [:package "my.package.ns"]
  [:enum "Enm1"
    [:option "allow_alias" :true]
    [:enumField "ZERO" 0 nil]
    [:enumField "ONE" 1 nil]
    [:enumField "ANOTHER_ONE" 1 [["deprecated" :true]]]]
  [:message "MsgA"
    [:field :optional "my.package.ns/Enm1" "field_a1" 1 nil]
    [:field :optional :sint32 "field_a2" 2 [["deprecated" :true]
                                            ["default" 5]]]
    [:message "MsgB"
      [:field :optional :uint32 "field_b1" 1 nil]
      [:extensions 1 2 [1000 2000]]
      [:field+ :optional :bool "my.package.ns/MsgA.ext_1" 1000 nil]]
    [:field :repeated "my.package.ns/MsgA.MsgB" "field_a3" 3 nil]]]}

With the following note worthy characteristics:

  • [:field ...] and [:enumField ...] has a nil at the end if it has no field option
  • [:field ...] of primitive type uses a keyword like :sint32
  • [:field ...] of message/enum type uses string like "MsgA"

And the following are due to :normalize true

  • [:field ...] of message/enum type resolved to a fully qualified name like "my.package.ns/Enm1"
  • [:field+ ...] is injected into message from field inside extend
  • [:field+ ...] name "my.package.ns/MsgA.ext_1" follows google's naming convention; long and verbose but an necessary evil to avoid name collision
  • extend is removed from AST since already injected

If :normalize false, you will get the following:

{"pb2_example.proto"
 [[:syntax "proto2"]
  [:package "my.package.ns"]
  [:enum "Enm1"
   [:option "allow_alias" :true]
   [:enumField "ZERO" 0 nil]
   [:enumField "ONE" 1 nil]
   [:enumField "ANOTHER_ONE" 1 [["deprecated" :true]]]]
  [:message "MsgA"
   [:field :optional "Enm1" "field_a1" 1 nil]
   [:field :optional :sint32 "field_a2" 2 [["deprecated" :true]
                                           ["default" 5]]]
   [:message "MsgB"
    [:field :optional :uint32 "field_b1" 1 nil]
    [:extensions 1 2 [1000 2000]]]
   [:field :repeated "MsgB" "field_a3" 3 nil]
   [:extend "MsgB"
    [:field :optional :bool "ext_1" 1000 nil]]]]}

AST Transformation

rubberbuf.ast-postprocessing provides transformation function that can be applied to above output.

rubberbuf.ast-postprocessing/unnest

Unnest nested message/enum out to top level, with its name replaced with a scoped name (.e.g MsgA.MsgB.MsgC).

Example:

(rubberbuf.ast-postprocessing/unnest
  {"p1.proto" [[:syntax "proto3"]
               [:package "a.b.c"]
               [:message "msg1"
                 [:message "msgA" [:field "a" :uint32 1 nil]
                 [:message "msgA" [:field "a" :uint32 1 nil]
                   [:message "msgA" [:field "a" :uint32 1 nil]
                   [:message "msgA" [:field "a" :uint32 1 nil]
                     [:enum "enmA" ["ZERO" 0] ["ONE" 1]]]]]]]
               [:message "msgA" [:field "m" "a.b.c/msg1" 1 nil]] [:message "msgB"]]})
; => {"p1.proto" [[:syntax "proto3"]
;                 [:package "a.b.c"]
;                 [:message "msg1"]
;                 [:message "msg1.msgA" [:field "a" :uint32 1 nil]]
;                 [:message "msg1.msgA.msgA" [:field "a" :uint32 1 nil]]
;                 [:message "msg1.msgA.msgA.msgA" [:field "a" :uint32 1 nil]]
;                 [:message "msg1.msgA.msgA.msgA.msgA" [:field "a" :uint32 1 nil]]
;                 [:enum "msg1.msgA.msgA.msgA.msgA.enmA" ["ZERO" 0] ["ONE" 1]]
;                 [:message "msgA" [:field "m" "a.b.c/msg1" 1 nil]]
;                 [:message "msgB"]]}

rubberbuf.ast-postprocessing/mapify

Transforms the vector structure of the AST into a map of maps; meant to be used after unnest.

Example:

(rubberbuf.ast-postprocessing/mapify
  {"p1.proto"
   [[:syntax "proto2"]
    [:package "my.package.ns"]
    [:message
     "Msg"
     [:field :required :string "msg" 1 nil]
     [:mapField :int32 :string "map_field" 2 nil]
     [:oneof "identifier" [:oneofField :string "name" 3 nil] [:oneofField :int32 "id" 4 nil]]]
    [:enum
     "enm"
     [:option "allow_alias" :true]
     [:enumField "ZERO" 0 nil]
     [:enumField "ONE" 1 nil]
     [:enumField "ANOTHER_ONE" 1 nil]
     [:enumField "TWO" 2 nil]
     [:enumField "THREE" 3 [["deprecated" :true]]]
     [:reserved-ranges 2 15 [9 11] [40 536870911]]
     [:reserved-names "FOO" "BAR"]]
    [:message "ReqABC"
     [:extensions [4 1000]]
     [:field+ :optional :string "my.ns.ext" 4 nil]]
    [:message "RespABC"]
    [:service
     "svc"
     [:option "deprecated" :true]
     [:rpc "method1" nil "simple/ReqABC" nil "simple/RespABC" [[:option "deprecated" :true]]]]]})
; =>
; {"my.package.ns/Msg"
;   {:context :message,
;    :fields
;    {"msg" {:context :required, :type :string, :fid 1, :options nil},
;     "map_field" {:context :map, :key-type :int32, :val-type :string, :fid 2, :options nil},
;     "identifier"
;     {:context :oneof,
;      :oneof-fields
;      {"name" {:context :oneof-field, :type :string, :fid 3, :options nil},
;       "id" {:context :oneof-field, :type :int32, :fid 4, :options nil}}}}},
;   "my.package.ns/enm"
;   {:context :enum,
;    :options [["allow_alias" :true]],
;    :enum-fields
;    {"ZERO" {:value 0, :options nil},
;     "ONE" {:value 1, :options nil},
;     "ANOTHER_ONE" {:value 1, :options nil},
;     "TWO" {:value 2, :options nil},
;     "THREE" {:value 3, :options [["deprecated" :true]]}},
;    :reserved-ranges [2 15 [9 11] [40 536870911]],
;    :reserved-names ["FOO" "BAR"]},
;   "my.package.ns/ReqABC"
;   {:context :message
;    :extensions [[4 1000]]
;    :fields {"my.ns.ext" {:context :optional, :type :string, :is-extension true,:fid 4, :options nil}}},
;   "my.package.ns/RespABC" {:context :message},
;   "my.package.ns/svc"
;   {:context :service,
;    :options [["deprecated" :true]],
;    :rpcs
;    {"method1"
;     {:context :rpc,
;      :request-spec nil,
;      :request "simple/ReqABC",
;      :response-spec nil,
;      :response "simple/RespABC",
;      :options [["deprecated" :true]]}}}}

Textformat

protobuf's textformat can be parsed using rubberbuf.parse-textformat/parse method.

(rubberbuf.parse-textformat/parse 
 "str_field: \"str\" 
  int_field: 123 
  float_field: 123.45 
  msg_field: {a: 1 b: 2} 
  repeated_field: {a: 1 b: 2} 
  repeated_field: {a: 3 b: 4}")
;; =>
;; {"str_field" "str",
;;  "int_field" 123,
;;  "float_field" 123.45,
;;  "msg_field" {"a" 1, "b" 2},
;;  "repeated_field" [{"a" 1, "b" 2} {"a" 3, "b" 4}]}

Unsupported feature

  • protobuf group type (deprecated by google)

Known Issues

  • no checking of semantic error (yet), i.e. no checking that same field number/name is reused in the same message, etc.
  • some types of protobuf syntax error can trick parser into looping indefinitely until out of memory

Above issues imply rubberbuf is not ready to be used as a standalone transpiler, but the issues can be worked around by using google's protoc (protocol buffer compiler) to validate the protobuf files before feeding into rubberbuf. This isn't ideal, but is a reasonable workaround especially if the need to invoke googles' protoc compiler already pre-exists.

Can you improve this documentation? These fine people already did:
st.cheewah & st-cheewah
Edit on GitHub

cljdoc is a website building & hosting documentation for Clojure/Script libraries

× close