/* GStreamer unit tests for discoverer * * Copyright (C) 2011 Stefan Kost * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library 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 * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public * License along with this library; if not, write to the * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, * Boston, MA 02110-1301, USA. */ #ifdef HAVE_CONFIG_H # include #endif #include #include #include #include #include static gboolean have_theora, have_ogg; GST_START_TEST (test_disco_init) { GError *err = NULL; GstDiscoverer *dc; dc = gst_discoverer_new (GST_SECOND, &err); fail_unless (dc != NULL); fail_unless (err == NULL); g_object_unref (dc); } GST_END_TEST; GST_START_TEST (test_disco_serializing) { GError *err = NULL; GstDiscoverer *dc; GstDiscovererInfo *info, *dinfo; gchar *uri; GVariant *serialized, *reserialized; GList *audio_streams; gchar *path = g_build_filename (GST_TEST_FILES_PATH, "theora-vorbis.ogg", NULL); /* high timeout, in case we're running under valgrind */ dc = gst_discoverer_new (30 * GST_SECOND, &err); fail_unless (dc != NULL); fail_unless (err == NULL); uri = gst_filename_to_uri (path, &err); g_free (path); fail_unless (err == NULL); info = gst_discoverer_discover_uri (dc, uri, &err); fail_unless (info); if (have_theora && have_ogg) { fail_unless_equals_int (gst_discoverer_info_get_result (info), GST_DISCOVERER_OK); } else { fail_unless_equals_int (gst_discoverer_info_get_result (info), GST_DISCOVERER_MISSING_PLUGINS); g_clear_error (&err); goto missing_plugins; } serialized = gst_discoverer_info_to_variant (info, GST_DISCOVERER_SERIALIZE_ALL); fail_unless (serialized); dinfo = gst_discoverer_info_from_variant (serialized); fail_unless (dinfo); audio_streams = gst_discoverer_info_get_audio_streams (dinfo); fail_unless_equals_int (g_list_length (audio_streams), 1); gst_discoverer_stream_info_list_free (audio_streams); reserialized = gst_discoverer_info_to_variant (dinfo, GST_DISCOVERER_SERIALIZE_ALL); fail_unless (g_variant_equal (serialized, reserialized)); gst_discoverer_info_unref (dinfo); g_variant_unref (serialized); g_variant_unref (reserialized); missing_plugins: gst_discoverer_info_unref (info); g_object_unref (dc); g_free (uri); } GST_END_TEST; GST_START_TEST (test_disco_sync) { GError *err = NULL; GstDiscoverer *dc; GstDiscovererInfo *info; GstDiscovererResult result; gchar *uri; /* high timeout, in case we're running under valgrind */ dc = gst_discoverer_new (30 * GST_SECOND, &err); fail_unless (dc != NULL); fail_unless (err == NULL); /* GST_TEST_FILE comes from makefile CFLAGS */ GST_INFO ("discovering file '%s'", GST_TEST_FILE); uri = g_filename_to_uri (GST_TEST_FILE, NULL, &err); fail_unless (err == NULL); GST_INFO ("discovering uri '%s'", uri); info = gst_discoverer_discover_uri (dc, uri, &err); result = gst_discoverer_info_get_result (info); GST_INFO ("result: %d", result); gst_discoverer_info_unref (info); g_free (uri); if (err) { /* we won't have the codec for the jpeg */ g_error_free (err); } g_object_unref (dc); } GST_END_TEST; static void test_disco_sync_reuse (const gchar * test_fn, guint num, GstClockTime timeout) { GError *err = NULL; GstDiscoverer *dc; GstDiscovererInfo *info; GstDiscovererResult result; gchar *uri, *path; int i; dc = gst_discoverer_new (timeout, &err); fail_unless (dc != NULL); fail_unless (err == NULL); /* GST_TEST_FILE comes from makefile CFLAGS */ path = g_build_filename (GST_TEST_FILES_PATH, test_fn, NULL); uri = gst_filename_to_uri (path, &err); g_free (path); fail_unless (err == NULL); for (i = 0; i < num; ++i) { GST_INFO ("[%02d] discovering uri '%s'", i, uri); info = gst_discoverer_discover_uri (dc, uri, &err); if (info) { result = gst_discoverer_info_get_result (info); GST_INFO ("result: %d", result); gst_discoverer_info_unref (info); } /* in case we don't have some of the elements needed */ if (err) { g_error_free (err); err = NULL; } } g_free (uri); g_object_unref (dc); } GST_START_TEST (test_disco_sync_reuse_ogg) { test_disco_sync_reuse ("theora-vorbis.ogg", 2, 10 * GST_SECOND); } GST_END_TEST; GST_START_TEST (test_disco_sync_reuse_mp3) { /* this will cause errors because -base doesn't do mp3 parsing or decoding */ test_disco_sync_reuse ("test.mp3", 3, 10 * GST_SECOND); } GST_END_TEST; GST_START_TEST (test_disco_sync_reuse_timeout) { /* set minimum timeout to test that, esp. leakage under valgrind */ /* FIXME: should really be even shorter */ test_disco_sync_reuse ("theora-vorbis.ogg", 2, GST_SECOND); } GST_END_TEST; GST_START_TEST (test_disco_missing_plugins) { const gchar *files[] = { "test.mkv", "test.mp3", "partialframe.mjpeg" }; GError *err = NULL; GstDiscoverer *dc; GstDiscovererInfo *info; GstDiscovererResult result; gchar *uri, *path; int i; for (i = 0; i < G_N_ELEMENTS (files); ++i) { dc = gst_discoverer_new (5 * GST_SECOND, &err); fail_unless (dc != NULL); fail_unless (err == NULL); /* GST_TEST_FILE comes from makefile CFLAGS */ path = g_build_filename (GST_TEST_FILES_PATH, files[i], NULL); uri = gst_filename_to_uri (path, &err); g_free (path); fail_unless (err == NULL); GST_INFO ("discovering uri '%s'", uri); info = gst_discoverer_discover_uri (dc, uri, &err); fail_unless (info != NULL); fail_unless (err != NULL); result = gst_discoverer_info_get_result (info); GST_INFO ("result: %d, error message: %s", result, err->message); fail_unless_equals_int (result, GST_DISCOVERER_MISSING_PLUGINS); #ifndef GST_DISABLE_DEPRECATED GST_INFO ("misc: %" GST_PTR_FORMAT, gst_discoverer_info_get_misc (info)); #endif gst_discoverer_info_unref (info); g_error_free (err); err = NULL; g_free (uri); g_object_unref (dc); } } GST_END_TEST; typedef struct _AsyncTestData { gchar *uri; GMainLoop *loop; GstDiscovererResult result; } AsyncTestData; static void discovered_cb (GstDiscoverer * discoverer, GstDiscovererInfo * info, GError * err, AsyncTestData * data) { const gchar *uri = gst_discoverer_info_get_uri (info); fail_unless_equals_string (data->uri, uri); /* cannot ensure GST_DISCOVERER_OK since there might be missing plugins */ data->result = gst_discoverer_info_get_result (info); g_main_loop_quit (data->loop); } static void test_disco_async_with_context (GMainContext * context, guint num) { GstDiscoverer *dc; GError *err = NULL; AsyncTestData data = { 0, }; gchar *path = g_build_filename (GST_TEST_FILES_PATH, "theora-vorbis.ogg", NULL); if (context) g_main_context_push_thread_default (context); data.uri = gst_filename_to_uri (path, &err); /* something wrong if we have error here */ fail_unless (err == NULL); g_free (path); data.loop = g_main_loop_new (context, FALSE); /* high timeout, in case we're running under valgrind */ dc = gst_discoverer_new (30 * GST_SECOND, &err); fail_unless (dc != NULL); fail_unless (err == NULL); g_signal_connect (dc, "discovered", G_CALLBACK (discovered_cb), &data); gst_discoverer_start (dc); for (guint i = 0; i < num; ++i) { fail_unless (gst_discoverer_discover_uri_async (dc, data.uri) == TRUE); g_main_loop_run (data.loop); if (have_theora && have_ogg) { fail_unless_equals_int (data.result, GST_DISCOVERER_OK); } else { fail_unless_equals_int (data.result, GST_DISCOVERER_MISSING_PLUGINS); } } gst_discoverer_stop (dc); g_object_unref (dc); g_free (data.uri); g_main_loop_unref (data.loop); if (context) g_main_context_pop_thread_default (context); } GST_START_TEST (test_disco_async) { /* use default GMainContext */ test_disco_async_with_context (NULL, 1); } GST_END_TEST; GST_START_TEST (test_disco_async_reuse) { /* use default GMainContext */ test_disco_async_with_context (NULL, 3); } GST_END_TEST; typedef struct _CustomContextData { GMutex lock; GCond cond; gboolean finish; } CustomContextData; static gpointer custom_context_thread_func (CustomContextData * data) { GMainContext *context; /* test async APIs with custom GMainContext */ context = g_main_context_new (); test_disco_async_with_context (context, 1); g_main_context_unref (context); data->finish = TRUE; g_cond_signal (&data->cond); return NULL; } GST_START_TEST (test_disco_async_custom_context) { GThread *thread; CustomContextData data; g_mutex_init (&data.lock); g_cond_init (&data.cond); data.finish = FALSE; /* ensure default context here, but we will use other thread default context * instead of this */ g_main_context_default (); thread = g_thread_new ("test-custom-context-thread", (GThreadFunc) custom_context_thread_func, &data); g_mutex_lock (&data.lock); while (!data.finish) g_cond_wait (&data.cond, &data.lock); g_mutex_unlock (&data.lock); g_thread_join (thread); g_mutex_clear (&data.lock); g_cond_clear (&data.cond); } GST_END_TEST; static Suite * discoverer_suite (void) { Suite *s = suite_create ("discoverer"); TCase *tc_chain = tcase_create ("general"); have_theora = gst_registry_check_feature_version (gst_registry_get (), "theoradec", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0); have_ogg = gst_registry_check_feature_version (gst_registry_get (), "oggdemux", GST_VERSION_MAJOR, GST_VERSION_MINOR, 0); suite_add_tcase (s, tc_chain); tcase_add_test (tc_chain, test_disco_init); tcase_add_test (tc_chain, test_disco_sync); tcase_add_test (tc_chain, test_disco_sync_reuse_ogg); tcase_add_test (tc_chain, test_disco_sync_reuse_mp3); tcase_add_test (tc_chain, test_disco_sync_reuse_timeout); tcase_add_test (tc_chain, test_disco_missing_plugins); tcase_add_test (tc_chain, test_disco_serializing); tcase_add_test (tc_chain, test_disco_async); tcase_add_test (tc_chain, test_disco_async_custom_context); tcase_add_test (tc_chain, test_disco_async_reuse); return s; } GST_CHECK_MAIN (discoverer);