2 Unix SMB/CIFS implementation.
4 local testing of registry library - registry backend
6 Copyright (C) Jelmer Vernooij 2005-2007
8 This program is free software; you can redistribute it and/or modify
9 it under the terms of the GNU General Public License as published by
10 the Free Software Foundation; either version 2 of the License, or
11 (at your option) any later version.
13 This program is distributed in the hope that it will be useful,
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 GNU General Public License for more details.
18 You should have received a copy of the GNU General Public License
19 along with this program; if not, write to the Free Software
20 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
24 #include "lib/registry/registry.h"
25 #include "lib/cmdline/popt_common.h"
26 #include "torture/torture.h"
27 #include "librpc/gen_ndr/winreg.h"
28 #include "libcli/security/security.h"
29 #include "system/filesys.h"
31 NTSTATUS torture_temp_dir(struct torture_context *tctx, const char *prefix,
32 const char **tempdir);
35 * Test obtaining a predefined key.
37 static bool test_get_predefined(struct torture_context *tctx,
40 const struct registry_context *rctx = _data;
41 struct registry_key *root;
44 error = reg_get_predefined_key(rctx, HKEY_CLASSES_ROOT, &root);
45 torture_assert_werr_ok(tctx, error,
46 "getting predefined key failed");
51 * Test obtaining a predefined key.
53 static bool test_get_predefined_unknown(struct torture_context *tctx,
56 const struct registry_context *rctx = _data;
57 struct registry_key *root;
60 error = reg_get_predefined_key(rctx, 1337, &root);
61 torture_assert_werr_equal(tctx, error, WERR_NOT_FOUND,
62 "getting predefined key failed");
66 static bool test_predef_key_by_name(struct torture_context *tctx,
69 const struct registry_context *rctx = _data;
70 struct registry_key *root;
73 error = reg_get_predefined_key_by_name(rctx, "HKEY_CLASSES_ROOT", &root);
74 torture_assert_werr_ok(tctx, error,
75 "getting predefined key failed");
77 error = reg_get_predefined_key_by_name(rctx, "HKEY_classes_ROOT", &root);
78 torture_assert_werr_ok(tctx, error,
79 "getting predefined key case insensitively failed");
84 static bool test_predef_key_by_name_invalid(struct torture_context *tctx,
87 const struct registry_context *rctx = _data;
88 struct registry_key *root;
91 error = reg_get_predefined_key_by_name(rctx, "BLA", &root);
92 torture_assert_werr_equal(tctx, error, WERR_BADFILE,
93 "getting predefined key failed");
98 * Test creating a new subkey
100 static bool test_create_subkey(struct torture_context *tctx,
103 const struct registry_context *rctx = _data;
104 struct registry_key *root, *newkey;
107 error = reg_get_predefined_key(rctx, HKEY_CLASSES_ROOT, &root);
108 torture_assert_werr_ok(tctx, error,
109 "getting predefined key failed");
111 error = reg_key_add_name(rctx, root, "Bad Bentheim", NULL, NULL, &newkey);
112 torture_assert_werr_ok(tctx, error, "Creating key return code");
113 torture_assert(tctx, newkey != NULL, "Creating new key");
119 * Test creating a new nested subkey
121 static bool test_create_nested_subkey(struct torture_context *tctx,
124 const struct registry_context *rctx = _data;
125 struct registry_key *root, *newkey1, *newkey2;
128 error = reg_get_predefined_key(rctx, HKEY_CLASSES_ROOT, &root);
129 torture_assert_werr_ok(tctx, error,
130 "getting predefined key failed");
132 error = reg_key_add_name(rctx, root, "Hamburg", NULL, NULL,
134 torture_assert_werr_ok(tctx, error, "Creating key return code");
135 torture_assert(tctx, newkey1 != NULL, "Creating new key");
137 error = reg_key_add_name(rctx, root, "Hamburg\\Hamburg", NULL, NULL,
139 torture_assert_werr_ok(tctx, error, "Creating key return code");
140 torture_assert(tctx, newkey2 != NULL, "Creating new key");
146 * Test creating a new subkey
148 static bool test_key_add_abs_top(struct torture_context *tctx,
151 const struct registry_context *rctx = _data;
152 struct registry_key *root;
155 error = reg_key_add_abs(tctx, rctx, "HKEY_CLASSES_ROOT", 0, NULL, &root);
156 torture_assert_werr_equal(tctx, error, WERR_ALREADY_EXISTS, "create top level");
162 * Test creating a new subkey
164 static bool test_key_add_abs(struct torture_context *tctx,
168 const struct registry_context *rctx = _data;
169 struct registry_key *root, *result1, *result2;
171 error = reg_key_add_abs(tctx, rctx, "HKEY_CLASSES_ROOT\\bloe", 0, NULL, &result1);
172 torture_assert_werr_ok(tctx, error, "create lowest");
174 error = reg_key_add_abs(tctx, rctx, "HKEY_CLASSES_ROOT\\bloe\\bla", 0, NULL, &result1);
175 torture_assert_werr_ok(tctx, error, "create nested");
177 error = reg_get_predefined_key(rctx, HKEY_CLASSES_ROOT, &root);
178 torture_assert_werr_ok(tctx, error,
179 "getting predefined key failed");
181 error = reg_open_key(tctx, root, "bloe", &result2);
182 torture_assert_werr_ok(tctx, error, "opening key");
184 error = reg_open_key(tctx, root, "bloe\\bla", &result2);
185 torture_assert_werr_ok(tctx, error, "opening key");
191 static bool test_del_key(struct torture_context *tctx, const void *_data)
193 const struct registry_context *rctx = _data;
194 struct registry_key *root, *newkey;
197 error = reg_get_predefined_key(rctx, HKEY_CLASSES_ROOT, &root);
198 torture_assert_werr_ok(tctx, error,
199 "getting predefined key failed");
201 error = reg_key_add_name(rctx, root, "Hamburg", NULL, NULL, &newkey);
203 torture_assert_werr_ok(tctx, error, "Creating key return code");
204 torture_assert(tctx, newkey != NULL, "Creating new key");
206 error = reg_key_del(root, "Hamburg");
207 torture_assert_werr_ok(tctx, error, "Delete key");
209 error = reg_key_del(root, "Hamburg");
210 torture_assert_werr_equal(tctx, error, WERR_NOT_FOUND,
211 "Delete missing key");
217 * Convenience function for opening the HKEY_CLASSES_ROOT hive and
218 * creating a single key for testing purposes.
220 static bool create_test_key(struct torture_context *tctx,
221 const struct registry_context *rctx,
223 struct registry_key **root,
224 struct registry_key **subkey)
228 error = reg_get_predefined_key(rctx, HKEY_CLASSES_ROOT, root);
229 torture_assert_werr_ok(tctx, error,
230 "getting predefined key failed");
232 error = reg_key_add_name(rctx, *root, name, NULL, NULL, subkey);
233 torture_assert_werr_ok(tctx, error, "Creating key return code");
239 static bool test_flush_key(struct torture_context *tctx, const void *_data)
241 const struct registry_context *rctx = _data;
242 struct registry_key *root, *subkey;
245 if (!create_test_key(tctx, rctx, "Munchen", &root, &subkey))
248 error = reg_key_flush(subkey);
249 torture_assert_werr_ok(tctx, error, "flush key");
251 torture_assert_werr_equal(tctx, reg_key_flush(NULL),
252 WERR_INVALID_PARAM, "flush key");
257 static bool test_query_key(struct torture_context *tctx, const void *_data)
259 const struct registry_context *rctx = _data;
260 struct registry_key *root, *subkey;
262 NTTIME last_changed_time;
263 uint32_t num_subkeys, num_values;
264 const char *classname;
266 if (!create_test_key(tctx, rctx, "Munchen", &root, &subkey))
269 error = reg_key_get_info(tctx, subkey, &classname,
270 &num_subkeys, &num_values,
273 torture_assert_werr_ok(tctx, error, "get info key");
274 torture_assert(tctx, classname == NULL, "classname");
275 torture_assert_int_equal(tctx, num_subkeys, 0, "num subkeys");
276 torture_assert_int_equal(tctx, num_values, 0, "num values");
281 static bool test_query_key_nums(struct torture_context *tctx, const void *_data)
283 const struct registry_context *rctx = _data;
284 struct registry_key *root, *subkey1, *subkey2;
286 uint32_t num_subkeys, num_values;
289 if (!create_test_key(tctx, rctx, "Berlin", &root, &subkey1))
292 error = reg_key_add_name(rctx, subkey1, "Bentheim", NULL, NULL, &subkey2);
293 torture_assert_werr_ok(tctx, error, "Creating key return code");
295 error = reg_val_set(subkey1, "Answer", REG_DWORD,
296 data_blob_talloc(tctx, &data, sizeof(data)));
297 torture_assert_werr_ok(tctx, error, "set value");
299 error = reg_key_get_info(tctx, subkey1, NULL, &num_subkeys,
302 torture_assert_werr_ok(tctx, error, "get info key");
303 torture_assert_int_equal(tctx, num_subkeys, 1, "num subkeys");
304 torture_assert_int_equal(tctx, num_values, 1, "num values");
310 * Test that the subkeys of a key can be enumerated, that
311 * the returned parameters for get_subkey_by_index are optional and
312 * that enumerating the parents of a non-top-level node works.
314 static bool test_list_subkeys(struct torture_context *tctx, const void *_data)
316 const struct registry_context *rctx = _data;
317 struct registry_key *subkey = NULL, *root;
319 NTTIME last_mod_time;
320 const char *classname, *name;
322 if (!create_test_key(tctx, rctx, "Goettingen", &root, &subkey))
325 error = reg_key_get_subkey_by_index(tctx, root, 0, &name, &classname,
328 torture_assert_werr_ok(tctx, error, "Enum keys return code");
329 torture_assert_str_equal(tctx, name, "Goettingen", "Enum keys data");
332 error = reg_key_get_subkey_by_index(tctx, root, 0, NULL, NULL, NULL);
334 torture_assert_werr_ok(tctx, error, "Enum keys with NULL arguments return code");
336 error = reg_key_get_subkey_by_index(tctx, root, 1, NULL, NULL, NULL);
338 torture_assert_werr_equal(tctx, error, WERR_NO_MORE_ITEMS,
339 "Invalid error for no more items");
341 error = reg_key_get_subkey_by_index(tctx, subkey, 0, NULL, NULL, NULL);
343 torture_assert_werr_equal(tctx, error, WERR_NO_MORE_ITEMS,
344 "Invalid error for no more items");
350 * Test setting a value
352 static bool test_set_value(struct torture_context *tctx, const void *_data)
354 const struct registry_context *rctx = _data;
355 struct registry_key *subkey = NULL, *root;
359 if (!create_test_key(tctx, rctx, "Dusseldorf", &root, &subkey))
362 error = reg_val_set(subkey, "Answer", REG_DWORD,
363 data_blob_talloc(tctx, &data, sizeof(data)));
364 torture_assert_werr_ok (tctx, error, "setting value");
370 * Test getting/setting security descriptors
372 static bool test_security(struct torture_context *tctx, const void *_data)
374 const struct registry_context *rctx = _data;
375 struct registry_key *subkey = NULL, *root;
377 struct security_descriptor *osd, *nsd;
379 if (!create_test_key(tctx, rctx, "Düsseldorf", &root, &subkey))
382 osd = security_descriptor_create(tctx,
384 SID_NT_AUTHENTICATED_USERS,
385 SEC_ACE_TYPE_ACCESS_ALLOWED,
387 SEC_ACE_FLAG_OBJECT_INHERIT,
390 error = reg_set_security(subkey, osd);
391 torture_assert_werr_ok(tctx, error, "setting security");
393 error = reg_get_security(tctx, subkey, &nsd);
394 torture_assert_werr_ok (tctx, error, "setting security");
396 torture_assert(tctx, security_descriptor_equal(osd, nsd),
397 "security descriptor changed!");
403 * Test getting a value
405 static bool test_get_value(struct torture_context *tctx, const void *_data)
407 const struct registry_context *rctx = _data;
408 struct registry_key *subkey = NULL, *root;
414 if (!create_test_key(tctx, rctx, "Duisburg", &root, &subkey))
417 error = reg_key_get_value_by_name(tctx, subkey, __FUNCTION__, &type,
419 torture_assert_werr_equal(tctx, error, WERR_NOT_FOUND,
420 "getting missing value");
422 error = reg_val_set(subkey, __FUNCTION__, REG_DWORD,
423 data_blob_talloc(tctx, &value, 4));
424 torture_assert_werr_ok (tctx, error, "setting value");
426 error = reg_key_get_value_by_name(tctx, subkey, __FUNCTION__, &type,
428 torture_assert_werr_ok(tctx, error, "getting value");
430 torture_assert_int_equal(tctx, 4, data.length, "value length ok");
431 torture_assert(tctx, memcmp(data.data, &value, 4) == 0, "value content ok");
432 torture_assert_int_equal(tctx, REG_DWORD, type, "value type");
438 * Test unsetting a value
440 static bool test_del_value(struct torture_context *tctx, const void *_data)
442 const struct registry_context *rctx = _data;
443 struct registry_key *subkey = NULL, *root;
449 if (!create_test_key(tctx, rctx, "Duisburg", &root, &subkey))
452 error = reg_key_get_value_by_name(tctx, subkey, __FUNCTION__, &type,
454 torture_assert_werr_equal(tctx, error, WERR_NOT_FOUND,
455 "getting missing value");
457 error = reg_val_set(subkey, __FUNCTION__, REG_DWORD,
458 data_blob_talloc(tctx, &value, 4));
459 torture_assert_werr_ok (tctx, error, "setting value");
461 error = reg_del_value(subkey, __FUNCTION__);
462 torture_assert_werr_ok (tctx, error, "unsetting value");
464 error = reg_key_get_value_by_name(tctx, subkey, __FUNCTION__, &type, &data);
465 torture_assert_werr_equal(tctx, error, WERR_NOT_FOUND,
466 "getting missing value");
472 * Test listing values
474 static bool test_list_values(struct torture_context *tctx, const void *_data)
476 const struct registry_context *rctx = _data;
477 struct registry_key *subkey = NULL, *root;
484 if (!create_test_key(tctx, rctx, "Bonn", &root, &subkey))
487 error = reg_val_set(subkey, "bar", REG_DWORD,
488 data_blob_talloc(tctx, &value, 4));
489 torture_assert_werr_ok (tctx, error, "setting value");
491 error = reg_key_get_value_by_index(tctx, subkey, 0, &name, &type, &data);
492 torture_assert_werr_ok(tctx, error, "getting value");
494 torture_assert_str_equal(tctx, name, "bar", "value name");
495 torture_assert_int_equal(tctx, 4, data.length, "value length");
496 torture_assert(tctx, memcmp(data.data, &value, 4) == 0, "value content");
497 torture_assert_int_equal(tctx, REG_DWORD, type, "value type");
499 error = reg_key_get_value_by_index(tctx, subkey, 1, &name, &type, &data);
500 torture_assert_werr_equal(tctx, error, WERR_NO_MORE_ITEMS,
501 "getting missing value");
506 static bool setup_local_registry(struct torture_context *tctx, void **data)
508 struct registry_context *rctx;
512 struct hive_key *hive_key;
513 const char *filename;
515 error = reg_open_local(tctx, &rctx, NULL, NULL);
516 torture_assert_werr_ok(tctx, error, "Opening local registry failed");
518 status = torture_temp_dir(tctx, "registry-local", &tempdir);
519 torture_assert_ntstatus_ok(tctx, status, "Creating temp dir failed");
521 filename = talloc_asprintf(tctx, "%s/classes_root.ldb", tempdir);
522 error = reg_open_ldb_file(tctx, filename, NULL, NULL, &hive_key);
523 torture_assert_werr_ok(tctx, error, "Opening classes_root file failed");
525 error = reg_mount_hive(rctx, hive_key, HKEY_CLASSES_ROOT, NULL);
526 torture_assert_werr_ok(tctx, error, "Mounting hive failed");
533 static void tcase_add_tests(struct torture_tcase *tcase)
535 torture_tcase_add_simple_test(tcase, "list_subkeys", test_list_subkeys);
536 torture_tcase_add_simple_test(tcase, "get_predefined_key", test_get_predefined);
537 torture_tcase_add_simple_test(tcase, "get_predefined_key", test_get_predefined_unknown);
538 torture_tcase_add_simple_test(tcase, "create_key", test_create_subkey);
539 torture_tcase_add_simple_test(tcase, "create_key", test_create_nested_subkey);
540 torture_tcase_add_simple_test(tcase, "key_add_abs", test_key_add_abs);
541 torture_tcase_add_simple_test(tcase, "key_add_abs_top", test_key_add_abs_top);
542 torture_tcase_add_simple_test(tcase, "set_value", test_set_value);
543 torture_tcase_add_simple_test(tcase, "get_value", test_get_value);
544 torture_tcase_add_simple_test(tcase, "list_values", test_list_values);
545 torture_tcase_add_simple_test(tcase, "del_key", test_del_key);
546 torture_tcase_add_simple_test(tcase, "del_value", test_del_value);
547 torture_tcase_add_simple_test(tcase, "flush_key", test_flush_key);
548 torture_tcase_add_simple_test(tcase, "query_key", test_query_key);
549 torture_tcase_add_simple_test(tcase, "query_key_nums", test_query_key_nums);
550 torture_tcase_add_simple_test(tcase, "test_predef_key_by_name",
551 test_predef_key_by_name);
552 torture_tcase_add_simple_test(tcase, "security", test_security);
553 torture_tcase_add_simple_test(tcase, "test_predef_key_by_name_invalid",
554 test_predef_key_by_name_invalid);
557 struct torture_suite *torture_registry_registry(TALLOC_CTX *mem_ctx)
559 struct torture_tcase *tcase;
560 struct torture_suite *suite = torture_suite_create(mem_ctx, "REGISTRY");
562 tcase = torture_suite_add_tcase(suite, "local");
563 torture_tcase_set_fixture(tcase, setup_local_registry, NULL);
564 tcase_add_tests(tcase);