Branch data Line data Source code
1 : : /*
2 : : * Copyright (C) 2008 Red Hat, Inc
3 : : *
4 : : * SPDX-License-Identifier: LGPL-2.1-or-later
5 : : *
6 : : * This library is free software; you can redistribute it and/or
7 : : * modify it under the terms of the GNU Lesser General Public
8 : : * License as published by the Free Software Foundation; either
9 : : * version 2.1 of the License, or (at your option) any later version.
10 : : *
11 : : * This library is distributed in the hope that it will be useful,
12 : : * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 : : * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 : : * Lesser General Public License for more details.
15 : : *
16 : : * You should have received a copy of the GNU Lesser General
17 : : * Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
18 : : *
19 : : * Author: Matthias Clasen
20 : : */
21 : :
22 : : #include <locale.h>
23 : :
24 : : #include <glib/glib.h>
25 : : #include <glib/gstdio.h>
26 : : #include <gio/gio.h>
27 : : #include <gio/gdesktopappinfo.h>
28 : : #include <gio/gunixinputstream.h>
29 : : #include <glib-unix.h>
30 : : #include <stdlib.h>
31 : : #include <string.h>
32 : : #include <unistd.h>
33 : : #include <sys/types.h>
34 : : #include <sys/stat.h>
35 : :
36 : 51 : G_DECLARE_FINAL_TYPE (TestLaunchContext, test_launch_context, TEST,
37 : : LAUNCH_CONTEXT, GAppLaunchContext);
38 : :
39 : : struct _TestLaunchContext {
40 : : GAppLaunchContext parent;
41 : :
42 : : char *overriden_startup_notify_id;
43 : : };
44 : :
45 : : struct _TestLaunchContextClass {
46 : : GAppLaunchContextClass parent;
47 : : };
48 : :
49 : 92 : G_DEFINE_FINAL_TYPE (TestLaunchContext, test_launch_context,
50 : : G_TYPE_APP_LAUNCH_CONTEXT);
51 : :
52 : : static void
53 : 39 : test_launch_context_init (TestLaunchContext *test_context)
54 : : {
55 : 39 : }
56 : :
57 : : static char *
58 : 8 : test_launch_context_get_startup_notify_id (GAppLaunchContext *context,
59 : : GAppInfo *app_info,
60 : : GList *files)
61 : : {
62 : 8 : TestLaunchContext *test_context = TEST_LAUNCH_CONTEXT (context);
63 : :
64 : 8 : if (test_context->overriden_startup_notify_id)
65 : 0 : return g_strdup (test_context->overriden_startup_notify_id);
66 : :
67 : 8 : if (g_app_info_get_id (app_info))
68 : 12 : return g_strdup (g_app_info_get_id (app_info));
69 : :
70 : 2 : if (g_app_info_get_display_name (app_info))
71 : 4 : return g_strdup (g_app_info_get_display_name (app_info));
72 : :
73 : 0 : return g_strdup (g_app_info_get_commandline (app_info));
74 : : }
75 : :
76 : : static void
77 : 39 : test_launch_context_get_startup_notify_dispose (GObject *object)
78 : : {
79 : 39 : TestLaunchContext *test_context = TEST_LAUNCH_CONTEXT (object);
80 : :
81 : 39 : g_clear_pointer (&test_context->overriden_startup_notify_id, g_free);
82 : 39 : G_OBJECT_CLASS (test_launch_context_parent_class)->dispose (object);
83 : 39 : }
84 : :
85 : : static void
86 : 1 : test_launch_context_class_init (TestLaunchContextClass *klass)
87 : : {
88 : 1 : G_APP_LAUNCH_CONTEXT_CLASS (klass)->get_startup_notify_id = test_launch_context_get_startup_notify_id;
89 : 1 : G_OBJECT_CLASS (klass)->dispose = test_launch_context_get_startup_notify_dispose;
90 : 1 : }
91 : :
92 : : static GAppInfo *
93 : 13 : create_command_line_app_info (const char *name,
94 : : const char *command_line,
95 : : const char *default_for_type)
96 : : {
97 : : GAppInfo *info;
98 : 13 : GError *error = NULL;
99 : :
100 : 13 : info = g_app_info_create_from_commandline (command_line,
101 : : name,
102 : : G_APP_INFO_CREATE_NONE,
103 : : &error);
104 : 13 : g_assert_no_error (error);
105 : :
106 : 13 : g_app_info_set_as_default_for_type (info, default_for_type, &error);
107 : 13 : g_assert_no_error (error);
108 : :
109 : 13 : return g_steal_pointer (&info);
110 : : }
111 : :
112 : : static GAppInfo *
113 : 11 : create_app_info (const char *name)
114 : : {
115 : 11 : GError *error = NULL;
116 : : GAppInfo *info;
117 : :
118 : 11 : info = create_command_line_app_info (name, "true blah", "application/x-blah");
119 : :
120 : : /* this is necessary to ensure that the info is saved */
121 : 11 : g_app_info_remove_supports_type (info, "application/x-blah", &error);
122 : 11 : g_assert_no_error (error);
123 : 11 : g_app_info_reset_type_associations ("application/x-blah");
124 : :
125 : 11 : return info;
126 : : }
127 : :
128 : : static gboolean
129 : 7 : skip_missing_update_desktop_database (void)
130 : : {
131 : 7 : gchar *path = g_find_program_in_path ("update-desktop-database");
132 : :
133 : 7 : if (path == NULL)
134 : : {
135 : 0 : g_test_skip ("update-desktop-database is required to run this test");
136 : 0 : return TRUE;
137 : : }
138 : 7 : g_free (path);
139 : 7 : return FALSE;
140 : : }
141 : :
142 : : static void
143 : 1 : test_delete (void)
144 : : {
145 : : GAppInfo *info;
146 : :
147 : : const char *id;
148 : : char *filename;
149 : : gboolean res;
150 : :
151 : 1 : if (skip_missing_update_desktop_database ())
152 : 0 : return;
153 : :
154 : 1 : info = create_app_info ("Blah");
155 : :
156 : 1 : id = g_app_info_get_id (info);
157 : 1 : g_assert_nonnull (id);
158 : :
159 : 1 : filename = g_build_filename (g_get_user_data_dir (), "applications", id, NULL);
160 : :
161 : 1 : res = g_file_test (filename, G_FILE_TEST_EXISTS);
162 : 1 : g_assert_true (res);
163 : :
164 : 1 : res = g_app_info_can_delete (info);
165 : 1 : g_assert_true (res);
166 : :
167 : 1 : res = g_app_info_delete (info);
168 : 1 : g_assert_true (res);
169 : :
170 : 1 : res = g_file_test (filename, G_FILE_TEST_EXISTS);
171 : 1 : g_assert_false (res);
172 : :
173 : 1 : g_object_unref (info);
174 : :
175 : 1 : if (g_file_test ("/usr/share/applications/gedit.desktop", G_FILE_TEST_EXISTS))
176 : : {
177 : 0 : info = (GAppInfo*)g_desktop_app_info_new_from_filename ("/usr/share/applications/gedit.desktop");
178 : 0 : g_assert_nonnull (info);
179 : :
180 : 0 : res = g_app_info_can_delete (info);
181 : 0 : g_assert_false (res);
182 : :
183 : 0 : res = g_app_info_delete (info);
184 : 0 : g_assert_false (res);
185 : : }
186 : :
187 : 1 : g_free (filename);
188 : : }
189 : :
190 : : static void
191 : 1 : test_default (void)
192 : : {
193 : : GAppInfo *info, *info1, *info2, *info3;
194 : : GList *list;
195 : 1 : GError *error = NULL;
196 : :
197 : 1 : if (skip_missing_update_desktop_database ())
198 : 0 : return;
199 : :
200 : 1 : info1 = create_app_info ("Blah1");
201 : 1 : info2 = create_app_info ("Blah2");
202 : 1 : info3 = create_app_info ("Blah3");
203 : :
204 : 1 : g_app_info_set_as_default_for_type (info1, "application/x-test", &error);
205 : 1 : g_assert_no_error (error);
206 : :
207 : 1 : g_app_info_set_as_default_for_type (info2, "application/x-test", &error);
208 : 1 : g_assert_no_error (error);
209 : :
210 : 1 : info = g_app_info_get_default_for_type ("application/x-test", FALSE);
211 : 1 : g_assert_nonnull (info);
212 : 1 : g_assert_cmpstr (g_app_info_get_id (info), ==, g_app_info_get_id (info2));
213 : 1 : g_object_unref (info);
214 : :
215 : 1 : g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_CRITICAL,
216 : : "*assertion*uri_scheme*failed*");
217 : 1 : g_assert_null (g_app_info_get_default_for_uri_scheme (NULL));
218 : 1 : g_test_assert_expected_messages ();
219 : :
220 : 1 : g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_CRITICAL,
221 : : "*assertion*uri_scheme*failed*");
222 : 1 : g_assert_null (g_app_info_get_default_for_uri_scheme (""));
223 : 1 : g_test_assert_expected_messages ();
224 : :
225 : 1 : g_app_info_set_as_default_for_type (info3, "x-scheme-handler/glib", &error);
226 : 1 : g_assert_no_error (error);
227 : 1 : info = g_app_info_get_default_for_uri_scheme ("glib");
228 : 1 : g_assert_nonnull (info);
229 : 1 : g_assert_true (g_app_info_equal (info, info3));
230 : 1 : g_object_unref (info);
231 : :
232 : : /* now try adding something, but not setting as default */
233 : 1 : g_app_info_add_supports_type (info3, "application/x-test", &error);
234 : 1 : g_assert_no_error (error);
235 : :
236 : : /* check that info2 is still default */
237 : 1 : info = g_app_info_get_default_for_type ("application/x-test", FALSE);
238 : 1 : g_assert_nonnull (info);
239 : 1 : g_assert_cmpstr (g_app_info_get_id (info), ==, g_app_info_get_id (info2));
240 : 1 : g_object_unref (info);
241 : :
242 : : /* now remove info1 again */
243 : 1 : g_app_info_remove_supports_type (info1, "application/x-test", &error);
244 : 1 : g_assert_no_error (error);
245 : :
246 : : /* and make sure info2 is still default */
247 : 1 : info = g_app_info_get_default_for_type ("application/x-test", FALSE);
248 : 1 : g_assert_nonnull (info);
249 : 1 : g_assert_cmpstr (g_app_info_get_id (info), ==, g_app_info_get_id (info2));
250 : 1 : g_object_unref (info);
251 : :
252 : : /* now clean it all up */
253 : 1 : g_app_info_reset_type_associations ("application/x-test");
254 : 1 : g_app_info_reset_type_associations ("x-scheme-handler/glib");
255 : :
256 : 1 : list = g_app_info_get_all_for_type ("application/x-test");
257 : 1 : g_assert_null (list);
258 : :
259 : 1 : list = g_app_info_get_all_for_type ("x-scheme-handler/glib");
260 : 1 : g_assert_null (list);
261 : :
262 : 1 : g_app_info_delete (info1);
263 : 1 : g_app_info_delete (info2);
264 : 1 : g_app_info_delete (info3);
265 : :
266 : 1 : g_object_unref (info1);
267 : 1 : g_object_unref (info2);
268 : 1 : g_object_unref (info3);
269 : : }
270 : :
271 : : typedef struct
272 : : {
273 : : GAppInfo *expected_info;
274 : : GMainLoop *loop;
275 : : } DefaultForTypeData;
276 : :
277 : : static void
278 : 6 : ensure_default_type_result (GAppInfo *info,
279 : : DefaultForTypeData *data,
280 : : GError *error)
281 : : {
282 : 6 : if (data->expected_info)
283 : : {
284 : 4 : g_assert_nonnull (info);
285 : 4 : g_assert_no_error (error);
286 : 4 : g_assert_true (g_app_info_equal (info, data->expected_info));
287 : : }
288 : : else
289 : : {
290 : 2 : g_assert_null (info);
291 : 2 : g_assert_error (error, G_IO_ERROR, G_IO_ERROR_NOT_FOUND);
292 : : }
293 : :
294 : 6 : g_main_loop_quit (data->loop);
295 : 6 : g_clear_object (&info);
296 : 6 : g_clear_error (&error);
297 : 6 : }
298 : :
299 : : static void
300 : 4 : on_default_for_type_cb (GObject *object,
301 : : GAsyncResult *result,
302 : : gpointer user_data)
303 : : {
304 : : GAppInfo *info;
305 : 4 : GError *error = NULL;
306 : 4 : DefaultForTypeData *data = user_data;
307 : :
308 : 4 : g_assert_null (object);
309 : :
310 : 4 : info = g_app_info_get_default_for_type_finish (result, &error);
311 : :
312 : 4 : ensure_default_type_result (info, data, error);
313 : 4 : }
314 : :
315 : : static void
316 : 2 : on_default_for_uri_cb (GObject *object,
317 : : GAsyncResult *result,
318 : : gpointer user_data)
319 : : {
320 : : GAppInfo *info;
321 : 2 : GError *error = NULL;
322 : 2 : DefaultForTypeData *data = user_data;
323 : :
324 : 2 : g_assert_null (object);
325 : :
326 : 2 : info = g_app_info_get_default_for_uri_scheme_finish (result, &error);
327 : :
328 : 2 : ensure_default_type_result (info, data, error);
329 : 2 : }
330 : :
331 : : static void
332 : 1 : test_default_async (void)
333 : : {
334 : : DefaultForTypeData data;
335 : : GAppInfo *info1, *info2, *info3;
336 : : GList *list;
337 : 1 : GError *error = NULL;
338 : :
339 : 1 : if (skip_missing_update_desktop_database ())
340 : 0 : return;
341 : :
342 : 1 : data.loop = g_main_loop_new (NULL, TRUE);
343 : :
344 : 1 : info1 = create_app_info ("Blah1");
345 : 1 : info2 = create_app_info ("Blah2");
346 : 1 : info3 = create_app_info ("Blah3");
347 : :
348 : 1 : g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_CRITICAL,
349 : : "*assertion*content_type*failed*");
350 : 1 : g_app_info_get_default_for_type_async (NULL, FALSE, NULL, NULL, NULL);
351 : 1 : g_test_assert_expected_messages ();
352 : :
353 : 1 : g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_CRITICAL,
354 : : "*assertion*content_type*failed*");
355 : 1 : g_app_info_get_default_for_type_async ("", FALSE, NULL, NULL, NULL);
356 : 1 : g_test_assert_expected_messages ();
357 : :
358 : 1 : g_app_info_set_as_default_for_type (info1, "application/x-test", &error);
359 : 1 : g_assert_no_error (error);
360 : :
361 : 1 : g_app_info_set_as_default_for_type (info2, "application/x-test", &error);
362 : 1 : g_assert_no_error (error);
363 : :
364 : 1 : data.expected_info = info2;
365 : 1 : g_app_info_get_default_for_type_async ("application/x-test", FALSE,
366 : : NULL, on_default_for_type_cb, &data);
367 : 1 : g_main_loop_run (data.loop);
368 : :
369 : : /* now try adding something, but not setting as default */
370 : 1 : g_app_info_add_supports_type (info3, "application/x-test", &error);
371 : 1 : g_assert_no_error (error);
372 : :
373 : : /* check that info2 is still default */
374 : 1 : data.expected_info = info2;
375 : 1 : g_app_info_get_default_for_type_async ("application/x-test", FALSE,
376 : : NULL, on_default_for_type_cb, &data);
377 : 1 : g_main_loop_run (data.loop);
378 : :
379 : : /* now remove info1 again */
380 : 1 : g_app_info_remove_supports_type (info1, "application/x-test", &error);
381 : 1 : g_assert_no_error (error);
382 : :
383 : : /* and make sure info2 is still default */
384 : 1 : data.expected_info = info2;
385 : 1 : g_app_info_get_default_for_type_async ("application/x-test", FALSE,
386 : : NULL, on_default_for_type_cb, &data);
387 : 1 : g_main_loop_run (data.loop);
388 : :
389 : 1 : g_app_info_set_as_default_for_type (info3, "x-scheme-handler/glib-async", &error);
390 : 1 : g_assert_no_error (error);
391 : :
392 : 1 : g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_CRITICAL,
393 : : "*assertion*uri_scheme*failed*");
394 : 1 : g_assert_null (g_app_info_get_default_for_uri_scheme (NULL));
395 : 1 : g_test_assert_expected_messages ();
396 : :
397 : 1 : g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_CRITICAL,
398 : : "*assertion*uri_scheme*failed*");
399 : 1 : g_assert_null (g_app_info_get_default_for_uri_scheme (""));
400 : 1 : g_test_assert_expected_messages ();
401 : :
402 : 1 : data.expected_info = info3;
403 : 1 : g_app_info_get_default_for_uri_scheme_async ("glib-async", NULL,
404 : : on_default_for_uri_cb, &data);
405 : 1 : g_main_loop_run (data.loop);
406 : :
407 : : /* now clean it all up */
408 : 1 : g_app_info_reset_type_associations ("application/x-test");
409 : :
410 : 1 : data.expected_info = NULL;
411 : 1 : g_app_info_get_default_for_type_async ("application/x-test", FALSE,
412 : : NULL, on_default_for_type_cb, &data);
413 : 1 : g_main_loop_run (data.loop);
414 : :
415 : 1 : g_app_info_reset_type_associations ("x-scheme-handler/glib-async");
416 : :
417 : 1 : data.expected_info = NULL;
418 : 1 : g_app_info_get_default_for_uri_scheme_async ("glib-async", NULL,
419 : : on_default_for_uri_cb, &data);
420 : 1 : g_main_loop_run (data.loop);
421 : :
422 : 1 : list = g_app_info_get_all_for_type ("application/x-test");
423 : 1 : g_assert_null (list);
424 : :
425 : 1 : g_app_info_delete (info1);
426 : 1 : g_app_info_delete (info2);
427 : 1 : g_app_info_delete (info3);
428 : :
429 : 1 : g_object_unref (info1);
430 : 1 : g_object_unref (info2);
431 : 1 : g_object_unref (info3);
432 : :
433 : 1 : g_main_loop_unref (data.loop);
434 : : }
435 : :
436 : : static void
437 : 1 : test_fallback (void)
438 : : {
439 : 1 : GAppInfo *info1, *info2, *app = NULL;
440 : : GList *apps, *recomm, *fallback, *list, *l, *m;
441 : 1 : GError *error = NULL;
442 : : gint old_length;
443 : :
444 : 1 : if (skip_missing_update_desktop_database ())
445 : 0 : return;
446 : :
447 : 1 : info1 = create_app_info ("Test1");
448 : 1 : info2 = create_app_info ("Test2");
449 : :
450 : 1 : g_assert_true (g_content_type_is_a ("text/x-python", "text/plain"));
451 : :
452 : 1 : apps = g_app_info_get_all_for_type ("text/x-python");
453 : 1 : old_length = g_list_length (apps);
454 : 1 : g_list_free_full (apps, g_object_unref);
455 : :
456 : 1 : g_app_info_add_supports_type (info1, "text/x-python", &error);
457 : 1 : g_assert_no_error (error);
458 : :
459 : 1 : g_app_info_add_supports_type (info2, "text/plain", &error);
460 : 1 : g_assert_no_error (error);
461 : :
462 : : /* check that both apps are registered */
463 : 1 : apps = g_app_info_get_all_for_type ("text/x-python");
464 : 1 : g_assert_cmpint (g_list_length (apps), ==, old_length + 2);
465 : :
466 : : /* check that Test1 is among the recommended apps */
467 : 1 : recomm = g_app_info_get_recommended_for_type ("text/x-python");
468 : 1 : g_assert_nonnull (recomm);
469 : 1 : for (l = recomm; l; l = l->next)
470 : : {
471 : 1 : app = l->data;
472 : 1 : if (g_app_info_equal (info1, app))
473 : 1 : break;
474 : : }
475 : 1 : g_assert_nonnull (app);
476 : 1 : g_assert_true (g_app_info_equal (info1, app));
477 : :
478 : : /* and that Test2 is among the fallback apps */
479 : 1 : fallback = g_app_info_get_fallback_for_type ("text/x-python");
480 : 1 : g_assert_nonnull (fallback);
481 : 1 : for (l = fallback; l; l = l->next)
482 : : {
483 : 1 : app = l->data;
484 : 1 : if (g_app_info_equal (info2, app))
485 : 1 : break;
486 : : }
487 : 1 : g_assert_cmpstr (g_app_info_get_name (app), ==, "Test2");
488 : :
489 : : /* check that recomm + fallback = all applications */
490 : 1 : list = g_list_concat (g_list_copy (recomm), g_list_copy (fallback));
491 : 1 : g_assert_cmpuint (g_list_length (list), ==, g_list_length (apps));
492 : :
493 : 3 : for (l = list, m = apps; l != NULL && m != NULL; l = l->next, m = m->next)
494 : : {
495 : 2 : g_assert_true (g_app_info_equal (l->data, m->data));
496 : : }
497 : :
498 : 1 : g_list_free (list);
499 : :
500 : 1 : g_list_free_full (apps, g_object_unref);
501 : 1 : g_list_free_full (recomm, g_object_unref);
502 : 1 : g_list_free_full (fallback, g_object_unref);
503 : :
504 : 1 : g_app_info_reset_type_associations ("text/x-python");
505 : 1 : g_app_info_reset_type_associations ("text/plain");
506 : :
507 : 1 : g_app_info_delete (info1);
508 : 1 : g_app_info_delete (info2);
509 : :
510 : 1 : g_object_unref (info1);
511 : 1 : g_object_unref (info2);
512 : : }
513 : :
514 : : static void
515 : 1 : test_last_used (void)
516 : : {
517 : : GList *applications;
518 : : GAppInfo *info1, *info2, *default_app;
519 : 1 : GError *error = NULL;
520 : :
521 : 1 : if (skip_missing_update_desktop_database ())
522 : 0 : return;
523 : :
524 : 1 : info1 = create_app_info ("Test1");
525 : 1 : info2 = create_app_info ("Test2");
526 : :
527 : 1 : g_app_info_set_as_default_for_type (info1, "application/x-test", &error);
528 : 1 : g_assert_no_error (error);
529 : :
530 : 1 : g_app_info_add_supports_type (info2, "application/x-test", &error);
531 : 1 : g_assert_no_error (error);
532 : :
533 : 1 : applications = g_app_info_get_recommended_for_type ("application/x-test");
534 : 1 : g_assert_cmpuint (g_list_length (applications), ==, 2);
535 : :
536 : : /* the first should be the default app now */
537 : 1 : g_assert_true (g_app_info_equal (g_list_nth_data (applications, 0), info1));
538 : 1 : g_assert_true (g_app_info_equal (g_list_nth_data (applications, 1), info2));
539 : :
540 : 1 : g_list_free_full (applications, g_object_unref);
541 : :
542 : 1 : g_app_info_set_as_last_used_for_type (info2, "application/x-test", &error);
543 : 1 : g_assert_no_error (error);
544 : :
545 : 1 : applications = g_app_info_get_recommended_for_type ("application/x-test");
546 : 1 : g_assert_cmpuint (g_list_length (applications), ==, 2);
547 : :
548 : 1 : default_app = g_app_info_get_default_for_type ("application/x-test", FALSE);
549 : 1 : g_assert_true (g_app_info_equal (default_app, info1));
550 : :
551 : : /* the first should be the other app now */
552 : 1 : g_assert_true (g_app_info_equal (g_list_nth_data (applications, 0), info2));
553 : 1 : g_assert_true (g_app_info_equal (g_list_nth_data (applications, 1), info1));
554 : :
555 : 1 : g_list_free_full (applications, g_object_unref);
556 : :
557 : 1 : g_app_info_reset_type_associations ("application/x-test");
558 : :
559 : 1 : g_app_info_delete (info1);
560 : 1 : g_app_info_delete (info2);
561 : :
562 : 1 : g_object_unref (info1);
563 : 1 : g_object_unref (info2);
564 : 1 : g_object_unref (default_app);
565 : : }
566 : :
567 : : static void
568 : 1 : test_extra_getters (void)
569 : : {
570 : : GDesktopAppInfo *appinfo;
571 : : const gchar *lang;
572 : : gchar *s;
573 : : gboolean b;
574 : :
575 : 1 : lang = setlocale (LC_ALL, NULL);
576 : 1 : g_setenv ("LANGUAGE", "de_DE.UTF8", TRUE);
577 : 1 : setlocale (LC_ALL, "");
578 : :
579 : 1 : appinfo = g_desktop_app_info_new_from_filename (g_test_get_filename (G_TEST_DIST, "appinfo-test-static.desktop", NULL));
580 : 1 : g_assert_nonnull (appinfo);
581 : :
582 : 1 : g_assert_true (g_desktop_app_info_has_key (appinfo, "Terminal"));
583 : 1 : g_assert_false (g_desktop_app_info_has_key (appinfo, "Bratwurst"));
584 : :
585 : 1 : s = g_desktop_app_info_get_string (appinfo, "StartupWMClass");
586 : 1 : g_assert_cmpstr (s, ==, "appinfo-class");
587 : 1 : g_free (s);
588 : :
589 : 1 : s = g_desktop_app_info_get_locale_string (appinfo, "X-JunkFood");
590 : 1 : g_assert_cmpstr (s, ==, "Bratwurst");
591 : 1 : g_free (s);
592 : :
593 : 1 : g_setenv ("LANGUAGE", "sv_SE.UTF8", TRUE);
594 : 1 : setlocale (LC_ALL, "");
595 : :
596 : 1 : s = g_desktop_app_info_get_locale_string (appinfo, "X-JunkFood");
597 : 1 : g_assert_cmpstr (s, ==, "Burger"); /* fallback */
598 : 1 : g_free (s);
599 : :
600 : 1 : b = g_desktop_app_info_get_boolean (appinfo, "Terminal");
601 : 1 : g_assert_true (b);
602 : :
603 : 1 : g_object_unref (appinfo);
604 : :
605 : 1 : g_setenv ("LANGUAGE", lang, TRUE);
606 : 1 : setlocale (LC_ALL, "");
607 : 1 : }
608 : :
609 : : static void
610 : 3 : wait_for_file (const gchar *want_this,
611 : : const gchar *but_not_this,
612 : : const gchar *or_this)
613 : : {
614 : 6 : while (access (want_this, F_OK) != 0)
615 : 3 : g_usleep (100000); /* 100ms */
616 : :
617 : 3 : g_assert_cmpuint (access (but_not_this, F_OK), !=, 0);
618 : 3 : g_assert_cmpuint (access (or_this, F_OK), !=, 0);
619 : :
620 : 3 : unlink (want_this);
621 : 3 : unlink (but_not_this);
622 : 3 : unlink (or_this);
623 : 3 : }
624 : :
625 : : static gboolean
626 : 2 : skip_missing_dbus_daemon (void)
627 : : {
628 : 2 : gchar *path = g_find_program_in_path ("dbus-daemon");
629 : 2 : if (path == NULL)
630 : : {
631 : 0 : g_test_skip ("dbus-daemon is required to run this test");
632 : 0 : return TRUE;
633 : : }
634 : 2 : g_free (path);
635 : 2 : return FALSE;
636 : : }
637 : :
638 : : static void
639 : 1 : test_actions (void)
640 : : {
641 : 1 : GTestDBus *bus = NULL;
642 : 1 : const char *expected[] = { "frob", "tweak", "twiddle", "broken", NULL };
643 : : const gchar * const *actions;
644 : : GDesktopAppInfo *appinfo;
645 : : const gchar *tmpdir;
646 : : gchar *name;
647 : : gchar *frob_path;
648 : : gchar *tweak_path;
649 : : gchar *twiddle_path;
650 : :
651 : 1 : if (skip_missing_dbus_daemon ())
652 : 0 : return;
653 : :
654 : : /* Set up a test session bus to keep D-Bus traffic off the real session bus. */
655 : 1 : bus = g_test_dbus_new (G_TEST_DBUS_NONE);
656 : 1 : g_test_dbus_up (bus);
657 : :
658 : 1 : appinfo = g_desktop_app_info_new_from_filename (g_test_get_filename (G_TEST_DIST, "appinfo-test-actions.desktop", NULL));
659 : 1 : g_assert_nonnull (appinfo);
660 : :
661 : 1 : actions = g_desktop_app_info_list_actions (appinfo);
662 : 5 : g_assert_cmpstrv (actions, expected);
663 : :
664 : 1 : name = g_desktop_app_info_get_action_name (appinfo, "frob");
665 : 1 : g_assert_cmpstr (name, ==, "Frobnicate");
666 : 1 : g_free (name);
667 : :
668 : 1 : name = g_desktop_app_info_get_action_name (appinfo, "tweak");
669 : 1 : g_assert_cmpstr (name, ==, "Tweak");
670 : 1 : g_free (name);
671 : :
672 : 1 : name = g_desktop_app_info_get_action_name (appinfo, "twiddle");
673 : 1 : g_assert_cmpstr (name, ==, "Twiddle");
674 : 1 : g_free (name);
675 : :
676 : 1 : name = g_desktop_app_info_get_action_name (appinfo, "broken");
677 : 1 : g_assert_nonnull (name);
678 : 1 : g_assert_true (g_utf8_validate (name, -1, NULL));
679 : 1 : g_free (name);
680 : :
681 : 1 : tmpdir = g_getenv ("G_TEST_TMPDIR");
682 : 1 : g_assert_nonnull (tmpdir);
683 : 1 : frob_path = g_build_filename (tmpdir, "frob", NULL);
684 : 1 : tweak_path = g_build_filename (tmpdir, "tweak", NULL);
685 : 1 : twiddle_path = g_build_filename (tmpdir, "twiddle", NULL);
686 : :
687 : 1 : g_assert_false (g_file_test (frob_path, G_FILE_TEST_EXISTS));
688 : 1 : g_assert_false (g_file_test (tweak_path, G_FILE_TEST_EXISTS));
689 : 1 : g_assert_false (g_file_test (twiddle_path, G_FILE_TEST_EXISTS));
690 : :
691 : 1 : g_desktop_app_info_launch_action (appinfo, "frob", NULL);
692 : 1 : wait_for_file (frob_path, tweak_path, twiddle_path);
693 : :
694 : 1 : g_desktop_app_info_launch_action (appinfo, "tweak", NULL);
695 : 1 : wait_for_file (tweak_path, frob_path, twiddle_path);
696 : :
697 : 1 : g_desktop_app_info_launch_action (appinfo, "twiddle", NULL);
698 : 1 : wait_for_file (twiddle_path, frob_path, tweak_path);
699 : :
700 : 1 : g_free (frob_path);
701 : 1 : g_free (tweak_path);
702 : 1 : g_free (twiddle_path);
703 : 1 : g_object_unref (appinfo);
704 : :
705 : 1 : g_test_dbus_down (bus);
706 : 1 : g_clear_object (&bus);
707 : : }
708 : :
709 : : static gchar *
710 : 48 : run_apps (const gchar *command,
711 : : const gchar *arg,
712 : : gboolean with_usr,
713 : : gboolean with_home,
714 : : const gchar *locale_name,
715 : : const gchar *language,
716 : : const gchar *xdg_current_desktop)
717 : : {
718 : : gboolean success;
719 : : gchar **envp;
720 : : gchar **argv;
721 : : gint status;
722 : : gchar *out;
723 : 48 : gchar *argv_str = NULL;
724 : :
725 : 48 : argv = g_new (gchar *, 4);
726 : 48 : argv[0] = g_test_build_filename (G_TEST_BUILT, "apps", NULL);
727 : 48 : argv[1] = g_strdup (command);
728 : 48 : argv[2] = g_strdup (arg);
729 : 48 : argv[3] = NULL;
730 : :
731 : 48 : g_assert_true (g_file_test (argv[0], G_FILE_TEST_IS_EXECUTABLE));
732 : 48 : envp = g_get_environ ();
733 : :
734 : 48 : if (with_usr)
735 : : {
736 : 39 : gchar *tmp = g_test_build_filename (G_TEST_DIST, "desktop-files", "usr", NULL);
737 : 39 : envp = g_environ_setenv (envp, "XDG_DATA_DIRS", tmp, TRUE);
738 : 39 : g_free (tmp);
739 : : }
740 : : else
741 : 9 : envp = g_environ_setenv (envp, "XDG_DATA_DIRS", "/does-not-exist", TRUE);
742 : :
743 : 48 : if (with_home)
744 : : {
745 : 32 : gchar *tmp = g_test_build_filename (G_TEST_DIST, "desktop-files", "home", NULL);
746 : 32 : envp = g_environ_setenv (envp, "XDG_DATA_HOME", tmp, TRUE);
747 : 32 : g_free (tmp);
748 : : }
749 : : else
750 : 16 : envp = g_environ_setenv (envp, "XDG_DATA_HOME", "/does-not-exist", TRUE);
751 : :
752 : 48 : if (locale_name)
753 : 2 : envp = g_environ_setenv (envp, "LC_ALL", locale_name, TRUE);
754 : : else
755 : 46 : envp = g_environ_setenv (envp, "LC_ALL", "C", TRUE);
756 : :
757 : 48 : if (language)
758 : 2 : envp = g_environ_setenv (envp, "LANGUAGE", language, TRUE);
759 : : else
760 : 46 : envp = g_environ_unsetenv (envp, "LANGUAGE");
761 : :
762 : 48 : if (xdg_current_desktop)
763 : 10 : envp = g_environ_setenv (envp, "XDG_CURRENT_DESKTOP", xdg_current_desktop, TRUE);
764 : : else
765 : 38 : envp = g_environ_unsetenv (envp, "XDG_CURRENT_DESKTOP");
766 : :
767 : 48 : envp = g_environ_setenv (envp, "G_MESSAGES_DEBUG", "", TRUE);
768 : :
769 : 48 : success = g_spawn_sync (NULL, argv, envp, 0, NULL, NULL, &out, NULL, &status, NULL);
770 : 48 : g_assert_true (success);
771 : 48 : g_assert_cmpuint (status, ==, 0);
772 : :
773 : 48 : argv_str = g_strjoinv (" ", argv);
774 : 48 : g_test_message ("%s: `%s` returned: %s", G_STRFUNC, argv_str, out);
775 : 48 : g_free (argv_str);
776 : :
777 : 48 : g_strfreev (envp);
778 : 48 : g_strfreev (argv);
779 : :
780 : 48 : return out;
781 : : }
782 : :
783 : : static void
784 : 50 : assert_strings_equivalent (const gchar *expected,
785 : : const gchar *result)
786 : 90 : {
787 : : gchar **expected_words;
788 : : gchar **result_words;
789 : : gint i, j;
790 : :
791 : 50 : expected_words = g_strsplit (expected, " ", 0);
792 : 50 : result_words = g_strsplit_set (result, " \n", 0);
793 : :
794 : 140 : for (i = 0; expected_words[i]; i++)
795 : : {
796 : 880 : for (j = 0; result_words[j]; j++)
797 : 880 : if (g_str_equal (expected_words[i], result_words[j]))
798 : 90 : goto got_it;
799 : :
800 : 0 : g_test_fail_printf ("Unable to find expected string '%s' in result '%s'", expected_words[i], result);
801 : :
802 : 90 : got_it:
803 : 90 : continue;
804 : : }
805 : :
806 : 50 : g_assert_cmpint (g_strv_length (expected_words), ==, g_strv_length (result_words));
807 : 50 : g_strfreev (expected_words);
808 : 50 : g_strfreev (result_words);
809 : 50 : }
810 : :
811 : : static void
812 : 4 : assert_list (const gchar *expected,
813 : : gboolean with_usr,
814 : : gboolean with_home,
815 : : const gchar *locale_name,
816 : : const gchar *language)
817 : : {
818 : : gchar *result;
819 : :
820 : 4 : result = run_apps ("list", NULL, with_usr, with_home, locale_name, language, NULL);
821 : 4 : g_strchomp (result);
822 : 4 : assert_strings_equivalent (expected, result);
823 : 4 : g_free (result);
824 : 4 : }
825 : :
826 : : static void
827 : 2 : assert_info (const gchar *desktop_id,
828 : : const gchar *expected,
829 : : gboolean with_usr,
830 : : gboolean with_home,
831 : : const gchar *locale_name,
832 : : const gchar *language)
833 : : {
834 : : gchar *result;
835 : :
836 : 2 : result = run_apps ("show-info", desktop_id, with_usr, with_home, locale_name, language, NULL);
837 : 2 : g_assert_cmpstr (result, ==, expected);
838 : 2 : g_free (result);
839 : 2 : }
840 : :
841 : : static void
842 : 26 : assert_search (const gchar *search_string,
843 : : const gchar *expected,
844 : : gboolean with_usr,
845 : : gboolean with_home,
846 : : const gchar *locale_name,
847 : : const gchar *language)
848 : : {
849 : : gchar **expected_lines;
850 : : gchar **result_lines;
851 : : gchar *result;
852 : : gint i;
853 : :
854 : 26 : expected_lines = g_strsplit (expected, "\n", -1);
855 : 26 : result = run_apps ("search", search_string, with_usr, with_home, locale_name, language, NULL);
856 : 26 : result_lines = g_strsplit (result, "\n", -1);
857 : 26 : g_assert_cmpint (g_strv_length (expected_lines), ==, g_strv_length (result_lines));
858 : 68 : for (i = 0; expected_lines[i]; i++)
859 : 42 : assert_strings_equivalent (expected_lines[i], result_lines[i]);
860 : 26 : g_strfreev (expected_lines);
861 : 26 : g_strfreev (result_lines);
862 : 26 : g_free (result);
863 : 26 : }
864 : :
865 : : static void
866 : 4 : assert_implementations (const gchar *interface,
867 : : const gchar *expected,
868 : : gboolean with_usr,
869 : : gboolean with_home)
870 : : {
871 : : gchar *result;
872 : :
873 : 4 : result = run_apps ("implementations", interface, with_usr, with_home, NULL, NULL, NULL);
874 : 4 : g_strchomp (result);
875 : 4 : assert_strings_equivalent (expected, result);
876 : 4 : g_free (result);
877 : 4 : }
878 : :
879 : : #define ALL_USR_APPS "evince-previewer.desktop nautilus-classic.desktop gnome-font-viewer.desktop " \
880 : : "baobab.desktop yelp.desktop eog.desktop cheese.desktop org.gnome.clocks.desktop " \
881 : : "gnome-contacts.desktop kde4-kate.desktop gcr-prompter.desktop totem.desktop " \
882 : : "gnome-terminal.desktop nautilus-autorun-software.desktop gcr-viewer.desktop " \
883 : : "nautilus-connect-server.desktop kde4-dolphin.desktop gnome-music.desktop " \
884 : : "kde4-konqbrowser.desktop gucharmap.desktop kde4-okular.desktop nautilus.desktop " \
885 : : "gedit.desktop evince.desktop file-roller.desktop dconf-editor.desktop glade.desktop " \
886 : : "invalid-desktop.desktop"
887 : : #define HOME_APPS "epiphany-weather-for-toronto-island-9c6a4e022b17686306243dada811d550d25eb1fb.desktop"
888 : : #define ALL_HOME_APPS HOME_APPS " eog.desktop"
889 : :
890 : : static void
891 : 1 : test_search (void)
892 : : {
893 : 1 : assert_list ("", FALSE, FALSE, NULL, NULL);
894 : 1 : assert_list (ALL_USR_APPS, TRUE, FALSE, NULL, NULL);
895 : 1 : assert_list (ALL_HOME_APPS, FALSE, TRUE, NULL, NULL);
896 : 1 : assert_list (ALL_USR_APPS " " HOME_APPS, TRUE, TRUE, NULL, NULL);
897 : :
898 : : /* The user has "installed" their own version of eog.desktop which
899 : : * calls it "Eye of GNOME". Do some testing based on that.
900 : : *
901 : : * We should always find "Pictures" keyword no matter where we look.
902 : : */
903 : 1 : assert_search ("Picture", "eog.desktop\n", TRUE, TRUE, NULL, NULL);
904 : 1 : assert_search ("Picture", "eog.desktop\n", TRUE, FALSE, NULL, NULL);
905 : 1 : assert_search ("Picture", "eog.desktop\n", FALSE, TRUE, NULL, NULL);
906 : 1 : assert_search ("Picture", "", FALSE, FALSE, NULL, NULL);
907 : :
908 : : /* We should only find it called "eye of gnome" when using the user's
909 : : * directory.
910 : : */
911 : 1 : assert_search ("eye gnome", "", TRUE, FALSE, NULL, NULL);
912 : 1 : assert_search ("eye gnome", "eog.desktop\n", FALSE, TRUE, NULL, NULL);
913 : 1 : assert_search ("eye gnome", "eog.desktop\n", TRUE, TRUE, NULL, NULL);
914 : :
915 : : /* We should only find it called "image viewer" when _not_ using the
916 : : * user's directory.
917 : : */
918 : 1 : assert_search ("image viewer", "eog.desktop\n", TRUE, FALSE, NULL, NULL);
919 : 1 : assert_search ("image viewer", "", FALSE, TRUE, NULL, NULL);
920 : 1 : assert_search ("image viewer", "", TRUE, TRUE, NULL, NULL);
921 : :
922 : : /* There're "flatpak" apps (clocks) installed as well - they should *not*
923 : : * match the prefix command ("/bin/sh") in the Exec= line though. Then with
924 : : * substring matching, Image Viewer (eog) should be in next group because it
925 : : * contains "Slideshow" in its keywords.
926 : : */
927 : 1 : assert_search ("sh", "gnome-terminal.desktop\n"
928 : : "eog.desktop\n", TRUE, FALSE, NULL, NULL);
929 : :
930 : : /* "frobnicator.desktop" is ignored by get_all() because the binary is
931 : : * missing, but search should still find it (to avoid either stale results
932 : : * from the cache or expensive stat() calls for each potential result)
933 : : */
934 : 1 : assert_search ("frobni", "frobnicator.desktop\n", TRUE, FALSE, NULL, NULL);
935 : :
936 : : /* Obvious multi-word search */
937 : 1 : assert_search ("doc hel", "yelp.desktop\n", TRUE, TRUE, NULL, NULL);
938 : :
939 : : /* Repeated search terms should do nothing... */
940 : 1 : assert_search ("files file fil fi f", "nautilus.desktop\n", TRUE, TRUE, NULL, NULL);
941 : :
942 : : /* "con" will match "connect" and "contacts" on name with prefix match in
943 : : * first group, then second group is a Keyword prefix match for "configuration" in dconf-editor.desktop
944 : : * and third group is a substring match for "Desktop Icons" in Name of nautilus-classic.desktop.
945 : : */
946 : 1 : assert_search ("con", "gnome-contacts.desktop nautilus-connect-server.desktop\n"
947 : : "dconf-editor.desktop\n"
948 : : "nautilus-classic.desktop\n", TRUE, TRUE, NULL, NULL);
949 : :
950 : : /* "gnome" will match "eye of gnome" from the user's directory, plus
951 : : * matching "GNOME Clocks" X-GNOME-FullName.
952 : : */
953 : 1 : assert_search ("gnome", "eog.desktop\n"
954 : : "org.gnome.clocks.desktop\n", TRUE, TRUE, NULL, NULL);
955 : :
956 : : /* eog has exec name 'false' in usr only */
957 : 1 : assert_search ("false", "eog.desktop\n", TRUE, FALSE, NULL, NULL);
958 : 1 : assert_search ("false", "", FALSE, TRUE, NULL, NULL);
959 : 1 : assert_search ("false", "", TRUE, TRUE, NULL, NULL);
960 : 1 : assert_search ("false", "", FALSE, FALSE, NULL, NULL);
961 : :
962 : : /* make sure we only search the first component */
963 : 1 : assert_search ("nonsearchable", "", TRUE, FALSE, NULL, NULL);
964 : :
965 : : /* "gnome con" will match only gnome contacts; via the name for
966 : : * "contacts" and keywords for "friend"
967 : : */
968 : 1 : assert_search ("friend con", "gnome-contacts.desktop\n", TRUE, TRUE, NULL, NULL);
969 : :
970 : : /* make sure we get the correct kde4- prefix on the application IDs
971 : : * from subdirectories
972 : : */
973 : 1 : assert_search ("konq", "kde4-konqbrowser.desktop\n", TRUE, TRUE, NULL, NULL);
974 : 1 : assert_search ("kate", "kde4-kate.desktop\n", TRUE, TRUE, NULL, NULL);
975 : :
976 : : /* make sure we can look up apps by name properly */
977 : 1 : assert_info ("kde4-kate.desktop",
978 : : "kde4-kate.desktop\n"
979 : : "Kate\n"
980 : : "Kate\n"
981 : : "nil\n", TRUE, TRUE, NULL, NULL);
982 : :
983 : 1 : assert_info ("nautilus.desktop",
984 : : "nautilus.desktop\n"
985 : : "Files\n"
986 : : "Files\n"
987 : : "Access and organize files\n", TRUE, TRUE, NULL, NULL);
988 : :
989 : : /* make sure localised searching works properly */
990 : 1 : assert_search ("foliumi", "nautilus.desktop\n"
991 : : "kde4-konqbrowser.desktop\n", TRUE, FALSE, "en_US.UTF-8", "eo");
992 : : /* the user's eog.desktop has no translations... */
993 : 1 : assert_search ("foliumi", "nautilus.desktop\n"
994 : : "kde4-konqbrowser.desktop\n", TRUE, TRUE, "en_US.UTF-8", "eo");
995 : 1 : }
996 : :
997 : : static void
998 : 1 : test_implements (void)
999 : : {
1000 : : /* Make sure we can find our search providers... */
1001 : 1 : assert_implementations ("org.gnome.Shell.SearchProvider2",
1002 : : "gnome-music.desktop gnome-contacts.desktop eog.desktop",
1003 : : TRUE, FALSE);
1004 : :
1005 : : /* And our image acquisition possibilities... */
1006 : 1 : assert_implementations ("org.freedesktop.ImageProvider",
1007 : : "cheese.desktop",
1008 : : TRUE, FALSE);
1009 : :
1010 : : /* Make sure the user's eog is properly masking the system one */
1011 : 1 : assert_implementations ("org.gnome.Shell.SearchProvider2",
1012 : : "gnome-music.desktop gnome-contacts.desktop",
1013 : : TRUE, TRUE);
1014 : :
1015 : : /* Make sure we get nothing if we have nothing */
1016 : 1 : assert_implementations ("org.gnome.Shell.SearchProvider2", "", FALSE, FALSE);
1017 : 1 : }
1018 : :
1019 : : static void
1020 : 11 : assert_shown (const gchar *desktop_id,
1021 : : gboolean expected,
1022 : : const gchar *xdg_current_desktop)
1023 : : {
1024 : : gchar *result;
1025 : :
1026 : 11 : result = run_apps ("should-show", desktop_id, TRUE, TRUE, NULL, NULL, xdg_current_desktop);
1027 : 11 : g_assert_cmpstr (result, ==, expected ? "true\n" : "false\n");
1028 : 11 : g_free (result);
1029 : 11 : }
1030 : :
1031 : : static void
1032 : 1 : test_show_in (void)
1033 : : {
1034 : 1 : assert_shown ("gcr-prompter.desktop", FALSE, NULL);
1035 : 1 : assert_shown ("gcr-prompter.desktop", FALSE, "GNOME");
1036 : 1 : assert_shown ("gcr-prompter.desktop", FALSE, "KDE");
1037 : 1 : assert_shown ("gcr-prompter.desktop", FALSE, "GNOME:GNOME-Classic");
1038 : 1 : assert_shown ("gcr-prompter.desktop", TRUE, "GNOME-Classic:GNOME");
1039 : 1 : assert_shown ("gcr-prompter.desktop", TRUE, "GNOME-Classic");
1040 : 1 : assert_shown ("gcr-prompter.desktop", TRUE, "GNOME-Classic:KDE");
1041 : 1 : assert_shown ("gcr-prompter.desktop", TRUE, "KDE:GNOME-Classic");
1042 : 1 : assert_shown ("invalid-desktop.desktop", TRUE, "GNOME");
1043 : 1 : assert_shown ("invalid-desktop.desktop", FALSE, "../invalid/desktop");
1044 : 1 : assert_shown ("invalid-desktop.desktop", FALSE, "../invalid/desktop:../invalid/desktop");
1045 : 1 : }
1046 : :
1047 : : static void
1048 : 4 : on_launch_started (GAppLaunchContext *context, GAppInfo *info, GVariant *platform_data, gpointer data)
1049 : : {
1050 : 4 : gboolean *invoked = data;
1051 : :
1052 : 4 : g_assert_true (G_IS_APP_LAUNCH_CONTEXT (context));
1053 : :
1054 : 4 : if (TEST_IS_LAUNCH_CONTEXT (context))
1055 : : {
1056 : : GVariantDict dict;
1057 : : const char *sni;
1058 : : char *expected_sni;
1059 : :
1060 : 4 : g_assert_nonnull (platform_data);
1061 : 4 : g_variant_dict_init (&dict, platform_data);
1062 : 4 : g_assert_true (
1063 : : g_variant_dict_lookup (&dict, "startup-notification-id", "&s", &sni));
1064 : 4 : expected_sni = g_app_launch_context_get_startup_notify_id (context, info, NULL);
1065 : 4 : g_assert_cmpstr (sni, ==, expected_sni);
1066 : :
1067 : 4 : g_free (expected_sni);
1068 : 4 : g_variant_dict_clear (&dict);
1069 : : }
1070 : : else
1071 : : {
1072 : : /* Our default context doesn't fill in any platform data */
1073 : 0 : g_assert_null (platform_data);
1074 : : }
1075 : :
1076 : 4 : g_assert_false (*invoked);
1077 : 4 : *invoked = TRUE;
1078 : 4 : }
1079 : :
1080 : : static void
1081 : 2 : on_launched (GAppLaunchContext *context, GAppInfo *info, GVariant *platform_data, gpointer data)
1082 : : {
1083 : 2 : gboolean *launched = data;
1084 : : GVariantDict dict;
1085 : : int pid;
1086 : :
1087 : 2 : g_assert_true (G_IS_APP_LAUNCH_CONTEXT (context));
1088 : 2 : g_assert_true (G_IS_APP_INFO (info));
1089 : 2 : g_assert_nonnull (platform_data);
1090 : 2 : g_variant_dict_init (&dict, platform_data);
1091 : 2 : g_assert_true (g_variant_dict_lookup (&dict, "pid", "i", &pid, NULL));
1092 : 2 : g_assert_cmpint (pid, >, 1);
1093 : :
1094 : 2 : g_assert_false (*launched);
1095 : 2 : *launched = TRUE;
1096 : :
1097 : 2 : g_variant_dict_clear (&dict);
1098 : 2 : }
1099 : :
1100 : : static void
1101 : 2 : on_launch_failed (GAppLaunchContext *context, const char *startup_notify_id, gpointer data)
1102 : : {
1103 : 2 : gboolean *invoked = data;
1104 : :
1105 : 2 : g_assert_true (G_IS_APP_LAUNCH_CONTEXT (context));
1106 : 2 : g_assert_nonnull (startup_notify_id);
1107 : 2 : g_test_message ("Application launch failed: %s", startup_notify_id);
1108 : :
1109 : 2 : g_assert_false (*invoked);
1110 : 2 : *invoked = TRUE;
1111 : 2 : }
1112 : :
1113 : : /* Test g_desktop_app_info_launch_uris_as_manager() and
1114 : : * g_desktop_app_info_launch_uris_as_manager_with_fds()
1115 : : */
1116 : : static void
1117 : 1 : test_launch_as_manager (void)
1118 : : {
1119 : : GDesktopAppInfo *appinfo;
1120 : 1 : GError *error = NULL;
1121 : : gboolean retval;
1122 : : const gchar *path;
1123 : 1 : gboolean invoked = FALSE;
1124 : 1 : gboolean launched = FALSE;
1125 : 1 : gboolean failed = FALSE;
1126 : : GAppLaunchContext *context;
1127 : :
1128 : 1 : path = g_test_get_filename (G_TEST_BUILT, "appinfo-test.desktop", NULL);
1129 : 1 : appinfo = g_desktop_app_info_new_from_filename (path);
1130 : 1 : g_assert_true (G_IS_APP_INFO (appinfo));
1131 : :
1132 : 1 : context = g_object_new (test_launch_context_get_type (), NULL);
1133 : 1 : g_signal_connect (context, "launch-started",
1134 : : G_CALLBACK (on_launch_started),
1135 : : &invoked);
1136 : 1 : g_signal_connect (context, "launched",
1137 : : G_CALLBACK (on_launched),
1138 : : &launched);
1139 : 1 : g_signal_connect (context, "launch-failed",
1140 : : G_CALLBACK (on_launch_failed),
1141 : : &failed);
1142 : 1 : retval = g_desktop_app_info_launch_uris_as_manager (appinfo, NULL, context, 0,
1143 : : NULL, NULL,
1144 : : NULL, NULL,
1145 : : &error);
1146 : 1 : g_assert_no_error (error);
1147 : 1 : g_assert_true (retval);
1148 : 1 : g_assert_true (invoked);
1149 : 1 : g_assert_true (launched);
1150 : 1 : g_assert_false (failed);
1151 : :
1152 : 1 : invoked = FALSE;
1153 : 1 : launched = FALSE;
1154 : 1 : failed = FALSE;
1155 : 1 : retval = g_desktop_app_info_launch_uris_as_manager_with_fds (appinfo,
1156 : : NULL, context, 0,
1157 : : NULL, NULL,
1158 : : NULL, NULL,
1159 : : -1, -1, -1,
1160 : : &error);
1161 : 1 : g_assert_no_error (error);
1162 : 1 : g_assert_true (retval);
1163 : 1 : g_assert_true (invoked);
1164 : 1 : g_assert_true (launched);
1165 : 1 : g_assert_false (failed);
1166 : :
1167 : 1 : g_object_unref (appinfo);
1168 : 1 : g_assert_finalize_object (context);
1169 : 1 : }
1170 : :
1171 : : static void
1172 : 1 : test_launch_as_manager_fail (void)
1173 : : {
1174 : : GAppLaunchContext *context;
1175 : : GDesktopAppInfo *appinfo;
1176 : 1 : GError *error = NULL;
1177 : : gboolean retval;
1178 : : const gchar *path;
1179 : 1 : gboolean launch_started = FALSE;
1180 : 1 : gboolean launched = FALSE;
1181 : 1 : gboolean failed = FALSE;
1182 : :
1183 : 1 : g_test_summary ("Tests that launch-errors are properly handled, we force " \
1184 : : "this by using invalid FD's values when launching as manager");
1185 : :
1186 : 1 : path = g_test_get_filename (G_TEST_BUILT, "appinfo-test.desktop", NULL);
1187 : 1 : appinfo = g_desktop_app_info_new_from_filename (path);
1188 : 1 : g_assert_true (G_IS_APP_INFO (appinfo));
1189 : :
1190 : 1 : context = g_object_new (test_launch_context_get_type (), NULL);
1191 : 1 : g_signal_connect (context, "launch-started",
1192 : : G_CALLBACK (on_launch_started),
1193 : : &launch_started);
1194 : 1 : g_signal_connect (context, "launched",
1195 : : G_CALLBACK (on_launched),
1196 : : &launched);
1197 : 1 : g_signal_connect (context, "launch-failed",
1198 : : G_CALLBACK (on_launch_failed),
1199 : : &failed);
1200 : :
1201 : 1 : retval = g_desktop_app_info_launch_uris_as_manager_with_fds (appinfo,
1202 : : NULL, context, 0,
1203 : : NULL, NULL,
1204 : : NULL, NULL,
1205 : : 3000, 3001, 3002,
1206 : : &error);
1207 : 1 : g_assert_error (error, G_SPAWN_ERROR, G_SPAWN_ERROR_FAILED);
1208 : 1 : g_assert_false (retval);
1209 : 1 : g_assert_true (launch_started);
1210 : 1 : g_assert_false (launched);
1211 : 1 : g_assert_true (failed);
1212 : :
1213 : 1 : g_clear_error (&error);
1214 : 1 : g_object_unref (appinfo);
1215 : 1 : g_assert_finalize_object (context);
1216 : 1 : }
1217 : :
1218 : : static GAppInfo *
1219 : 2 : create_app_info_toucher (const char *name,
1220 : : const char *touched_file_name,
1221 : : const char *handled_type,
1222 : : char **out_file_path)
1223 : : {
1224 : 2 : GError *error = NULL;
1225 : : GAppInfo *info;
1226 : : gchar *command_line;
1227 : : gchar *file_path;
1228 : : gchar *tmpdir;
1229 : :
1230 : 2 : g_assert_nonnull (out_file_path);
1231 : :
1232 : 2 : tmpdir = g_dir_make_tmp ("desktop-app-info-launch-XXXXXX", &error);
1233 : 2 : g_assert_no_error (error);
1234 : :
1235 : 2 : file_path = g_build_filename (tmpdir, touched_file_name, NULL);
1236 : 2 : command_line = g_strdup_printf ("touch %s", file_path);
1237 : :
1238 : 2 : info = create_command_line_app_info (name, command_line, handled_type);
1239 : 2 : *out_file_path = g_steal_pointer (&file_path);
1240 : :
1241 : 2 : g_free (tmpdir);
1242 : 2 : g_free (command_line);
1243 : :
1244 : 2 : return info;
1245 : : }
1246 : :
1247 : : static void
1248 : 1 : test_default_uri_handler (void)
1249 : : {
1250 : 1 : GError *error = NULL;
1251 : 1 : gchar *file_path = NULL;
1252 : : GAppInfo *info;
1253 : :
1254 : 1 : if (skip_missing_update_desktop_database ())
1255 : 0 : return;
1256 : :
1257 : 1 : info = create_app_info_toucher ("Touch Handled", "handled",
1258 : : "x-scheme-handler/glib-touch",
1259 : : &file_path);
1260 : 1 : g_assert_true (G_IS_APP_INFO (info));
1261 : 1 : g_assert_nonnull (file_path);
1262 : :
1263 : 1 : g_assert_true (g_app_info_launch_default_for_uri ("glib-touch://touch-me",
1264 : : NULL, &error));
1265 : 1 : g_assert_no_error (error);
1266 : :
1267 : 838 : while (!g_file_test (file_path, G_FILE_TEST_IS_REGULAR));
1268 : 1 : g_assert_true (g_file_test (file_path, G_FILE_TEST_IS_REGULAR));
1269 : :
1270 : 1 : g_assert_false (g_app_info_launch_default_for_uri ("glib-INVALID-touch://touch-me",
1271 : : NULL, &error));
1272 : 1 : g_assert_error (error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED);
1273 : 1 : g_clear_error (&error);
1274 : :
1275 : 1 : g_object_unref (info);
1276 : 1 : g_free (file_path);
1277 : : }
1278 : :
1279 : : static void
1280 : 1 : on_launch_default_for_uri_success_cb (GObject *object,
1281 : : GAsyncResult *result,
1282 : : gpointer user_data)
1283 : : {
1284 : 1 : GError *error = NULL;
1285 : 1 : gboolean *called = user_data;
1286 : :
1287 : 1 : g_assert_true (g_app_info_launch_default_for_uri_finish (result, &error));
1288 : 1 : g_assert_no_error (error);
1289 : :
1290 : 1 : *called = TRUE;
1291 : 1 : }
1292 : :
1293 : : static void
1294 : 1 : on_launch_default_for_uri_not_found_cb (GObject *object,
1295 : : GAsyncResult *result,
1296 : : gpointer user_data)
1297 : : {
1298 : 1 : GError *error = NULL;
1299 : 1 : GMainLoop *loop = user_data;
1300 : :
1301 : 1 : g_assert_false (g_app_info_launch_default_for_uri_finish (result, &error));
1302 : 1 : g_assert_error (error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED);
1303 : 1 : g_clear_error (&error);
1304 : :
1305 : 1 : g_main_loop_quit (loop);
1306 : 1 : }
1307 : :
1308 : : static void
1309 : 1 : on_launch_default_for_uri_cancelled_cb (GObject *object,
1310 : : GAsyncResult *result,
1311 : : gpointer user_data)
1312 : : {
1313 : 1 : GError *error = NULL;
1314 : 1 : GMainLoop *loop = user_data;
1315 : :
1316 : 1 : g_assert_false (g_app_info_launch_default_for_uri_finish (result, &error));
1317 : 1 : g_assert_error (error, G_IO_ERROR, G_IO_ERROR_CANCELLED);
1318 : 1 : g_clear_error (&error);
1319 : :
1320 : 1 : g_main_loop_quit (loop);
1321 : 1 : }
1322 : :
1323 : : static void
1324 : 1 : test_default_uri_handler_async (void)
1325 : : {
1326 : : GCancellable *cancellable;
1327 : 1 : gchar *file_path = NULL;
1328 : : GAppInfo *info;
1329 : : GMainLoop *loop;
1330 : 1 : gboolean called = FALSE;
1331 : : gint64 start_time, touch_time;
1332 : :
1333 : 1 : if (skip_missing_update_desktop_database ())
1334 : 0 : return;
1335 : :
1336 : 1 : loop = g_main_loop_new (NULL, FALSE);
1337 : 1 : info = create_app_info_toucher ("Touch Handled", "handled-async",
1338 : : "x-scheme-handler/glib-async-touch",
1339 : : &file_path);
1340 : 1 : g_assert_true (G_IS_APP_INFO (info));
1341 : 1 : g_assert_nonnull (file_path);
1342 : :
1343 : 1 : start_time = g_get_real_time ();
1344 : 1 : g_app_info_launch_default_for_uri_async ("glib-async-touch://touch-me", NULL,
1345 : : NULL,
1346 : : on_launch_default_for_uri_success_cb,
1347 : : &called);
1348 : :
1349 : 413 : while (!g_file_test (file_path, G_FILE_TEST_IS_REGULAR) || !called)
1350 : 412 : g_main_context_iteration (NULL, FALSE);
1351 : :
1352 : 1 : touch_time = g_get_real_time () - start_time;
1353 : 1 : g_assert_true (called);
1354 : 1 : g_assert_true (g_file_test (file_path, G_FILE_TEST_IS_REGULAR));
1355 : :
1356 : 1 : g_unlink (file_path);
1357 : 1 : g_assert_false (g_file_test (file_path, G_FILE_TEST_IS_REGULAR));
1358 : :
1359 : 1 : g_app_info_launch_default_for_uri_async ("glib-async-INVALID-touch://touch-me",
1360 : : NULL, NULL,
1361 : : on_launch_default_for_uri_not_found_cb,
1362 : : loop);
1363 : 1 : g_main_loop_run (loop);
1364 : :
1365 : 1 : cancellable = g_cancellable_new ();
1366 : 1 : g_app_info_launch_default_for_uri_async ("glib-async-touch://touch-me", NULL,
1367 : : cancellable,
1368 : : on_launch_default_for_uri_cancelled_cb,
1369 : : loop);
1370 : 1 : g_cancellable_cancel (cancellable);
1371 : 1 : g_main_loop_run (loop);
1372 : :
1373 : : /* If started, our touch app would take some time to actually write the
1374 : : * file to disk, so let's wait a bit here to ensure that the file isn't
1375 : : * inadvertently getting created when a launch operation is canceled up
1376 : : * front. Give it 3× as long as the successful case took, to allow for
1377 : : * some variance.
1378 : : */
1379 : 1 : g_usleep (touch_time * 3);
1380 : 1 : g_assert_false (g_file_test (file_path, G_FILE_TEST_IS_REGULAR));
1381 : :
1382 : 1 : g_object_unref (info);
1383 : 1 : g_main_loop_unref (loop);
1384 : 1 : g_free (file_path);
1385 : : }
1386 : :
1387 : : /* Test if Desktop-File Id is correctly formed */
1388 : : static void
1389 : 1 : test_id (void)
1390 : : {
1391 : : gchar *result;
1392 : :
1393 : 1 : result = run_apps ("default-for-type", "application/vnd.kde.okular-archive",
1394 : : TRUE, FALSE, NULL, NULL, NULL);
1395 : 1 : g_assert_cmpstr (result, ==, "kde4-okular.desktop\n");
1396 : 1 : g_free (result);
1397 : 1 : }
1398 : :
1399 : : static const char *
1400 : 102 : get_terminal_divider (const char *terminal_name)
1401 : : {
1402 : 102 : if (g_str_equal (terminal_name, "xdg-terminal-exec"))
1403 : 3 : return NULL;
1404 : 99 : if (g_str_equal (terminal_name, "kgx"))
1405 : 9 : return "-e";
1406 : 90 : if (g_str_equal (terminal_name, "gnome-terminal"))
1407 : 9 : return "--";
1408 : 81 : if (g_str_equal (terminal_name, "tilix"))
1409 : 9 : return "-e";
1410 : 72 : if (g_str_equal (terminal_name, "konsole"))
1411 : 9 : return "-e";
1412 : 63 : if (g_str_equal (terminal_name, "nxterm"))
1413 : 9 : return "-e";
1414 : 54 : if (g_str_equal (terminal_name, "color-xterm"))
1415 : 9 : return "-e";
1416 : 45 : if (g_str_equal (terminal_name, "rxvt"))
1417 : 9 : return "-e";
1418 : 36 : if (g_str_equal (terminal_name, "dtterm"))
1419 : 9 : return "-e";
1420 : 27 : if (g_str_equal (terminal_name, "xterm"))
1421 : 9 : return "-e";
1422 : 18 : if (g_str_equal (terminal_name, "mate-terminal"))
1423 : 9 : return "-x";
1424 : 9 : if (g_str_equal (terminal_name, "xfce4-terminal"))
1425 : 9 : return "-x";
1426 : :
1427 : : g_return_val_if_reached (NULL);
1428 : : }
1429 : :
1430 : : typedef enum {
1431 : : TERMINAL_LAUNCH_TYPE_COMMAND_LINE_WITH_PATH_OVERRIDE,
1432 : : TERMINAL_LAUNCH_TYPE_COMMAND_LINE_WITH_CONTEXT,
1433 : : TERMINAL_LAUNCH_TYPE_KEY_FILE_WITH_PATH,
1434 : : } TerminalLaunchType;
1435 : :
1436 : : typedef struct {
1437 : : const char *exec;
1438 : : TerminalLaunchType type;
1439 : : } TerminalLaunchData;
1440 : :
1441 : : static TerminalLaunchData *
1442 : 36 : terminal_launch_data_new (const char *exec, TerminalLaunchType type)
1443 : : {
1444 : 36 : TerminalLaunchData *d = NULL;
1445 : :
1446 : 36 : d = g_new0 (TerminalLaunchData, 1);
1447 : 36 : d->exec = exec;
1448 : 36 : d->type = type;
1449 : :
1450 : 36 : return d;
1451 : : }
1452 : :
1453 : : static void
1454 : 36 : test_launch_uris_with_terminal (gconstpointer data)
1455 : : {
1456 : : int fd;
1457 : : int ret;
1458 : : int flags;
1459 : : int terminal_divider_arg_length;
1460 : 36 : const TerminalLaunchData *launch_data = data;
1461 : 36 : const char *terminal_exec = launch_data->exec;
1462 : 36 : char *old_path = NULL;
1463 : : char *command_line;
1464 : : char *bin_path;
1465 : : char *terminal_path;
1466 : : char *output_fd_path;
1467 : : char *script_contents;
1468 : 36 : char *output_contents = NULL;
1469 : : char *sh;
1470 : : GAppInfo *app_info;
1471 : : GList *uris;
1472 : : GList *paths;
1473 : : GStrv output_args;
1474 : 36 : GError *error = NULL;
1475 : : GInputStream *input_stream;
1476 : : GDataInputStream *data_input_stream;
1477 : : GAppLaunchContext *launch_context;
1478 : :
1479 : 36 : sh = g_find_program_in_path ("sh");
1480 : 36 : g_assert_nonnull (sh);
1481 : :
1482 : 36 : bin_path = g_dir_make_tmp ("bin-path-XXXXXX", &error);
1483 : 36 : g_assert_no_error (error);
1484 : :
1485 : 36 : launch_context = g_object_new (test_launch_context_get_type (), NULL);
1486 : :
1487 : 36 : switch (launch_data->type)
1488 : : {
1489 : 12 : case TERMINAL_LAUNCH_TYPE_COMMAND_LINE_WITH_PATH_OVERRIDE:
1490 : 12 : old_path = g_strdup (g_getenv ("PATH"));
1491 : 12 : g_assert_true (g_setenv ("PATH", bin_path, TRUE));
1492 : 12 : break;
1493 : :
1494 : 12 : case TERMINAL_LAUNCH_TYPE_COMMAND_LINE_WITH_CONTEXT:
1495 : 12 : g_app_launch_context_setenv (launch_context, "PATH", bin_path);
1496 : 12 : break;
1497 : :
1498 : 12 : case TERMINAL_LAUNCH_TYPE_KEY_FILE_WITH_PATH:
1499 : 12 : g_app_launch_context_setenv (launch_context, "PATH", "/not/valid");
1500 : 12 : break;
1501 : :
1502 : 0 : default:
1503 : : g_assert_not_reached ();
1504 : : }
1505 : :
1506 : 36 : terminal_path = g_build_filename (bin_path, terminal_exec, NULL);
1507 : 36 : output_fd_path = g_build_filename (bin_path, "fifo", NULL);
1508 : :
1509 : 36 : ret = mkfifo (output_fd_path, 0600);
1510 : 36 : g_assert_cmpint (ret, ==, 0);
1511 : :
1512 : 36 : fd = g_open (output_fd_path, O_RDONLY | O_CLOEXEC | O_NONBLOCK, 0);
1513 : 36 : g_assert_cmpint (fd, >=, 0);
1514 : :
1515 : 36 : flags = fcntl (fd, F_GETFL);
1516 : 36 : g_assert_cmpint (flags, >=, 0);
1517 : :
1518 : 36 : ret = fcntl (fd, F_SETFL, flags & ~O_NONBLOCK);
1519 : 36 : g_assert_cmpint (ret, ==, 0);
1520 : :
1521 : 36 : input_stream = g_unix_input_stream_new (fd, TRUE);
1522 : 36 : data_input_stream = g_data_input_stream_new (input_stream);
1523 : 36 : script_contents = g_strdup_printf ("#!%s\n" \
1524 : : "out='%s'\n"
1525 : : "printf '%%s\\n' \"$*\" > \"$out\"\n",
1526 : : sh,
1527 : : output_fd_path);
1528 : 36 : g_file_set_contents (terminal_path, script_contents, -1, &error);
1529 : 36 : g_assert_no_error (error);
1530 : 36 : g_assert_cmpint (g_chmod (terminal_path, 0500), ==, 0);
1531 : :
1532 : 36 : g_test_message ("Fake '%s' terminal created as: %s", terminal_exec, terminal_path);
1533 : :
1534 : 36 : command_line = g_strdup_printf ("true %s-argument", terminal_exec);
1535 : :
1536 : 36 : if (launch_data->type == TERMINAL_LAUNCH_TYPE_KEY_FILE_WITH_PATH)
1537 : : {
1538 : : GKeyFile *key_file;
1539 : : char *key_file_contents;
1540 : 12 : const char base_file[] =
1541 : : "[Desktop Entry]\n"
1542 : : "Type=Application\n"
1543 : : "Name=terminal launched app\n"
1544 : : "Terminal=true\n"
1545 : : "Path=%s\n"
1546 : : "Exec=%s\n";
1547 : :
1548 : 12 : key_file = g_key_file_new ();
1549 : 12 : key_file_contents = g_strdup_printf (base_file, bin_path, command_line);
1550 : :
1551 : 12 : g_assert_true (
1552 : : g_key_file_load_from_data (key_file, key_file_contents, -1,
1553 : : G_KEY_FILE_NONE, NULL));
1554 : :
1555 : 12 : app_info = (GAppInfo*) g_desktop_app_info_new_from_keyfile (key_file);
1556 : 12 : g_assert_true (G_IS_DESKTOP_APP_INFO (app_info));
1557 : 12 : g_assert_true (
1558 : : g_desktop_app_info_get_boolean (G_DESKTOP_APP_INFO (app_info), "Terminal"));
1559 : :
1560 : 12 : g_key_file_unref (key_file);
1561 : 12 : g_free (key_file_contents);
1562 : : }
1563 : : else
1564 : : {
1565 : 24 : app_info = g_app_info_create_from_commandline (command_line,
1566 : : "Test App on Terminal",
1567 : : G_APP_INFO_CREATE_NEEDS_TERMINAL |
1568 : : G_APP_INFO_CREATE_SUPPORTS_URIS,
1569 : : &error);
1570 : 24 : g_assert_no_error (error);
1571 : : }
1572 : :
1573 : 36 : paths = g_list_prepend (NULL, bin_path);
1574 : 36 : uris = g_list_prepend (NULL, g_filename_to_uri (bin_path, NULL, &error));
1575 : 36 : g_assert_no_error (error);
1576 : :
1577 : 36 : paths = g_list_prepend (paths, (gpointer) g_get_user_data_dir ());
1578 : 36 : uris = g_list_append (uris, g_filename_to_uri (g_get_user_data_dir (), NULL, &error));
1579 : 36 : g_assert_no_error (error);
1580 : :
1581 : 36 : g_assert_cmpint (g_list_length (paths), ==, 2);
1582 : 36 : g_app_info_launch_uris (app_info, uris, launch_context, &error);
1583 : 36 : g_assert_no_error (error);
1584 : :
1585 : 108 : while (output_contents == NULL)
1586 : : {
1587 : 36 : output_contents =
1588 : 36 : g_data_input_stream_read_upto (data_input_stream, "\n", 1, NULL, NULL, &error);
1589 : 36 : g_assert_no_error (error);
1590 : :
1591 : 36 : if (output_contents == NULL)
1592 : 0 : g_usleep (G_USEC_PER_SEC / 10);
1593 : : }
1594 : 36 : g_test_message ("'%s' called with arguments: '%s'", terminal_exec, output_contents);
1595 : :
1596 : 36 : g_data_input_stream_read_byte (data_input_stream, NULL, &error);
1597 : 36 : g_assert_no_error (error);
1598 : :
1599 : 36 : output_args = g_strsplit (output_contents, " ", -1);
1600 : 36 : g_clear_pointer (&output_contents, g_free);
1601 : :
1602 : 36 : terminal_divider_arg_length = (get_terminal_divider (terminal_exec) != NULL) ? 1 : 0;
1603 : 36 : g_assert_cmpuint (g_strv_length (output_args), ==, 3 + terminal_divider_arg_length);
1604 : 36 : if (terminal_divider_arg_length == 1)
1605 : : {
1606 : 33 : g_assert_cmpstr (output_args[0], ==, get_terminal_divider (terminal_exec));
1607 : 33 : g_assert_cmpstr (output_args[1], ==, "true");
1608 : 33 : g_assert_cmpstr (output_args[2], ==, command_line + 5);
1609 : : }
1610 : : else
1611 : : {
1612 : 3 : g_assert_cmpstr (output_args[0], ==, "true");
1613 : 3 : g_assert_cmpstr (output_args[1], ==, command_line + 5);
1614 : : }
1615 : 36 : paths = g_list_delete_link (paths,
1616 : 36 : g_list_find_custom (paths, output_args[2 + terminal_divider_arg_length], g_str_equal));
1617 : 36 : g_assert_cmpint (g_list_length (paths), ==, 1);
1618 : 36 : g_clear_pointer (&output_args, g_strfreev);
1619 : :
1620 : 144 : while (output_contents == NULL)
1621 : : {
1622 : 72 : output_contents =
1623 : 72 : g_data_input_stream_read_upto (data_input_stream, "\n", 1, NULL, NULL, &error);
1624 : 72 : g_assert_no_error (error);
1625 : :
1626 : 72 : if (output_contents == NULL)
1627 : 36 : g_usleep (G_USEC_PER_SEC / 10);
1628 : : }
1629 : 36 : g_test_message ("'%s' called with arguments: '%s'", terminal_exec, output_contents);
1630 : :
1631 : 36 : g_data_input_stream_read_byte (data_input_stream, NULL, &error);
1632 : 36 : g_assert_no_error (error);
1633 : :
1634 : 36 : output_args = g_strsplit (output_contents, " ", -1);
1635 : 36 : g_clear_pointer (&output_contents, g_free);
1636 : 36 : g_assert_cmpuint (g_strv_length (output_args), ==, 3 + terminal_divider_arg_length);
1637 : 36 : if (terminal_divider_arg_length > 0)
1638 : : {
1639 : 33 : g_assert_cmpstr (output_args[0], ==, get_terminal_divider (terminal_exec));
1640 : 33 : g_assert_cmpstr (output_args[1], ==, "true");
1641 : 33 : g_assert_cmpstr (output_args[2], ==, command_line + 5);
1642 : : }
1643 : : else
1644 : : {
1645 : 3 : g_assert_cmpstr (output_args[0], ==, "true");
1646 : 3 : g_assert_cmpstr (output_args[1], ==, command_line + 5);
1647 : : }
1648 : 36 : paths = g_list_delete_link (paths,
1649 : 36 : g_list_find_custom (paths, output_args[2 + terminal_divider_arg_length], g_str_equal));
1650 : 36 : g_assert_cmpint (g_list_length (paths), ==, 0);
1651 : 36 : g_clear_pointer (&output_args, g_strfreev);
1652 : :
1653 : 36 : g_assert_null (paths);
1654 : :
1655 : 36 : if (launch_data->type == TERMINAL_LAUNCH_TYPE_COMMAND_LINE_WITH_PATH_OVERRIDE)
1656 : 12 : g_assert_true (g_setenv ("PATH", old_path, TRUE));
1657 : :
1658 : 36 : g_close (fd, &error);
1659 : 36 : g_assert_no_error (error);
1660 : :
1661 : 36 : g_free (sh);
1662 : 36 : g_free (command_line);
1663 : 36 : g_free (bin_path);
1664 : 36 : g_free (terminal_path);
1665 : 36 : g_free (output_fd_path);
1666 : 36 : g_free (script_contents);
1667 : 36 : g_free (old_path);
1668 : 36 : g_clear_pointer (&output_args, g_strfreev);
1669 : 36 : g_clear_pointer (&output_contents, g_free);
1670 : 36 : g_clear_object (&data_input_stream);
1671 : 36 : g_clear_object (&input_stream);
1672 : 36 : g_clear_object (&app_info);
1673 : 36 : g_clear_object (&launch_context);
1674 : 36 : g_clear_error (&error);
1675 : 36 : g_clear_list (&paths, NULL);
1676 : 36 : g_clear_list (&uris, g_free);
1677 : 36 : }
1678 : :
1679 : : static void
1680 : 1 : test_launch_uris_with_invalid_terminal (void)
1681 : : {
1682 : : char *old_path;
1683 : : char *bin_path;
1684 : : GAppInfo *app_info;
1685 : 1 : GError *error = NULL;
1686 : :
1687 : 1 : bin_path = g_dir_make_tmp ("bin-path-XXXXXX", &error);
1688 : 1 : g_assert_no_error (error);
1689 : :
1690 : 1 : old_path = g_strdup (g_getenv ("PATH"));
1691 : 1 : g_assert_true (g_setenv ("PATH", bin_path, TRUE));
1692 : :
1693 : 1 : app_info = g_app_info_create_from_commandline ("true invalid-glib-terminal",
1694 : : "Test App on Invalid Terminal",
1695 : : G_APP_INFO_CREATE_NEEDS_TERMINAL |
1696 : : G_APP_INFO_CREATE_SUPPORTS_URIS,
1697 : : &error);
1698 : 1 : g_assert_no_error (error);
1699 : :
1700 : 1 : g_app_info_launch_uris (app_info, NULL, NULL, &error);
1701 : 1 : g_assert_error (error, G_IO_ERROR, G_IO_ERROR_FAILED);
1702 : 1 : g_clear_error (&error);
1703 : :
1704 : 1 : g_assert_true (g_setenv ("PATH", old_path, TRUE));
1705 : :
1706 : 1 : g_clear_object (&app_info);
1707 : 1 : g_clear_error (&error);
1708 : 1 : g_free (bin_path);
1709 : 1 : g_free (old_path);
1710 : 1 : }
1711 : :
1712 : : static void
1713 : 1 : test_app_path (void)
1714 : : {
1715 : : GDesktopAppInfo *appinfo;
1716 : : const char *desktop_path;
1717 : :
1718 : 1 : desktop_path = g_test_get_filename (G_TEST_BUILT, "appinfo-test-path.desktop", NULL);
1719 : 1 : appinfo = g_desktop_app_info_new_from_filename (desktop_path);
1720 : :
1721 : 1 : g_assert_true (G_IS_DESKTOP_APP_INFO (appinfo));
1722 : :
1723 : 1 : g_clear_object (&appinfo);
1724 : 1 : }
1725 : :
1726 : : static void
1727 : 1 : test_app_path_wrong (void)
1728 : : {
1729 : : GKeyFile *key_file;
1730 : : GDesktopAppInfo *appinfo;
1731 : 1 : const gchar bad_try_exec_file_contents[] =
1732 : : "[Desktop Entry]\n"
1733 : : "Type=Application\n"
1734 : : "Name=appinfo-test\n"
1735 : : "TryExec=appinfo-test\n"
1736 : : "Path=this-must-not-exist‼\n"
1737 : : "Exec=true\n";
1738 : 1 : const gchar bad_exec_file_contents[] =
1739 : : "[Desktop Entry]\n"
1740 : : "Type=Application\n"
1741 : : "Name=appinfo-test\n"
1742 : : "TryExec=true\n"
1743 : : "Path=this-must-not-exist‼\n"
1744 : : "Exec=appinfo-test\n";
1745 : :
1746 : 1 : g_assert_true (
1747 : : g_file_test (g_test_get_filename (G_TEST_BUILT, "appinfo-test", NULL),
1748 : : G_FILE_TEST_IS_REGULAR | G_FILE_TEST_IS_EXECUTABLE));
1749 : :
1750 : 1 : key_file = g_key_file_new ();
1751 : :
1752 : 1 : g_assert_true (
1753 : : g_key_file_load_from_data (key_file, bad_try_exec_file_contents, -1,
1754 : : G_KEY_FILE_NONE, NULL));
1755 : :
1756 : 1 : appinfo = g_desktop_app_info_new_from_keyfile (key_file);
1757 : 1 : g_assert_false (G_IS_DESKTOP_APP_INFO (appinfo));
1758 : :
1759 : 1 : g_assert_true (
1760 : : g_key_file_load_from_data (key_file, bad_exec_file_contents, -1,
1761 : : G_KEY_FILE_NONE, NULL));
1762 : :
1763 : 1 : appinfo = g_desktop_app_info_new_from_keyfile (key_file);
1764 : 1 : g_assert_false (G_IS_DESKTOP_APP_INFO (appinfo));
1765 : :
1766 : 1 : g_clear_pointer (&key_file, g_key_file_unref);
1767 : 1 : g_clear_object (&appinfo);
1768 : 1 : }
1769 : :
1770 : : static void
1771 : 1 : test_launch_startup_notify_fail (void)
1772 : : {
1773 : : GAppInfo *app_info;
1774 : : GAppLaunchContext *context;
1775 : 1 : GError *error = NULL;
1776 : : gboolean launch_started;
1777 : : gboolean launch_failed;
1778 : : gboolean launched;
1779 : : GList *uris;
1780 : :
1781 : 1 : app_info = g_app_info_create_from_commandline ("this-must-not-exist‼",
1782 : : "failing app",
1783 : : G_APP_INFO_CREATE_NONE |
1784 : : G_APP_INFO_CREATE_SUPPORTS_STARTUP_NOTIFICATION,
1785 : : &error);
1786 : 1 : g_assert_no_error (error);
1787 : :
1788 : 1 : context = g_object_new (test_launch_context_get_type (), NULL);
1789 : 1 : g_signal_connect (context, "launch-started",
1790 : : G_CALLBACK (on_launch_started),
1791 : : &launch_started);
1792 : 1 : g_signal_connect (context, "launched",
1793 : : G_CALLBACK (on_launch_started),
1794 : : &launched);
1795 : 1 : g_signal_connect (context, "launch-failed",
1796 : : G_CALLBACK (on_launch_failed),
1797 : : &launch_failed);
1798 : :
1799 : 1 : launch_started = FALSE;
1800 : 1 : launch_failed = FALSE;
1801 : 1 : launched = FALSE;
1802 : 1 : uris = g_list_prepend (NULL, g_file_new_for_uri ("foo://bar"));
1803 : 1 : uris = g_list_prepend (uris, g_file_new_for_uri ("bar://foo"));
1804 : 1 : g_assert_false (g_app_info_launch (app_info, uris, context, &error));
1805 : 1 : g_assert_error (error, G_SPAWN_ERROR, G_SPAWN_ERROR_NOENT);
1806 : 1 : g_assert_true (launch_started);
1807 : 1 : g_assert_true (launch_failed);
1808 : 1 : g_assert_false (launched);
1809 : :
1810 : 1 : g_clear_error (&error);
1811 : 1 : g_clear_object (&app_info);
1812 : 1 : g_clear_object (&context);
1813 : 1 : g_clear_list (&uris, g_object_unref);
1814 : 1 : }
1815 : :
1816 : : static void
1817 : 1 : test_launch_fail (void)
1818 : : {
1819 : : GAppInfo *app_info;
1820 : 1 : GError *error = NULL;
1821 : :
1822 : 1 : app_info = g_app_info_create_from_commandline ("this-must-not-exist‼",
1823 : : "failing app",
1824 : : G_APP_INFO_CREATE_NONE,
1825 : : &error);
1826 : 1 : g_assert_no_error (error);
1827 : :
1828 : 1 : g_assert_false (g_app_info_launch (app_info, NULL, NULL, &error));
1829 : 1 : g_assert_error (error, G_SPAWN_ERROR, G_SPAWN_ERROR_NOENT);
1830 : :
1831 : 1 : g_clear_error (&error);
1832 : 1 : g_clear_object (&app_info);
1833 : 1 : }
1834 : :
1835 : : static void
1836 : 1 : test_launch_fail_absolute_path (void)
1837 : : {
1838 : : GAppInfo *app_info;
1839 : 1 : GError *error = NULL;
1840 : :
1841 : 1 : app_info = g_app_info_create_from_commandline ("/nothing/of/this-must-exist‼",
1842 : : NULL,
1843 : : G_APP_INFO_CREATE_NONE,
1844 : : &error);
1845 : 1 : g_assert_no_error (error);
1846 : :
1847 : 1 : g_assert_false (g_app_info_launch (app_info, NULL, NULL, &error));
1848 : 1 : g_assert_error (error, G_SPAWN_ERROR, G_SPAWN_ERROR_NOENT);
1849 : :
1850 : 1 : g_clear_error (&error);
1851 : 1 : g_clear_object (&app_info);
1852 : :
1853 : 1 : app_info = g_app_info_create_from_commandline ("/",
1854 : : NULL,
1855 : : G_APP_INFO_CREATE_NONE,
1856 : : &error);
1857 : 1 : g_assert_no_error (error);
1858 : :
1859 : 1 : g_assert_false (g_app_info_launch (app_info, NULL, NULL, &error));
1860 : 1 : g_assert_error (error, G_SPAWN_ERROR, G_SPAWN_ERROR_NOENT);
1861 : :
1862 : 1 : g_clear_error (&error);
1863 : 1 : g_clear_object (&app_info);
1864 : 1 : }
1865 : :
1866 : : static void
1867 : 1 : async_result_cb (GObject *source_object,
1868 : : GAsyncResult *result,
1869 : : gpointer user_data)
1870 : : {
1871 : 1 : GAsyncResult **result_out = user_data;
1872 : :
1873 : 1 : g_assert (*result_out == NULL);
1874 : 1 : *result_out = g_object_ref (result);
1875 : 1 : g_main_context_wakeup (g_main_context_get_thread_default ());
1876 : 1 : }
1877 : :
1878 : : static void
1879 : 1 : test_launch_fail_dbus (void)
1880 : : {
1881 : 1 : GTestDBus *bus = NULL;
1882 : 1 : GDesktopAppInfo *app_info = NULL;
1883 : 1 : GAppLaunchContext *context = NULL;
1884 : 1 : GAsyncResult *result = NULL;
1885 : 1 : GError *error = NULL;
1886 : :
1887 : 1 : if (skip_missing_dbus_daemon ())
1888 : 0 : return;
1889 : :
1890 : : /* Set up a test session bus to ensure that launching the app happens using
1891 : : * D-Bus rather than spawning. */
1892 : 1 : bus = g_test_dbus_new (G_TEST_DBUS_NONE);
1893 : 1 : g_test_dbus_up (bus);
1894 : :
1895 : 1 : app_info = g_desktop_app_info_new_from_filename (g_test_get_filename (G_TEST_DIST, "org.gtk.test.dbusappinfo.desktop", NULL));
1896 : 1 : g_assert_nonnull (app_info);
1897 : :
1898 : 1 : g_assert_true (g_desktop_app_info_has_key (app_info, "DBusActivatable"));
1899 : :
1900 : 1 : context = g_app_launch_context_new ();
1901 : :
1902 : 1 : g_app_info_launch_uris_async (G_APP_INFO (app_info), NULL, context, NULL, async_result_cb, &result);
1903 : :
1904 : 4 : while (result == NULL)
1905 : 3 : g_main_context_iteration (NULL, TRUE);
1906 : :
1907 : 1 : g_assert_false (g_app_info_launch_uris_finish (G_APP_INFO (app_info), result, &error));
1908 : 1 : g_assert_error (error, G_DBUS_ERROR, G_DBUS_ERROR_SERVICE_UNKNOWN);
1909 : :
1910 : 1 : g_test_dbus_down (bus);
1911 : 1 : g_clear_object (&bus);
1912 : :
1913 : 1 : g_clear_error (&error);
1914 : 1 : g_clear_object (&result);
1915 : 1 : g_clear_object (&context);
1916 : 1 : g_clear_object (&app_info);
1917 : : }
1918 : :
1919 : : int
1920 : 1 : main (int argc,
1921 : : char *argv[])
1922 : : {
1923 : : guint i;
1924 : 1 : const gchar *supported_terminals[] = {
1925 : : "xdg-terminal-exec",
1926 : : "kgx",
1927 : : "gnome-terminal",
1928 : : "mate-terminal",
1929 : : "xfce4-terminal",
1930 : : "tilix",
1931 : : "konsole",
1932 : : "nxterm",
1933 : : "color-xterm",
1934 : : "rxvt",
1935 : : "dtterm",
1936 : : "xterm",
1937 : : };
1938 : :
1939 : : /* While we use %G_TEST_OPTION_ISOLATE_DIRS to create temporary directories
1940 : : * for each of the tests, we want to use the system MIME registry, assuming
1941 : : * that it exists and correctly has shared-mime-info installed. */
1942 : 1 : g_content_type_set_mime_dirs (NULL);
1943 : :
1944 : 1 : g_test_init (&argc, &argv, G_TEST_OPTION_ISOLATE_DIRS, NULL);
1945 : :
1946 : 1 : g_test_add_func ("/desktop-app-info/delete", test_delete);
1947 : 1 : g_test_add_func ("/desktop-app-info/default", test_default);
1948 : 1 : g_test_add_func ("/desktop-app-info/default-async", test_default_async);
1949 : 1 : g_test_add_func ("/desktop-app-info/fallback", test_fallback);
1950 : 1 : g_test_add_func ("/desktop-app-info/lastused", test_last_used);
1951 : 1 : g_test_add_func ("/desktop-app-info/extra-getters", test_extra_getters);
1952 : 1 : g_test_add_func ("/desktop-app-info/actions", test_actions);
1953 : 1 : g_test_add_func ("/desktop-app-info/search", test_search);
1954 : 1 : g_test_add_func ("/desktop-app-info/implements", test_implements);
1955 : 1 : g_test_add_func ("/desktop-app-info/show-in", test_show_in);
1956 : 1 : g_test_add_func ("/desktop-app-info/app-path", test_app_path);
1957 : 1 : g_test_add_func ("/desktop-app-info/app-path/wrong", test_app_path_wrong);
1958 : 1 : g_test_add_func ("/desktop-app-info/launch/fail", test_launch_fail);
1959 : 1 : g_test_add_func ("/desktop-app-info/launch/fail-absolute-path", test_launch_fail_absolute_path);
1960 : 1 : g_test_add_func ("/desktop-app-info/launch/fail-startup-notify", test_launch_startup_notify_fail);
1961 : 1 : g_test_add_func ("/desktop-app-info/launch/fail-dbus", test_launch_fail_dbus);
1962 : 1 : g_test_add_func ("/desktop-app-info/launch-as-manager", test_launch_as_manager);
1963 : 1 : g_test_add_func ("/desktop-app-info/launch-as-manager/fail", test_launch_as_manager_fail);
1964 : 1 : g_test_add_func ("/desktop-app-info/launch-default-uri-handler", test_default_uri_handler);
1965 : 1 : g_test_add_func ("/desktop-app-info/launch-default-uri-handler-async", test_default_uri_handler_async);
1966 : 1 : g_test_add_func ("/desktop-app-info/id", test_id);
1967 : :
1968 : 13 : for (i = 0; i < G_N_ELEMENTS (supported_terminals); i++)
1969 : : {
1970 : : char *path;
1971 : :
1972 : 12 : path = g_strdup_printf ("/desktop-app-info/launch-uris-with-terminal/with-path/%s",
1973 : : supported_terminals[i]);
1974 : 12 : g_test_add_data_func_full (path,
1975 : 12 : terminal_launch_data_new (supported_terminals[i],
1976 : : TERMINAL_LAUNCH_TYPE_COMMAND_LINE_WITH_PATH_OVERRIDE),
1977 : : test_launch_uris_with_terminal, g_free);
1978 : 12 : g_clear_pointer (&path, g_free);
1979 : :
1980 : 12 : path = g_strdup_printf ("/desktop-app-info/launch-uris-with-terminal/with-context/%s",
1981 : : supported_terminals[i]);
1982 : 12 : g_test_add_data_func_full (path,
1983 : 12 : terminal_launch_data_new (supported_terminals[i],
1984 : : TERMINAL_LAUNCH_TYPE_COMMAND_LINE_WITH_CONTEXT),
1985 : : test_launch_uris_with_terminal, g_free);
1986 : 12 : g_clear_pointer (&path, g_free);
1987 : :
1988 : 12 : path = g_strdup_printf ("/desktop-app-info/launch-uris-with-terminal/with-desktop-path/%s",
1989 : : supported_terminals[i]);
1990 : 12 : g_test_add_data_func_full (path,
1991 : 12 : terminal_launch_data_new (supported_terminals[i],
1992 : : TERMINAL_LAUNCH_TYPE_KEY_FILE_WITH_PATH),
1993 : : test_launch_uris_with_terminal, g_free);
1994 : 12 : g_clear_pointer (&path, g_free);
1995 : : }
1996 : :
1997 : 1 : g_test_add_func ("/desktop-app-info/launch-uris-with-terminal/invalid-glib-terminal",
1998 : : test_launch_uris_with_invalid_terminal);
1999 : :
2000 : 1 : return g_test_run ();
2001 : : }
|