s3: Fix bug #9085.
[Samba.git] / source3 / utils / net_registry.c
blobc118b7331074e5bff43af230c096d0d9fac0bb36
1 /*
2 * Samba Unix/Linux SMB client library
3 * Distributed SMB/CIFS Server Management Utility
4 * Local registry interface
6 * Copyright (C) Michael Adam 2008
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 3 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, see <http://www.gnu.org/licenses/>.
22 #include "includes.h"
23 #include "utils/net.h"
24 #include "utils/net_registry_util.h"
29 * Helper functions
33 /**
34 * split given path into hive and remaining path and open the hive key
36 static WERROR open_hive(TALLOC_CTX *ctx, const char *path,
37 uint32 desired_access,
38 struct registry_key **hive,
39 char **subkeyname)
41 WERROR werr;
42 NT_USER_TOKEN *token = NULL;
43 char *hivename = NULL;
44 char *tmp_subkeyname = NULL;
45 TALLOC_CTX *tmp_ctx = talloc_stackframe();
47 if ((hive == NULL) || (subkeyname == NULL)) {
48 werr = WERR_INVALID_PARAM;
49 goto done;
52 werr = split_hive_key(tmp_ctx, path, &hivename, &tmp_subkeyname);
53 if (!W_ERROR_IS_OK(werr)) {
54 goto done;
56 *subkeyname = talloc_strdup(ctx, tmp_subkeyname);
57 if (*subkeyname == NULL) {
58 werr = WERR_NOMEM;
59 goto done;
62 werr = ntstatus_to_werror(registry_create_admin_token(tmp_ctx, &token));
63 if (!W_ERROR_IS_OK(werr)) {
64 goto done;
67 werr = reg_openhive(ctx, hivename, desired_access, token, hive);
68 if (!W_ERROR_IS_OK(werr)) {
69 goto done;
72 werr = WERR_OK;
74 done:
75 TALLOC_FREE(tmp_ctx);
76 return werr;
79 static WERROR open_key(TALLOC_CTX *ctx, const char *path,
80 uint32 desired_access,
81 struct registry_key **key)
83 WERROR werr;
84 char *subkey_name = NULL;
85 struct registry_key *hive = NULL;
86 TALLOC_CTX *tmp_ctx = talloc_stackframe();
88 if ((path == NULL) || (key == NULL)) {
89 return WERR_INVALID_PARAM;
92 werr = open_hive(tmp_ctx, path, desired_access, &hive, &subkey_name);
93 if (!W_ERROR_IS_OK(werr)) {
94 d_fprintf(stderr, _("open_hive failed: %s\n"),
95 win_errstr(werr));
96 goto done;
99 werr = reg_openkey(ctx, hive, subkey_name, desired_access, key);
100 if (!W_ERROR_IS_OK(werr)) {
101 d_fprintf(stderr, _("reg_openkey failed: %s\n"),
102 win_errstr(werr));
103 goto done;
106 werr = WERR_OK;
108 done:
109 TALLOC_FREE(tmp_ctx);
110 return werr;
115 * the main "net registry" function implementations
119 static int net_registry_enumerate(struct net_context *c, int argc,
120 const char **argv)
122 WERROR werr;
123 struct registry_key *key = NULL;
124 TALLOC_CTX *ctx = talloc_stackframe();
125 char *subkey_name;
126 NTTIME modtime;
127 uint32_t count;
128 char *valname = NULL;
129 struct registry_value *valvalue = NULL;
130 int ret = -1;
132 if (argc != 1 || c->display_usage) {
133 d_printf("%s\n%s",
134 _("Usage:"),
135 _("net registry enumerate <path>\n"));
136 d_printf("%s\n%s",
137 _("Example:"),
138 _("net registry enumerate 'HKLM\\Software\\Samba'\n"));
139 goto done;
142 werr = open_key(ctx, argv[0], REG_KEY_READ, &key);
143 if (!W_ERROR_IS_OK(werr)) {
144 d_fprintf(stderr, _("open_key failed: %s\n"), win_errstr(werr));
145 goto done;
148 for (count = 0;
149 werr = reg_enumkey(ctx, key, count, &subkey_name, &modtime),
150 W_ERROR_IS_OK(werr);
151 count++)
153 print_registry_key(subkey_name, &modtime);
155 if (!W_ERROR_EQUAL(WERR_NO_MORE_ITEMS, werr)) {
156 goto done;
159 for (count = 0;
160 werr = reg_enumvalue(ctx, key, count, &valname, &valvalue),
161 W_ERROR_IS_OK(werr);
162 count++)
164 print_registry_value_with_name(valname, valvalue);
166 if (!W_ERROR_EQUAL(WERR_NO_MORE_ITEMS, werr)) {
167 goto done;
170 ret = 0;
171 done:
172 TALLOC_FREE(ctx);
173 return ret;
176 static int net_registry_createkey(struct net_context *c, int argc,
177 const char **argv)
179 WERROR werr;
180 enum winreg_CreateAction action;
181 char *subkeyname;
182 struct registry_key *hivekey = NULL;
183 struct registry_key *subkey = NULL;
184 TALLOC_CTX *ctx = talloc_stackframe();
185 int ret = -1;
187 if (argc != 1 || c->display_usage) {
188 d_printf("%s\n%s",
189 _("Usage:"),
190 _("net registry createkey <path>\n"));
191 d_printf("%s\n%s",
192 _("Example:"),
193 _("net registry createkey "
194 "'HKLM\\Software\\Samba\\smbconf.127.0.0.1'\n"));
195 goto done;
197 if (strlen(argv[0]) == 0) {
198 d_fprintf(stderr, _("error: zero length key name given\n"));
199 goto done;
202 werr = open_hive(ctx, argv[0], REG_KEY_WRITE, &hivekey, &subkeyname);
203 if (!W_ERROR_IS_OK(werr)) {
204 d_fprintf(stderr, _("open_hive failed: %s\n"),
205 win_errstr(werr));
206 goto done;
209 werr = reg_createkey(ctx, hivekey, subkeyname, REG_KEY_WRITE,
210 &subkey, &action);
211 if (!W_ERROR_IS_OK(werr)) {
212 d_fprintf(stderr, _("reg_createkey failed: %s\n"),
213 win_errstr(werr));
214 goto done;
216 switch (action) {
217 case REG_ACTION_NONE:
218 d_printf(_("createkey did nothing -- huh?\n"));
219 break;
220 case REG_CREATED_NEW_KEY:
221 d_printf(_("createkey created %s\n"), argv[0]);
222 break;
223 case REG_OPENED_EXISTING_KEY:
224 d_printf(_("createkey opened existing %s\n"), argv[0]);
225 break;
228 ret = 0;
230 done:
231 TALLOC_FREE(ctx);
232 return ret;
235 static int net_registry_deletekey(struct net_context *c, int argc,
236 const char **argv)
238 WERROR werr;
239 char *subkeyname;
240 struct registry_key *hivekey = NULL;
241 TALLOC_CTX *ctx = talloc_stackframe();
242 int ret = -1;
244 if (argc != 1 || c->display_usage) {
245 d_printf("%s\n%s",
246 _("Usage:"),
247 _("net registry deletekey <path>\n"));
248 d_printf("%s\n%s",
249 _("Example:"),
250 _("net registry deletekey "
251 "'HKLM\\Software\\Samba\\smbconf.127.0.0.1'\n"));
252 goto done;
254 if (strlen(argv[0]) == 0) {
255 d_fprintf(stderr, _("error: zero length key name given\n"));
256 goto done;
259 werr = open_hive(ctx, argv[0], REG_KEY_WRITE, &hivekey, &subkeyname);
260 if (!W_ERROR_IS_OK(werr)) {
261 d_fprintf(stderr, "open_hive %s: %s\n", _("failed"),
262 win_errstr(werr));
263 goto done;
266 werr = reg_deletekey(hivekey, subkeyname);
267 if (!W_ERROR_IS_OK(werr)) {
268 d_fprintf(stderr, "reg_deletekey %s: %s\n", _("failed"),
269 win_errstr(werr));
270 goto done;
273 ret = 0;
275 done:
276 TALLOC_FREE(ctx);
277 return ret;
280 static int net_registry_getvalue_internal(struct net_context *c, int argc,
281 const char **argv, bool raw)
283 WERROR werr;
284 int ret = -1;
285 struct registry_key *key = NULL;
286 struct registry_value *value = NULL;
287 TALLOC_CTX *ctx = talloc_stackframe();
289 if (argc != 2 || c->display_usage) {
290 d_fprintf(stderr, "%s\n%s",
291 _("Usage:"),
292 _("net rpc registry getvalue <key> <valuename>\n"));
293 goto done;
296 werr = open_key(ctx, argv[0], REG_KEY_READ, &key);
297 if (!W_ERROR_IS_OK(werr)) {
298 d_fprintf(stderr, _("open_key failed: %s\n"), win_errstr(werr));
299 goto done;
302 werr = reg_queryvalue(ctx, key, argv[1], &value);
303 if (!W_ERROR_IS_OK(werr)) {
304 d_fprintf(stderr, _("reg_queryvalue failed: %s\n"),
305 win_errstr(werr));
306 goto done;
309 print_registry_value(value, raw);
311 ret = 0;
313 done:
314 TALLOC_FREE(ctx);
315 return ret;
318 static int net_registry_getvalue(struct net_context *c, int argc,
319 const char **argv)
321 return net_registry_getvalue_internal(c, argc, argv, false);
324 static int net_registry_getvalueraw(struct net_context *c, int argc,
325 const char **argv)
327 return net_registry_getvalue_internal(c, argc, argv, true);
330 static int net_registry_setvalue(struct net_context *c, int argc,
331 const char **argv)
333 WERROR werr;
334 struct registry_value value;
335 struct registry_key *key = NULL;
336 int ret = -1;
337 TALLOC_CTX *ctx = talloc_stackframe();
339 if (argc < 4 || c->display_usage) {
340 d_fprintf(stderr, "%s\n%s",
341 _("Usage:"),
342 _("net rpc registry setvalue <key> <valuename> "
343 "<type> [<val>]+\n"));
344 goto done;
347 if (!strequal(argv[2], "multi_sz") && (argc != 4)) {
348 d_fprintf(stderr, _("Too many args for type %s\n"), argv[2]);
349 goto done;
352 if (strequal(argv[2], "dword")) {
353 value.type = REG_DWORD;
354 value.v.dword = strtoul(argv[3], NULL, 10);
355 } else if (strequal(argv[2], "sz")) {
356 value.type = REG_SZ;
357 value.v.sz.len = strlen(argv[3])+1;
358 value.v.sz.str = CONST_DISCARD(char *, argv[3]);
359 } else if (strequal(argv[2], "multi_sz")) {
360 value.type = REG_MULTI_SZ;
361 value.v.multi_sz.num_strings = argc - 3;
362 value.v.multi_sz.strings = (char **)(argv + 3);
363 } else {
364 d_fprintf(stderr, _("type \"%s\" not implemented\n"), argv[2]);
365 goto done;
368 werr = open_key(ctx, argv[0], REG_KEY_WRITE, &key);
369 if (!W_ERROR_IS_OK(werr)) {
370 d_fprintf(stderr, _("open_key failed: %s\n"), win_errstr(werr));
371 goto done;
374 werr = reg_setvalue(key, argv[1], &value);
375 if (!W_ERROR_IS_OK(werr)) {
376 d_fprintf(stderr, _("reg_setvalue failed: %s\n"),
377 win_errstr(werr));
378 goto done;
381 ret = 0;
383 done:
384 TALLOC_FREE(ctx);
385 return ret;
388 static int net_registry_deletevalue(struct net_context *c, int argc,
389 const char **argv)
391 WERROR werr;
392 struct registry_key *key = NULL;
393 TALLOC_CTX *ctx = talloc_stackframe();
394 int ret = -1;
396 if (argc != 2 || c->display_usage) {
397 d_fprintf(stderr, "%s\n%s",
398 _("Usage:"),
399 _("net rpc registry deletevalue <key> <valuename>\n"));
400 goto done;
403 werr = open_key(ctx, argv[0], REG_KEY_WRITE, &key);
404 if (!W_ERROR_IS_OK(werr)) {
405 d_fprintf(stderr, _("open_key failed: %s\n"), win_errstr(werr));
406 goto done;
409 werr = reg_deletevalue(key, argv[1]);
410 if (!W_ERROR_IS_OK(werr)) {
411 d_fprintf(stderr, _("reg_deletekey failed: %s\n"),
412 win_errstr(werr));
413 goto done;
416 ret = 0;
418 done:
419 TALLOC_FREE(ctx);
420 return ret;
423 static int net_registry_getsd(struct net_context *c, int argc,
424 const char **argv)
426 WERROR werr;
427 int ret = -1;
428 struct registry_key *key = NULL;
429 struct security_descriptor *secdesc = NULL;
430 TALLOC_CTX *ctx = talloc_stackframe();
431 uint32_t access_mask = REG_KEY_READ |
432 SEC_FLAG_MAXIMUM_ALLOWED |
433 SEC_FLAG_SYSTEM_SECURITY;
436 * net_rpc_regsitry uses SEC_FLAG_SYSTEM_SECURITY, but access
437 * is denied with these perms right now...
439 access_mask = REG_KEY_READ;
441 if (argc != 1 || c->display_usage) {
442 d_printf("%s\n%s",
443 _("Usage:"),
444 _("net registry getsd <path>\n"));
445 d_printf("%s\n%s",
446 _("Example:"),
447 _("net registry getsd 'HKLM\\Software\\Samba'\n"));
448 goto done;
450 if (strlen(argv[0]) == 0) {
451 d_fprintf(stderr, "error: zero length key name given\n");
452 goto done;
455 werr = open_key(ctx, argv[0], access_mask, &key);
456 if (!W_ERROR_IS_OK(werr)) {
457 d_fprintf(stderr, _("open_key failed: %s\n"), win_errstr(werr));
458 goto done;
461 werr = reg_getkeysecurity(ctx, key, &secdesc);
462 if (!W_ERROR_IS_OK(werr)) {
463 d_fprintf(stderr, _("reg_getkeysecurity failed: %s\n"),
464 win_errstr(werr));
465 goto done;
468 display_sec_desc(secdesc);
470 ret = 0;
472 done:
473 TALLOC_FREE(ctx);
474 return ret;
477 int net_registry(struct net_context *c, int argc, const char **argv)
479 int ret = -1;
481 struct functable func[] = {
483 "enumerate",
484 net_registry_enumerate,
485 NET_TRANSPORT_LOCAL,
486 N_("Enumerate registry keys and values"),
487 N_("net registry enumerate\n"
488 " Enumerate registry keys and values")
491 "createkey",
492 net_registry_createkey,
493 NET_TRANSPORT_LOCAL,
494 N_("Create a new registry key"),
495 N_("net registry createkey\n"
496 " Create a new registry key")
499 "deletekey",
500 net_registry_deletekey,
501 NET_TRANSPORT_LOCAL,
502 N_("Delete a registry key"),
503 N_("net registry deletekey\n"
504 " Delete a registry key")
507 "getvalue",
508 net_registry_getvalue,
509 NET_TRANSPORT_LOCAL,
510 N_("Print a registry value"),
511 N_("net registry getvalue\n"
512 " Print a registry value")
515 "getvalueraw",
516 net_registry_getvalueraw,
517 NET_TRANSPORT_LOCAL,
518 N_("Print a registry value (raw format)"),
519 N_("net registry getvalueraw\n"
520 " Print a registry value (raw format)")
523 "setvalue",
524 net_registry_setvalue,
525 NET_TRANSPORT_LOCAL,
526 N_("Set a new registry value"),
527 N_("net registry setvalue\n"
528 " Set a new registry value")
531 "deletevalue",
532 net_registry_deletevalue,
533 NET_TRANSPORT_LOCAL,
534 N_("Delete a registry value"),
535 N_("net registry deletevalue\n"
536 " Delete a registry value")
539 "getsd",
540 net_registry_getsd,
541 NET_TRANSPORT_LOCAL,
542 N_("Get security descriptor"),
543 N_("net registry getsd\n"
544 " Get security descriptor")
546 { NULL, NULL, 0, NULL, NULL }
549 if (!W_ERROR_IS_OK(registry_init_basic())) {
550 return -1;
553 ret = net_run_function(c, argc, argv, "net registry", func);
555 return ret;