1
0
mirror of https://github.com/profanity-im/profanity.git synced 2024-11-03 19:37:16 -05:00

Use hashtables for plugin disco features

This commit is contained in:
James Booth 2016-08-17 23:21:48 +01:00
parent 8b6549b36c
commit 9945246a64
6 changed files with 214 additions and 28 deletions

View File

@ -126,6 +126,7 @@ unittest_sources = \
tests/unittests/test_cmd_roster.c tests/unittests/test_cmd_roster.h \ tests/unittests/test_cmd_roster.c tests/unittests/test_cmd_roster.h \
tests/unittests/test_cmd_disconnect.c tests/unittests/test_cmd_disconnect.h \ tests/unittests/test_cmd_disconnect.c tests/unittests/test_cmd_disconnect.h \
tests/unittests/test_callbacks.c tests/unittests/test_callbacks.h \ tests/unittests/test_callbacks.c tests/unittests/test_callbacks.h \
tests/unittests/test_plugins_disco.c tests/unittests/test_plugins_disco.h \
tests/unittests/unittests.c tests/unittests/unittests.c
functionaltest_sources = \ functionaltest_sources = \

View File

@ -37,76 +37,112 @@
#include <glib.h> #include <glib.h>
// features to reference count map
static GHashTable *features = NULL;
// plugin to feature map
static GHashTable *plugin_to_features = NULL; static GHashTable *plugin_to_features = NULL;
static void static void
_free_features(GList *features) _free_features(GHashTable *features)
{ {
g_list_free_full(features, free); g_hash_table_destroy(features);
} }
void void
disco_add_feature(const char *plugin_name, char* feature) disco_add_feature(const char *plugin_name, char *feature)
{ {
if (feature == NULL || plugin_name == NULL) { if (feature == NULL || plugin_name == NULL) {
return; return;
} }
if (!features) {
features = g_hash_table_new_full(g_str_hash, g_str_equal, free, NULL);
}
if (!plugin_to_features) { if (!plugin_to_features) {
plugin_to_features = g_hash_table_new_full(g_str_hash, g_str_equal, free, (GDestroyNotify)_free_features); plugin_to_features = g_hash_table_new_full(g_str_hash, g_str_equal, free, (GDestroyNotify)_free_features);
} }
GList *features = g_hash_table_lookup(plugin_to_features, plugin_name); GHashTable *plugin_features = g_hash_table_lookup(plugin_to_features, plugin_name);
if (!features) { gboolean added = FALSE;
features = g_list_append(features, strdup(feature)); if (plugin_features == NULL) {
g_hash_table_insert(plugin_to_features, strdup(plugin_name), features); plugin_features = g_hash_table_new_full(g_str_hash, g_str_equal, free, NULL);
g_hash_table_add(plugin_features, strdup(feature));
g_hash_table_insert(plugin_to_features, strdup(plugin_name), plugin_features);
added = TRUE;
} else if (!g_hash_table_contains(plugin_features, feature)) {
g_hash_table_add(plugin_features, strdup(feature));
added = TRUE;
}
if (added == FALSE) {
return;
}
if (!g_hash_table_contains(features, feature)) {
g_hash_table_insert(features, strdup(feature), GINT_TO_POINTER(1));
} else { } else {
features = g_list_append(features, strdup(feature)); void *refcountp = g_hash_table_lookup(features, feature);
int refcount = GPOINTER_TO_INT(refcountp);
refcount++;
g_hash_table_replace(features, strdup(feature), GINT_TO_POINTER(refcount));
} }
} }
void void
disco_remove_features(const char *plugin_name) disco_remove_features(const char *plugin_name)
{ {
if (!features) {
return;
}
if (!plugin_to_features) { if (!plugin_to_features) {
return; return;
} }
if (!g_hash_table_contains(plugin_to_features, plugin_name)) { GHashTable *plugin_features_set = g_hash_table_lookup(plugin_to_features, plugin_name);
if (!plugin_features_set) {
return; return;
} }
g_hash_table_remove(plugin_to_features, plugin_name); GList *plugin_feature_list = g_hash_table_get_keys(plugin_features_set);
GList *curr = plugin_feature_list;
while (curr) {
char *feature = curr->data;
if (g_hash_table_contains(features, feature)) {
void *refcountp = g_hash_table_lookup(features, feature);
int refcount = GPOINTER_TO_INT(refcountp);
if (refcount == 1) {
g_hash_table_remove(features, feature);
} else {
refcount--;
g_hash_table_replace(features, strdup(feature), GINT_TO_POINTER(refcount));
}
}
curr = g_list_next(curr);
}
g_list_free(plugin_feature_list);
} }
GList* GList*
disco_get_features(void) disco_get_features(void)
{ {
GList *result = NULL; if (features == NULL) {
if (!plugin_to_features) { return NULL;
return result;
} }
GList *lists = g_hash_table_get_values(plugin_to_features); return g_hash_table_get_keys(features);
GList *curr_list = lists;
while (curr_list) {
GList *features = curr_list->data;
GList *curr = features;
while (curr) {
result = g_list_append(result, curr->data);
curr = g_list_next(curr);
}
curr_list = g_list_next(curr_list);
}
g_list_free(lists);
return result;
} }
void void
disco_close(void) disco_close(void)
{ {
if (features) {
g_hash_table_destroy(features);
features = NULL;
}
if (plugin_to_features) { if (plugin_to_features) {
g_hash_table_destroy(plugin_to_features); g_hash_table_destroy(plugin_to_features);
plugin_to_features = NULL; plugin_to_features = NULL;

View File

@ -35,6 +35,8 @@
#ifndef PLUGINS_DISCO_H #ifndef PLUGINS_DISCO_H
#define PLUGINS_DISCO_H #define PLUGINS_DISCO_H
#include <glib.h>
void disco_add_feature(const char* plugin_name, char *feature); void disco_add_feature(const char* plugin_name, char *feature);
void disco_remove_features(const char *plugin_name); void disco_remove_features(const char *plugin_name);
GList* disco_get_features(void); GList* disco_get_features(void);

View File

@ -0,0 +1,131 @@
#include <stdarg.h>
#include <stddef.h>
#include <setjmp.h>
#include <cmocka.h>
#include <stdlib.h>
#include "plugins/disco.h"
void
returns_empty_list_when_none(void **state)
{
disco_close();
GList *features = disco_get_features();
assert_int_equal(g_list_length(features), 0);
g_list_free(features);
disco_close();
}
void
returns_added_feature(void **state)
{
disco_close();
disco_add_feature("my_plugin", "some:feature:example");
GList *features = disco_get_features();
assert_int_equal(g_list_length(features), 1);
char *feature = features->data;
assert_string_equal(feature, "some:feature:example");
g_list_free(features);
disco_close();
}
void
resets_features_on_close(void **state)
{
disco_close();
disco_add_feature("my_plugin", "some:feature:example");
GList *features = disco_get_features();
assert_int_equal(g_list_length(features), 1);
g_list_free(features);
disco_close();
features = disco_get_features();
assert_int_equal(g_list_length(features), 0);
g_list_free(features);
disco_close();
}
void
returns_all_added_features(void **state)
{
disco_close();
disco_add_feature("first_plugin", "first:feature");
disco_add_feature("first_plugin", "second:feature");
disco_add_feature("second_plugin", "third:feature");
disco_add_feature("third_plugin", "fourth:feature");
disco_add_feature("third_plugin", "fifth:feature");
GList *features = disco_get_features();
assert_int_equal(g_list_length(features), 5);
assert_true(g_list_find_custom(features, "first:feature", g_strcmp0));
assert_true(g_list_find_custom(features, "second:feature", g_strcmp0));
assert_true(g_list_find_custom(features, "third:feature", g_strcmp0));
assert_true(g_list_find_custom(features, "fourth:feature", g_strcmp0));
assert_true(g_list_find_custom(features, "fifth:feature", g_strcmp0));
g_list_free(features);
disco_close();
}
void
does_not_add_duplicate_feature(void **state)
{
disco_close();
disco_add_feature("my_plugin", "my:feature");
disco_add_feature("some_other_plugin", "my:feature");
GList *features = disco_get_features();
assert_int_equal(g_list_length(features), 1);
g_list_free(features);
disco_close();
}
void
removes_plugin_features(void **state)
{
disco_close();
disco_add_feature("plugin1", "plugin1:feature1");
disco_add_feature("plugin1", "plugin1:feature2");
disco_add_feature("plugin2", "plugin2:feature1");
GList *features = disco_get_features();
assert_int_equal(g_list_length(features), 3);
g_list_free(features);
disco_remove_features("plugin1");
features = disco_get_features();
assert_int_equal(g_list_length(features), 1);
g_list_free(features);
disco_close();
}
void
does_not_remove_feature_when_more_than_one_reference(void **state)
{
disco_close();
disco_add_feature("plugin1", "feature1");
disco_add_feature("plugin1", "feature2");
disco_add_feature("plugin2", "feature1");
GList *features = disco_get_features();
assert_int_equal(g_list_length(features), 2);
g_list_free(features);
disco_remove_features("plugin1");
features = disco_get_features();
assert_int_equal(g_list_length(features), 1);
g_list_free(features);
disco_close();
}

View File

@ -0,0 +1,7 @@
void returns_empty_list_when_none(void **state);
void returns_added_feature(void **state);
void resets_features_on_close(void **state);
void returns_all_added_features(void **state);
void does_not_add_duplicate_feature(void **state);
void removes_plugin_features(void **state);
void does_not_remove_feature_when_more_than_one_reference(void **state);

View File

@ -34,6 +34,7 @@
#include "test_cmd_disconnect.h" #include "test_cmd_disconnect.h"
#include "test_form.h" #include "test_form.h"
#include "test_callbacks.h" #include "test_callbacks.h"
#include "test_plugins_disco.h"
int main(int argc, char* argv[]) { int main(int argc, char* argv[]) {
const UnitTest all_tests[] = { const UnitTest all_tests[] = {
@ -587,6 +588,14 @@ int main(int argc, char* argv[]) {
unit_test(returns_no_commands), unit_test(returns_no_commands),
unit_test(returns_commands), unit_test(returns_commands),
unit_test(returns_empty_list_when_none),
unit_test(returns_added_feature),
unit_test(resets_features_on_close),
unit_test(returns_all_added_features),
unit_test(does_not_add_duplicate_feature),
unit_test(removes_plugin_features),
unit_test(does_not_remove_feature_when_more_than_one_reference),
}; };
return run_tests(all_tests); return run_tests(all_tests);