2013-09-13 03:37:23 +00:00
|
|
|
# JSON serialization support for Google's protobuf Messages
|
|
|
|
# Copyright (c) 2009, Paul Dovbush
|
|
|
|
# All rights reserved.
|
|
|
|
# http://code.google.com/p/protobuf-json/
|
|
|
|
#
|
|
|
|
# Redistribution and use in source and binary forms, with or without
|
|
|
|
# modification, are permitted provided that the following conditions are
|
|
|
|
# met:
|
|
|
|
#
|
|
|
|
# * Redistributions of source code must retain the above copyright
|
|
|
|
# notice, this list of conditions and the following disclaimer.
|
|
|
|
# * Redistributions in binary form must reproduce the above
|
|
|
|
# copyright notice, this list of conditions and the following disclaimer
|
|
|
|
# in the documentation and/or other materials provided with the
|
|
|
|
# distribution.
|
|
|
|
# * Neither the name of <ORGANIZATION> nor the names of its
|
|
|
|
# contributors may be used to endorse or promote products derived from
|
|
|
|
# this software without specific prior written permission.
|
|
|
|
#
|
|
|
|
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
|
|
# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
|
|
# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
|
# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
|
|
# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
|
|
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
|
|
|
'''
|
|
|
|
Provide serialization and de-serialization of Google's protobuf Messages into/from JSON format.
|
|
|
|
'''
|
|
|
|
|
|
|
|
# groups are deprecated and not supported;
|
|
|
|
# Note that preservation of unknown fields is currently not available for Python (c) google docs
|
|
|
|
# extensions is not supported from 0.0.5 (due to gpb2.3 changes)
|
|
|
|
|
2017-06-23 19:31:42 +00:00
|
|
|
__version__ = '0.0.6'
|
|
|
|
__author__ = 'Paul Dovbush <dpp@dpp.su>'
|
2013-09-13 03:37:23 +00:00
|
|
|
|
2016-11-25 23:10:45 +00:00
|
|
|
|
|
|
|
from google.protobuf.descriptor import FieldDescriptor as FD
|
|
|
|
from functools import partial
|
2016-05-26 15:20:44 +00:00
|
|
|
|
2017-06-23 19:31:42 +00:00
|
|
|
|
|
|
|
class ParseError(Exception):
|
|
|
|
pass
|
2013-09-13 03:37:23 +00:00
|
|
|
|
|
|
|
|
2016-11-25 23:10:45 +00:00
|
|
|
def json2pb(pb, js, useFieldNumber=False):
|
2013-09-13 03:37:23 +00:00
|
|
|
''' convert JSON string to google.protobuf.descriptor instance '''
|
|
|
|
for field in pb.DESCRIPTOR.fields:
|
2016-11-25 23:10:45 +00:00
|
|
|
if useFieldNumber:
|
|
|
|
key = field.number
|
|
|
|
else:
|
|
|
|
key = field.name
|
|
|
|
if key not in js:
|
2013-09-13 03:37:23 +00:00
|
|
|
continue
|
|
|
|
if field.type == FD.TYPE_MESSAGE:
|
|
|
|
pass
|
|
|
|
elif field.type in _js2ftype:
|
|
|
|
ftype = _js2ftype[field.type]
|
|
|
|
else:
|
2014-06-27 15:02:44 +00:00
|
|
|
raise ParseError("Field %s.%s of type '%d' is not supported" % (pb.__class__.__name__, field.name, field.type, ))
|
2016-11-25 23:10:45 +00:00
|
|
|
value = js[key]
|
2013-09-13 03:37:23 +00:00
|
|
|
if field.label == FD.LABEL_REPEATED:
|
|
|
|
pb_value = getattr(pb, field.name, None)
|
|
|
|
for v in value:
|
|
|
|
if field.type == FD.TYPE_MESSAGE:
|
2016-11-25 23:10:45 +00:00
|
|
|
json2pb(pb_value.add(), v, useFieldNumber=useFieldNumber)
|
2013-09-13 03:37:23 +00:00
|
|
|
else:
|
|
|
|
pb_value.append(ftype(v))
|
|
|
|
else:
|
|
|
|
if field.type == FD.TYPE_MESSAGE:
|
2016-11-25 23:10:45 +00:00
|
|
|
json2pb(getattr(pb, field.name, None), value, useFieldNumber=useFieldNumber)
|
2013-09-13 03:37:23 +00:00
|
|
|
else:
|
|
|
|
setattr(pb, field.name, ftype(value))
|
|
|
|
return pb
|
|
|
|
|
|
|
|
|
2016-11-25 23:10:45 +00:00
|
|
|
def pb2json(pb, useFieldNumber=False):
|
2013-09-13 03:37:23 +00:00
|
|
|
''' convert google.protobuf.descriptor instance to JSON string '''
|
2014-06-27 15:02:44 +00:00
|
|
|
js = {}
|
2017-06-23 19:31:42 +00:00
|
|
|
# fields = pb.DESCRIPTOR.fields # all fields
|
|
|
|
fields = pb.ListFields() # only filled (including extensions)
|
|
|
|
for field, value in fields:
|
2016-11-25 23:10:45 +00:00
|
|
|
if useFieldNumber:
|
|
|
|
key = field.number
|
|
|
|
else:
|
|
|
|
key = field.name
|
2013-09-13 03:37:23 +00:00
|
|
|
if field.type == FD.TYPE_MESSAGE:
|
2016-11-25 23:10:45 +00:00
|
|
|
ftype = partial(pb2json, useFieldNumber=useFieldNumber)
|
|
|
|
# ---- monkey patching ----
|
2014-07-26 15:15:12 +00:00
|
|
|
elif field.type == FD.TYPE_ENUM:
|
2017-06-23 19:31:42 +00:00
|
|
|
def ftype(x):
|
|
|
|
return field.enum_type.values[x].name
|
2016-11-25 23:10:45 +00:00
|
|
|
# ---- end of monkey patching ----
|
2013-09-13 03:37:23 +00:00
|
|
|
elif field.type in _ftype2js:
|
|
|
|
ftype = _ftype2js[field.type]
|
|
|
|
else:
|
2014-06-27 15:02:44 +00:00
|
|
|
raise ParseError("Field %s.%s of type '%d' is not supported" % (pb.__class__.__name__, field.name, field.type, ))
|
2013-09-13 03:37:23 +00:00
|
|
|
if field.label == FD.LABEL_REPEATED:
|
|
|
|
js_value = []
|
|
|
|
for v in value:
|
|
|
|
js_value.append(ftype(v))
|
|
|
|
else:
|
|
|
|
js_value = ftype(value)
|
2016-11-25 23:10:45 +00:00
|
|
|
js[key] = js_value
|
2013-09-13 03:37:23 +00:00
|
|
|
return js
|
|
|
|
|
|
|
|
|
|
|
|
_ftype2js = {
|
|
|
|
FD.TYPE_DOUBLE: float,
|
|
|
|
FD.TYPE_FLOAT: float,
|
2017-06-05 19:28:41 +00:00
|
|
|
FD.TYPE_INT64: int,
|
|
|
|
FD.TYPE_UINT64: int,
|
2013-09-13 03:37:23 +00:00
|
|
|
FD.TYPE_INT32: int,
|
|
|
|
FD.TYPE_FIXED64: float,
|
|
|
|
FD.TYPE_FIXED32: float,
|
|
|
|
FD.TYPE_BOOL: bool,
|
2017-06-05 19:28:41 +00:00
|
|
|
FD.TYPE_STRING: str,
|
2017-06-23 19:31:42 +00:00
|
|
|
# FD.TYPE_MESSAGE: pb2json, #handled specially
|
2016-11-25 23:10:45 +00:00
|
|
|
FD.TYPE_BYTES: lambda x: x.encode('string_escape'),
|
2013-09-13 03:37:23 +00:00
|
|
|
FD.TYPE_UINT32: int,
|
2016-11-25 23:10:45 +00:00
|
|
|
FD.TYPE_ENUM: int,
|
2013-09-13 03:37:23 +00:00
|
|
|
FD.TYPE_SFIXED32: float,
|
|
|
|
FD.TYPE_SFIXED64: float,
|
|
|
|
FD.TYPE_SINT32: int,
|
2017-06-05 19:28:41 +00:00
|
|
|
FD.TYPE_SINT64: int,
|
2013-09-13 03:37:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
_js2ftype = {
|
|
|
|
FD.TYPE_DOUBLE: float,
|
|
|
|
FD.TYPE_FLOAT: float,
|
2017-06-05 19:28:41 +00:00
|
|
|
FD.TYPE_INT64: int,
|
|
|
|
FD.TYPE_UINT64: int,
|
2013-09-13 03:37:23 +00:00
|
|
|
FD.TYPE_INT32: int,
|
|
|
|
FD.TYPE_FIXED64: float,
|
|
|
|
FD.TYPE_FIXED32: float,
|
|
|
|
FD.TYPE_BOOL: bool,
|
2017-06-05 19:28:41 +00:00
|
|
|
FD.TYPE_STRING: str,
|
2017-06-23 19:31:42 +00:00
|
|
|
# FD.TYPE_MESSAGE: json2pb, # handled specially
|
2016-11-25 23:10:45 +00:00
|
|
|
FD.TYPE_BYTES: lambda x: x.decode('string_escape'),
|
2013-09-13 03:37:23 +00:00
|
|
|
FD.TYPE_UINT32: int,
|
2016-11-25 23:10:45 +00:00
|
|
|
FD.TYPE_ENUM: int,
|
2013-09-13 03:37:23 +00:00
|
|
|
FD.TYPE_SFIXED32: float,
|
|
|
|
FD.TYPE_SFIXED64: float,
|
|
|
|
FD.TYPE_SINT32: int,
|
2017-06-05 19:28:41 +00:00
|
|
|
FD.TYPE_SINT64: int,
|
2013-09-13 03:37:23 +00:00
|
|
|
}
|
2016-11-25 23:10:45 +00:00
|
|
|
|
|
|
|
# more monkey patching
|
|
|
|
|
|
|
|
import binascii
|
|
|
|
from . import types_pb2 as types
|
|
|
|
|
|
|
|
_ftype2js[FD.TYPE_BYTES] = lambda x: binascii.hexlify(x)
|
2017-06-23 19:31:42 +00:00
|
|
|
del _ftype2js[FD.TYPE_ENUM] # handled specially
|
2016-11-25 23:10:45 +00:00
|
|
|
|
|
|
|
_js2ftype[FD.TYPE_BYTES] = lambda x: binascii.unhexlify(x)
|
|
|
|
_js2ftype[FD.TYPE_ENUM] = lambda x: getattr(types, x)
|