]> CyberLeo.Net >> Repos - FreeBSD/releng/10.0.git/blob - usr.sbin/sysrc/sysrc
- Copy stable/10 (r259064) to releng/10.0 as part of the
[FreeBSD/releng/10.0.git] / usr.sbin / sysrc / sysrc
1 #!/bin/sh
2 #-
3 # Copyright (c) 2010-2013 Devin Teske
4 # All rights reserved.
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 AUTHOR 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 AUTHOR 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 # $FreeBSD$
28 #
29 ############################################################ INCLUDES
30
31 BSDCFG_SHARE="/usr/share/bsdconfig"
32 [ "$_COMMON_SUBR" ] || . $BSDCFG_SHARE/common.subr || exit 1
33 [ "$_SYSRC_SUBR"  ] || f_include $BSDCFG_SHARE/sysrc.subr
34
35 ############################################################ GLOBALS
36
37 #
38 # Options
39 #
40 DELETE=
41 DESCRIBE=
42 IGNORE_UNKNOWNS=
43 JAIL=
44 QUIET=
45 ROOTDIR=
46 SHOW_ALL=
47 SHOW_EQUALS=
48 SHOW_FILE=
49 SHOW_NAME=1
50 SHOW_VALUE=1
51 SYSRC_VERBOSE=
52
53 ############################################################ FUNCTIONS
54
55 # die [ $fmt [ $opts ... ]]
56 #
57 # Optionally print a message to stderr before exiting with failure status.
58 #
59 die()
60 {
61         local fmt="$1"
62         [ $# -gt 0 ] && shift 1
63         [  "$fmt"  ] && f_err "$fmt\n" "$@"
64
65         exit $FAILURE
66 }
67
68 # usage
69 #
70 # Prints a short syntax statement and exits.
71 #
72 usage()
73 {
74         f_err "Usage: %s [OPTIONS] name[=value] ...\n" "$pgm"
75         f_err "Try \`%s --help' for more information.\n" "$pgm"
76         die
77 }
78
79 # help
80 #
81 # Prints a full syntax statement and exits.
82 #
83 help()
84 {
85         local optfmt="\t%-11s%s\n"
86         local envfmt="\t%-17s%s\n"
87
88         f_err "Usage: %s [OPTIONS] name[=value] ...\n" "$pgm"
89
90         f_err "OPTIONS:\n"
91         f_err "$optfmt" "-a" \
92               "Dump a list of all non-default configuration variables."
93         f_err "$optfmt" "-A" \
94               "Dump a list of all configuration variables (incl. defaults)."
95         f_err "$optfmt" "-d" \
96               "Print a description of the given variable."
97         f_err "$optfmt" "-D" \
98               "Show default value(s) only (this is the same as setting"
99         f_err "$optfmt" "" \
100               "RC_CONFS to NULL or passing \`-f' with a NULL file-argument)."
101         f_err "$optfmt" "-e" \
102               "Print query results as \`var=value' (useful for producing"
103         f_err "$optfmt" "" \
104               "output to be fed back in). Ignored if \`-n' is specified."
105         f_err "$optfmt" "-f file" \
106               "Operate on the specified file(s) instead of rc_conf_files."
107         f_err "$optfmt" "" \
108               "Can be specified multiple times for additional files."
109         f_err "$optfmt" "-F" \
110               "Show only the last rc.conf(5) file each directive is in."
111         f_err "$optfmt" "-h" \
112               "Print a short usage statement to stderr and exit."
113         f_err "$optfmt" "--help" \
114               "Print this message to stderr and exit."
115         f_err "$optfmt" "-i" \
116               "Ignore unknown variables."
117         f_err "$optfmt" "-j jail" \
118               "The jid or name of the jail to operate within (overrides"
119         f_err "$optfmt" "" \
120               "\`-R dir'; requires jexec(8))."
121         f_err "$optfmt" "-n" \
122               "Show only variable values, not their names."
123         f_err "$optfmt" "-N" \
124               "Show only variable names, not their values."
125         f_err "$optfmt" "-q" \
126               "Quiet. Ignore previous \`-v' and/or SYSRC_VERBOSE."
127         f_err "$optfmt" "-R dir" \
128               "Operate within the root directory \`dir' rather than \`/'."
129         f_err "$optfmt" "-v" \
130               "Verbose. Print the pathname of the specific rc.conf(5)"
131         f_err "$optfmt" "" \
132               "file where the directive was found."
133         f_err "$optfmt" "-x" \
134               "Remove variable(s) from specified file(s)."
135         f_err "\n"
136
137         f_err "ENVIRONMENT:\n"
138         f_err "$envfmt" "RC_CONFS" \
139               "Override default rc_conf_files (even if set to NULL)."
140         f_err "$envfmt" "RC_DEFAULTS" \
141               "Location of \`/etc/defaults/rc.conf' file."
142         f_err "$envfmt" "SYSRC_VERBOSE" \
143               "Default verbosity. Set to non-NULL to enable."
144
145         die
146 }
147
148 # jail_depend
149 #
150 # Dump dependencies such as language-file variables and include files to stdout
151 # to be piped-into sh(1) running via jexec(8)/chroot(8). As a security measure,
152 # this prevents existing language files and library files from being loaded in
153 # the jail. This also relaxes the requirement to have these files in every jail
154 # before sysrc can be used on said jail.
155 #
156 jail_depend()
157 {
158         #
159         # Indicate that we are jailed
160         #
161         echo export _SYSRC_JAILED=1
162
163         #
164         # Print i18n language variables (their current values are sanitized
165         # and re-printed for interpretation so that the i18n language files
166         # do not need to exist within the jail).
167         #
168         local var val
169         for var in \
170                 msg_cannot_create_permission_denied \
171                 msg_permission_denied \
172                 msg_previous_syntax_errors \
173         ; do
174                 val=$( eval echo \"\$$var\" |
175                         awk '{ gsub(/'\''/, "'\''\\'\'\''"); print }' )
176                 echo $var="'$val'"
177         done
178
179         #
180         # Print include dependencies
181         #
182         cat $BSDCFG_SHARE/common.subr
183         cat $BSDCFG_SHARE/sysrc.subr
184 }
185
186 ############################################################ MAIN SOURCE
187
188 #
189 # Perform sanity checks
190 #
191 [ $# -gt 0 ] || usage
192
193 #
194 # Check for `--help' command-line option
195 #
196 ( # Operate in sub-shell to protect $@ in parent
197         while [ $# -gt 0 ]; do
198                 case "$1" in
199                 --help) exit 1;;
200                 -[fRj]) # These flags take an argument
201                         shift 1;;
202                 esac
203                 shift 1
204         done
205         exit 0
206 ) || help
207
208 #
209 # Process command-line flags
210 #
211 while getopts aAdDef:Fhij:nNqR:vxX flag; do
212         case "$flag" in
213         a) SHOW_ALL=${SHOW_ALL:-1};;
214         A) SHOW_ALL=2;;
215         d) DESCRIBE=1;;
216         D) RC_CONFS=;;
217         e) SHOW_EQUALS=1;;
218         f) RC_CONFS="$RC_CONFS${RC_CONFS:+ }$OPTARG";;
219         F) SHOW_FILE=1;;
220         h) usage;;
221         i) IGNORE_UNKNOWNS=1;;
222         j) [ "$OPTARG" ] || die \
223                 "%s: Missing or null argument to \`-j' flag" "$pgm"
224            JAIL="$OPTARG";;
225         n) SHOW_NAME=;;
226         N) SHOW_VALUE=;;
227         q) QUIET=1 SYSRC_VERBOSE=;;
228         R) [ "$OPTARG" ] || die \
229                 "%s: Missing or null argument to \`-R' flag" "$pgm"
230            ROOTDIR="$OPTARG";;
231         v) SYSRC_VERBOSE=1 QUIET=;;
232         x) DELETE=${DELETE:-1};;
233         X) DELETE=2;;
234         \?) usage;;
235         esac
236 done
237 shift $(( $OPTIND - 1 ))
238
239 #
240 # [More] Sanity checks (e.g., "sysrc --")
241 #
242 [ $# -eq 0 -a ! "$SHOW_ALL" ] && usage
243
244 #
245 # Taint-check all rc.conf(5) files
246 #
247 errmsg="$pgm: Exiting due to previous syntax errors"
248 if [ "${RC_CONFS+set}" ]; then
249         ( for i in $RC_CONFS; do
250                 [ -e "$i" ] || continue
251                 /bin/sh -n "$i" || exit $FAILURE
252           done
253           exit $SUCCESS
254         ) || die "$errmsg"
255 else
256         /bin/sh -n "$RC_DEFAULTS" || die "$errmsg"
257         ( . "$RC_DEFAULTS"
258           for i in $rc_conf_files; do
259                 [ -e "$i" ] || continue
260                 /bin/sh -n "$i" || exit $FAILURE
261           done
262           exit $SUCCESS
263         ) || die "$errmsg"
264 fi
265
266 #
267 # Process `-x' (and secret `-X') command-line options
268 #
269 errmsg="$pgm: \`-x' option incompatible with \`-a'/\`-A' options"
270 errmsg="$errmsg (use \`-X' to override)"
271 if [ "$DELETE" -a "$SHOW_ALL" ]; then
272         [ "$DELETE" = "2" ] || die "$errmsg"
273 fi
274
275 #
276 # Process `-e', `-n', and `-N' command-line options
277 #
278 SEP=': '
279 [ "$SHOW_EQUALS" ] && SEP='="'
280 [ "$SHOW_NAME" ] || SHOW_EQUALS=
281 [ "$SYSRC_VERBOSE" = "0" ] && SYSRC_VERBOSE=
282 if [ ! "$SHOW_VALUE" ]; then
283         SHOW_NAME=1
284         SHOW_EQUALS=
285 fi
286
287 #
288 # Process `-j jail' and `-R dir' command-line options
289 #
290 if [ "$JAIL" -o "$ROOTDIR" ]; then
291         #
292         # Reconstruct the arguments that we want to carry-over
293         #
294         args="
295                 ${SYSRC_VERBOSE:+-v}
296                 ${QUIET:+-q}
297                 $( [ "$DELETE" = "1" ] && echo \ -x )
298                 $( [ "$DELETE" = "2" ] && echo \ -X )
299                 $( [ "$SHOW_ALL" = "1" ] && echo \ -a )
300                 $( [ "$SHOW_ALL" = "2" ] && echo \ -A )
301                 ${DESCRIBE:+-d}
302                 ${SHOW_EQUALS:+-e}
303                 ${IGNORE_UNKNOWNS:+-i}
304                 $( [ "$SHOW_NAME"  ] || echo \ -n )
305                 $( [ "$SHOW_VALUE" ] || echo \ -N )
306                 $( [ "$SHOW_FILE"  ] && echo \ -F )
307         "
308         if [ "${RC_CONFS+set}" ]; then
309                 args="$args -f '$RC_CONFS'"
310         fi
311         for arg in "$@"; do
312                 args="$args '$arg'"
313         done
314
315         #
316         # If both are supplied, `-j jail' supercedes `-R dir'
317         #
318         if [ "$JAIL" ]; then
319                 #
320                 # Re-execute ourselves with sh(1) via jexec(8)
321                 #
322                 ( echo set -- $args
323                   jail_depend
324                   cat $0
325                 ) | env - RC_DEFAULTS="$RC_DEFAULTS" \
326                         /usr/sbin/jexec "$JAIL" /bin/sh
327                 exit $?
328         elif [ "$ROOTDIR" ]; then
329                 #
330                 # Make sure that the root directory specified is not to any
331                 # running jails.
332                 #
333                 # NOTE: To maintain backward compatibility with older jails on
334                 # older systems, we will not perform this check if either the
335                 # jls(1) or jexec(8) utilities are missing.
336                 #
337                 if f_have jexec && f_have jls; then
338                         jid="`jls jid path | \
339                         (
340                                 while read JID JROOT; do
341                                         [ "$JROOT" = "$ROOTDIR" ] || continue
342                                         echo $JID
343                                 done
344                         )`"
345
346                         #
347                         # If multiple running jails match the specified root
348                         # directory, exit with error.
349                         #
350                         if [ "$jid" -a "${jid%[$IFS]*}" != "$jid" ]; then
351                                 die "%s: %s: %s" "$pgm" "$ROOTDIR" \
352                                     "$( echo "Multiple jails claim this" \
353                                              "directory as their root." \
354                                              "(use \`-j jail' instead)" )"
355                         fi
356
357                         #
358                         # If only a single running jail matches the specified
359                         # root directory, implicitly use `-j jail'.
360                         #
361                         if [ "$jid" ]; then
362                                 #
363                                 # Re-execute outselves with sh(1) via jexec(8)
364                                 #
365                                 ( echo set -- $args
366                                   jail_depend
367                                   cat $0
368                                 ) | env - RC_DEFAULTS="$RC_DEFAULTS" \
369                                         /usr/sbin/jexec "$jid" /bin/sh
370                                 exit $?
371                         fi
372
373                         # Otherwise, fall through and allow chroot(8)
374                 fi
375
376                 #
377                 # Re-execute ourselves with sh(1) via chroot(8)
378                 #
379                 ( echo set -- $args
380                   jail_depend
381                   cat $0
382                 ) | env - RC_DEFAULTS="$RC_DEFAULTS" \
383                         /usr/sbin/chroot "$ROOTDIR" /bin/sh
384                 exit $?
385         fi
386 fi
387
388 #
389 # Process `-a' or `-A' command-line options
390 #
391 if [ "$SHOW_ALL" ]; then
392         #
393         # Get a list of variables that are currently set in the rc.conf(5)
394         # files (included `/etc/defaults/rc.conf') by performing a call to
395         # source_rc_confs() in a clean environment.
396         #
397         ( # Operate in a sub-shell to protect the parent environment
398                 #
399                 # Set which variables we want to preserve in the environment.
400                 # Append the pipe-character (|) to the list of internal field
401                 # separation (IFS) characters, allowing us to use the below
402                 # list both as an extended grep (-E) pattern and argument list
403                 # (required to first get f_clean_env() to preserve these in the
404                 # environment and then later to prune them from the list of
405                 # variables produced by set(1)).
406                 #
407                 IFS="$IFS|"
408                 EXCEPT="IFS|EXCEPT|PATH|RC_DEFAULTS|OPTIND|DESCRIBE|SEP"
409                 EXCEPT="$EXCEPT|DELETE|SHOW_ALL|SHOW_EQUALS|SHOW_NAME"
410                 EXCEPT="$EXCEPT|SHOW_VALUE|SHOW_FILE|SYSRC_VERBOSE|RC_CONFS"
411                 EXCEPT="$EXCEPT|pgm|SUCCESS|FAILURE"
412                 EXCEPT="$EXCEPT|f_sysrc_desc_awk|f_sysrc_delete_awk"
413
414                 #
415                 # Clean the environment (except for our required variables)
416                 # and then source the required files.
417                 #
418                 f_clean_env --except $EXCEPT
419                 if [ -f "$RC_DEFAULTS" -a -r "$RC_DEFAULTS" ]; then
420                         . "$RC_DEFAULTS"
421
422                         #
423                         # If passed `-a' (rather than `-A'), re-purge the
424                         # environment, removing the rc.conf(5) defaults.
425                         #
426                         [ "$SHOW_ALL" = "1" ] \
427                                 && f_clean_env --except rc_conf_files $EXCEPT
428
429                         #
430                         # If `-f file' was passed, set $rc_conf_files to an
431                         # explicit value, modifying the default behavior of
432                         # source_rc_confs().
433                         #
434                         [ "${RC_CONFS+set}" ] && rc_conf_files="$RC_CONFS"
435
436                         source_rc_confs
437
438                         #
439                         # If passed `-a' (rather than `-A'), remove
440                         # `rc_conf_files' unless it was defined somewhere
441                         # other than rc.conf(5) defaults.
442                         #
443                         [ "$SHOW_ALL" = "1" -a \
444                           "$( f_sysrc_find rc_conf_files )" = "$RC_DEFAULTS" \
445                         ] \
446                         && unset rc_conf_files
447                 fi
448
449                 for NAME in $( set |
450                         awk -F= '/^[[:alpha:]_][[:alnum:]_]*=/ {print $1}' |
451                         grep -Ev "^($EXCEPT)$"
452                 ); do
453                         #
454                         # If enabled, describe rather than expand value
455                         #
456                         if [ "$DESCRIBE" ]; then
457                                 echo "$NAME: $( f_sysrc_desc "$NAME" )"
458                                 continue
459                         fi
460
461                         #
462                         # If `-F' is passed, find it and move on
463                         #
464                         if [ "$SHOW_FILE" ]; then
465                                 [ "$SHOW_NAME" ] && echo -n "$NAME: "
466                                 f_sysrc_find "$NAME"
467                                 continue
468                         fi
469
470                         #
471                         # If `-X' is passed, delete the variables
472                         #
473                         if [ "$DELETE" = "2" ]; then
474                                 f_sysrc_delete "$NAME"
475                                 continue
476                         fi
477
478                         [ "$SYSRC_VERBOSE" ] && \
479                                 echo -n "$( f_sysrc_find "$NAME" ): "
480
481                         #
482                         # If `-N' is passed, simplify the output
483                         #
484                         if [ ! "$SHOW_VALUE" ]; then
485                                 echo "$NAME"
486                                 continue
487                         fi
488
489                         echo "${SHOW_NAME:+$NAME$SEP}$(
490                               f_sysrc_get "$NAME" )${SHOW_EQUALS:+\"}"
491
492                 done
493         )
494
495         #
496         # Ignore the remainder of positional arguments.
497         #
498         exit $SUCCESS
499 fi
500
501 #
502 # Process command-line arguments
503 #
504 while [ $# -gt 0 ]; do
505         NAME="${1%%=*}"
506
507         [ "$DESCRIBE" ] && \
508                 echo "$NAME: $( f_sysrc_desc "$NAME" )"
509
510         case "$1" in
511         *=*)
512                 #
513                 # Like sysctl(8), if both `-d' AND "name=value" is passed,
514                 # first describe, then attempt to set
515                 #
516
517                 if [ "$SYSRC_VERBOSE" ]; then
518                         file=$( f_sysrc_find "$NAME" )
519                         [ "$file" = "$RC_DEFAULTS" -o ! "$file" ] && \
520                                 file=$( f_sysrc_get 'rc_conf_files%%[$IFS]*' )
521                         echo -n "$file: "
522                 fi
523
524                 #
525                 # If `-x' or `-X' is passed, delete the variable and ignore the
526                 # desire to set some value
527                 #
528                 if [ "$DELETE" ]; then
529                         f_sysrc_delete "$NAME"
530                         shift 1
531                         continue
532                 fi
533
534                 #
535                 # If `-N' is passed, simplify the output
536                 #
537                 if [ ! "$SHOW_VALUE" ]; then
538                         echo "$NAME"
539                         f_sysrc_set "$NAME" "${1#*}"
540                 else
541                         if [ "$SHOW_FILE" ]; then
542                                 before=$( f_sysrc_find "$NAME" )
543                         else
544                                 before=$( f_sysrc_get "$NAME" )
545                         fi
546                         if f_sysrc_set "$NAME" "${1#*=}"; then
547                                 if [ "$SHOW_FILE" ]; then
548                                         after=$( f_sysrc_find "$NAME" )
549                                         echo -n "${SHOW_NAME:+$NAME$SEP}"
550                                         echo -n "$before${SHOW_EQUALS:+\"}"
551                                         echo " -> $after"
552                                 else
553                                         after=$( f_sysrc_get "$NAME" )
554                                         echo -n "${SHOW_NAME:+$NAME$SEP}"
555                                         echo "$before -> $after"
556                                 fi
557                         fi
558                 fi
559                 ;;
560         *)
561                 if ! IGNORED="$( f_sysrc_get "$NAME?" )"; then
562                         [ "$IGNORE_UNKNOWNS" ] \
563                                 || echo "$pgm: unknown variable '$NAME'"
564                         shift 1
565                         continue
566                 fi
567
568                 #
569                 # Like sysctl(8), when `-d' is passed,
570                 # desribe it rather than expanding it
571                 #
572
573                 if [ "$DESCRIBE" ]; then
574                         shift 1
575                         continue
576                 fi
577
578                 #
579                 # If `-x' or `-X' is passed, delete the variable
580                 #
581                 if [ "$DELETE" ]; then
582                         f_sysrc_delete "$NAME"
583                         shift 1
584                         continue
585                 fi
586
587                 #
588                 # If `-F' is passed, find it and move on
589                 #
590                 if [ "$SHOW_FILE" ]; then
591                         [ "$SHOW_NAME" ] && echo -n "$NAME: "
592                         f_sysrc_find "$NAME"
593                         shift 1
594                         continue
595                 fi
596
597                 [ "$SYSRC_VERBOSE" ] && \
598                         echo -n "$( f_sysrc_find "$NAME" ): "
599
600                 #
601                 # If `-N' is passed, simplify the output
602                 #
603                 if [ ! "$SHOW_VALUE" ]; then
604                         echo "$NAME"
605                 else
606                         echo "${SHOW_NAME:+$NAME$SEP}$(
607                               f_sysrc_get "$NAME" )${SHOW_EQUALS:+\"}"
608                 fi
609         esac
610         shift 1
611 done