Added a minimal test and fixed some bugs.
diff --git a/python/BUILD b/python/BUILD
index 7c4088f..4cbc17a 100644
--- a/python/BUILD
+++ b/python/BUILD
@@ -28,3 +28,19 @@
"@python_headers",
],
)
+
+# Copy the extension into the location recognized by Python.
+genrule(
+ name = "message_ext",
+ srcs = [":message"],
+ outs = ["google/protobuf/pyext/_message.cpython-39-x86_64-linux-gnu.so"],
+ cmd = "cp $< $@",
+)
+
+py_test(
+ name = "minimal_test",
+ srcs = ["minimal_test.py"],
+ data = [":message_ext"],
+ imports = ["."],
+ legacy_create_init = False,
+)
diff --git a/python/descriptor.c b/python/descriptor.c
index b700fbb..767c368 100644
--- a/python/descriptor.c
+++ b/python/descriptor.c
@@ -56,11 +56,16 @@
static PyObject *PyUpb_DescriptorBase_NewInternal(PyTypeObject *type,
const void *def,
PyObject *pool) {
- PyUpb_DescriptorBase *base = PyObject_New(PyUpb_DescriptorBase, type);
- base->pool = pool;
- base->def = def;
- Py_INCREF(pool);
- PyUpb_ObjCache_Add(def, &base->ob_base);
+ PyUpb_DescriptorBase *base = (PyUpb_DescriptorBase*)PyUpb_ObjCache_Get(def);
+
+ if (!base) {
+ base = PyObject_New(PyUpb_DescriptorBase, type);
+ base->pool = pool;
+ base->def = def;
+ Py_INCREF(pool);
+ PyUpb_ObjCache_Add(def, &base->ob_base);
+ }
+
return &base->ob_base;
}
@@ -79,11 +84,6 @@
// FieldDescriptor
// -----------------------------------------------------------------------------
-typedef struct {
- PyObject_HEAD
- upb_fielddef *fielddef;
-} PyUpb_FieldDescriptor;
-
static PyObject *PyUpb_FieldDescriptor_GetType(PyUpb_DescriptorBase *self,
void *closure) {
return PyLong_FromLong(upb_fielddef_descriptortype(self->def));
@@ -94,6 +94,11 @@
return PyLong_FromLong(upb_fielddef_label(self->def));
}
+static PyObject *PyUpb_FieldDescriptor_GetNumber(PyUpb_DescriptorBase *self,
+ void *closure) {
+ return PyLong_FromLong(upb_fielddef_number(self->def));
+}
+
static PyGetSetDef PyUpb_FieldDescriptor_Getters[] = {
/*
{ "full_name", (getter)GetFullName, NULL, "Full name"},
@@ -107,8 +112,8 @@
{ "cpp_type", (getter)PyUpb_FieldDescriptor_GetCppType, NULL, "C++ Type"},
*/
{ "label", (getter)PyUpb_FieldDescriptor_GetLabel, NULL, "Label"},
+ { "number", (getter)PyUpb_FieldDescriptor_GetNumber, NULL, "Number"},
/*
- { "number", (getter)GetNumber, NULL, "Number"},
{ "index", (getter)GetIndex, NULL, "Index"},
{ "default_value", (getter)GetDefaultValue, NULL, "Default Value"},
{ "has_default_value", (getter)HasDefaultValue},
@@ -148,10 +153,10 @@
};
static PyType_Spec PyUpb_FieldDescriptor_Spec = {
- PYUPB_MODULE_NAME ".FieldDescriptor", // tp_name
- sizeof(PyUpb_FieldDescriptor), // tp_basicsize
+ PYUPB_MODULE_NAME ".FieldDescriptor",
+ sizeof(PyUpb_DescriptorBase),
0, // tp_itemsize
- Py_TPFLAGS_DEFAULT, // tp_flags
+ Py_TPFLAGS_DEFAULT,
PyUpb_FieldDescriptor_Slots,
};
diff --git a/python/minimal_test.py b/python/minimal_test.py
new file mode 100644
index 0000000..b9303f4
--- /dev/null
+++ b/python/minimal_test.py
@@ -0,0 +1,29 @@
+"""A bare-bones unit test, to be removed once upb can pass existing unit tests."""
+
+
+import unittest
+from google.protobuf.pyext import _message
+
+class TestMessageExtension(unittest.TestCase):
+
+ def test_descriptor_pool(self):
+ serialized_desc = b'\n\ntest.proto\"\x0e\n\x02M1*\x08\x08\x01\x10\x80\x80\x80\x80\x02:\x15\n\x08test_ext\x12\x03.M1\x18\x01 \x01(\x05'
+ pool = _message.DescriptorPool()
+ file_desc = pool.AddSerializedFile(serialized_desc)
+ self.assertEqual("test.proto", file_desc.name)
+ ext_desc = pool.FindExtensionByName("test_ext")
+ self.assertEqual(1, ext_desc.number)
+
+ # Test object cache: repeatedly retrieving the same descriptor
+ # should result in the same object
+ self.assertIs(ext_desc, pool.FindExtensionByName("test_ext"))
+
+
+ def test_lib_is_upb(self):
+ # Ensure we are not pulling in a different protobuf library on the
+ # system.
+ self.assertTrue(_message._IS_UPB)
+
+
+if __name__ == '__main__':
+ unittest.main()
diff --git a/python/protobuf.c b/python/protobuf.c
index 62398cf..eeea3b3 100644
--- a/python/protobuf.c
+++ b/python/protobuf.c
@@ -27,10 +27,11 @@
#include "protobuf.h"
+#include "descriptor.h"
#include "descriptor_pool.h"
-static void PyUpb_ModuleDealloc(void *_s) {
- PyUpb_ModuleState *s = _s;
+static void PyUpb_ModuleDealloc(void *module) {
+ PyUpb_ModuleState *s = PyModule_GetState(module);
upb_arena_free(s->obj_cache_arena);
}
@@ -114,10 +115,14 @@
state->obj_cache_arena = upb_arena_new();
upb_inttable_init(&state->obj_cache, state->obj_cache_arena);
- if (!PyUpb_InitDescriptorPool(m)) {
+ if (!PyUpb_InitDescriptorPool(m) || !PyUpb_InitDescriptor(m)) {
Py_DECREF(m);
return NULL;
}
+ // Temporary: an cookie we can use in the tests to ensure we are testing upb
+ // and not another protobuf library on the system.
+ PyModule_AddObject(m, "_IS_UPB", Py_True);
+
return m;
}