mutter/src/tests/meta-wayland-test-utils.c
Sebastian Wick 8c3c29d0c0 tests/wayland-test-utils: Keep track of process exit
Whenever a MetaWaylandTestClient exists without success the calling test
will fail. This fixes a bunch of cases where the test would get stuck
waiting for some event from the client when it already died and won't be
able to send the event.

Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3348>
2023-11-03 10:34:27 +00:00

160 lines
4.6 KiB
C

/*
* Copyright (C) 2021-2022 Red Hat, Inc.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License as
* published by the Free Software Foundation; either version 2 of the
* License, or (at your option) any later version.
*
* 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, see <http://www.gnu.org/licenses/>.
*/
#include "config.h"
#include "tests/meta-wayland-test-utils.h"
#include <gio/gio.h>
#include "core/display-private.h"
#include "wayland/meta-wayland.h"
struct _MetaWaylandTestClient
{
GSubprocess *subprocess;
char *path;
gboolean finished;
};
static char *
get_test_client_path (const char *test_client_name)
{
return g_test_build_filename (G_TEST_BUILT,
"src",
"tests",
"wayland-test-clients",
test_client_name,
NULL);
}
static void
wayland_test_client_finished (GObject *source_object,
GAsyncResult *res,
gpointer user_data)
{
MetaWaylandTestClient *wayland_test_client = user_data;
GError *error = NULL;
if (!g_subprocess_wait_finish (wayland_test_client->subprocess,
res,
&error))
{
g_error ("Failed to wait for Wayland test client '%s': %s",
wayland_test_client->path, error->message);
}
g_assert_true (g_subprocess_get_successful (wayland_test_client->subprocess));
wayland_test_client->finished = TRUE;
}
MetaWaylandTestClient *
meta_wayland_test_client_new (MetaContext *context,
const char *test_client_name)
{
MetaWaylandCompositor *compositor;
const char *wayland_display_name;
g_autofree char *test_client_path = NULL;
g_autoptr (GSubprocessLauncher) launcher = NULL;
GSubprocess *subprocess;
GError *error = NULL;
MetaWaylandTestClient *wayland_test_client;
compositor = meta_context_get_wayland_compositor (context);
wayland_display_name = meta_wayland_get_wayland_display_name (compositor);
test_client_path = get_test_client_path (test_client_name);
launcher = g_subprocess_launcher_new (G_SUBPROCESS_FLAGS_NONE);
g_subprocess_launcher_setenv (launcher,
"WAYLAND_DISPLAY", wayland_display_name,
TRUE);
subprocess = g_subprocess_launcher_spawn (launcher,
&error,
test_client_path,
NULL);
if (!subprocess)
{
g_error ("Failed to launch Wayland test client '%s': %s",
test_client_path, error->message);
}
wayland_test_client = g_new0 (MetaWaylandTestClient, 1);
wayland_test_client->subprocess = subprocess;
wayland_test_client->path = g_strdup (test_client_name);
g_subprocess_wait_async (wayland_test_client->subprocess, NULL,
wayland_test_client_finished,
wayland_test_client);
return wayland_test_client;
}
static void
wayland_test_client_destroy (MetaWaylandTestClient *wayland_test_client)
{
g_free (wayland_test_client->path);
g_object_unref (wayland_test_client->subprocess);
g_free (wayland_test_client);
}
void
meta_wayland_test_client_finish (MetaWaylandTestClient *wayland_test_client)
{
while (!wayland_test_client->finished)
g_main_context_iteration (NULL, TRUE);
wayland_test_client_destroy (wayland_test_client);
}
MetaWindow *
meta_find_client_window (MetaContext *context,
const char *title)
{
MetaDisplay *display = meta_context_get_display (context);
g_autoptr (GSList) windows = NULL;
GSList *l;
windows = meta_display_list_windows (display, META_LIST_DEFAULT);
for (l = windows; l; l = l->next)
{
MetaWindow *window = l->data;
if (g_strcmp0 (meta_window_get_title (window), title) == 0)
return window;
}
return NULL;
}
MetaWindow *
meta_wait_for_client_window (MetaContext *context,
const char *title)
{
while (TRUE)
{
MetaWindow *window;
window = meta_find_client_window (context, title);
if (window)
return window;
g_main_context_iteration (NULL, TRUE);
}
}