/* Copyright 2012 Canonical Ltd. Authors: Ted Gould This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License version 3, 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 warranties of MERCHANTABILITY, SATISFACTORY QUALITY, 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, see . */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #include "glib-compat.h" #include "dbus-test.h" typedef struct { gchar * filename; gchar * executable; guint watch; GIOChannel * stderr; GIOChannel * file; GPid pid; gboolean crashed; } DbusTestBustlePrivate; static void dbus_test_bustle_class_init (DbusTestBustleClass *klass); static void dbus_test_bustle_init (DbusTestBustle *self); static void dbus_test_bustle_dispose (GObject *object); static void dbus_test_bustle_finalize (GObject *object); static void process_run (DbusTestTask * task); static DbusTestTaskState get_state (DbusTestTask * task); static gboolean get_passed (DbusTestTask * task); static gboolean bustle_write_error (GIOChannel * channel, GIOCondition condition, gpointer data); G_DEFINE_TYPE_WITH_PRIVATE (DbusTestBustle, dbus_test_bustle, DBUS_TEST_TYPE_TASK); static void dbus_test_bustle_class_init (DbusTestBustleClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); object_class->dispose = dbus_test_bustle_dispose; object_class->finalize = dbus_test_bustle_finalize; DbusTestTaskClass * task_class = DBUS_TEST_TASK_CLASS(klass); task_class->run = process_run; task_class->get_state = get_state; task_class->get_passed = get_passed; return; } static void dbus_test_bustle_init (DbusTestBustle *self) { gchar * current_dir = g_get_current_dir(); DbusTestBustlePrivate *priv = dbus_test_bustle_get_instance_private(self); priv->filename = g_strconcat(current_dir, G_DIR_SEPARATOR_S, "bustle.log", NULL); priv->executable = g_strdup(BUSTLE_DUAL_MONITOR); priv->watch = 0; priv->stderr = NULL; priv->file = NULL; priv->pid = 0; priv->crashed = FALSE; g_free (current_dir); return; } static void dbus_test_bustle_dispose (GObject *object) { g_return_if_fail(DBUS_TEST_IS_BUSTLE(object)); DbusTestBustle * bustler = DBUS_TEST_BUSTLE(object); DbusTestBustlePrivate *priv = dbus_test_bustle_get_instance_private(bustler); if (priv->watch != 0) { g_source_remove(priv->watch); priv->watch = 0; } if (priv->pid != 0) { gchar * command = g_strdup_printf("kill -INT %d", priv->pid); g_spawn_command_line_sync(command, NULL, NULL, NULL, NULL); g_free(command); g_spawn_close_pid(priv->pid); } if (priv->stderr != NULL) { while (G_IO_IN & g_io_channel_get_buffer_condition(priv->stderr)) { bustle_write_error(priv->stderr, 0 /* unused */, bustler); } g_clear_pointer(&priv->stderr, g_io_channel_unref); } if (priv->file != NULL) { g_io_channel_shutdown(priv->file, TRUE, NULL); g_clear_pointer(&priv->file, g_io_channel_unref); } G_OBJECT_CLASS (dbus_test_bustle_parent_class)->dispose (object); return; } static void dbus_test_bustle_finalize (GObject *object) { g_return_if_fail(DBUS_TEST_IS_BUSTLE(object)); DbusTestBustle * bustler = DBUS_TEST_BUSTLE(object); DbusTestBustlePrivate *priv = dbus_test_bustle_get_instance_private(bustler); g_free(priv->filename); g_free(priv->executable); G_OBJECT_CLASS (dbus_test_bustle_parent_class)->finalize (object); return; } DbusTestBustle * dbus_test_bustle_new (const gchar * filename) { g_return_val_if_fail(filename != NULL, NULL); DbusTestBustle * bustler = g_object_new(DBUS_TEST_TYPE_BUSTLE, NULL); DbusTestBustlePrivate *priv = dbus_test_bustle_get_instance_private(bustler); g_free(priv->filename); priv->filename = g_strdup(filename); dbus_test_task_set_name(DBUS_TEST_TASK(bustler), "Bustle"); return bustler; } void dbus_test_bustle_set_executable (DbusTestBustle * bustle, const gchar * executable) { g_return_if_fail(DBUS_TEST_IS_BUSTLE(bustle)); g_return_if_fail(executable != NULL); DbusTestBustlePrivate *priv = dbus_test_bustle_get_instance_private(bustle); g_free(priv->executable); priv->executable = g_strdup(executable); return; } static void bustle_watcher (GPid pid, G_GNUC_UNUSED gint status, gpointer data) { g_critical("Bustle Monitor exited abruptly!"); DbusTestBustle * bustler = DBUS_TEST_BUSTLE(data); DbusTestBustlePrivate *priv = dbus_test_bustle_get_instance_private(bustler); if (priv->pid != 0) { g_spawn_close_pid(pid); priv->pid = 0; } priv->crashed = TRUE; g_signal_emit_by_name(G_OBJECT(bustler), DBUS_TEST_TASK_SIGNAL_STATE_CHANGED, DBUS_TEST_TASK_STATE_FINISHED, NULL); return; } static gboolean bustle_write_error (GIOChannel * channel, G_GNUC_UNUSED GIOCondition condition, gpointer data) { gchar * line; gsize termloc; do { GIOStatus status = g_io_channel_read_line (channel, &line, NULL, &termloc, NULL); if (status == G_IO_STATUS_EOF) { return FALSE; } if (status != G_IO_STATUS_NORMAL) { continue; } line[termloc] = '\0'; dbus_test_task_print(DBUS_TEST_TASK(data), line); g_free(line); } while (G_IO_IN & g_io_channel_get_buffer_condition(channel)); return TRUE; } static void process_run (DbusTestTask * task) { g_return_if_fail(DBUS_TEST_IS_BUSTLE(task)); DbusTestBustle * bustler = DBUS_TEST_BUSTLE(task); DbusTestBustlePrivate *priv = dbus_test_bustle_get_instance_private(bustler); if (priv->pid != 0) { return; } GError * error = NULL; priv->file = g_io_channel_new_file(priv->filename, "w", &error); if (error != NULL) { g_critical("Unable to open bustle file '%s': %s", priv->filename, error->message); g_error_free(error); priv->crashed = TRUE; g_signal_emit_by_name(G_OBJECT(bustler), DBUS_TEST_TASK_SIGNAL_STATE_CHANGED, DBUS_TEST_TASK_STATE_FINISHED, NULL); return; } gint bustle_stderr_num; gchar * current_dir = g_get_current_dir(); gchar ** bustle_monitor = g_new0(gchar *, 3); bustle_monitor[0] = (gchar *)priv->executable; bustle_monitor[1] = (gchar *)priv->filename; g_spawn_async_with_pipes(current_dir, bustle_monitor, /* argv */ NULL, /* envp */ /* G_SPAWN_SEARCH_PATH | G_SPAWN_STDERR_TO_DEV_NULL, */ /* flags */ G_SPAWN_SEARCH_PATH | G_SPAWN_DO_NOT_REAP_CHILD, /* flags */ NULL, /* child setup func */ NULL, /* child setup data */ &priv->pid, /* PID */ NULL, /* stdin */ NULL, /* stdout */ &bustle_stderr_num, /* stderr */ &error); /* error */ g_free(current_dir); g_free(bustle_monitor); if (error != NULL) { g_critical("Unable to start bustling data: %s", error->message); g_error_free(error); priv->pid = 0; /* ensure this */ priv->crashed = TRUE; g_signal_emit_by_name(G_OBJECT(bustler), DBUS_TEST_TASK_SIGNAL_STATE_CHANGED, DBUS_TEST_TASK_STATE_FINISHED, NULL); return; } if (TRUE) { gchar * start = g_strdup_printf("Starting bustle monitor. PID: %d", priv->pid); dbus_test_task_print(DBUS_TEST_TASK(bustler), start); g_free(start); } priv->watch = g_child_watch_add(priv->pid, bustle_watcher, bustler); priv->stderr = g_io_channel_unix_new(bustle_stderr_num); g_io_add_watch(priv->stderr, G_IO_IN | G_IO_HUP | G_IO_ERR, /* conditions */ bustle_write_error, /* func */ bustler); /* func data */ return; } static DbusTestTaskState get_state (DbusTestTask * task) { /* We're always finished, but we want an error */ g_return_val_if_fail(DBUS_TEST_IS_BUSTLE(task), DBUS_TEST_TASK_STATE_FINISHED); return DBUS_TEST_TASK_STATE_FINISHED; } static gboolean get_passed (DbusTestTask * task) { g_return_val_if_fail(DBUS_TEST_IS_BUSTLE(task), FALSE); DbusTestBustle * bustler = DBUS_TEST_BUSTLE(task); DbusTestBustlePrivate *priv = dbus_test_bustle_get_instance_private(bustler); if (priv->crashed) { return FALSE; } else { return TRUE; } }