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