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