[evolution-data-server] Added concurrent view test: test-client-view-operations.c



commit 4fe9f583fc00be7f613467343aa3d12e808c04d1
Author: Tristan Van Berkom <tristanvb openismus com>
Date:   Fri Feb 15 18:34:05 2013 +0900

    Added concurrent view test: test-client-view-operations.c
    
    This tests concurrent access to EBookClientView, 20 threads each
    race to receive notifications from the same addressbook and the
    test asserts that all 5 contacts in the test addressbook are found.
    
    This is particularly interesting with DEBUG_DIRECT set (Direct Read Access),
    as it ensures that concurrent direct reads of the addressbook do not
    cause any lockups.

 tests/libebook/client/Makefile.am                  |    3 +
 .../libebook/client/test-client-view-operations.c  |  259 ++++++++++++++++++++
 2 files changed, 262 insertions(+), 0 deletions(-)
---
diff --git a/tests/libebook/client/Makefile.am b/tests/libebook/client/Makefile.am
index f23c5aa..8bff7c5 100644
--- a/tests/libebook/client/Makefile.am
+++ b/tests/libebook/client/Makefile.am
@@ -37,6 +37,7 @@ TESTS =                                                               \
        test-client-write-write                                 \
        test-client-uid-only-view                               \
        test-client-revision-view                               \
+       test-client-view-operations                             \
        test-client-suppress-notifications                      \
        test-client-modify-contact                              \
        test-client-remove-contact                              \
@@ -99,6 +100,8 @@ test_client_uid_only_view_LDADD=$(TEST_LIBS)
 test_client_uid_only_view_CPPFLAGS=$(TEST_CPPFLAGS)
 test_client_revision_view_LDADD=$(TEST_LIBS)
 test_client_revision_view_CPPFLAGS=$(TEST_CPPFLAGS)
+test_client_view_operations_LDADD=$(TEST_LIBS)
+test_client_view_operations_CPPFLAGS=$(TEST_CPPFLAGS)
 test_client_suppress_notifications_LDADD=$(TEST_LIBS)
 test_client_suppress_notifications_CPPFLAGS=$(TEST_CPPFLAGS)
 test_client_modify_contact_LDADD=$(TEST_LIBS)
diff --git a/tests/libebook/client/test-client-view-operations.c 
b/tests/libebook/client/test-client-view-operations.c
new file mode 100644
index 0000000..9d470fd
--- /dev/null
+++ b/tests/libebook/client/test-client-view-operations.c
@@ -0,0 +1,259 @@
+/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
+
+#include <stdlib.h>
+#include <locale.h>
+#include <libebook/libebook.h>
+#include <libedata-book/libedata-book.h>
+
+#include "client-test-utils.h"
+#include "e-test-server-utils.h"
+
+static ETestServerClosure book_closure = { E_TEST_SERVER_ADDRESS_BOOK, NULL, 0 };
+
+#define N_THREADS  5
+#define N_CONTACTS 5
+
+typedef struct {
+       GThread         *thread;
+       const gchar     *book_uid;
+       EBookClient     *client;
+       EBookClientView *view;
+
+       GMainLoop       *loop;
+
+       GMutex           complete_mutex;
+       GCond            complete_cond;
+       gboolean         complete;
+       gint             n_contacts;
+} ThreadData;
+
+static void
+objects_added (EBookClientView *view,
+               const GSList *contacts,
+              ThreadData *data)
+{
+       const GSList *l;
+
+       for (l = contacts; l; l = l->next) {
+               /* print_email (l->data); */
+
+               data->n_contacts++;
+       }
+}
+
+static void
+objects_modified (EBookClientView *view,
+                 const GSList *contacts,
+                 ThreadData *data)
+{
+       const GSList *l;
+
+       for (l = contacts; l; l = l->next) {
+               /* print_email (l->data); */
+       }
+}
+
+static void
+objects_removed (EBookClientView *view,
+                 const GSList *ids,
+                ThreadData *data)
+{
+       const GSList *l;
+
+       for (l = ids; l; l = l->next) {
+               /* printf ("   Removed contact: %s\n", (gchar *) l->data); */
+
+               data->n_contacts--;
+       }
+}
+
+static void
+complete (EBookClientView *view,
+          const GError *error,
+         ThreadData *data)
+{
+       g_mutex_lock (&data->complete_mutex);
+       data->complete = TRUE;
+       g_cond_signal (&data->complete_cond);
+       g_mutex_unlock (&data->complete_mutex);
+}
+
+static gboolean
+start_view (ThreadData *data)
+{
+       EBookQuery   *query;
+       gchar        *sexp;
+       GError *error = NULL;
+
+       query = e_book_query_any_field_contains ("");
+       sexp = e_book_query_to_string (query);
+
+       if (!e_book_client_get_view_sync (data->client, sexp,
+                                         &(data->view), NULL, &error))
+               g_error ("Error getting view: %s", error->message);
+
+       g_signal_connect (data->view, "objects-added", G_CALLBACK (objects_added), data);
+       g_signal_connect (data->view, "objects-modified", G_CALLBACK (objects_modified), data);
+       g_signal_connect (data->view, "objects-removed", G_CALLBACK (objects_removed), data);
+       g_signal_connect (data->view, "complete", G_CALLBACK (complete), data);
+
+       e_book_client_view_set_fields_of_interest (data->view, NULL, &error);
+       if (error)
+               g_error ("set fields of interest: %s", error->message);
+
+       e_book_client_view_start (data->view, &error);
+       if (error)
+               g_error ("start view: %s", error->message);
+
+       e_book_query_unref (query);
+       g_free (sexp);
+
+       return FALSE;
+}
+
+static void
+start_thread_test (ThreadData *data)
+{
+       GMainContext *context;
+       GSource      *source;
+
+       context = g_main_loop_get_context (data->loop);
+       source  = g_idle_source_new ();
+
+       g_source_set_callback (source, (GSourceFunc)start_view, data, NULL);
+       g_source_attach (source, context);
+}
+
+static void
+finish_thread_test (ThreadData *data)
+{
+       g_assert_cmpint (data->n_contacts, ==, N_CONTACTS);
+
+       g_main_loop_quit (data->loop);
+       g_thread_join (data->thread);
+       g_mutex_clear (&data->complete_mutex);
+       g_cond_clear (&data->complete_cond);
+       g_slice_free (ThreadData, data);
+}
+
+static gpointer
+test_view_thread (ThreadData *data)
+{
+       GMainContext    *context;
+       ESourceRegistry *registry;
+       ESource         *source;
+       GError          *error = NULL;
+
+       context    = g_main_context_new ();
+       data->loop = g_main_loop_new (context, FALSE);
+       g_main_context_push_thread_default (context);
+
+       /* Open the test book client in this thread */
+       registry = e_source_registry_new_sync (NULL, &error);
+       if (!registry)
+               g_error ("Unable to create the registry: %s", error->message);
+
+       source = e_source_registry_ref_source (registry, data->book_uid);
+       if (!source)
+               g_error ("Unable to fetch source uid '%s' from the registry", data->book_uid);
+
+       if (g_getenv ("DEBUG_DIRECT") != NULL)
+               data->client = e_book_client_connect_direct_sync (registry, source, NULL, &error);
+       else
+               data->client = e_book_client_connect_sync (source, NULL, &error);
+
+       if (!data->client)
+               g_error ("Unable to create EBookClient for uid '%s': %s", data->book_uid, error->message);
+
+       start_thread_test (data);
+
+       g_main_loop_run (data->loop);
+
+       g_object_unref (source);
+       g_object_unref (registry);
+
+       g_object_unref (data->client);
+       g_main_context_pop_thread_default (context);
+       g_main_loop_unref (data->loop);
+       g_main_context_unref (context);
+
+       return NULL;
+}
+
+static ThreadData *
+create_test_thread (const gchar   *book_uid)
+{
+       ThreadData  *data = g_slice_new0 (ThreadData);
+
+       data->book_uid    = book_uid;
+
+       g_mutex_init (&data->complete_mutex);
+       g_cond_init (&data->complete_cond);
+
+       data->thread = g_thread_new ("test-thread", (GThreadFunc)test_view_thread, data);
+
+       return data;
+}
+
+static void
+test_concurrent_views (ETestServerFixture *fixture,
+                      gconstpointer       user_data)
+{
+       EBookClient *main_client;
+       ESource *source;
+       const gchar *book_uid = NULL;
+       ThreadData **tests;
+       gint i;
+
+       main_client = E_TEST_SERVER_UTILS_SERVICE (fixture, EBookClient);
+       source = e_client_get_source (E_CLIENT (main_client));
+       book_uid = e_source_get_uid (source);
+
+       /* Create out test contacts */
+       if (!add_contact_from_test_case_verify (main_client, "custom-1", NULL) ||
+           !add_contact_from_test_case_verify (main_client, "custom-2", NULL) ||
+           !add_contact_from_test_case_verify (main_client, "custom-3", NULL) ||
+           !add_contact_from_test_case_verify (main_client, "custom-4", NULL) ||
+           !add_contact_from_test_case_verify (main_client, "custom-5", NULL)) {
+               g_object_unref (main_client);
+               g_error ("Failed to create contacts for testing");
+       }
+
+       /* Create all concurrent threads accessing the same addressbook */
+       tests = g_new0 (ThreadData *, N_THREADS);
+       for (i = 0; i < N_THREADS; i++)
+               tests[i] = create_test_thread (book_uid);
+
+
+       /* Wait for all threads to receive the complete signal */
+       for (i = 0; i < N_THREADS; i++) {
+               g_mutex_lock (&(tests[i]->complete_mutex));
+               while (!tests[i]->complete)
+                       g_cond_wait (&(tests[i]->complete_cond),
+                                    &(tests[i]->complete_mutex));
+               g_mutex_unlock (&(tests[i]->complete_mutex));
+       }
+
+       /* Finish all tests */
+       for (i = 0; i < N_THREADS; i++)
+               finish_thread_test (tests[i]);
+
+       /* Cleanup */
+       g_free (tests);
+}
+
+gint
+main (gint argc,
+      gchar **argv)
+{
+#if !GLIB_CHECK_VERSION (2, 35, 1)
+       g_type_init ();
+#endif
+       g_test_init (&argc, &argv, NULL);
+       setlocale (LC_ALL, "en_US.UTF-8");
+
+       g_test_add ("/EBookClient/ConcurrentViews", ETestServerFixture, &book_closure,
+                   e_test_server_utils_setup, test_concurrent_views, e_test_server_utils_teardown);
+
+       return e_test_server_utils_run ();
+}


[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]