]>
Commit | Line | Data |
---|---|---|
7b8c51ad LC |
1 | /* |
2 | * QDict unit-tests. | |
3 | * | |
4 | * Copyright (C) 2009 Red Hat Inc. | |
5 | * | |
6 | * Authors: | |
7 | * Luiz Capitulino <[email protected]> | |
41836a9f LC |
8 | * |
9 | * This work is licensed under the terms of the GNU LGPL, version 2.1 or later. | |
10 | * See the COPYING.LIB file in the top-level directory. | |
7b8c51ad | 11 | */ |
7b8c51ad | 12 | |
47e6b297 | 13 | #include "qemu/osdep.h" |
7b1b5d19 | 14 | #include "qapi/qmp/qdict.h" |
7b8c51ad LC |
15 | |
16 | /* | |
17 | * Public Interface test-cases | |
18 | * | |
19 | * (with some violations to access 'private' data) | |
20 | */ | |
21 | ||
ac531cb6 | 22 | static void qdict_new_test(void) |
7b8c51ad LC |
23 | { |
24 | QDict *qdict; | |
25 | ||
26 | qdict = qdict_new(); | |
ac531cb6 AL |
27 | g_assert(qdict != NULL); |
28 | g_assert(qdict_size(qdict) == 0); | |
29 | g_assert(qdict->base.refcnt == 1); | |
30 | g_assert(qobject_type(QOBJECT(qdict)) == QTYPE_QDICT); | |
7b8c51ad | 31 | |
cb3e7f08 | 32 | qobject_unref(qdict); |
7b8c51ad | 33 | } |
7b8c51ad | 34 | |
ac531cb6 | 35 | static void qdict_put_obj_test(void) |
7b8c51ad | 36 | { |
01b2ffce | 37 | QNum *qn; |
7b8c51ad LC |
38 | QDict *qdict; |
39 | QDictEntry *ent; | |
40 | const int num = 42; | |
41 | ||
42 | qdict = qdict_new(); | |
43 | ||
44 | // key "" will have tdb hash 12345 | |
46f5ac20 | 45 | qdict_put_int(qdict, "", num); |
7b8c51ad | 46 | |
ac531cb6 | 47 | g_assert(qdict_size(qdict) == 1); |
c8bc3cd7 | 48 | ent = QLIST_FIRST(&qdict->table[12345 % QDICT_BUCKET_MAX]); |
7dc847eb | 49 | qn = qobject_to(QNum, ent->value); |
01b2ffce | 50 | g_assert_cmpint(qnum_get_int(qn), ==, num); |
7b8c51ad | 51 | |
cb3e7f08 | 52 | qobject_unref(qdict); |
7b8c51ad | 53 | } |
7b8c51ad | 54 | |
ac531cb6 | 55 | static void qdict_destroy_simple_test(void) |
7b8c51ad LC |
56 | { |
57 | QDict *qdict; | |
58 | ||
59 | qdict = qdict_new(); | |
46f5ac20 EB |
60 | qdict_put_int(qdict, "num", 0); |
61 | qdict_put_str(qdict, "str", "foo"); | |
7b8c51ad | 62 | |
cb3e7f08 | 63 | qobject_unref(qdict); |
7b8c51ad | 64 | } |
7b8c51ad | 65 | |
ac531cb6 | 66 | static void qdict_get_test(void) |
7b8c51ad | 67 | { |
01b2ffce | 68 | QNum *qn; |
7b8c51ad LC |
69 | QObject *obj; |
70 | const int value = -42; | |
71 | const char *key = "test"; | |
ac531cb6 | 72 | QDict *tests_dict = qdict_new(); |
7b8c51ad | 73 | |
46f5ac20 | 74 | qdict_put_int(tests_dict, key, value); |
7b8c51ad LC |
75 | |
76 | obj = qdict_get(tests_dict, key); | |
ac531cb6 | 77 | g_assert(obj != NULL); |
7b8c51ad | 78 | |
7dc847eb | 79 | qn = qobject_to(QNum, obj); |
01b2ffce | 80 | g_assert_cmpint(qnum_get_int(qn), ==, value); |
ac531cb6 | 81 | |
cb3e7f08 | 82 | qobject_unref(tests_dict); |
7b8c51ad | 83 | } |
7b8c51ad | 84 | |
ac531cb6 | 85 | static void qdict_get_int_test(void) |
7b8c51ad LC |
86 | { |
87 | int ret; | |
88 | const int value = 100; | |
89 | const char *key = "int"; | |
ac531cb6 | 90 | QDict *tests_dict = qdict_new(); |
7b8c51ad | 91 | |
46f5ac20 | 92 | qdict_put_int(tests_dict, key, value); |
7b8c51ad LC |
93 | |
94 | ret = qdict_get_int(tests_dict, key); | |
ac531cb6 AL |
95 | g_assert(ret == value); |
96 | ||
cb3e7f08 | 97 | qobject_unref(tests_dict); |
7b8c51ad | 98 | } |
7b8c51ad | 99 | |
ac531cb6 | 100 | static void qdict_get_try_int_test(void) |
7b8c51ad LC |
101 | { |
102 | int ret; | |
103 | const int value = 100; | |
104 | const char *key = "int"; | |
ac531cb6 | 105 | QDict *tests_dict = qdict_new(); |
7b8c51ad | 106 | |
46f5ac20 | 107 | qdict_put_int(tests_dict, key, value); |
269c20b2 | 108 | qdict_put_str(tests_dict, "string", "test"); |
7b8c51ad LC |
109 | |
110 | ret = qdict_get_try_int(tests_dict, key, 0); | |
ac531cb6 AL |
111 | g_assert(ret == value); |
112 | ||
269c20b2 MAL |
113 | ret = qdict_get_try_int(tests_dict, "missing", -42); |
114 | g_assert_cmpuint(ret, ==, -42); | |
115 | ||
116 | ret = qdict_get_try_int(tests_dict, "string", -42); | |
117 | g_assert_cmpuint(ret, ==, -42); | |
118 | ||
cb3e7f08 | 119 | qobject_unref(tests_dict); |
7b8c51ad | 120 | } |
7b8c51ad | 121 | |
ac531cb6 | 122 | static void qdict_get_str_test(void) |
7b8c51ad LC |
123 | { |
124 | const char *p; | |
125 | const char *key = "key"; | |
126 | const char *str = "string"; | |
ac531cb6 | 127 | QDict *tests_dict = qdict_new(); |
7b8c51ad | 128 | |
46f5ac20 | 129 | qdict_put_str(tests_dict, key, str); |
7b8c51ad LC |
130 | |
131 | p = qdict_get_str(tests_dict, key); | |
ac531cb6 AL |
132 | g_assert(p != NULL); |
133 | g_assert(strcmp(p, str) == 0); | |
134 | ||
cb3e7f08 | 135 | qobject_unref(tests_dict); |
7b8c51ad | 136 | } |
7b8c51ad | 137 | |
ac531cb6 | 138 | static void qdict_get_try_str_test(void) |
7b8c51ad LC |
139 | { |
140 | const char *p; | |
141 | const char *key = "key"; | |
142 | const char *str = "string"; | |
ac531cb6 | 143 | QDict *tests_dict = qdict_new(); |
7b8c51ad | 144 | |
46f5ac20 | 145 | qdict_put_str(tests_dict, key, str); |
7b8c51ad LC |
146 | |
147 | p = qdict_get_try_str(tests_dict, key); | |
ac531cb6 AL |
148 | g_assert(p != NULL); |
149 | g_assert(strcmp(p, str) == 0); | |
150 | ||
cb3e7f08 | 151 | qobject_unref(tests_dict); |
7b8c51ad | 152 | } |
7b8c51ad | 153 | |
ac531cb6 | 154 | static void qdict_haskey_not_test(void) |
7b8c51ad | 155 | { |
ac531cb6 AL |
156 | QDict *tests_dict = qdict_new(); |
157 | g_assert(qdict_haskey(tests_dict, "test") == 0); | |
158 | ||
cb3e7f08 | 159 | qobject_unref(tests_dict); |
7b8c51ad | 160 | } |
7b8c51ad | 161 | |
ac531cb6 | 162 | static void qdict_haskey_test(void) |
7b8c51ad LC |
163 | { |
164 | const char *key = "test"; | |
ac531cb6 | 165 | QDict *tests_dict = qdict_new(); |
7b8c51ad | 166 | |
46f5ac20 | 167 | qdict_put_int(tests_dict, key, 0); |
ac531cb6 AL |
168 | g_assert(qdict_haskey(tests_dict, key) == 1); |
169 | ||
cb3e7f08 | 170 | qobject_unref(tests_dict); |
7b8c51ad | 171 | } |
7b8c51ad | 172 | |
ac531cb6 | 173 | static void qdict_del_test(void) |
7b8c51ad LC |
174 | { |
175 | const char *key = "key test"; | |
ac531cb6 | 176 | QDict *tests_dict = qdict_new(); |
7b8c51ad | 177 | |
46f5ac20 | 178 | qdict_put_str(tests_dict, key, "foo"); |
ac531cb6 | 179 | g_assert(qdict_size(tests_dict) == 1); |
7b8c51ad LC |
180 | |
181 | qdict_del(tests_dict, key); | |
182 | ||
ac531cb6 AL |
183 | g_assert(qdict_size(tests_dict) == 0); |
184 | g_assert(qdict_haskey(tests_dict, key) == 0); | |
185 | ||
cb3e7f08 | 186 | qobject_unref(tests_dict); |
7b8c51ad | 187 | } |
7b8c51ad | 188 | |
ac531cb6 | 189 | static void qobject_to_qdict_test(void) |
7b8c51ad | 190 | { |
ac531cb6 | 191 | QDict *tests_dict = qdict_new(); |
7dc847eb | 192 | g_assert(qobject_to(QDict, QOBJECT(tests_dict)) == tests_dict); |
ac531cb6 | 193 | |
cb3e7f08 | 194 | qobject_unref(tests_dict); |
7b8c51ad | 195 | } |
7b8c51ad | 196 | |
ac531cb6 | 197 | static void qdict_iterapi_test(void) |
d02c6bd4 LC |
198 | { |
199 | int count; | |
200 | const QDictEntry *ent; | |
ac531cb6 | 201 | QDict *tests_dict = qdict_new(); |
d02c6bd4 | 202 | |
ac531cb6 | 203 | g_assert(qdict_first(tests_dict) == NULL); |
d02c6bd4 | 204 | |
46f5ac20 EB |
205 | qdict_put_int(tests_dict, "key1", 1); |
206 | qdict_put_int(tests_dict, "key2", 2); | |
207 | qdict_put_int(tests_dict, "key3", 3); | |
d02c6bd4 LC |
208 | |
209 | count = 0; | |
210 | for (ent = qdict_first(tests_dict); ent; ent = qdict_next(tests_dict, ent)){ | |
ac531cb6 | 211 | g_assert(qdict_haskey(tests_dict, qdict_entry_key(ent)) == 1); |
d02c6bd4 LC |
212 | count++; |
213 | } | |
214 | ||
ac531cb6 | 215 | g_assert(count == qdict_size(tests_dict)); |
d02c6bd4 LC |
216 | |
217 | /* Do it again to test restarting */ | |
218 | count = 0; | |
219 | for (ent = qdict_first(tests_dict); ent; ent = qdict_next(tests_dict, ent)){ | |
ac531cb6 | 220 | g_assert(qdict_haskey(tests_dict, qdict_entry_key(ent)) == 1); |
d02c6bd4 LC |
221 | count++; |
222 | } | |
223 | ||
ac531cb6 AL |
224 | g_assert(count == qdict_size(tests_dict)); |
225 | ||
cb3e7f08 | 226 | qobject_unref(tests_dict); |
d02c6bd4 | 227 | } |
d02c6bd4 | 228 | |
7b8c51ad LC |
229 | /* |
230 | * Errors test-cases | |
231 | */ | |
232 | ||
ac531cb6 | 233 | static void qdict_put_exists_test(void) |
7b8c51ad LC |
234 | { |
235 | int value; | |
236 | const char *key = "exists"; | |
ac531cb6 | 237 | QDict *tests_dict = qdict_new(); |
7b8c51ad | 238 | |
46f5ac20 EB |
239 | qdict_put_int(tests_dict, key, 1); |
240 | qdict_put_int(tests_dict, key, 2); | |
7b8c51ad LC |
241 | |
242 | value = qdict_get_int(tests_dict, key); | |
ac531cb6 | 243 | g_assert(value == 2); |
29ec3156 | 244 | |
ac531cb6 AL |
245 | g_assert(qdict_size(tests_dict) == 1); |
246 | ||
cb3e7f08 | 247 | qobject_unref(tests_dict); |
7b8c51ad | 248 | } |
7b8c51ad | 249 | |
ac531cb6 | 250 | static void qdict_get_not_exists_test(void) |
7b8c51ad | 251 | { |
ac531cb6 AL |
252 | QDict *tests_dict = qdict_new(); |
253 | g_assert(qdict_get(tests_dict, "foo") == NULL); | |
254 | ||
cb3e7f08 | 255 | qobject_unref(tests_dict); |
7b8c51ad | 256 | } |
7b8c51ad LC |
257 | |
258 | /* | |
259 | * Stress test-case | |
260 | * | |
261 | * This is a lot big for a unit-test, but there is no other place | |
262 | * to have it. | |
263 | */ | |
264 | ||
265 | static void remove_dots(char *string) | |
266 | { | |
267 | char *p = strchr(string, ':'); | |
268 | if (p) | |
269 | *p = '\0'; | |
270 | } | |
271 | ||
272 | static QString *read_line(FILE *file, char *key) | |
273 | { | |
274 | char value[128]; | |
275 | ||
7464f058 | 276 | if (fscanf(file, "%127s%127s", key, value) == EOF) { |
7b8c51ad | 277 | return NULL; |
7464f058 | 278 | } |
7b8c51ad LC |
279 | remove_dots(key); |
280 | return qstring_from_str(value); | |
281 | } | |
282 | ||
283 | #define reset_file(file) fseek(file, 0L, SEEK_SET) | |
284 | ||
ac531cb6 | 285 | static void qdict_stress_test(void) |
7b8c51ad LC |
286 | { |
287 | size_t lines; | |
288 | char key[128]; | |
289 | FILE *test_file; | |
290 | QDict *qdict; | |
291 | QString *value; | |
292 | const char *test_file_path = "qdict-test-data.txt"; | |
293 | ||
294 | test_file = fopen(test_file_path, "r"); | |
ac531cb6 | 295 | g_assert(test_file != NULL); |
7b8c51ad LC |
296 | |
297 | // Create the dict | |
298 | qdict = qdict_new(); | |
ac531cb6 | 299 | g_assert(qdict != NULL); |
7b8c51ad LC |
300 | |
301 | // Add everything from the test file | |
302 | for (lines = 0;; lines++) { | |
303 | value = read_line(test_file, key); | |
304 | if (!value) | |
305 | break; | |
306 | ||
307 | qdict_put(qdict, key, value); | |
308 | } | |
ac531cb6 | 309 | g_assert(qdict_size(qdict) == lines); |
7b8c51ad LC |
310 | |
311 | // Check if everything is really in there | |
312 | reset_file(test_file); | |
313 | for (;;) { | |
314 | const char *str1, *str2; | |
315 | ||
316 | value = read_line(test_file, key); | |
317 | if (!value) | |
318 | break; | |
319 | ||
320 | str1 = qstring_get_str(value); | |
321 | ||
322 | str2 = qdict_get_str(qdict, key); | |
ac531cb6 | 323 | g_assert(str2 != NULL); |
7b8c51ad | 324 | |
ac531cb6 | 325 | g_assert(strcmp(str1, str2) == 0); |
7b8c51ad | 326 | |
cb3e7f08 | 327 | qobject_unref(value); |
7b8c51ad LC |
328 | } |
329 | ||
330 | // Delete everything | |
331 | reset_file(test_file); | |
332 | for (;;) { | |
333 | value = read_line(test_file, key); | |
334 | if (!value) | |
335 | break; | |
336 | ||
337 | qdict_del(qdict, key); | |
cb3e7f08 | 338 | qobject_unref(value); |
7b8c51ad | 339 | |
ac531cb6 | 340 | g_assert(qdict_haskey(qdict, key) == 0); |
7b8c51ad LC |
341 | } |
342 | fclose(test_file); | |
343 | ||
ac531cb6 | 344 | g_assert(qdict_size(qdict) == 0); |
cb3e7f08 | 345 | qobject_unref(qdict); |
7b8c51ad | 346 | } |
7b8c51ad | 347 | |
ac531cb6 | 348 | int main(int argc, char **argv) |
7b8c51ad | 349 | { |
ac531cb6 | 350 | g_test_init(&argc, &argv, NULL); |
7b8c51ad | 351 | |
ac531cb6 AL |
352 | g_test_add_func("/public/new", qdict_new_test); |
353 | g_test_add_func("/public/put_obj", qdict_put_obj_test); | |
354 | g_test_add_func("/public/destroy_simple", qdict_destroy_simple_test); | |
7b8c51ad LC |
355 | |
356 | /* Continue, but now with fixtures */ | |
ac531cb6 AL |
357 | g_test_add_func("/public/get", qdict_get_test); |
358 | g_test_add_func("/public/get_int", qdict_get_int_test); | |
359 | g_test_add_func("/public/get_try_int", qdict_get_try_int_test); | |
360 | g_test_add_func("/public/get_str", qdict_get_str_test); | |
361 | g_test_add_func("/public/get_try_str", qdict_get_try_str_test); | |
362 | g_test_add_func("/public/haskey_not", qdict_haskey_not_test); | |
363 | g_test_add_func("/public/haskey", qdict_haskey_test); | |
364 | g_test_add_func("/public/del", qdict_del_test); | |
365 | g_test_add_func("/public/to_qdict", qobject_to_qdict_test); | |
366 | g_test_add_func("/public/iterapi", qdict_iterapi_test); | |
367 | ||
368 | g_test_add_func("/errors/put_exists", qdict_put_exists_test); | |
369 | g_test_add_func("/errors/get_not_exists", qdict_get_not_exists_test); | |
7b8c51ad LC |
370 | |
371 | /* The Big one */ | |
ac531cb6 AL |
372 | if (g_test_slow()) { |
373 | g_test_add_func("/stress/test", qdict_stress_test); | |
374 | } | |
7b8c51ad | 375 | |
ac531cb6 | 376 | return g_test_run(); |
7b8c51ad | 377 | } |