tests: Use fixture in GIRepository tests

This deduplicates some code and will allow writing further tests more
easily.
This commit is contained in:
Philip Chimento 2024-01-21 10:08:17 -08:00
parent d41f05e9cf
commit b8a65edb01
5 changed files with 160 additions and 164 deletions

View File

@ -20,47 +20,22 @@
* Author: Philip Withnall <pwithnall@gnome.org> * Author: Philip Withnall <pwithnall@gnome.org>
*/ */
#include "glib.h"
#include "girepository.h" #include "girepository.h"
#include "girffi.h" #include "girffi.h"
#include "glib.h"
static GIRepository * #include "test-common.h"
load_typelib_from_builddir (const char *name,
const char *version)
{
GIRepository *repository;
char *gobject_typelib_dir = NULL;
GITypelib *typelib = NULL;
GError *local_error = NULL;
gobject_typelib_dir = g_test_build_filename (G_TEST_BUILT, "..", "..", "introspection", NULL);
g_test_message ("Using GI_TYPELIB_DIR = %s", gobject_typelib_dir);
gi_repository_prepend_search_path (gobject_typelib_dir);
g_free (gobject_typelib_dir);
repository = gi_repository_new ();
g_assert_nonnull (repository);
typelib = gi_repository_require (repository, name, version, 0, &local_error);
g_assert_no_error (local_error);
g_assert_nonnull (typelib);
return g_steal_pointer (&repository);
}
static void static void
test_function_info_invoker (void) test_function_info_invoker (RepositoryFixture *fx,
const void *unused)
{ {
GIRepository *repository;
GIFunctionInfo *function_info = NULL; GIFunctionInfo *function_info = NULL;
GIFunctionInvoker invoker; GIFunctionInvoker invoker;
GError *local_error = NULL; GError *local_error = NULL;
g_test_summary ("Test preparing a function invoker"); g_test_summary ("Test preparing a function invoker");
repository = load_typelib_from_builddir ("GLib", "2.0"); function_info = (GIFunctionInfo *) gi_repository_find_by_name (fx->repository, "GLib", "get_locale_variants");
function_info = (GIFunctionInfo *) gi_repository_find_by_name (repository, "GLib", "get_locale_variants");
g_assert_nonnull (function_info); g_assert_nonnull (function_info);
gi_function_info_prep_invoker (function_info, &invoker, &local_error); gi_function_info_prep_invoker (function_info, &invoker, &local_error);
@ -68,21 +43,15 @@ test_function_info_invoker (void)
gi_function_invoker_clear (&invoker); gi_function_invoker_clear (&invoker);
g_clear_pointer (&function_info, gi_base_info_unref); g_clear_pointer (&function_info, gi_base_info_unref);
g_clear_object (&repository);
} }
int int
main (int argc, main (int argc,
char *argv[]) char *argv[])
{ {
/* Isolate from the system typelibs and GIRs. */ repository_init (&argc, &argv);
g_setenv ("GI_TYPELIB_PATH", "/dev/null", TRUE);
g_setenv ("GI_GIR_PATH", "/dev/null", TRUE);
g_test_init (&argc, &argv, G_TEST_OPTION_ISOLATE_DIRS, NULL); ADD_REPOSITORY_TEST ("/function-info/invoker", test_function_info_invoker, &typelib_load_spec_glib);
g_test_add_func ("/function-info/invoker", test_function_info_invoker);
return g_test_run (); return g_test_run ();
} }

View File

@ -58,7 +58,7 @@ foreach test_name, extra_args : girepository_tests
) )
endif endif
exe = executable(test_name, source, exe = executable(test_name, source, 'test-common.c',
c_args: test_cargs + extra_args.get('c_args', []), c_args: test_cargs + extra_args.get('c_args', []),
cpp_args: test_cpp_args + extra_args.get('cpp_args', []), cpp_args: test_cpp_args + extra_args.get('cpp_args', []),
link_args: extra_args.get('link_args', []), link_args: extra_args.get('link_args', []),

View File

@ -21,60 +21,27 @@
#include "glib.h" #include "glib.h"
#include "girepository.h" #include "girepository.h"
#include "test-common.h"
static GIRepository *
load_typelib_from_builddir (const char *name,
const char *version)
{
GIRepository *repository;
char *gobject_typelib_dir = NULL;
GITypelib *typelib = NULL;
GError *local_error = NULL;
gobject_typelib_dir = g_test_build_filename (G_TEST_BUILT, "..", "..", "introspection", NULL);
g_test_message ("Using GI_TYPELIB_DIR = %s", gobject_typelib_dir);
gi_repository_prepend_search_path (gobject_typelib_dir);
g_free (gobject_typelib_dir);
repository = gi_repository_new ();
g_assert_nonnull (repository);
typelib = gi_repository_require (repository, name, version, 0, &local_error);
g_assert_no_error (local_error);
g_assert_nonnull (typelib);
return g_steal_pointer (&repository);
}
static void static void
test_repository_basic (void) test_repository_basic (RepositoryFixture *fx,
const void *unused)
{ {
GIRepository *repository;
char *gobject_typelib_dir = NULL;
const char * const * search_paths; const char * const * search_paths;
GITypelib *typelib = NULL;
char **namespaces = NULL; char **namespaces = NULL;
const char *expected_namespaces[] = { "GLib", NULL }; const char *expected_namespaces[] = { "GLib", NULL };
GError *local_error = NULL;
char **versions; char **versions;
size_t n_versions; size_t n_versions;
g_test_summary ("Test basic opening of a repository and requiring a typelib"); g_test_summary ("Test basic opening of a repository and requiring a typelib");
gobject_typelib_dir = g_test_build_filename (G_TEST_BUILT, "..", "..", "introspection", NULL); versions = gi_repository_enumerate_versions (fx->repository, "SomeInvalidNamespace", &n_versions);
g_test_message ("Using GI_TYPELIB_DIR = %s", gobject_typelib_dir);
gi_repository_prepend_search_path (gobject_typelib_dir);
repository = gi_repository_new ();
g_assert_nonnull (repository);
versions = gi_repository_enumerate_versions (repository, "SomeInvalidNamespace", &n_versions);
g_assert_nonnull (versions); g_assert_nonnull (versions);
g_assert_cmpstrv (versions, ((char *[]){NULL})); g_assert_cmpstrv (versions, ((char *[]){NULL}));
g_assert_cmpuint (n_versions, ==, 0); g_assert_cmpuint (n_versions, ==, 0);
g_clear_pointer (&versions, g_strfreev); g_clear_pointer (&versions, g_strfreev);
versions = gi_repository_enumerate_versions (repository, "GLib", NULL); versions = gi_repository_enumerate_versions (fx->repository, "GLib", NULL);
g_assert_nonnull (versions); g_assert_nonnull (versions);
g_assert_cmpstrv (versions, ((char *[]){"2.0", NULL})); g_assert_cmpstrv (versions, ((char *[]){"2.0", NULL}));
g_clear_pointer (&versions, g_strfreev); g_clear_pointer (&versions, g_strfreev);
@ -82,46 +49,24 @@ test_repository_basic (void)
search_paths = gi_repository_get_search_path (NULL); search_paths = gi_repository_get_search_path (NULL);
g_assert_nonnull (search_paths); g_assert_nonnull (search_paths);
g_assert_cmpuint (g_strv_length ((char **) search_paths), >, 0); g_assert_cmpuint (g_strv_length ((char **) search_paths), >, 0);
g_assert_cmpstr (search_paths[0], ==, gobject_typelib_dir); g_assert_cmpstr (search_paths[0], ==, fx->gobject_typelib_dir);
typelib = gi_repository_require (repository, "GLib", "2.0", 0, &local_error); namespaces = gi_repository_get_loaded_namespaces (fx->repository);
g_assert_no_error (local_error);
g_assert_nonnull (typelib);
namespaces = gi_repository_get_loaded_namespaces (repository);
g_assert_cmpstrv (namespaces, expected_namespaces); g_assert_cmpstrv (namespaces, expected_namespaces);
g_strfreev (namespaces); g_strfreev (namespaces);
g_free (gobject_typelib_dir);
g_clear_object (&repository);
} }
static void static void
test_repository_info (void) test_repository_info (RepositoryFixture *fx,
const void *unused)
{ {
GIRepository *repository;
char *gobject_typelib_dir = NULL;
GITypelib *typelib = NULL;
GIObjectInfo *object_info = NULL; GIObjectInfo *object_info = NULL;
GISignalInfo *signal_info = NULL; GISignalInfo *signal_info = NULL;
GIFunctionInfo *method_info = NULL; GIFunctionInfo *method_info = NULL;
GError *local_error = NULL;
g_test_summary ("Test retrieving some basic info blobs from a typelib"); g_test_summary ("Test retrieving some basic info blobs from a typelib");
gobject_typelib_dir = g_test_build_filename (G_TEST_BUILT, "..", "..", "introspection", NULL); object_info = (GIObjectInfo *) gi_repository_find_by_name (fx->repository, "GObject", "Object");
g_test_message ("Using GI_TYPELIB_DIR = %s", gobject_typelib_dir);
gi_repository_prepend_search_path (gobject_typelib_dir);
g_free (gobject_typelib_dir);
repository = gi_repository_new ();
g_assert_nonnull (repository);
typelib = gi_repository_require (repository, "GObject", "2.0", 0, &local_error);
g_assert_no_error (local_error);
g_assert_nonnull (typelib);
object_info = (GIObjectInfo *) gi_repository_find_by_name (repository, "GObject", "Object");
g_assert_nonnull (object_info); g_assert_nonnull (object_info);
g_assert_cmpint (gi_base_info_get_info_type ((GIBaseInfo *) object_info), ==, GI_INFO_TYPE_OBJECT); g_assert_cmpint (gi_base_info_get_info_type ((GIBaseInfo *) object_info), ==, GI_INFO_TYPE_OBJECT);
@ -150,45 +95,29 @@ test_repository_info (void)
gi_base_info_unref (signal_info); gi_base_info_unref (signal_info);
gi_base_info_unref (object_info); gi_base_info_unref (object_info);
g_clear_object (&repository);
} }
static void static void
test_repository_dependencies (void) test_repository_dependencies (RepositoryFixture *fx,
const void *unused)
{ {
GIRepository *repository;
GITypelib *typelib;
GError *error = NULL; GError *error = NULL;
char *gobject_typelib_dir = NULL;
char **dependencies; char **dependencies;
g_test_summary ("Test ensures namespace dependencies are correctly exposed"); g_test_summary ("Test ensures namespace dependencies are correctly exposed");
gobject_typelib_dir = g_test_build_filename (G_TEST_BUILT, "..", "..", "gobject", NULL); dependencies = gi_repository_get_dependencies (fx->repository, "GObject");
g_test_message ("Using GI_TYPELIB_DIR = %s", gobject_typelib_dir);
gi_repository_prepend_search_path (gobject_typelib_dir);
g_free (gobject_typelib_dir);
repository = gi_repository_new ();
g_assert_nonnull (repository);
typelib = gi_repository_require (repository, "GObject", "2.0", 0, &error);
g_assert_no_error (error);
g_assert_nonnull (typelib);
dependencies = gi_repository_get_dependencies (repository, "GObject");
g_assert_cmpuint (g_strv_length (dependencies), ==, 1); g_assert_cmpuint (g_strv_length (dependencies), ==, 1);
g_assert_true (g_strv_contains ((const char **) dependencies, "GLib-2.0")); g_assert_true (g_strv_contains ((const char **) dependencies, "GLib-2.0"));
g_clear_error (&error); g_clear_error (&error);
g_clear_object (&repository);
g_clear_pointer (&dependencies, g_strfreev); g_clear_pointer (&dependencies, g_strfreev);
} }
static void static void
test_repository_arg_info (void) test_repository_arg_info (RepositoryFixture *fx,
const void *unused)
{ {
GIRepository *repository;
GIObjectInfo *object_info = NULL; GIObjectInfo *object_info = NULL;
GIStructInfo *struct_info = NULL; GIStructInfo *struct_info = NULL;
GIFunctionInfo *method_info = NULL; GIFunctionInfo *method_info = NULL;
@ -198,12 +127,10 @@ test_repository_arg_info (void)
g_test_summary ("Test retrieving GIArgInfos from a typelib"); g_test_summary ("Test retrieving GIArgInfos from a typelib");
repository = load_typelib_from_builddir ("GObject", "2.0");
/* Test all the methods of GIArgInfo. Here were looking at the /* Test all the methods of GIArgInfo. Here were looking at the
* `const char *property_name` argument of g_object_get_property(). (The * `const char *property_name` argument of g_object_get_property(). (The
* self argument is not exposed through gi_callable_info_get_arg().) */ * self argument is not exposed through gi_callable_info_get_arg().) */
object_info = (GIObjectInfo *) gi_repository_find_by_name (repository, "GObject", "Object"); object_info = (GIObjectInfo *) gi_repository_find_by_name (fx->repository, "GObject", "Object");
g_assert_nonnull (object_info); g_assert_nonnull (object_info);
method_info = gi_object_info_find_method (object_info, "get_property"); method_info = gi_object_info_find_method (object_info, "get_property");
@ -239,7 +166,7 @@ test_repository_arg_info (void)
/* Test an (out) argument. Here its the `guint *n_properties` from /* Test an (out) argument. Here its the `guint *n_properties` from
* g_object_class_list_properties(). */ * g_object_class_list_properties(). */
struct_info = (GIStructInfo *) gi_repository_find_by_name (repository, "GObject", "ObjectClass"); struct_info = (GIStructInfo *) gi_repository_find_by_name (fx->repository, "GObject", "ObjectClass");
g_assert_nonnull (struct_info); g_assert_nonnull (struct_info);
method_info = gi_struct_info_find_method (struct_info, "list_properties"); method_info = gi_struct_info_find_method (struct_info, "list_properties");
@ -256,33 +183,27 @@ test_repository_arg_info (void)
g_clear_pointer (&arg_info, gi_base_info_unref); g_clear_pointer (&arg_info, gi_base_info_unref);
g_clear_pointer (&method_info, gi_base_info_unref); g_clear_pointer (&method_info, gi_base_info_unref);
g_clear_pointer (&struct_info, gi_base_info_unref); g_clear_pointer (&struct_info, gi_base_info_unref);
g_clear_object (&repository);
} }
static void static void
test_repository_boxed_info (void) test_repository_boxed_info (RepositoryFixture *fx,
const void *unused)
{ {
GIRepository *repository;
GIBoxedInfo *boxed_info = NULL; GIBoxedInfo *boxed_info = NULL;
g_test_summary ("Test retrieving GIBoxedInfos from a typelib"); g_test_summary ("Test retrieving GIBoxedInfos from a typelib");
repository = load_typelib_from_builddir ("GObject", "2.0");
/* Test all the methods of GIBoxedInfo. This is simple, because there are none. */ /* Test all the methods of GIBoxedInfo. This is simple, because there are none. */
boxed_info = (GIBoxedInfo *) gi_repository_find_by_name (repository, "GObject", "BookmarkFile"); boxed_info = (GIBoxedInfo *) gi_repository_find_by_name (fx->repository, "GObject", "BookmarkFile");
g_assert_nonnull (boxed_info); g_assert_nonnull (boxed_info);
g_clear_pointer (&boxed_info, gi_base_info_unref); g_clear_pointer (&boxed_info, gi_base_info_unref);
g_clear_object (&repository);
} }
static void static void
test_repository_callable_info (void) test_repository_callable_info (RepositoryFixture *fx,
const void *unused)
{ {
GIRepository *repository;
GIObjectInfo *object_info = NULL; GIObjectInfo *object_info = NULL;
GIFunctionInfo *method_info = NULL; GIFunctionInfo *method_info = NULL;
GICallableInfo *callable_info; GICallableInfo *callable_info;
@ -293,11 +214,9 @@ test_repository_callable_info (void)
g_test_summary ("Test retrieving GICallableInfos from a typelib"); g_test_summary ("Test retrieving GICallableInfos from a typelib");
repository = load_typelib_from_builddir ("GObject", "2.0");
/* Test all the methods of GICallableInfo. Here were looking at /* Test all the methods of GICallableInfo. Here were looking at
* g_object_get_qdata(). */ * g_object_get_qdata(). */
object_info = (GIObjectInfo *) gi_repository_find_by_name (repository, "GObject", "Object"); object_info = (GIObjectInfo *) gi_repository_find_by_name (fx->repository, "GObject", "Object");
g_assert_nonnull (object_info); g_assert_nonnull (object_info);
method_info = gi_object_info_find_method (object_info, "get_qdata"); method_info = gi_object_info_find_method (object_info, "get_qdata");
@ -333,46 +252,36 @@ test_repository_callable_info (void)
g_clear_pointer (&method_info, gi_base_info_unref); g_clear_pointer (&method_info, gi_base_info_unref);
g_clear_pointer (&object_info, gi_base_info_unref); g_clear_pointer (&object_info, gi_base_info_unref);
g_clear_object (&repository);
} }
static void static void
test_repository_callback_info (void) test_repository_callback_info (RepositoryFixture *fx,
const void *unused)
{ {
GIRepository *repository;
GICallbackInfo *callback_info = NULL; GICallbackInfo *callback_info = NULL;
g_test_summary ("Test retrieving GICallbackInfos from a typelib"); g_test_summary ("Test retrieving GICallbackInfos from a typelib");
repository = load_typelib_from_builddir ("GObject", "2.0");
/* Test all the methods of GICallbackInfo. This is simple, because there are none. */ /* Test all the methods of GICallbackInfo. This is simple, because there are none. */
callback_info = (GICallbackInfo *) gi_repository_find_by_name (repository, "GObject", "ObjectFinalizeFunc"); callback_info = (GICallbackInfo *) gi_repository_find_by_name (fx->repository, "GObject", "ObjectFinalizeFunc");
g_assert_nonnull (callback_info); g_assert_nonnull (callback_info);
g_clear_pointer (&callback_info, gi_base_info_unref); g_clear_pointer (&callback_info, gi_base_info_unref);
g_clear_object (&repository);
} }
int int
main (int argc, main (int argc,
char *argv[]) char *argv[])
{ {
/* Isolate from the system typelibs and GIRs. */ repository_init (&argc, &argv);
g_setenv ("GI_TYPELIB_PATH", "/dev/null", TRUE);
g_setenv ("GI_GIR_PATH", "/dev/null", TRUE);
g_test_init (&argc, &argv, G_TEST_OPTION_ISOLATE_DIRS, NULL); ADD_REPOSITORY_TEST ("/repository/basic", test_repository_basic, &typelib_load_spec_glib);
ADD_REPOSITORY_TEST ("/repository/info", test_repository_info, &typelib_load_spec_gobject);
g_test_add_func ("/repository/basic", test_repository_basic); ADD_REPOSITORY_TEST ("/repository/dependencies", test_repository_dependencies, &typelib_load_spec_gobject);
g_test_add_func ("/repository/info", test_repository_info); ADD_REPOSITORY_TEST ("/repository/arg-info", test_repository_arg_info, &typelib_load_spec_gobject);
g_test_add_func ("/repository/dependencies", test_repository_dependencies); ADD_REPOSITORY_TEST ("/repository/boxed-info", test_repository_boxed_info, &typelib_load_spec_gobject);
g_test_add_func ("/repository/arg-info", test_repository_arg_info); ADD_REPOSITORY_TEST ("/repository/callable-info", test_repository_callable_info, &typelib_load_spec_gobject);
g_test_add_func ("/repository/boxed-info", test_repository_boxed_info); ADD_REPOSITORY_TEST ("/repository/callback-info", test_repository_callback_info, &typelib_load_spec_gobject);
g_test_add_func ("/repository/callable-info", test_repository_callable_info);
g_test_add_func ("/repository/callback-info", test_repository_callback_info);
return g_test_run (); return g_test_run ();
} }

View File

@ -0,0 +1,67 @@
/*
* Copyright 2023 GNOME Foundation, Inc.
* Copyright 2024 Philip Chimento <philip.chimento@gmail.com>
*
* SPDX-License-Identifier: LGPL-2.1-or-later
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General
* Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
*/
#include "config.h"
#include "girepository.h"
#include "glib.h"
#include "test-common.h"
void
repository_init (int *argc,
char **argv[])
{
/* Isolate from the system typelibs and GIRs. */
g_setenv ("GI_TYPELIB_PATH", "/dev/null", TRUE);
g_setenv ("GI_GIR_PATH", "/dev/null", TRUE);
g_test_init (argc, argv, G_TEST_OPTION_ISOLATE_DIRS, NULL);
}
void
repository_setup (RepositoryFixture *fx,
const void *data)
{
GITypelib *typelib = NULL;
GError *local_error = NULL;
TypelibLoadSpec *load_spec = (TypelibLoadSpec *) data;
fx->gobject_typelib_dir = g_test_build_filename (G_TEST_BUILT, "..", "..", "introspection", NULL);
g_test_message ("Using GI_TYPELIB_DIR = %s", fx->gobject_typelib_dir);
gi_repository_prepend_search_path (fx->gobject_typelib_dir);
fx->repository = gi_repository_new ();
g_assert_nonnull (fx->repository);
if (load_spec)
{
typelib = gi_repository_require (fx->repository, load_spec->name, load_spec->version, 0, &local_error);
g_assert_no_error (local_error);
g_assert_nonnull (typelib);
}
}
void
repository_teardown (RepositoryFixture *fx,
const void *unused)
{
g_clear_pointer (&fx->gobject_typelib_dir, g_free);
g_clear_object (&fx->repository);
}

View File

@ -0,0 +1,51 @@
/*
* Copyright 2024 Philip Chimento <philip.chimento@gmail.com>
*
* SPDX-License-Identifier: LGPL-2.1-or-later
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General
* Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
*/
#pragma once
#include "config.h"
#include "girepository.h"
typedef struct
{
GIRepository *repository;
char *gobject_typelib_dir;
} RepositoryFixture;
typedef struct
{
const char *name;
const char *version;
} TypelibLoadSpec;
static const TypelibLoadSpec typelib_load_spec_glib = { "GLib", "2.0" };
static const TypelibLoadSpec typelib_load_spec_gobject = { "GObject", "2.0" };
void repository_init (int *argc,
char **argv[]);
void repository_setup (RepositoryFixture *fx,
const void *data);
void repository_teardown (RepositoryFixture *fx,
const void *unused);
#define ADD_REPOSITORY_TEST(testpath, ftest, load_spec) \
g_test_add ((testpath), RepositoryFixture, load_spec, repository_setup, (ftest), repository_teardown)