+2005-08-31 John (J5) Palmieri <johnp@redhat.com>
+
+ * python/Makefile.am: Break on pyrexc errors instead of ignoring them
+
+ * python/dbus_bindings.pyx: Memory management foo
+ (global): remove hacky _user_data_references global list
+ (GIL_safe_cunregister_function_handler): userdata now stuffed into
+ tuples. Unref user_data
+ (GIL_safe_cmessage_function_handler): userdata now stuffed into tuples
+ (Connection::__del__): Remove and replace with __dealloc__ method
+ (Connection::add_filter): Stuff user_data into a tuple. Use Py_INCREF
+ to keep tuple from being deallocated instead of the global var hack
+ (Connection::register_object_path): Stuff user_data into a tuple.
+ Use Py_INCREF to keep tuple from being deallocated instead of the
+ global var hack
+ (Connection::register_fallback): Stuff user_data into a tuple.
+ Use Py_INCREF to keep tuple from being deallocated instead of the
+ global var hack
+ (GIL_safe_pending_call_notification): Don't unref the message
+ because it gets unreffed when going out of scope. Py_XDECREF
+ the user_data
+ (PendingCall::__del__): Remove and replace with __dealloc__ method
+ (PendingCall::set_notify): ref the pending call because we will
+ need it to stick around for when the notify callback gets called
+ (Message::__del__): Remove and replace with __dealloc__ method
+
+ * python/dbus_glib_bindings.pyx (init_gthreads): Changed to
+ gthreads_init to match up with the dbus call
+
+ * python/glib.py (init_threads): Changed to threads_init to match
+ up with gobject.threads_init(). init_threads is kept for backwards
+ compat but will most likely be deprecated in the future
+
+ * test/python/test-client.py:
+ - revamp to use Python's unittest functionality
+ - add async call tests
+ - setup threads in glib and dbus so we make sure locks are working
+
2005-08-30 John (J5) Palmieri <johnp@redhat.com>
* python/dbus_bindings.pyx
-$(PYTHON) $(srcdir)/extract.py $(srcdir)/dbus_bindings.pxd.in -I$(srcdir)/$(top_builddir) -I$(srcdir) > $@.tmp && mv $@.tmp $@
dbus_bindings.c: $(srcdir)/dbus_bindings.pyx dbus_bindings.pxd
- -pyrexc $(srcdir)/dbus_bindings.pyx -I. -o ./dbus_bindings.c
+ pyrexc $(srcdir)/dbus_bindings.pyx -I. -o ./dbus_bindings.c
dbus_glib_bindings.c: $(srcdir)/dbus_glib_bindings.pyx dbus_bindings.pxd
- -pyrexc $(srcdir)/dbus_glib_bindings.pyx -I. -o ./dbus_glib_bindings.c
+ pyrexc $(srcdir)/dbus_glib_bindings.pyx -I. -o ./dbus_glib_bindings.c
void (* dbus_internal_pad3) (void *)
void (* dbus_internal_pad4) (void *)
-
-_user_data_references = [ ]
-
class DBusException(Exception):
pass
void *user_data):
cdef Connection conn
- itup = <object>user_data
- assert (type(tup) == list)
+ tup = <object>user_data
+ assert (type(tup) == tuple)
function = tup[1]
conn = Connection()
conn.__cinit__(None, connection)
args = (conn)
function(*args)
+ Py_XDECREF(tup)
cdef void cunregister_function_handler (DBusConnection *connection,
void *user_data):
cdef Message message
tup = <object>user_data
- assert (type(tup) == list)
+ assert (type(tup) == tuple)
function = tup[0]
message = EmptyMessage()
conn.__cinit__(None, connection)
args = (conn,
message)
-
+
retval = function(*args)
if (retval == None):
if dbus_error_is_set(&error):
raise DBusException, error.message
- def __del__(self):
+ def __dealloc__(self):
if self.conn != NULL:
dbus_connection_unref(self.conn)
# FIXME: set_dispatch_status_function, get_unix_user, set_unix_user_function
def add_filter(self, filter_function):
- user_data = [ filter_function ]
- global _user_data_references
- _user_data_references.append(user_data)
-
+ user_data = (filter_function,)
+ Py_XINCREF(user_data)
+
return dbus_connection_add_filter(self.conn,
cmessage_function_handler,
<void*>user_data,
cvtable.unregister_function = cunregister_function_handler
cvtable.message_function = cmessage_function_handler
- user_data = [message_cb, unregister_cb]
- global _user_data_references
- _user_data_references.append(user_data)
+ user_data = (message_cb, unregister_cb)
+ Py_XINCREF(user_data)
return dbus_connection_register_object_path(self.conn, path, &cvtable,
<void*>user_data)
cvtable.unregister_function = cunregister_function_handler
cvtable.message_function = cmessage_function_handler
- user_data = [message_cb, unregister_cb]
- global _user_data_references
- _user_data_references.append(user_data)
-
+ user_data = (message_cb, unregister_cb)
+ Py_XINCREF(user_data)
+
return dbus_connection_register_fallback(self.conn, path, &cvtable,
<void*>user_data)
else:
error_handler(DBusException('Unexpected Message Type: ' + message.type_to_name(type)))
- dbus_message_unref(dbus_message)
dbus_pending_call_unref(pending_call)
+ Py_XDECREF(<object>user_data)
cdef void _pending_call_notification(DBusPendingCall *pending_call,
void *user_data):
self.pending_call = _pending_call
dbus_pending_call_ref(self.pending_call)
- def __del__(self):
+ def __dealloc__(self):
if self.pending_call != NULL:
dbus_pending_call_unref(self.pending_call)
def set_notify(self, reply_handler, error_handler):
user_data = (reply_handler, error_handler)
Py_XINCREF(user_data)
+ dbus_pending_call_ref(self.pending_call)
dbus_pending_call_set_notify(self.pending_call, _pending_call_notification,
<void *>user_data, _pending_call_free_user_data)
self.msg = dbus_message_new_error(cmsg, error_name, error_message)
- def __del__(self):
+ def __dealloc__(self):
if self.msg != NULL:
dbus_message_unref(self.msg)
connection = conn
dbus_connection_setup_with_g_main(connection._get_conn(), NULL)
-def init_gthreads ():
+def gthreads_init ():
dbus_g_thread_init ()
dbus_glib_bindings.setup_with_g_main(conn._connection)
_dbus_gthreads_initialized = False
-def init_threads():
+def threads_init():
global _dbus_gthreads_initialized
if not _dbus_gthreads_initialized:
- dbus_glib_bindings.init_gthreads ()
+ dbus_glib_bindings.gthreads_init()
_dbus_gthreads_initialized = True
+def init_threads():
+ threads_init()
setattr(dbus, "_dbus_mainloop_setup_function", _setup_with_g_main)
#!/usr/bin/env python
import sys
import os
+import unittest
builddir = os.environ["DBUS_TOP_BUILDDIR"]
pydir = builddir + "/python"
import dbus
import dbus_bindings
+import gobject
+import dbus.glib
if not dbus.__file__.startswith(pydir):
raise Exception("DBus modules are not being picked up from the package")
-bus = dbus.SessionBus()
-remote_object = bus.get_object("org.freedesktop.DBus.TestSuitePythonService", "/org/freedesktop/DBus/TestSuitePythonObject")
-iface = dbus.Interface(remote_object, "org.freedesktop.DBus.TestSuiteInterface")
-
-try:
- #test dbus_interface parameter
- print remote_object.Echo("dbus_interface test Passed", dbus_interface = "org.freedesktop.DBus.TestSuiteInterface")
+if not dbus_bindings.__file__.startswith(pydir):
+ raise Exception("DBus modules are not being picked up from the package")
- #test introspection
- print "\n********* Introspection Test ************"
- print remote_object.Introspect(dbus_interface="org.freedesktop.DBus.Introspectable")
- print "Introspection test passed"
+class TestDBusBindings(unittest.TestCase):
+ def setUp(self):
+ self.bus = dbus.SessionBus()
+ self.remote_object = self.bus.get_object("org.freedesktop.DBus.TestSuitePythonService", "/org/freedesktop/DBus/TestSuitePythonObject")
+ self.iface = dbus.Interface(self.remote_object, "org.freedesktop.DBus.TestSuiteInterface")
- #test sending python types and getting them back
- print "\n********* Testing Python Types ***********"
- test_vals = [1, 12323231, 3.14159265, 99999999.99,
+ self.test_types_vals = [1, 12323231, 3.14159265, 99999999.99,
"dude", "123", "What is all the fuss about?", "gob@gob.com",
[1,2,3], ["how", "are", "you"], [1.23,2.3], [1], ["Hello"],
(1,2,3), (1,), (1,"2",3), ("2", "what"), ("you", 1.2),
{1:"a", 2:"b"}, {"a":1, "b":2}, {1:1.1, 2:2.2}, {1.1:"a", 1.2:"b"},
[[1,2,3],[2,3,4]], [["a","b"],["c","d"]],
- ([1,2,3],"c", 1.2, ["a","b","c"], {"a": (1,"v"), "b": (2,"d")})]
+ ([1,2,3],"c", 1.2, ["a","b","c"], {"a": (1,"v"), "b": (2,"d")})
+ ]
+
+ def testInterfaceKeyword(self):
+ #test dbus_interface parameter
+ print self.remote_object.Echo("dbus_interface on Proxy test Passed", dbus_interface = "org.freedesktop.DBus.TestSuiteInterface")
+ print self.iface.Echo("dbus_interface on Interface test Passed", dbus_interface = "org.freedesktop.DBus.TestSuiteInterface")
+ self.assert_(True)
+
+ def testIntrospection(self):
+ #test introspection
+ print "\n********* Introspection Test ************"
+ print self.remote_object.Introspect(dbus_interface="org.freedesktop.DBus.Introspectable")
+ print "Introspection test passed"
+ self.assert_(True)
+
+ def testPythonTypes(self):
+ #test sending python types and getting them back
+ print "\n********* Testing Python Types ***********"
- for send_val in test_vals:
- print "Testing %s"% str(send_val)
- recv_val = iface.Echo(send_val)
- #TODO: is this right in python - construct a better comparison
- # method
- if send_val != recv_val:
- raise Exception("Python type tests: %s does not equal %s"%(str(send_val), str(recv_val)))
-
-
-
-except Exception, e:
- print e
- sys.exit(1)
-
-sys.exit(0)
+ for send_val in self.test_types_vals:
+ print "Testing %s"% str(send_val)
+ recv_val = self.iface.Echo(send_val)
+ self.assertEquals(send_val, recv_val)
+
+ def testAsyncCalls(self):
+ #test sending python types and getting them back async
+ print "\n********* Testing Async Calls ***********"
+
+
+ main_loop = gobject.MainLoop()
+ class async_check:
+ def __init__(self, test_controler, expected_result, do_exit):
+ self.expected_result = expected_result
+ self.do_exit = do_exit
+ self.test_controler = test_controler
+
+ def callback(self, val):
+ if self.do_exit:
+ main_loop.quit()
+
+ self.test_controler.assertEquals(val, self.expected_result)
+
+ def error_handler(error):
+ print error
+ if self.do_exit:
+ main_loop.quit()
+
+ self.test_controler.assert_(val, False)
+
+ last_type = self.test_types_vals[-1]
+ for send_val in self.test_types_vals:
+ print "Testing %s"% str(send_val)
+ check = async_check(self, send_val, last_type == send_val)
+ recv_val = self.iface.Echo(send_val,
+ reply_handler = check.callback,
+ error_handler = check.error_handler)
+
+ main_loop.run()
+
+if __name__ == '__main__':
+ gobject.threads_init()
+ dbus.glib.init_threads()
+
+ unittest.main()
+