我正在使用protorpc和端点 - proto-datastore。
我想从提供的结构中构建自定义消息。
例如,它是以下密钥列表:['id1', 'id2', 'id3']
每个键都分配给一个名为MessageField
的CustomField
。
我想从Message
和一个包含所有key
的类中选择。
def create_custom_container(key_list):
class cls():
pass
for i, k in enumerate(key_list):
setattr(cls, k, MessageField(CustomField, i))
return cls
class CustomMessage(Message, create_custom_container(key_list)):
pass
但它没有用,我得到了:MessageDefinitionError: Message types may only inherit from Message
我从protorpc source code看到Message
使用元类来防止它被遗传或者在飞行中被归类。
所以,我不知道如何动态创建我的自定义消息。
该库在定义Message类的约束方面有很长的路要走 - 强制它来强制新属性可能会导致消息根本无法正常工作。
幸运的是,不是使用class CustomMessage
语句对类主体进行硬编码,而是使用调用创建自定义类 - 允许您以编程方式定义内容。这样,您不需要在继承树上使用多个类。
您所要做的就是使用适当的参数调用Message
的元类,而不是通常调用type
,并作为类命名空间传递 -
所以你可以重写你的身体创造功能:
def create_custom_body(key_list):
dct = {}
for i, k in enumerate(key_list):
dct[k] = MessageField(CustomField, i)
return dct
CustomClass = Message.__class__("CustomClass", (Message,), create_custom_body(key_list))
这将适用于这种情况。如果库的元类将使用自定义命名空间(即它将具有__prepare__
方法),但是,您需要修改它以使用types.new_class
和适当的回调:
from types import new_class
def create_custom_body(dct, key_list):
for i, k in enumerate(key_list):
dct[k] = MessageField(CustomField, i)
return dct
CustomClass = types.new_class(
"CustomClass", (Message,),
exec_body=(lambda namespace: create_custom_body(namespace, key_list))
)