NetBSD/usr.sbin/sysinst/configmenu.c
dholland 50dbef1a09 Move sysinst sources to usr.sbin.
This commit only physically moves the sources - there are no other
changes, to maximize the probability that this will be treated as a
rename if we ever do manage to migrate away from CVS.

Moving sysinst has been discussed on and off for years and has two
goals: making it easier to work on sysinst, and also making sysinst
available on running systems for use installing chroots and VM images
and other such things. None of the latter is possible yet, but as they
say, one thing at a time.

Doing this now was approved in an impromptu fashion by mrg, riz,
riastradh, me, and groo.
2014-07-26 19:30:39 +00:00

460 lines
11 KiB
C

/* $NetBSD: configmenu.c,v 1.1 2014/07/26 19:30:44 dholland Exp $ */
/*-
* Copyright (c) 2012 The NetBSD Foundation, Inc.
* All rights reserved.
*
* This code is derived from software contributed to The NetBSD Foundation
* by Jeffrey C. Rizzo
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*/
/* configmenu.c -- post-installation system configuration menu. */
#include <stdio.h>
#include <curses.h>
#include <unistd.h>
#include "defs.h"
#include "msg_defs.h"
#include "menu_defs.h"
static int set_network(struct menudesc*, void *);
static int set_timezone_menu(struct menudesc *, void *);
static int set_root_shell(struct menudesc *, void *);
static int change_root_password(struct menudesc *, void *);
static int add_new_user(struct menudesc *, void *);
static int set_binpkg(struct menudesc *, void *);
static int set_pkgsrc(struct menudesc *, void *);
static void config_list_init(void);
static void get_rootsh(void);
static int toggle_rcvar(struct menudesc *, void *);
static void configmenu_hdr(struct menudesc *, void *);
static int check_root_password(void);
char pkgpath[STRSIZE];
char pkgsrcpath[STRSIZE];
extern const char *tz_default;
enum {
CONFIGOPT_NETCONF,
CONFIGOPT_TZ,
CONFIGOPT_ROOTSH,
CONFIGOPT_ROOTPW,
CONFIGOPT_BINPKG,
CONFIGOPT_PKGSRC,
CONFIGOPT_SSHD,
CONFIGOPT_NTPD,
CONFIGOPT_NTPDATE,
CONFIGOPT_MDNSD,
CONFIGOPT_ADDUSER,
CONFIGOPT_LAST
};
typedef struct configinfo {
const char *optname;
uint opt;
const char *rcvar;
int (*action)(struct menudesc *, void *);
const char *setting;
} configinfo;
configinfo config_list[] = {
{MSG_Configure_network, CONFIGOPT_NETCONF, NULL, set_network, MSG_configure},
{MSG_timezone, CONFIGOPT_TZ, NULL, set_timezone_menu, NULL},
{MSG_Root_shell, CONFIGOPT_ROOTSH, NULL, set_root_shell, NULL},
{MSG_change_rootpw, CONFIGOPT_ROOTPW, NULL, change_root_password, MSG_change},
{MSG_enable_binpkg, CONFIGOPT_BINPKG, NULL, set_binpkg, MSG_configure},
{MSG_get_pkgsrc, CONFIGOPT_PKGSRC, NULL, set_pkgsrc, MSG_install},
{MSG_enable_sshd, CONFIGOPT_SSHD, "sshd", toggle_rcvar, NULL},
{MSG_enable_ntpd, CONFIGOPT_NTPD, "ntpd", toggle_rcvar, NULL},
{MSG_run_ntpdate, CONFIGOPT_NTPDATE, "ntpdate", toggle_rcvar, NULL},
{MSG_enable_mdnsd, CONFIGOPT_MDNSD, "mdnsd", toggle_rcvar, NULL},
{MSG_add_a_user, CONFIGOPT_ADDUSER, NULL, add_new_user, ""},
{NULL, CONFIGOPT_LAST, NULL, NULL, NULL}
};
static void
config_list_init(void)
{
int i;
for (i=0; i < CONFIGOPT_LAST; i++) {
switch (i) {
case CONFIGOPT_TZ:
get_tz_default();
config_list[CONFIGOPT_TZ].setting = tz_default;
break;
case CONFIGOPT_ROOTSH:
get_rootsh();
break;
case CONFIGOPT_ROOTPW:
if (check_root_password())
config_list[i].setting = MSG_password_set;
else
config_list[i].setting = MSG_empty;
break;
default:
if (config_list[i].rcvar != NULL) {
if (check_rcvar(config_list[i].rcvar))
config_list[i].setting = MSG_YES;
else
config_list[i].setting = MSG_NO;
}
break;
}
}
}
static void
get_rootsh(void)
{
static char *buf = NULL;
if (buf != NULL)
free(buf);
if (target_already_root())
collect(T_OUTPUT, &buf,
"/usr/bin/awk -F: '$1==\"root\" { print $NF; exit }'"
" /etc/passwd");
else
collect(T_OUTPUT, &buf,
"chroot %s /usr/bin/awk -F: '$1==\"root\" { print $NF; exit }'"
" /etc/passwd",target_prefix());
config_list[CONFIGOPT_ROOTSH].setting = (const char *)buf;
}
static void
set_config(menudesc *menu, int opt, void *arg)
{
configinfo **configp = arg;
configinfo *config = configp[opt];
const char *optname, *setting;
optname = config->optname;
setting = msg_string(config->setting);
wprintw(menu->mw, "%-50s %-10s", msg_string(optname), setting);
}
static int
init_config_menu(configinfo *conf, menu_ent *me, configinfo **ce)
{
int opt;
int configopts;
for (configopts = 0; ; conf++) {
opt = conf->opt;
if (opt == CONFIGOPT_LAST)
break;
*ce = conf;
me->opt_menu = OPT_NOMENU;
me->opt_flags = 0;
me->opt_name = NULL; /* NULL so set_config will draw */
me->opt_action = conf->action;
configopts++;
ce++;
me++;
}
return configopts;
}
static int
/*ARGSUSED*/
set_timezone_menu(struct menudesc *menu, void *arg)
{
configinfo **confp = arg;
set_timezone();
get_tz_default();
confp[menu->cursel]->setting = tz_default;
return 0;
}
static int
set_root_shell(struct menudesc *menu, void *arg)
{
configinfo **confp = arg;
process_menu(MENU_rootsh, &confp[menu->cursel]->setting);
run_program(RUN_PROGRESS | RUN_CHROOT,
"chpass -s %s root", confp[menu->cursel]->setting);
return 0;
}
static int
set_network(struct menudesc *menu, void *arg)
{
network_up = 0;
if (config_network())
mnt_net_config();
return 0;
}
static int
check_root_password(void)
{
char *buf;
int rval;
if (target_already_root())
collect(T_OUTPUT, &buf, "getent passwd root | cut -d: -f2");
else
collect(T_OUTPUT, &buf, "chroot %s getent passwd root | "
"chroot %s cut -d: -f2",
target_prefix(), target_prefix());
if (logfp)
fprintf(logfp,"buf %s strlen(buf) %zu\n", buf, strlen(buf));
if (strlen(buf) <= 1) /* newline */
rval = 0;
else
rval = 1;
free(buf);
return rval;
}
static int
add_new_user(struct menudesc *menu, void *arg)
{
char username[STRSIZE];
int inwheel=0;
msg_prompt(MSG_addusername, NULL, username, sizeof username -1);
process_menu(MENU_yesno, deconst(MSG_addusertowheel));
inwheel = yesno;
ushell = "/bin/csh";
process_menu(MENU_usersh, NULL);
if (inwheel)
run_program(RUN_PROGRESS | RUN_CHROOT,
"/usr/sbin/useradd -m -s %s -G wheel %s",
ushell, username);
else
run_program(RUN_PROGRESS | RUN_CHROOT,
"/usr/sbin/useradd -m -s %s %s", ushell, username);
run_program(RUN_DISPLAY | RUN_PROGRESS | RUN_CHROOT,
"passwd -l %s", username);
return 0;
}
static int
change_root_password(struct menudesc *menu, void *arg)
{
configinfo **confp = arg;
msg_display(MSG_rootpw);
process_menu(MENU_yesno, NULL);
if (yesno)
run_program(RUN_DISPLAY | RUN_PROGRESS | RUN_CHROOT,
"passwd -l root");
confp[menu->cursel]->setting = MSG_password_set;
return 0;
}
static int
set_binpkg(struct menudesc *menu, void *arg)
{
configinfo **confp = arg;
char pattern[STRSIZE];
/* binary pkg config requires network at this point, so if
it's not already configured, do it. */
if (network_up == 0) {
if (config_network())
mnt_net_config();
}
process_menu(MENU_binpkg, NULL);
make_url(pkgpath, &pkg, pkg_dir);
if ( run_program(RUN_DISPLAY | RUN_PROGRESS | RUN_CHROOT,
"pkg_add %s/pkgin", pkgpath) != 0) {
msg_display(MSG_pkgin_failed);
process_menu(MENU_ok, NULL);
confp[menu->cursel]->setting = MSG_failed;
return 0;
}
/* configure pkgin to use $pkgpath as a repository */
snprintf(pattern, STRSIZE, "s,^[^#].*$,%s,", pkgpath);
replace("/usr/pkg/etc/pkgin/repositories.conf", pattern);
run_program(RUN_DISPLAY | RUN_PROGRESS | RUN_CHROOT,
"/usr/pkg/bin/pkgin -y update");
msg_display(MSG_binpkg_installed);
process_menu(MENU_ok, NULL);
confp[menu->cursel]->setting = MSG_DONE;
return 0;
}
static int
set_pkgsrc(struct menudesc *menu, void *arg)
{
configinfo **confp = arg;
distinfo dist;
dist.name = "pkgsrc";
dist.set = SET_PKGSRC;
dist.desc = "source for 3rd-party packages";
dist.marker_file = NULL;
int status = SET_RETRY;
do {
status = get_pkgsrc();
if (status == SET_OK) {
status = extract_file(&dist, 0);
continue;
} else if (status == SET_SKIP) {
confp[menu->cursel]->setting = MSG_abandoned;
return 0;
}
process_menu(MENU_yesno, deconst(MSG_retry_pkgsrc_network));
if (!yesno) {
confp[menu->cursel]->setting = MSG_abandoned;
return 1;
}
}
while (status == SET_RETRY);
confp[menu->cursel]->setting = MSG_DONE;
return 0;
}
static int
toggle_rcvar(struct menudesc *menu, void *arg)
{
configinfo **confp = arg;
int s;
const char *setting, *varname;
char pattern[STRSIZE];
char buf[STRSIZE];
char *cp;
int found = 0;
FILE *fp;
varname = confp[menu->cursel]->rcvar;
s = check_rcvar(varname);
/* we're toggling, so invert the sense */
if (s) {
confp[menu->cursel]->setting = MSG_NO;
setting = "NO";
} else {
confp[menu->cursel]->setting = MSG_YES;
setting = "YES";
}
if (!(fp = fopen(target_expand("/etc/rc.conf"), "r"))) {
msg_display(MSG_rcconf_delete_failed, varname);
process_menu(MENU_ok, NULL);
return -1;
}
while (fgets(buf, sizeof buf, fp) != NULL) {
cp = buf + strspn(buf, " \t"); /* Skip initial spaces */
if (strncmp(cp, varname, strlen(varname)) == 0) {
cp += strlen(varname);
if (*cp != '=')
continue;
buf[strlen(buf) - 1] = 0;
snprintf(pattern, sizeof pattern,
"s,^%s$,%s=%s,",
buf, varname, setting);
found = 1;
break;
}
}
fclose(fp);
if (!found) {
add_rc_conf("%s=%s\n", varname, setting);
if (logfp) {
fprintf(logfp, "adding %s=%s\n", varname, setting);
fflush(logfp);
}
} else {
if (logfp) {
fprintf(logfp, "replacement pattern is %s\n", pattern);
fflush(logfp);
}
replace("/etc/rc.conf", pattern);
}
return 0;
}
static void
configmenu_hdr(struct menudesc *menu, void *arg)
{
msg_display(MSG_configmenu);
}
void
do_configmenu()
{
int menu_no;
int opts;
menu_ent me[CONFIGOPT_LAST];
configinfo *ce[CONFIGOPT_LAST];
wrefresh(curscr);
wmove(stdscr, 0, 0);
wclear(stdscr);
wrefresh(stdscr);
/* if the target isn't mounted already, figure it out. */
if (target_mounted() == 0) {
if (find_disks(msg_string(MSG_configure_prior)) < 0)
return;
if (mount_disks() != 0)
return;
}
config_list_init();
make_url(pkgpath, &pkg, pkg_dir);
opts = init_config_menu(config_list, me, ce);
menu_no = new_menu(NULL, me, opts, 0, -4, 0, 70,
MC_SCROLL | MC_NOBOX | MC_DFLTEXIT,
configmenu_hdr, set_config, NULL, "XXX Help String",
MSG_doneconfig);
process_menu(menu_no, ce);
free_menu(menu_no);
sanity_check();
}