xref: /freebsd/sbin/bectl/bectl.c (revision 78adacd4eab39a3508bd8c65f0aba94fc6b907ce)
1 /*-
2  * SPDX-License-Identifier: BSD-2-Clause-FreeBSD
3  *
4  * Copyright (c) 2017 Kyle J. Kneitinger <kyle@kneit.in>
5  *
6  * Redistribution and use in source and binary forms, with or without
7  * modification, are permitted provided that the following conditions
8  * are met:
9  * 1. Redistributions of source code must retain the above copyright
10  *    notice, this list of conditions and the following disclaimer.
11  * 2. Redistributions in binary form must reproduce the above copyright
12  *    notice, this list of conditions and the following disclaimer in the
13  *    documentation and/or other materials provided with the distribution.
14  *
15  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
16  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
17  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
18  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
19  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
20  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
21  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
22  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
23  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
24  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
25  * SUCH DAMAGE.
26  */
27 
28 #include <sys/cdefs.h>
29 __FBSDID("$FreeBSD$");
30 
31 #include <sys/param.h>
32 #include <sys/mount.h>
33 #include <errno.h>
34 #include <libutil.h>
35 #include <stdbool.h>
36 #include <stdio.h>
37 #include <stdint.h>
38 #include <stdlib.h>
39 #include <string.h>
40 #include <sysexits.h>
41 #include <time.h>
42 #include <unistd.h>
43 
44 #include <be.h>
45 
46 #include "bectl.h"
47 
48 static int bectl_cmd_activate(int argc, char *argv[]);
49 static int bectl_cmd_check(int argc, char *argv[]);
50 static int bectl_cmd_create(int argc, char *argv[]);
51 static int bectl_cmd_destroy(int argc, char *argv[]);
52 static int bectl_cmd_export(int argc, char *argv[]);
53 static int bectl_cmd_import(int argc, char *argv[]);
54 #if SOON
55 static int bectl_cmd_add(int argc, char *argv[]);
56 #endif
57 static int bectl_cmd_mount(int argc, char *argv[]);
58 static int bectl_cmd_rename(int argc, char *argv[]);
59 static int bectl_cmd_unmount(int argc, char *argv[]);
60 
61 libbe_handle_t *be;
62 
63 int aok;
64 
65 int
66 usage(bool explicit)
67 {
68 	FILE *fp;
69 
70 	fp =  explicit ? stdout : stderr;
71 	fprintf(fp, "%s",
72 	    "Usage:\tbectl {-h | -? | subcommand [args...]}\n"
73 #if SOON
74 	    "\tbectl add (path)*\n"
75 #endif
76 	    "\tbectl activate [-t] beName\n"
77 	    "\tbectl activate [-T]\n"
78 	    "\tbectl check\n"
79 	    "\tbectl create [-r] [-e {nonActiveBe | beName@snapshot}] beName\n"
80 	    "\tbectl create [-r] beName@snapshot\n"
81 	    "\tbectl destroy [-Fo] {beName | beName@snapshot}\n"
82 	    "\tbectl export sourceBe\n"
83 	    "\tbectl import targetBe\n"
84 	    "\tbectl jail {-b | -U} [{-o key=value | -u key}]... "
85 	    "{jailID | jailName}\n"
86 	    "\t      bootenv [utility [argument ...]]\n"
87 	    "\tbectl list [-DHas] [{-c property | -C property}]\n"
88 	    "\tbectl mount beName [mountpoint]\n"
89 	    "\tbectl rename origBeName newBeName\n"
90 	    "\tbectl {ujail | unjail} {jailID | jailName} bootenv\n"
91 	    "\tbectl {umount | unmount} [-f] beName\n");
92 
93 	return (explicit ? 0 : EX_USAGE);
94 }
95 
96 
97 /*
98  * Represents a relationship between the command name and the parser action
99  * that handles it.
100  */
101 struct command_map_entry {
102 	const char *command;
103 	int (*fn)(int argc, char *argv[]);
104 	/* True if libbe_print_on_error should be disabled */
105 	bool silent;
106 };
107 
108 static struct command_map_entry command_map[] =
109 {
110 	{ "activate", bectl_cmd_activate,false   },
111 	{ "create",   bectl_cmd_create,  false   },
112 	{ "destroy",  bectl_cmd_destroy, false   },
113 	{ "export",   bectl_cmd_export,  false   },
114 	{ "import",   bectl_cmd_import,  false   },
115 #if SOON
116 	{ "add",      bectl_cmd_add,     false   },
117 #endif
118 	{ "jail",     bectl_cmd_jail,    false   },
119 	{ "list",     bectl_cmd_list,    false   },
120 	{ "mount",    bectl_cmd_mount,   false   },
121 	{ "rename",   bectl_cmd_rename,  false   },
122 	{ "unjail",   bectl_cmd_unjail,  false   },
123 	{ "unmount",  bectl_cmd_unmount, false   },
124 	{ "check",    bectl_cmd_check,   true    },
125 };
126 
127 static struct command_map_entry *
128 get_cmd_info(const char *cmd)
129 {
130 	size_t i;
131 
132 	for (i = 0; i < nitems(command_map); ++i) {
133 		if (strcmp(cmd, command_map[i].command) == 0)
134 			return (&command_map[i]);
135 	}
136 
137 	return (NULL);
138 }
139 
140 
141 static int
142 bectl_cmd_activate(int argc, char *argv[])
143 {
144 	int err, opt;
145 	bool temp, reset;
146 
147 	temp = false;
148 	reset = false;
149 	while ((opt = getopt(argc, argv, "tT")) != -1) {
150 		switch (opt) {
151 		case 't':
152 			if (reset)
153 				return (usage(false));
154 			temp = true;
155 			break;
156 		case 'T':
157 			if (temp)
158 				return (usage(false));
159 			reset = true;
160 			break;
161 		default:
162 			fprintf(stderr, "bectl activate: unknown option '-%c'\n",
163 			    optopt);
164 			return (usage(false));
165 		}
166 	}
167 
168 	argc -= optind;
169 	argv += optind;
170 
171 	if (argc != 1 && (!reset || argc != 0)) {
172 		fprintf(stderr, "bectl activate: wrong number of arguments\n");
173 		return (usage(false));
174 	}
175 
176 	if (reset) {
177 		if ((err = be_deactivate(be, NULL, reset)) == 0)
178 			printf("Temporary activation removed\n");
179 		else
180 			printf("Failed to remove temporary activation\n");
181 		return (err);
182 	}
183 
184 	/* activate logic goes here */
185 	if ((err = be_activate(be, argv[0], temp)) != 0)
186 		/* XXX TODO: more specific error msg based on err */
187 		printf("Did not successfully activate boot environment %s\n",
188 		    argv[0]);
189 	else
190 		printf("Successfully activated boot environment %s\n", argv[0]);
191 
192 	if (temp)
193 		printf("for next boot\n");
194 
195 	return (err);
196 }
197 
198 
199 /*
200  * TODO: when only one arg is given, and it contains an "@" the this should
201  * create that snapshot
202  */
203 static int
204 bectl_cmd_create(int argc, char *argv[])
205 {
206 	char snapshot[BE_MAXPATHLEN];
207 	char *atpos, *bootenv, *snapname;
208 	int err, opt;
209 	bool recursive;
210 
211 	snapname = NULL;
212 	recursive = false;
213 	while ((opt = getopt(argc, argv, "e:r")) != -1) {
214 		switch (opt) {
215 		case 'e':
216 			snapname = optarg;
217 			break;
218 		case 'r':
219 			recursive = true;
220 			break;
221 		default:
222 			fprintf(stderr, "bectl create: unknown option '-%c'\n",
223 			    optopt);
224 			return (usage(false));
225 		}
226 	}
227 
228 	argc -= optind;
229 	argv += optind;
230 
231 	if (argc != 1) {
232 		fprintf(stderr, "bectl create: wrong number of arguments\n");
233 		return (usage(false));
234 	}
235 
236 	bootenv = *argv;
237 
238 	err = BE_ERR_SUCCESS;
239 	if ((atpos = strchr(bootenv, '@')) != NULL) {
240 		/*
241 		 * This is the "create a snapshot variant". No new boot
242 		 * environment is to be created here.
243 		 */
244 		*atpos++ = '\0';
245 		err = be_snapshot(be, bootenv, atpos, recursive, NULL);
246 	} else {
247 		if (snapname == NULL)
248 			/* Create from currently booted BE */
249 			err = be_snapshot(be, be_active_path(be), NULL,
250 			    recursive, snapshot);
251 		else if (strchr(snapname, '@') != NULL)
252 			/* Create from given snapshot */
253 			strlcpy(snapshot, snapname, sizeof(snapshot));
254 		else
255 			/* Create from given BE */
256 			err = be_snapshot(be, snapname, NULL, recursive,
257 			    snapshot);
258 
259 		if (err == BE_ERR_SUCCESS)
260 			err = be_create_depth(be, bootenv, snapshot,
261 					      recursive == true ? -1 : 0);
262 	}
263 
264 	switch (err) {
265 	case BE_ERR_SUCCESS:
266 		break;
267 	default:
268 		if (atpos != NULL)
269 			fprintf(stderr,
270 			    "Failed to create a snapshot '%s' of '%s'\n",
271 			    atpos, bootenv);
272 		else if (snapname == NULL)
273 			fprintf(stderr,
274 			    "Failed to create bootenv %s\n", bootenv);
275 		else
276 			fprintf(stderr,
277 			    "Failed to create bootenv %s from snapshot %s\n",
278 			    bootenv, snapname);
279 	}
280 
281 	return (err);
282 }
283 
284 
285 static int
286 bectl_cmd_export(int argc, char *argv[])
287 {
288 	char *bootenv;
289 
290 	if (argc == 1) {
291 		fprintf(stderr, "bectl export: missing boot environment name\n");
292 		return (usage(false));
293 	}
294 
295 	if (argc > 2) {
296 		fprintf(stderr, "bectl export: extra arguments provided\n");
297 		return (usage(false));
298 	}
299 
300 	bootenv = argv[1];
301 
302 	if (isatty(STDOUT_FILENO)) {
303 		fprintf(stderr, "bectl export: must redirect output\n");
304 		return (EX_USAGE);
305 	}
306 
307 	be_export(be, bootenv, STDOUT_FILENO);
308 
309 	return (0);
310 }
311 
312 
313 static int
314 bectl_cmd_import(int argc, char *argv[])
315 {
316 	char *bootenv;
317 	int err;
318 
319 	if (argc == 1) {
320 		fprintf(stderr, "bectl import: missing boot environment name\n");
321 		return (usage(false));
322 	}
323 
324 	if (argc > 2) {
325 		fprintf(stderr, "bectl import: extra arguments provided\n");
326 		return (usage(false));
327 	}
328 
329 	bootenv = argv[1];
330 
331 	if (isatty(STDIN_FILENO)) {
332 		fprintf(stderr, "bectl import: input can not be from terminal\n");
333 		return (EX_USAGE);
334 	}
335 
336 	err = be_import(be, bootenv, STDIN_FILENO);
337 
338 	return (err);
339 }
340 
341 #if SOON
342 static int
343 bectl_cmd_add(int argc, char *argv[])
344 {
345 
346 	if (argc < 2) {
347 		fprintf(stderr, "bectl add: must provide at least one path\n");
348 		return (usage(false));
349 	}
350 
351 	for (int i = 1; i < argc; ++i) {
352 		printf("arg %d: %s\n", i, argv[i]);
353 		/* XXX TODO catch err */
354 		be_add_child(be, argv[i], true);
355 	}
356 
357 	return (0);
358 }
359 #endif
360 
361 static int
362 bectl_cmd_destroy(int argc, char *argv[])
363 {
364 	nvlist_t *props;
365 	char *origin, *target, targetds[BE_MAXPATHLEN];
366 	int err, flags, opt;
367 
368 	flags = 0;
369 	while ((opt = getopt(argc, argv, "Fo")) != -1) {
370 		switch (opt) {
371 		case 'F':
372 			flags |= BE_DESTROY_FORCE;
373 			break;
374 		case 'o':
375 			flags |= BE_DESTROY_ORIGIN;
376 			break;
377 		default:
378 			fprintf(stderr, "bectl destroy: unknown option '-%c'\n",
379 			    optopt);
380 			return (usage(false));
381 		}
382 	}
383 
384 	argc -= optind;
385 	argv += optind;
386 
387 	if (argc != 1) {
388 		fprintf(stderr, "bectl destroy: wrong number of arguments\n");
389 		return (usage(false));
390 	}
391 
392 	target = argv[0];
393 
394 	/* We'll emit a notice if there's an origin to be cleaned up */
395 	if ((flags & BE_DESTROY_ORIGIN) == 0 && strchr(target, '@') == NULL) {
396 		flags |= BE_DESTROY_AUTOORIGIN;
397 		if (be_root_concat(be, target, targetds) != 0)
398 			goto destroy;
399 		if (be_prop_list_alloc(&props) != 0)
400 			goto destroy;
401 		if (be_get_dataset_props(be, targetds, props) != 0) {
402 			be_prop_list_free(props);
403 			goto destroy;
404 		}
405 		if (nvlist_lookup_string(props, "origin", &origin) == 0 &&
406 		    !be_is_auto_snapshot_name(be, origin))
407 			fprintf(stderr, "bectl destroy: leaving origin '%s' intact\n",
408 			    origin);
409 		be_prop_list_free(props);
410 	}
411 
412 destroy:
413 	err = be_destroy(be, target, flags);
414 
415 	return (err);
416 }
417 
418 static int
419 bectl_cmd_mount(int argc, char *argv[])
420 {
421 	char result_loc[BE_MAXPATHLEN];
422 	char *bootenv, *mountpoint;
423 	int err, mntflags;
424 
425 	/* XXX TODO: Allow shallow */
426 	mntflags = BE_MNT_DEEP;
427 	if (argc < 2) {
428 		fprintf(stderr, "bectl mount: missing argument(s)\n");
429 		return (usage(false));
430 	}
431 
432 	if (argc > 3) {
433 		fprintf(stderr, "bectl mount: too many arguments\n");
434 		return (usage(false));
435 	}
436 
437 	bootenv = argv[1];
438 	mountpoint = ((argc == 3) ? argv[2] : NULL);
439 
440 	err = be_mount(be, bootenv, mountpoint, mntflags, result_loc);
441 
442 	switch (err) {
443 	case BE_ERR_SUCCESS:
444 		printf("Successfully mounted %s at %s\n", bootenv, result_loc);
445 		break;
446 	default:
447 		fprintf(stderr,
448 		    (argc == 3) ? "Failed to mount bootenv %s at %s\n" :
449 		    "Failed to mount bootenv %s at temporary path %s\n",
450 		    bootenv, mountpoint);
451 	}
452 
453 	return (err);
454 }
455 
456 
457 static int
458 bectl_cmd_rename(int argc, char *argv[])
459 {
460 	char *dest, *src;
461 	int err;
462 
463 	if (argc < 3) {
464 		fprintf(stderr, "bectl rename: missing argument\n");
465 		return (usage(false));
466 	}
467 
468 	if (argc > 3) {
469 		fprintf(stderr, "bectl rename: too many arguments\n");
470 		return (usage(false));
471 	}
472 
473 	src = argv[1];
474 	dest = argv[2];
475 
476 	err = be_rename(be, src, dest);
477 
478 	switch (err) {
479 	case BE_ERR_SUCCESS:
480 		break;
481 	default:
482 		fprintf(stderr, "Failed to rename bootenv %s to %s\n",
483 		    src, dest);
484 	}
485 
486 	return (0);
487 }
488 
489 static int
490 bectl_cmd_unmount(int argc, char *argv[])
491 {
492 	char *bootenv, *cmd;
493 	int err, flags, opt;
494 
495 	/* Store alias used */
496 	cmd = argv[0];
497 
498 	flags = 0;
499 	while ((opt = getopt(argc, argv, "f")) != -1) {
500 		switch (opt) {
501 		case 'f':
502 			flags |= BE_MNT_FORCE;
503 			break;
504 		default:
505 			fprintf(stderr, "bectl %s: unknown option '-%c'\n",
506 			    cmd, optopt);
507 			return (usage(false));
508 		}
509 	}
510 
511 	argc -= optind;
512 	argv += optind;
513 
514 	if (argc != 1) {
515 		fprintf(stderr, "bectl %s: wrong number of arguments\n", cmd);
516 		return (usage(false));
517 	}
518 
519 	bootenv = argv[0];
520 
521 	err = be_unmount(be, bootenv, flags);
522 
523 	switch (err) {
524 	case BE_ERR_SUCCESS:
525 		break;
526 	default:
527 		fprintf(stderr, "Failed to unmount bootenv %s\n", bootenv);
528 	}
529 
530 	return (err);
531 }
532 
533 static int
534 bectl_cmd_check(int argc, char *argv[] __unused)
535 {
536 
537 	/* The command is left as argv[0] */
538 	if (argc != 1) {
539 		fprintf(stderr, "bectl check: wrong number of arguments\n");
540 		return (usage(false));
541 	}
542 
543 	return (0);
544 }
545 
546 int
547 main(int argc, char *argv[])
548 {
549 	struct command_map_entry *cmd;
550 	const char *command;
551 	char *root;
552 	int rc;
553 
554 	cmd = NULL;
555 	root = NULL;
556 	if (argc < 2)
557 		return (usage(false));
558 
559 	if (strcmp(argv[1], "-r") == 0) {
560 		if (argc < 4)
561 			return (usage(false));
562 		root = strdup(argv[2]);
563 		command = argv[3];
564 		argc -= 3;
565 		argv += 3;
566 	} else {
567 		command = argv[1];
568 		argc -= 1;
569 		argv += 1;
570 	}
571 
572 	/* Handle command aliases */
573 	if (strcmp(command, "umount") == 0)
574 		command = "unmount";
575 
576 	if (strcmp(command, "ujail") == 0)
577 		command = "unjail";
578 
579 	if ((strcmp(command, "-?") == 0) || (strcmp(command, "-h") == 0))
580 		return (usage(true));
581 
582 	if ((cmd = get_cmd_info(command)) == NULL) {
583 		fprintf(stderr, "Unknown command: %s\n", command);
584 		return (usage(false));
585 	}
586 
587 	if ((be = libbe_init(root)) == NULL) {
588 		if (!cmd->silent)
589 			fprintf(stderr, "libbe_init(\"%s\") failed.\n",
590 			    root != NULL ? root : "");
591 		return (-1);
592 	}
593 
594 	libbe_print_on_error(be, !cmd->silent);
595 
596 	rc = cmd->fn(argc, argv);
597 
598 	libbe_close(be);
599 	return (rc);
600 }
601