glib/tests/keyfile-test.c
Matthias Clasen 58f83683e9 Don't leak memory here.
2005-06-23  Matthias Clasen  <mclasen@redhat.com>

	* tests/keyfile-test.c (test_group_remove): Don't leak
	memory here.

	* glib/gkeyfile.c (g_key_file_load_from_data_dirs): Don't
	leak output_path.  (#308546, Kjartan Maraas)
2005-06-23 16:28:55 +00:00

797 lines
19 KiB
C

#include <glib.h>
#include <locale.h>
#include <string.h>
#include <stdlib.h>
static GKeyFile *
load_data (const gchar *data,
GKeyFileFlags flags)
{
GKeyFile *keyfile;
GError *error = NULL;
keyfile = g_key_file_new ();
g_key_file_load_from_data (keyfile, data, -1, flags, &error);
if (error)
{
g_print ("Could not load data: %s\n", error->message);
exit (1);
}
return keyfile;
}
static void
check_error (GError **error,
GQuark domain,
gint code)
{
if (*error == NULL)
{
g_print ("Missing an error\n");
exit (1);
}
if ((*error)->domain != domain)
{
g_print ("Wrong error domain: got %s, expected %s\n",
g_quark_to_string ((*error)->domain),
g_quark_to_string (domain));
exit (1);
}
if ((*error)->code != code)
{
g_print ("Wrong error code: got %d, expected %d\n",
(*error)->code, code);
exit (1);
}
g_error_free (*error);
*error = NULL;
}
static void
check_no_error (GError **error)
{
if (*error != NULL)
{
g_print ("Unexpected error: (%s, %d) %s\n",
g_quark_to_string ((*error)->domain),
(*error)->code, (*error)->message);
exit (1);
}
}
static void
check_string_value (GKeyFile *keyfile,
const gchar *group,
const gchar *key,
const gchar *expected)
{
GError *error = NULL;
gchar *value;
value = g_key_file_get_string (keyfile, group, key, &error);
check_no_error (&error);
g_assert (value != NULL);
if (strcmp (value, expected) != 0)
{
g_print ("Group %s key %s: "
"expected string value '%s', actual value '%s'\n",
group, key, expected, value);
exit (1);
}
g_free (value);
}
static void
check_locale_string_value (GKeyFile *keyfile,
const gchar *group,
const gchar *key,
const gchar *locale,
const gchar *expected)
{
GError *error = NULL;
gchar *value;
value = g_key_file_get_locale_string (keyfile, group, key, locale, &error);
check_no_error (&error);
g_assert (value != NULL);
if (strcmp (value, expected) != 0)
{
g_print ("Group %s key %s locale %s: "
"expected string value '%s', actual value '%s'\n",
group, key, locale, expected, value);
exit (1);
}
g_free (value);
}
static void
check_string_list_value (GKeyFile *keyfile,
const gchar *group,
const gchar *key,
...)
{
gint i;
gchar *v, **value;
va_list args;
gsize len;
GError *error = NULL;
value = g_key_file_get_string_list (keyfile, group, key, &len, &error);
check_no_error (&error);
g_assert (value != NULL);
va_start (args, key);
i = 0;
v = va_arg (args, gchar*);
while (v)
{
if (value[i] == NULL)
{
g_print ("Group %s key %s: list too short (%d)\n",
group, key, i);
exit (1);
}
if (strcmp (v, value[i]) != 0)
{
g_print ("Group %s key %s: mismatch at %d, expected %s, got %s\n",
group, key, i, v, value[i]);
exit (1);
}
i++;
v = va_arg (args, gchar*);
}
va_end (args);
g_strfreev (value);
}
static void
check_integer_list_value (GKeyFile *keyfile,
const gchar *group,
const gchar *key,
...)
{
gint i;
gint v, *value;
va_list args;
gsize len;
GError *error = NULL;
value = g_key_file_get_integer_list (keyfile, group, key, &len, &error);
check_no_error (&error);
g_assert (value != NULL);
va_start (args, key);
i = 0;
v = va_arg (args, gint);
while (v != -100)
{
if (i == len)
{
g_print ("Group %s key %s: list too short (%d)\n",
group, key, i);
exit (1);
}
if (value[i] != v)
{
g_print ("Group %s key %s: mismatch at %d, expected %d, got %d\n",
group, key, i, v, value[i]);
exit (1);
}
i++;
v = va_arg (args, gint);
}
va_end (args);
g_free (value);
}
static void
check_boolean_list_value (GKeyFile *keyfile,
const gchar *group,
const gchar *key,
...)
{
gint i;
gboolean v, *value;
va_list args;
gsize len;
GError *error = NULL;
value = g_key_file_get_boolean_list (keyfile, group, key, &len, &error);
check_no_error (&error);
g_assert (value != NULL);
va_start (args, key);
i = 0;
v = va_arg (args, gboolean);
while (v != -100)
{
if (i == len)
{
g_print ("Group %s key %s: list too short (%d)\n",
group, key, i);
exit (1);
}
if (value[i] != v)
{
g_print ("Group %s key %s: mismatch at %d, expected %d, got %d\n",
group, key, i, v, value[i]);
exit (1);
}
i++;
v = va_arg (args, gboolean);
}
va_end (args);
g_free (value);
}
static void
check_boolean_value (GKeyFile *keyfile,
const gchar *group,
const gchar *key,
gboolean expected)
{
GError *error = NULL;
gboolean value;
value = g_key_file_get_boolean (keyfile, group, key, &error);
check_no_error (&error);
if (value != expected)
{
g_print ("Group %s key %s: "
"expected boolean value '%s', actual value '%s'\n",
group, key,
expected ? "true" : "false",
value ? "true" : "false");
exit (1);
}
}
static void
check_integer_value (GKeyFile *keyfile,
const gchar *group,
const gchar *key,
gint expected)
{
GError *error = NULL;
gint value;
value = g_key_file_get_integer (keyfile, group, key, &error);
check_no_error (&error);
if (value != expected)
{
g_print ("Group %s key %s: "
"expected integer value %d, actual value %d\n",
group, key, expected, value);
exit (1);
}
}
static void
check_name (const gchar *what,
const gchar *value,
const gchar *expected,
gint position)
{
if (strcmp (expected, value) != 0)
{
g_print ("Wrong %s returned: got %s at %d, expected %s\n",
what, value, position, expected);
exit (1);
}
}
static void
check_length (const gchar *what,
gint n_items,
gint length,
gint expected)
{
if (n_items != length || length != expected)
{
g_print ("Wrong number of %s returned: got %d items, length %d, expected %d\n",
what, n_items, length, expected);
exit (1);
}
}
/* check that both \n and \r\n are accepted as line ends,
* and that stray \r are passed through
*/
static void
test_line_ends (void)
{
GKeyFile *keyfile;
const gchar *data =
"[group1]\n"
"key1=value1\n"
"key2=value2\r\n"
"[group2]\r\n"
"key3=value3\r\r\n"
"key4=value4\n";
keyfile = load_data (data, 0);
check_string_value (keyfile, "group1", "key1", "value1");
check_string_value (keyfile, "group1", "key2", "value2");
check_string_value (keyfile, "group2", "key3", "value3\r");
check_string_value (keyfile, "group2", "key4", "value4");
g_key_file_free (keyfile);
}
/* check handling of whitespace
*/
static void
test_whitespace (void)
{
GKeyFile *keyfile;
const gchar *data =
"[group1]\n"
"key1 = value1\n"
"key2\t=\tvalue2\n"
" [ group2 ] \n"
"key3 = value3 \n"
"key4 = value \t4\n"
" key5 = value5\n";
keyfile = load_data (data, 0);
check_string_value (keyfile, "group1", "key1", "value1");
check_string_value (keyfile, "group1", "key2", "value2");
check_string_value (keyfile, " group2 ", "key3", "value3 ");
check_string_value (keyfile, " group2 ", "key4", "value \t4");
check_string_value (keyfile, " group2 ", "key5", "value5");
g_key_file_free (keyfile);
}
/* check key and group listing */
static void
test_listing (void)
{
GKeyFile *keyfile;
gchar **names;
gsize len;
gchar *start;
GError *error = NULL;
const gchar *data =
"[group1]\n"
"key1=value1\n"
"key2=value2\n"
"[group2]\n"
"key3=value3\n"
"key4=value4\n";
keyfile = load_data (data, 0);
names = g_key_file_get_groups (keyfile, &len);
if (names == NULL)
{
g_print ("Error listing groups\n");
exit (1);
}
check_length ("groups", g_strv_length (names), len, 2);
check_name ("group name", names[0], "group1", 0);
check_name ("group name", names[1], "group2", 1);
g_strfreev (names);
names = g_key_file_get_keys (keyfile, "group1", &len, &error);
check_no_error (&error);
check_length ("keys", g_strv_length (names), len, 2);
check_name ("key", names[0], "key1", 0);
check_name ("key", names[1], "key2", 1);
g_strfreev (names);
names = g_key_file_get_keys (keyfile, "no-such-group", &len, &error);
check_error (&error, G_KEY_FILE_ERROR, G_KEY_FILE_ERROR_GROUP_NOT_FOUND);
g_strfreev (names);
if (!g_key_file_has_group (keyfile, "group1") ||
!g_key_file_has_group (keyfile, "group2") ||
g_key_file_has_group (keyfile, "group10") ||
g_key_file_has_group (keyfile, "group2 "))
{
g_print ("Group finding trouble\n");
exit (1);
}
start = g_key_file_get_start_group (keyfile);
if (!start || strcmp (start, "group1") != 0)
{
g_print ("Start group finding trouble\n");
exit (1);
}
g_free (start);
if (!g_key_file_has_key (keyfile, "group1", "key1", &error) ||
!g_key_file_has_key (keyfile, "group2", "key3", &error) ||
g_key_file_has_key (keyfile, "group2", "no-such-key", &error))
{
g_print ("Key finding trouble\n");
exit (1);
}
check_no_error (&error);
g_key_file_has_key (keyfile, "no-such-group", "key", &error);
check_error (&error, G_KEY_FILE_ERROR, G_KEY_FILE_ERROR_GROUP_NOT_FOUND);
g_key_file_free (keyfile);
}
/* check parsing of string values */
static void
test_string (void)
{
GKeyFile *keyfile;
GError *error = NULL;
gchar *value;
const gchar *data =
"[valid]\n"
"key1=\\s\\n\\t\\r\\\\\n"
"key2=\"quoted\"\n"
"key3='quoted'\n"
"key4=\xe2\x89\xa0\xe2\x89\xa0\n"
"[invalid]\n"
"key1=\\a\\b\\0800xff\n"
"key2=blabla\\\n";
keyfile = load_data (data, 0);
check_string_value (keyfile, "valid", "key1", " \n\t\r\\");
check_string_value (keyfile, "valid", "key2", "\"quoted\"");
check_string_value (keyfile, "valid", "key3", "'quoted'");
check_string_value (keyfile, "valid", "key4", "\xe2\x89\xa0\xe2\x89\xa0");
value = g_key_file_get_string (keyfile, "invalid", "key1", &error);
check_error (&error, G_KEY_FILE_ERROR, G_KEY_FILE_ERROR_INVALID_VALUE);
g_free (value);
value = g_key_file_get_string (keyfile, "invalid", "key2", &error);
check_error (&error, G_KEY_FILE_ERROR, G_KEY_FILE_ERROR_INVALID_VALUE);
g_free (value);
g_key_file_free (keyfile);
}
/* check parsing of boolean values */
static void
test_boolean (void)
{
GKeyFile *keyfile;
GError *error = NULL;
const gchar *data =
"[valid]\n"
"key1=true\n"
"key2=false\n"
"key3=1\n"
"key4=0\n"
"[invalid]\n"
"key1=t\n"
"key2=f\n"
"key3=yes\n"
"key4=no\n";
keyfile = load_data (data, 0);
check_boolean_value (keyfile, "valid", "key1", TRUE);
check_boolean_value (keyfile, "valid", "key2", FALSE);
check_boolean_value (keyfile, "valid", "key3", TRUE);
check_boolean_value (keyfile, "valid", "key4", FALSE);
g_key_file_get_boolean (keyfile, "invalid", "key1", &error);
check_error (&error, G_KEY_FILE_ERROR, G_KEY_FILE_ERROR_INVALID_VALUE);
g_key_file_get_boolean (keyfile, "invalid", "key2", &error);
check_error (&error, G_KEY_FILE_ERROR, G_KEY_FILE_ERROR_INVALID_VALUE);
g_key_file_get_boolean (keyfile, "invalid", "key3", &error);
check_error (&error, G_KEY_FILE_ERROR, G_KEY_FILE_ERROR_INVALID_VALUE);
g_key_file_get_boolean (keyfile, "invalid", "key4", &error);
check_error (&error, G_KEY_FILE_ERROR, G_KEY_FILE_ERROR_INVALID_VALUE);
g_key_file_free (keyfile);
}
/* check parsing of integer values */
static void
test_integer (void)
{
GKeyFile *keyfile;
GError *error = NULL;
const gchar *data =
"[valid]\n"
"key1=0\n"
"key2=1\n"
"key3=-1\n"
"key4=2324431\n"
"key5=-2324431\n"
"key6=000111\n"
"[invalid]\n"
"key1=0xffff\n"
"key2=0.5\n"
"key3=1e37\n"
"key4=ten\n";
keyfile = load_data (data, 0);
check_integer_value (keyfile, "valid", "key1", 0);
check_integer_value (keyfile, "valid", "key2", 1);
check_integer_value (keyfile, "valid", "key3", -1);
check_integer_value (keyfile, "valid", "key4", 2324431);
check_integer_value (keyfile, "valid", "key5", -2324431);
check_integer_value (keyfile, "valid", "key6", 111);
g_key_file_get_integer (keyfile, "invalid", "key1", &error);
check_error (&error, G_KEY_FILE_ERROR, G_KEY_FILE_ERROR_INVALID_VALUE);
g_key_file_get_integer (keyfile, "invalid", "key2", &error);
check_error (&error, G_KEY_FILE_ERROR, G_KEY_FILE_ERROR_INVALID_VALUE);
g_key_file_get_integer (keyfile, "invalid", "key3", &error);
check_error (&error, G_KEY_FILE_ERROR, G_KEY_FILE_ERROR_INVALID_VALUE);
g_key_file_get_integer (keyfile, "invalid", "key4", &error);
check_error (&error, G_KEY_FILE_ERROR, G_KEY_FILE_ERROR_INVALID_VALUE);
g_key_file_free (keyfile);
}
/* check handling of translated strings */
static void
test_locale_string (void)
{
GKeyFile *keyfile;
GError *error = NULL;
gchar *value;
const gchar *data =
"[valid]\n"
"key1=v1\n"
"key1[de]=v1-de\n"
"key1[de_DE]=v1-de_DE\n"
"key1[de_DE.UTF8]=v1-de_DE.UTF8\n"
"key1[fr]=v1-fr\n"
"key1[en] =v1-en\n"
"[invalid]\n"
"key1[de=v1\n"
"key1[fr]]=v2\n"
"key1 [en]=v3\n";
keyfile = load_data (data, G_KEY_FILE_KEEP_TRANSLATIONS);
check_locale_string_value (keyfile, "valid", "key1", "it", "v1");
check_locale_string_value (keyfile, "valid", "key1", "de", "v1-de");
check_locale_string_value (keyfile, "valid", "key1", "de_DE", "v1-de_DE");
check_locale_string_value (keyfile, "valid", "key1", "de_DE.UTF8", "v1-de_DE.UTF8");
check_locale_string_value (keyfile, "valid", "key1", "fr", "v1-fr");
check_locale_string_value (keyfile, "valid", "key1", "fr_FR", "v1-fr");
check_locale_string_value (keyfile, "valid", "key1", "en", "v1-en");
value = g_key_file_get_locale_string (keyfile, "invalid", "key1", "de", &error);
check_error (&error, G_KEY_FILE_ERROR, G_KEY_FILE_ERROR_KEY_NOT_FOUND);
g_free (value);
value = g_key_file_get_locale_string (keyfile, "invalid", "key1", "fr", &error);
check_error (&error, G_KEY_FILE_ERROR, G_KEY_FILE_ERROR_KEY_NOT_FOUND);
g_free (value);
value = g_key_file_get_locale_string (keyfile, "invalid", "key1", "en", &error);
check_error (&error, G_KEY_FILE_ERROR, G_KEY_FILE_ERROR_KEY_NOT_FOUND);
g_free (value);
g_key_file_free (keyfile);
/* now test that translations are thrown away */
g_setenv ("LANGUAGE", "de", TRUE);
setlocale (LC_ALL, "");
keyfile = load_data (data, 0);
check_locale_string_value (keyfile, "valid", "key1", "it", "v1");
check_locale_string_value (keyfile, "valid", "key1", "de", "v1-de");
check_locale_string_value (keyfile, "valid", "key1", "de_DE", "v1-de");
check_locale_string_value (keyfile, "valid", "key1", "de_DE.UTF8", "v1-de");
check_locale_string_value (keyfile, "valid", "key1", "fr", "v1");
check_locale_string_value (keyfile, "valid", "key1", "fr_FR", "v1");
check_locale_string_value (keyfile, "valid", "key1", "en", "v1");
g_key_file_free (keyfile);
}
static void
test_lists (void)
{
GKeyFile *keyfile;
const gchar *data =
"[valid]\n"
"key1=v1;v2\n"
"key2=v1;v2;\n"
"key3=v1,v2\n"
"key4=v1\\;v2\n"
"key5=true;false\n"
"key6=1;0;-1\n"
"key7= 1 ; 0 ; -1 \n"
"key8=v1\\,v2\n";
keyfile = load_data (data, 0);
check_string_list_value (keyfile, "valid", "key1", "v1", "v2", NULL);
check_string_list_value (keyfile, "valid", "key2", "v1", "v2", NULL);
check_string_list_value (keyfile, "valid", "key3", "v1,v2", NULL);
check_string_list_value (keyfile, "valid", "key4", "v1;v2", NULL);
check_boolean_list_value (keyfile, "valid", "key5", TRUE, FALSE, -100);
check_integer_list_value (keyfile, "valid", "key6", 1, 0, -1, -100);
/* maybe these should be valid */
/* check_integer_list_value (keyfile, "valid", "key7", 1, 0, -1, -100);*/
/* check_string_list_value (keyfile, "valid", "key8", "v1\\,v2", NULL);*/
g_key_file_free (keyfile);
/* Now check an alternate separator */
keyfile = load_data (data, 0);
g_key_file_set_list_separator (keyfile, ',');
check_string_list_value (keyfile, "valid", "key1", "v1;v2", NULL);
check_string_list_value (keyfile, "valid", "key2", "v1;v2;", NULL);
check_string_list_value (keyfile, "valid", "key3", "v1", "v2", NULL);
g_key_file_free (keyfile);
}
/* http://bugzilla.gnome.org/show_bug.cgi?id=165887 */
static void
test_group_remove (void)
{
GKeyFile *keyfile;
gchar **names;
gsize len;
GError *error = NULL;
const gchar *data =
"[group1]\n"
"[group2]\n"
"key1=bla\n"
"key2=bla\n"
"[group3]\n"
"key1=bla\n"
"key2=bla\n";
keyfile = load_data (data, 0);
names = g_key_file_get_groups (keyfile, &len);
if (names == NULL)
{
g_print ("Error listing groups\n");
exit (1);
}
check_length ("groups", g_strv_length (names), len, 3);
check_name ("group name", names[0], "group1", 0);
check_name ("group name", names[1], "group2", 1);
check_name ("group name", names[2], "group3", 2);
g_key_file_remove_group (keyfile, "group1", &error);
check_no_error (&error);
g_strfreev (names);
names = g_key_file_get_groups (keyfile, &len);
if (names == NULL)
{
g_print ("Error listing groups\n");
exit (1);
}
check_length ("groups", g_strv_length (names), len, 2);
check_name ("group name", names[0], "group2", 0);
check_name ("group name", names[1], "group3", 1);
g_key_file_remove_group (keyfile, "group2", &error);
check_no_error (&error);
g_strfreev (names);
names = g_key_file_get_groups (keyfile, &len);
if (names == NULL)
{
g_print ("Error listing groups\n");
exit (1);
}
check_length ("groups", g_strv_length (names), len, 1);
check_name ("group name", names[0], "group3", 0);
g_key_file_remove_group (keyfile, "no such group", &error);
check_error (&error, G_KEY_FILE_ERROR, G_KEY_FILE_ERROR_GROUP_NOT_FOUND);
g_strfreev (names);
g_key_file_free (keyfile);
}
/* http://bugzilla.gnome.org/show_bug.cgi?id=165980 */
static void
test_key_remove (void)
{
GKeyFile *keyfile;
gchar *value;
GError *error = NULL;
const gchar *data =
"[group1]\n"
"key1=bla\n"
"key2=bla\n";
keyfile = load_data (data, 0);
check_string_value (keyfile, "group1", "key1", "bla");
g_key_file_remove_key (keyfile, "group1", "key1", &error);
check_no_error (&error);
value = g_key_file_get_string (keyfile, "group1", "key1", &error);
check_error (&error, G_KEY_FILE_ERROR, G_KEY_FILE_ERROR_KEY_NOT_FOUND);
g_free (value);
g_key_file_remove_key (keyfile, "group1", "key1", &error);
check_error (&error, G_KEY_FILE_ERROR, G_KEY_FILE_ERROR_KEY_NOT_FOUND);
g_key_file_remove_key (keyfile, "no such group", "key1", &error);
check_error (&error, G_KEY_FILE_ERROR, G_KEY_FILE_ERROR_GROUP_NOT_FOUND);
g_key_file_free (keyfile);
}
int
main (int argc, char *argv[])
{
test_line_ends ();
test_whitespace ();
test_listing ();
test_string ();
test_boolean ();
test_integer ();
test_locale_string ();
test_lists ();
test_group_remove ();
test_key_remove ();
return 0;
}