1
0
mirror of https://git.tartarus.org/simon/putty.git synced 2025-01-25 01:02:24 +00:00

Uppity: allow running multiple independent servers.

I've moved all the results of the command-line config options into a
small struct instead of having them be local variables of main(). We
maintain an array of those structs; most command-line options modify
the last element in the array; and we respond to the new special
option '--and' by appending a fresh struct to the end of the array and
initialising it to default values.

So now, if I want two or three SSH servers running on different ports
with separately configured host keys, banners, etc, I can do that with
a single command line along the lines of:

  ./uppity --listen 2222 --hostkey this.ppk --bannertext "this" \
     --and --listen 2223 --hostkey that.ppk --bannertext "that"

There's a single number space of connections used in log messages, and
each new connection reports which of the servers it connects to.

This is only a marginally useful feature: there's not much it does
that couldn't have been done just as well by running multiple Uppitys
each in their own process. But when I do want several servers at once
(which I've been using recently to test the jump-host system), it's
quite nice to have them all producing a single combined stream of log
data and all conveniently killable with a single ^C.
This commit is contained in:
Simon Tatham 2021-09-17 16:10:29 +01:00
parent fb663d4761
commit d489c64f48

View File

@ -130,6 +130,8 @@ struct server_instance {
}; };
struct server_config { struct server_config {
unsigned config_id;
Conf *conf; Conf *conf;
const SshServerConfig *ssc; const SshServerConfig *ssc;
@ -140,16 +142,16 @@ struct server_config {
struct AuthPolicyShared *ap_shared; struct AuthPolicyShared *ap_shared;
unsigned next_id;
Socket *listening_socket; Socket *listening_socket;
Plug listening_plug; Plug listening_plug;
}; };
static unsigned next_id = 0;
static void log_to_stderr(unsigned id, const char *msg) static void log_to_stderr(unsigned id, const char *msg)
{ {
if (id != (unsigned)-1) if (id != (unsigned)-1)
fprintf(stderr, "#%u: ", id); fprintf(stderr, "conn#%u: ", id);
fputs(msg, stderr); fputs(msg, stderr);
fputc('\n', stderr); fputc('\n', stderr);
fflush(stderr); fflush(stderr);
@ -348,7 +350,7 @@ static void show_help(FILE *fp)
{ {
safety_warning(fp); safety_warning(fp);
fputs("\n" fputs("\n"
"usage: uppity [options]\n" "usage: uppity [options] [--and <options>...]\n"
"options: --listen [PORT|PATH] listen to a port on localhost, or Unix socket\n" "options: --listen [PORT|PATH] listen to a port on localhost, or Unix socket\n"
" --listen-once (with --listen) stop after one " " --listen-once (with --listen) stop after one "
"connection\n" "connection\n"
@ -385,6 +387,7 @@ static void show_help(FILE *fp)
" --sshlog FILE write SSH packet log to FILE\n" " --sshlog FILE write SSH packet log to FILE\n"
" --sshrawlog FILE write SSH packets + raw data log" " --sshrawlog FILE write SSH packets + raw data log"
" to FILE\n" " to FILE\n"
" --and run a separate server on another port\n"
"also: uppity --help show this text\n" "also: uppity --help show this text\n"
" uppity --version show version information\n" " uppity --version show version information\n"
"\n", fp); "\n", fp);
@ -461,7 +464,7 @@ static Plug *server_conn_plug(
memset(inst, 0, sizeof(*inst)); memset(inst, 0, sizeof(*inst));
inst->id = cfg->next_id++; inst->id = next_id++;
inst->ap.shared = cfg->ap_shared; inst->ap.shared = cfg->ap_shared;
if (cfg->ssc->stunt_allow_trivial_ki_auth) if (cfg->ssc->stunt_allow_trivial_ki_auth)
inst->ap.kbdint_state = 1; inst->ap.kbdint_state = 1;
@ -504,7 +507,7 @@ static int server_accepting(Plug *p, accept_fn_t constructor, accept_ctx_t ctx)
cfg->listening_socket = NULL; cfg->listening_socket = NULL;
} }
unsigned old_next_id = cfg->next_id; unsigned old_next_id = next_id;
Plug *plug = server_conn_plug(cfg, &inst); Plug *plug = server_conn_plug(cfg, &inst);
s = constructor(ctx, plug); s = constructor(ctx, plug);
@ -514,15 +517,17 @@ static int server_accepting(Plug *p, accept_fn_t constructor, accept_ctx_t ctx)
SocketPeerInfo *pi = sk_peer_info(s); SocketPeerInfo *pi = sk_peer_info(s);
if (pi->addressfamily != ADDRTYPE_LOCAL && !sk_peer_trusted(s)) { if (pi->addressfamily != ADDRTYPE_LOCAL && !sk_peer_trusted(s)) {
fprintf(stderr, "rejected connection from %s (untrustworthy peer)\n", fprintf(stderr, "rejected connection to serv#%u "
pi->log_text); "from %s (untrustworthy peer)\n",
cfg->config_id, pi->log_text);
sk_free_peer_info(pi); sk_free_peer_info(pi);
sk_close(s); sk_close(s);
cfg->next_id = old_next_id; next_id = old_next_id;
return 1; return 1;
} }
char *msg = dupprintf("new connection from %s", pi->log_text); char *msg = dupprintf("new connection to serv#%u from %s",
cfg->config_id, pi->log_text);
log_to_stderr(inst->id, msg); log_to_stderr(inst->id, msg);
sfree(msg); sfree(msg);
sk_free_peer_info(pi); sk_free_peer_info(pi);
@ -538,29 +543,87 @@ static const PlugVtable server_plugvt = {
.accepting = server_accepting, .accepting = server_accepting,
}; };
int main(int argc, char **argv) struct cmdline_instance {
{ int listen_port;
int listen_port = -1; const char *listen_socket;
const char *listen_socket = NULL; ssh_key **hostkeys;
size_t nhostkeys, hostkeysize;
ssh_key **hostkeys = NULL; RSAKey *hostkey1;
size_t nhostkeys = 0, hostkeysize = 0;
RSAKey *hostkey1 = NULL;
struct AuthPolicyShared aps; struct AuthPolicyShared aps;
SshServerConfig ssc; SshServerConfig ssc;
Conf *conf;
};
Conf *conf = make_ssh_server_conf(); static void init_cmdline_instance(struct cmdline_instance *ci)
{
ci->listen_port = -1;
ci->listen_socket = NULL;
aps.ssh1keys = NULL; ci->hostkeys = NULL;
aps.ssh2keys = NULL; ci->nhostkeys = ci->hostkeysize = 0;
ci->hostkey1 = NULL;
memset(&ssc, 0, sizeof(ssc)); ci->conf = make_ssh_server_conf();
ssc.application_name = "Uppity"; ci->aps.ssh1keys = NULL;
ssc.session_starting_dir = getenv("HOME"); ci->aps.ssh2keys = NULL;
ssc.ssh1_cipher_mask = SSH1_SUPPORTED_CIPHER_MASK;
ssc.ssh1_allow_compression = true; memset(&ci->ssc, 0, sizeof(ci->ssc));
ci->ssc.application_name = "Uppity";
ci->ssc.session_starting_dir = getenv("HOME");
ci->ssc.ssh1_cipher_mask = SSH1_SUPPORTED_CIPHER_MASK;
ci->ssc.ssh1_allow_compression = true;
}
static void cmdline_instance_start(struct cmdline_instance *ci)
{
static unsigned next_server_config_id = 0;
struct server_config *scfg = snew(struct server_config);
scfg->config_id = next_server_config_id++;
scfg->conf = ci->conf;
scfg->ssc = &ci->ssc;
scfg->hostkeys = ci->hostkeys;
scfg->nhostkeys = ci->nhostkeys;
scfg->hostkey1 = ci->hostkey1;
scfg->ap_shared = &ci->aps;
if (ci->listen_port >= 0 || ci->listen_socket) {
listening = true;
scfg->listening_plug.vt = &server_plugvt;
char *msg;
if (ci->listen_port >= 0) {
scfg->listening_socket = sk_newlistener(
NULL, ci->listen_port, &scfg->listening_plug, true,
ADDRTYPE_UNSPEC);
msg = dupprintf("serv#%u: listening on port %d",
scfg->config_id, ci->listen_port);
} else {
SockAddr *addr = unix_sock_addr(ci->listen_socket);
scfg->listening_socket = new_unix_listener(
addr, &scfg->listening_plug);
msg = dupprintf("serv#%u: listening on Unix socket %s",
scfg->config_id, ci->listen_socket);
}
log_to_stderr(-1, msg);
sfree(msg);
} else {
struct server_instance *inst;
Plug *plug = server_conn_plug(scfg, &inst);
ssh_server_start(plug, make_fd_socket(0, 1, -1, NULL, 0, plug));
log_to_stderr(inst->id, "speaking SSH on stdio");
}
}
int main(int argc, char **argv)
{
size_t ninstances = 0, instancessize = 8;
struct cmdline_instance *instances = snewn(
instancessize, struct cmdline_instance);
struct cmdline_instance *ci = &instances[ninstances++];
init_cmdline_instance(ci);
if (argc <= 1) { if (argc <= 1) {
/* /*
@ -584,13 +647,17 @@ int main(int argc, char **argv)
show_version_and_exit(); show_version_and_exit();
} else if (longoptnoarg(arg, "--verbose") || !strcmp(arg, "-v")) { } else if (longoptnoarg(arg, "--verbose") || !strcmp(arg, "-v")) {
verbose = true; verbose = true;
} else if (longoptnoarg(arg, "--and")) {
sgrowarray(instances, instancessize, ninstances);
ci = &instances[ninstances++];
init_cmdline_instance(ci);
} else if (longoptarg(arg, "--listen", &val, &argc, &argv)) { } else if (longoptarg(arg, "--listen", &val, &argc, &argv)) {
if (val[0] == '/') { if (val[0] == '/') {
listen_port = -1; ci->listen_port = -1;
listen_socket = val; ci->listen_socket = val;
} else { } else {
listen_port = atoi(val); ci->listen_port = atoi(val);
listen_socket = NULL; ci->listen_socket = NULL;
} }
} else if (!strcmp(arg, "--listen-once")) { } else if (!strcmp(arg, "--listen-once")) {
listen_once = true; listen_once = true;
@ -622,24 +689,24 @@ int main(int argc, char **argv)
sfree(uk->comment); sfree(uk->comment);
sfree(uk); sfree(uk);
for (int i = 0; i < nhostkeys; i++) for (int i = 0; i < ci->nhostkeys; i++)
if (ssh_key_alg(hostkeys[i]) == ssh_key_alg(key)) { if (ssh_key_alg(ci->hostkeys[i]) == ssh_key_alg(key)) {
fprintf(stderr, "%s: host key '%s' duplicates key " fprintf(stderr, "%s: host key '%s' duplicates key "
"type %s\n", appname, val, "type %s\n", appname, val,
ssh_key_alg(key)->ssh_id); ssh_key_alg(key)->ssh_id);
exit(1); exit(1);
} }
sgrowarray(hostkeys, hostkeysize, nhostkeys); sgrowarray(ci->hostkeys, ci->hostkeysize, ci->nhostkeys);
hostkeys[nhostkeys++] = key; ci->hostkeys[ci->nhostkeys++] = key;
} else if (keytype == SSH_KEYTYPE_SSH1) { } else if (keytype == SSH_KEYTYPE_SSH1) {
if (hostkey1) { if (ci->hostkey1) {
fprintf(stderr, "%s: host key '%s' is a redundant " fprintf(stderr, "%s: host key '%s' is a redundant "
"SSH-1 host key\n", appname, val); "SSH-1 host key\n", appname, val);
exit(1); exit(1);
} }
hostkey1 = snew(RSAKey); ci->hostkey1 = snew(RSAKey);
if (!rsa1_load_f(keyfile, hostkey1, NULL, &error)) { if (!rsa1_load_f(keyfile, ci->hostkey1, NULL, &error)) {
fprintf(stderr, "%s: unable to load host key '%s': " fprintf(stderr, "%s: unable to load host key '%s': "
"%s\n", appname, val, error); "%s\n", appname, val, error);
exit(1); exit(1);
@ -665,19 +732,19 @@ int main(int argc, char **argv)
exit(1); exit(1);
} }
if (ssc.rsa_kex_key) { if (ci->ssc.rsa_kex_key) {
freersakey(ssc.rsa_kex_key); freersakey(ci->ssc.rsa_kex_key);
} else { } else {
ssc.rsa_kex_key = snew(RSAKey); ci->ssc.rsa_kex_key = snew(RSAKey);
} }
if (!rsa1_load_f(keyfile, ssc.rsa_kex_key, NULL, &error)) { if (!rsa1_load_f(keyfile, ci->ssc.rsa_kex_key, NULL, &error)) {
fprintf(stderr, "%s: unable to load RSA kex key '%s': " fprintf(stderr, "%s: unable to load RSA kex key '%s': "
"%s\n", appname, val, error); "%s\n", appname, val, error);
exit(1); exit(1);
} }
ssc.rsa_kex_key->sshk.vt = &ssh_rsa; ci->ssc.rsa_kex_key->sshk.vt = &ssh_rsa;
} else if (longoptarg(arg, "--userkey", &val, &argc, &argv)) { } else if (longoptarg(arg, "--userkey", &val, &argc, &argv)) {
Filename *keyfile; Filename *keyfile;
int keytype; int keytype;
@ -704,8 +771,8 @@ int main(int argc, char **argv)
memcpy(blob, sb->u, sb->len); memcpy(blob, sb->u, sb->len);
node->public_blob = make_ptrlen(blob, sb->len); node->public_blob = make_ptrlen(blob, sb->len);
node->next = aps.ssh2keys; node->next = ci->aps.ssh2keys;
aps.ssh2keys = node; ci->aps.ssh2keys = node;
strbuf_free(sb); strbuf_free(sb);
} else if (keytype == SSH_KEYTYPE_SSH1_PUBLIC) { } else if (keytype == SSH_KEYTYPE_SSH1_PUBLIC) {
@ -724,8 +791,8 @@ int main(int argc, char **argv)
BinarySource_BARE_INIT(src, sb->u, sb->len); BinarySource_BARE_INIT(src, sb->u, sb->len);
get_rsa_ssh1_pub(src, &node->key, RSA_SSH1_EXPONENT_FIRST); get_rsa_ssh1_pub(src, &node->key, RSA_SSH1_EXPONENT_FIRST);
node->next = aps.ssh1keys; node->next = ci->aps.ssh1keys;
aps.ssh1keys = node; ci->aps.ssh1keys = node;
strbuf_free(sb); strbuf_free(sb);
} else { } else {
@ -747,27 +814,27 @@ int main(int argc, char **argv)
exit(1); exit(1);
} }
fclose(fp); fclose(fp);
ssc.banner = ptrlen_from_strbuf(sb); ci->ssc.banner = ptrlen_from_strbuf(sb);
} else if (longoptarg(arg, "--bannertext", &val, &argc, &argv)) { } else if (longoptarg(arg, "--bannertext", &val, &argc, &argv)) {
ssc.banner = ptrlen_from_asciz(val); ci->ssc.banner = ptrlen_from_asciz(val);
} else if (longoptarg(arg, "--sessiondir", &val, &argc, &argv)) { } else if (longoptarg(arg, "--sessiondir", &val, &argc, &argv)) {
ssc.session_starting_dir = val; ci->ssc.session_starting_dir = val;
} else if (longoptarg(arg, "--kexinit-kex", &val, &argc, &argv)) { } else if (longoptarg(arg, "--kexinit-kex", &val, &argc, &argv)) {
ssc.kex_override[KEXLIST_KEX] = ptrlen_from_asciz(val); ci->ssc.kex_override[KEXLIST_KEX] = ptrlen_from_asciz(val);
} else if (longoptarg(arg, "--kexinit-hostkey", &val, &argc, &argv)) { } else if (longoptarg(arg, "--kexinit-hostkey", &val, &argc, &argv)) {
ssc.kex_override[KEXLIST_HOSTKEY] = ptrlen_from_asciz(val); ci->ssc.kex_override[KEXLIST_HOSTKEY] = ptrlen_from_asciz(val);
} else if (longoptarg(arg, "--kexinit-cscipher", &val, &argc, &argv)) { } else if (longoptarg(arg, "--kexinit-cscipher", &val, &argc, &argv)) {
ssc.kex_override[KEXLIST_CSCIPHER] = ptrlen_from_asciz(val); ci->ssc.kex_override[KEXLIST_CSCIPHER] = ptrlen_from_asciz(val);
} else if (longoptarg(arg, "--kexinit-csmac", &val, &argc, &argv)) { } else if (longoptarg(arg, "--kexinit-csmac", &val, &argc, &argv)) {
ssc.kex_override[KEXLIST_CSMAC] = ptrlen_from_asciz(val); ci->ssc.kex_override[KEXLIST_CSMAC] = ptrlen_from_asciz(val);
} else if (longoptarg(arg, "--kexinit-cscomp", &val, &argc, &argv)) { } else if (longoptarg(arg, "--kexinit-cscomp", &val, &argc, &argv)) {
ssc.kex_override[KEXLIST_CSCOMP] = ptrlen_from_asciz(val); ci->ssc.kex_override[KEXLIST_CSCOMP] = ptrlen_from_asciz(val);
} else if (longoptarg(arg, "--kexinit-sccipher", &val, &argc, &argv)) { } else if (longoptarg(arg, "--kexinit-sccipher", &val, &argc, &argv)) {
ssc.kex_override[KEXLIST_SCCIPHER] = ptrlen_from_asciz(val); ci->ssc.kex_override[KEXLIST_SCCIPHER] = ptrlen_from_asciz(val);
} else if (longoptarg(arg, "--kexinit-scmac", &val, &argc, &argv)) { } else if (longoptarg(arg, "--kexinit-scmac", &val, &argc, &argv)) {
ssc.kex_override[KEXLIST_SCMAC] = ptrlen_from_asciz(val); ci->ssc.kex_override[KEXLIST_SCMAC] = ptrlen_from_asciz(val);
} else if (longoptarg(arg, "--kexinit-sccomp", &val, &argc, &argv)) { } else if (longoptarg(arg, "--kexinit-sccomp", &val, &argc, &argv)) {
ssc.kex_override[KEXLIST_SCCOMP] = ptrlen_from_asciz(val); ci->ssc.kex_override[KEXLIST_SCCOMP] = ptrlen_from_asciz(val);
} else if (longoptarg(arg, "--ssh1-ciphers", &val, &argc, &argv)) { } else if (longoptarg(arg, "--ssh1-ciphers", &val, &argc, &argv)) {
ptrlen list = ptrlen_from_asciz(val); ptrlen list = ptrlen_from_asciz(val);
ptrlen word; ptrlen word;
@ -786,45 +853,58 @@ int main(int argc, char **argv)
appname, PTRLEN_PRINTF(word)); appname, PTRLEN_PRINTF(word));
exit(1); exit(1);
} }
ssc.ssh1_cipher_mask = mask; ci->ssc.ssh1_cipher_mask = mask;
} else if (longoptnoarg(arg, "--ssh1-no-compression")) { } else if (longoptnoarg(arg, "--ssh1-no-compression")) {
ssc.ssh1_allow_compression = false; ci->ssc.ssh1_allow_compression = false;
} else if (longoptnoarg(arg, "--exitsignum")) { } else if (longoptnoarg(arg, "--exitsignum")) {
ssc.exit_signal_numeric = true; ci->ssc.exit_signal_numeric = true;
} else if (longoptarg(arg, "--sshlog", &val, &argc, &argv) || } else if (longoptarg(arg, "--sshlog", &val, &argc, &argv) ||
longoptarg(arg, "-sshlog", &val, &argc, &argv)) { longoptarg(arg, "-sshlog", &val, &argc, &argv)) {
Filename *logfile = filename_from_str(val); Filename *logfile = filename_from_str(val);
conf_set_filename(conf, CONF_logfilename, logfile); conf_set_filename(ci->conf, CONF_logfilename, logfile);
filename_free(logfile); filename_free(logfile);
conf_set_int(conf, CONF_logtype, LGTYP_PACKETS); conf_set_int(ci->conf, CONF_logtype, LGTYP_PACKETS);
conf_set_int(conf, CONF_logxfovr, LGXF_OVR); conf_set_int(ci->conf, CONF_logxfovr, LGXF_OVR);
} else if (longoptarg(arg, "--sshrawlog", &val, &argc, &argv) || } else if (longoptarg(arg, "--sshrawlog", &val, &argc, &argv) ||
longoptarg(arg, "-sshrawlog", &val, &argc, &argv)) { longoptarg(arg, "-sshrawlog", &val, &argc, &argv)) {
Filename *logfile = filename_from_str(val); Filename *logfile = filename_from_str(val);
conf_set_filename(conf, CONF_logfilename, logfile); conf_set_filename(ci->conf, CONF_logfilename, logfile);
filename_free(logfile); filename_free(logfile);
conf_set_int(conf, CONF_logtype, LGTYP_SSHRAW); conf_set_int(ci->conf, CONF_logtype, LGTYP_SSHRAW);
conf_set_int(conf, CONF_logxfovr, LGXF_OVR); conf_set_int(ci->conf, CONF_logxfovr, LGXF_OVR);
} else if (!strcmp(arg, "--pretend-to-accept-any-pubkey")) { } else if (!strcmp(arg, "--pretend-to-accept-any-pubkey")) {
ssc.stunt_pretend_to_accept_any_pubkey = true; ci->ssc.stunt_pretend_to_accept_any_pubkey = true;
} else if (!strcmp(arg, "--open-unconditional-agent-socket")) { } else if (!strcmp(arg, "--open-unconditional-agent-socket")) {
ssc.stunt_open_unconditional_agent_socket = true; ci->ssc.stunt_open_unconditional_agent_socket = true;
} else if (!strcmp(arg, "--allow-none-auth")) { } else if (!strcmp(arg, "--allow-none-auth")) {
ssc.stunt_allow_none_auth = true; ci->ssc.stunt_allow_none_auth = true;
} else if (!strcmp(arg, "--allow-trivial-ki-auth")) { } else if (!strcmp(arg, "--allow-trivial-ki-auth")) {
ssc.stunt_allow_trivial_ki_auth = true; ci->ssc.stunt_allow_trivial_ki_auth = true;
} else if (!strcmp(arg, "--return-success-to-pubkey-offer")) { } else if (!strcmp(arg, "--return-success-to-pubkey-offer")) {
ssc.stunt_return_success_to_pubkey_offer = true; ci->ssc.stunt_return_success_to_pubkey_offer = true;
} else { } else {
fprintf(stderr, "%s: unrecognised option '%s'\n", appname, arg); fprintf(stderr, "%s: unrecognised option '%s'\n", appname, arg);
exit(1); exit(1);
} }
} }
if (nhostkeys == 0 && !hostkey1) { if (ninstances > 1 && listen_once) {
fprintf(stderr, "%s: cannot listen once only with multiple server "
"instances\n", appname);
exit(1);
}
for (size_t i = 0; i < ninstances; i++) {
ci = &instances[i];
if (ci->nhostkeys == 0 && !ci->hostkey1) {
fprintf(stderr, "%s: specify at least one host key\n", appname); fprintf(stderr, "%s: specify at least one host key\n", appname);
exit(1); exit(1);
} }
if (ninstances > 1 && !(ci->listen_port >= 0 || ci->listen_socket)) {
fprintf(stderr, "%s: cannot talk to stdio with multiple server "
"instances\n", appname);
exit(1);
}
}
random_ref(); random_ref();
@ -837,41 +917,8 @@ int main(int argc, char **argv)
sk_init(); sk_init();
uxsel_init(); uxsel_init();
struct server_config scfg; for (size_t i = 0; i < ninstances; i++)
scfg.conf = conf; cmdline_instance_start(&instances[i]);
scfg.ssc = &ssc;
scfg.hostkeys = hostkeys;
scfg.nhostkeys = nhostkeys;
scfg.hostkey1 = hostkey1;
scfg.ap_shared = &aps;
scfg.next_id = 0;
if (listen_port >= 0 || listen_socket) {
listening = true;
scfg.listening_plug.vt = &server_plugvt;
char *msg;
if (listen_port >= 0) {
scfg.listening_socket = sk_newlistener(
NULL, listen_port, &scfg.listening_plug, true,
ADDRTYPE_UNSPEC);
msg = dupprintf("%s: listening on port %d",
appname, listen_port);
} else {
SockAddr *addr = unix_sock_addr(listen_socket);
scfg.listening_socket = new_unix_listener(
addr, &scfg.listening_plug);
msg = dupprintf("%s: listening on Unix socket %s",
appname, listen_socket);
}
log_to_stderr(-1, msg);
sfree(msg);
} else {
struct server_instance *inst;
Plug *plug = server_conn_plug(&scfg, &inst);
ssh_server_start(plug, make_fd_socket(0, 1, -1, NULL, 0, plug));
log_to_stderr(inst->id, "speaking SSH on stdio");
}
cli_main_loop(cliloop_no_pw_setup, cliloop_no_pw_check, cli_main_loop(cliloop_no_pw_setup, cliloop_no_pw_check,
cliloop_always_continue, NULL); cliloop_always_continue, NULL);