blob: f41459f6aa959f296681897de62975c07726d4a4 [file] [log] [blame]
/* libnih
*
* Copyright © 2009 Scott James Remnant <scott@netsplit.com>.
* Copyright © 2009 Canonical Ltd.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2, as
* published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
#ifndef NIH_TEST_DBUS_H
#define NIH_TEST_DBUS_H
#include <sys/types.h>
#include <sys/wait.h>
#include <assert.h>
#include <signal.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <dbus/dbus.h>
/**
* TEST_DBUS:
* @_pid: variable to store pid in.
*
* Spawns a D-Bus bus daemon which may be used for testing purposes. You
* may create connections to the daemon with either the system bus or session
* bus addresses, which will both contain its address; alternatively use
* TEST_DBUS_OPEN() to establish a connection.
*
* The pid of the daemon process is stored in @_pid.
*
* The daemon should be killed with TEST_DBUS_END() once you have finished
* with it.
**/
#define TEST_DBUS(_pid) \
do { \
int _test_fds[2]; \
assert0 (pipe (_test_fds)); \
assert ((_pid = fork ()) >= 0); \
if (_pid > 0) { \
char _test_address[128] = { 0 }; \
close (_test_fds[1]); \
assert (read (_test_fds[0], _test_address, \
sizeof (_test_address)) > 0); \
close (_test_fds[0]); \
\
while (strrchr (_test_address, '\n')) \
*(strrchr (_test_address, '\n')) = '\0'; \
\
assert0 (setenv ("DBUS_SYSTEM_BUS_ADDRESS", \
_test_address, TRUE)); \
assert0 (setenv ("DBUS_SESSION_BUS_ADDRESS", \
_test_address, TRUE)); \
} else if (_pid == 0) { \
close (_test_fds[0]); \
assert (dup2 (_test_fds[1], STDOUT_FILENO) >= 0); \
assert0 (execlp ("dbus-daemon", "test_dbus-daemon", \
"--session", "--print-address", NULL)); \
exit (255); \
} \
} while (0)
/**
* TEST_DBUS_OPEN:
* @_conn: variable to store connection in.
*
* Creates a new connection to the temporary D-Bus server, ensuring that
* it won't exit on disconnect and removing the NameAcquired signal from
* the incoming queue.
*
* The connection may be closed again using TEST_DBUS_CLOSE().
**/
#define TEST_DBUS_OPEN(_conn) \
do { \
DBusMessage *_test_message; \
\
assert ((_conn = dbus_bus_get_private (DBUS_BUS_SYSTEM, NULL)) != NULL); \
dbus_connection_set_exit_on_disconnect (_conn, FALSE); \
\
while (! (_test_message = dbus_connection_pop_message (_conn))) \
dbus_connection_read_write (_conn, -1); \
\
assert (dbus_message_is_signal ( \
_test_message, DBUS_INTERFACE_DBUS, \
"NameAcquired")); \
\
dbus_message_unref (_test_message); \
} while (0)
/**
* TEST_DBUS_MESSAGE:
* @_conn: connection,
* @_message: variable to store message in.
*
* Waits for a single message to arrive and pops it from the incoming queue,
* placing it in @_message.
**/
#define TEST_DBUS_MESSAGE(_conn, _message) \
while (! (_message = dbus_connection_pop_message (_conn))) \
dbus_connection_read_write (_conn, -1);
/**
* TEST_DBUS_DISPATCH:
* @_conn: connection.
*
* Reads and writes from the connection @_conn and dispatches whatever
* messages were received or sent in that pulse.
**/
#define TEST_DBUS_DISPATCH(_conn) \
do { \
dbus_connection_read_write (_conn, -1); \
while (dbus_connection_dispatch (_conn) != DBUS_DISPATCH_COMPLETE) \
; \
} while (0)
/**
* TEST_DBUS_CLOSE:
* @_conn: connection to close.
*
* Closes the connection opened with TEST_DBUS_OPEN().
**/
#define TEST_DBUS_CLOSE(_conn) \
do { \
dbus_connection_close (_conn); \
dbus_connection_unref (_conn); \
} while (0)
/**
* TEST_DBUS_END:
* @_pid: pid of dbus daemon.
*
* Terminates a dbus daemon that has been used for testing.
**/
#define TEST_DBUS_END(_pid) \
do { \
kill (_pid, SIGTERM); \
waitpid (_pid, NULL, 0); \
unsetenv ("DBUS_SESSION_BUS_ADDRESS"); \
unsetenv ("DBUS_SYSTEM_BUS_ADDRESS"); \
} while (0)
#endif /* NIH_TEST_DBUS_H */