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

Reorganise host key checking and confirmation.

Previously, checking the host key against the persistent cache managed
by the storage.h API was done as part of the seat_verify_ssh_host_key
method, i.e. separately by each Seat.

Now that check is done by verify_ssh_host_key(), which is a new
function in ssh/common.c that centralises all the parts of host key
checking that don't need an interactive prompt. It subsumes the
previous verify_ssh_manual_host_key() that checked against the Conf,
and it does the check against the storage API that each Seat was
previously doing separately. If it can't confirm or definitively
reject the host key by itself, _then_ it calls out to the Seat, once
an interactive prompt is definitely needed.

The main point of doing this is so that when SshProxy forwards a Seat
call from the proxy SSH connection to the primary Seat, it won't print
an announcement of which connection is involved unless it's actually
going to do something interactive. (Not that we're printing those
announcements _yet_ anyway, but this is a piece of groundwork that
works towards doing so.)

But while I'm at it, I've also taken the opportunity to clean things
up a bit by renaming functions sensibly. Previously we had three very
similarly named functions verify_ssh_manual_host_key(), SeatVtable's
'verify_ssh_host_key' method, and verify_host_key() in storage.h. Now
the Seat method is called 'confirm' rather than 'verify' (since its
job is now always to print an interactive prompt, so it looks more
like the other confirm_foo methods), and the storage.h function is
called check_stored_host_key(), which goes better with store_host_key
and avoids having too many functions with similar names. And the
'manual' function is subsumed into the new centralised code, so
there's now just *one* host key function with 'verify' in the name.

Several functions are reindented in this commit. Best viewed with
whitespace changes ignored.
This commit is contained in:
Simon Tatham 2021-10-25 18:12:17 +01:00
parent f1746d69b1
commit efa89573ae
26 changed files with 240 additions and 266 deletions

2
pscp.c
View File

@ -75,7 +75,7 @@ static const SeatVtable pscp_seat_vt = {
.update_specials_menu = nullseat_update_specials_menu, .update_specials_menu = nullseat_update_specials_menu,
.get_ttymode = nullseat_get_ttymode, .get_ttymode = nullseat_get_ttymode,
.set_busy_status = nullseat_set_busy_status, .set_busy_status = nullseat_set_busy_status,
.verify_ssh_host_key = console_verify_ssh_host_key, .confirm_ssh_host_key = console_confirm_ssh_host_key,
.confirm_weak_crypto_primitive = console_confirm_weak_crypto_primitive, .confirm_weak_crypto_primitive = console_confirm_weak_crypto_primitive,
.confirm_weak_cached_hostkey = console_confirm_weak_cached_hostkey, .confirm_weak_cached_hostkey = console_confirm_weak_cached_hostkey,
.is_utf8 = nullseat_is_never_utf8, .is_utf8 = nullseat_is_never_utf8,

View File

@ -56,7 +56,7 @@ static const SeatVtable psftp_seat_vt = {
.update_specials_menu = nullseat_update_specials_menu, .update_specials_menu = nullseat_update_specials_menu,
.get_ttymode = nullseat_get_ttymode, .get_ttymode = nullseat_get_ttymode,
.set_busy_status = nullseat_set_busy_status, .set_busy_status = nullseat_set_busy_status,
.verify_ssh_host_key = console_verify_ssh_host_key, .confirm_ssh_host_key = console_confirm_ssh_host_key,
.confirm_weak_crypto_primitive = console_confirm_weak_crypto_primitive, .confirm_weak_crypto_primitive = console_confirm_weak_crypto_primitive,
.confirm_weak_cached_hostkey = console_confirm_weak_cached_hostkey, .confirm_weak_cached_hostkey = console_confirm_weak_cached_hostkey,
.is_utf8 = nullseat_is_never_utf8, .is_utf8 = nullseat_is_never_utf8,

View File

@ -523,8 +523,8 @@ void psocks_start(psocks_state *ps)
* Some stubs that are needed to link against PuTTY modules. * Some stubs that are needed to link against PuTTY modules.
*/ */
int verify_host_key(const char *hostname, int port, int check_stored_host_key(const char *hostname, int port,
const char *keytype, const char *key) const char *keytype, const char *key)
{ {
unreachable("host keys not handled in this tool"); unreachable("host keys not handled in this tool");
} }

47
putty.h
View File

@ -1081,31 +1081,43 @@ struct SeatVtable {
/* /*
* Ask the seat whether a given SSH host key should be accepted. * Ask the seat whether a given SSH host key should be accepted.
* This may return immediately after checking saved configuration * This is called after we've already checked it by any means we
* or command-line options, or it may have to present a prompt to * can do ourselves, such as checking against host key
* the user and return asynchronously later. * fingerprints in the Conf or the host key cache on disk: once we
* call this function, we've already decided there's nothing for
* it but to prompt the user.
*
* 'mismatch' reports the result of checking the host key cache:
* it is true if the server has presented a host key different
* from the one we expected, and false if we had no expectation in
* the first place.
*
* This call may prompt the user synchronously and not return
* until the answer is available, or it may present the prompt and
* return immediately, giving the answer later via the provided
* callback.
* *
* Return values: * Return values:
* *
* - +1 means `key was OK' (either already known or the user just * - +1 means `user approved the key, so continue with the
* approved it) `so continue with the connection' * connection'
* *
* - 0 means `key was not OK, abandon the connection' * - 0 means `user rejected the key, abandon the connection'
* *
* - -1 means `I've initiated enquiries, please wait to be called * - -1 means `I've initiated enquiries, please wait to be called
* back via the provided function with a result that's either 0 * back via the provided function with a result that's either 0
* or +1'. * or +1'.
*/ */
int (*verify_ssh_host_key)( int (*confirm_ssh_host_key)(
Seat *seat, const char *host, int port, const char *keytype, Seat *seat, const char *host, int port, const char *keytype,
char *keystr, const char *keydisp, char **key_fingerprints, char *keystr, const char *keydisp, char **key_fingerprints,
void (*callback)(void *ctx, int result), void *ctx); bool mismatch, void (*callback)(void *ctx, int result), void *ctx);
/* /*
* Check with the seat whether it's OK to use a cryptographic * Check with the seat whether it's OK to use a cryptographic
* primitive from below the 'warn below this line' threshold in * primitive from below the 'warn below this line' threshold in
* the input Conf. Return values are the same as * the input Conf. Return values are the same as
* verify_ssh_host_key above. * confirm_ssh_host_key above.
*/ */
int (*confirm_weak_crypto_primitive)( int (*confirm_weak_crypto_primitive)(
Seat *seat, const char *algtype, const char *algname, Seat *seat, const char *algtype, const char *algname,
@ -1229,11 +1241,12 @@ static inline char *seat_get_ttymode(Seat *seat, const char *mode)
{ return seat->vt->get_ttymode(seat, mode); } { return seat->vt->get_ttymode(seat, mode); }
static inline void seat_set_busy_status(Seat *seat, BusyStatus status) static inline void seat_set_busy_status(Seat *seat, BusyStatus status)
{ seat->vt->set_busy_status(seat, status); } { seat->vt->set_busy_status(seat, status); }
static inline int seat_verify_ssh_host_key( static inline int seat_confirm_ssh_host_key(
Seat *seat, const char *h, int p, const char *ktyp, char *kstr, Seat *seat, const char *h, int p, const char *ktyp, char *kstr,
const char *kdsp, char **fps, void (*cb)(void *ctx, int result), void *ctx) const char *kdsp, char **fps, bool mis,
{ return seat->vt->verify_ssh_host_key(seat, h, p, ktyp, kstr, kdsp, fps, void (*cb)(void *ctx, int result), void *ctx)
cb, ctx); } { return seat->vt->confirm_ssh_host_key(seat, h, p, ktyp, kstr, kdsp, fps,
mis, cb, ctx); }
static inline int seat_confirm_weak_crypto_primitive( static inline int seat_confirm_weak_crypto_primitive(
Seat *seat, const char *atyp, const char *aname, Seat *seat, const char *atyp, const char *aname,
void (*cb)(void *ctx, int result), void *ctx) void (*cb)(void *ctx, int result), void *ctx)
@ -1308,9 +1321,9 @@ void nullseat_connection_fatal(Seat *seat, const char *message);
void nullseat_update_specials_menu(Seat *seat); void nullseat_update_specials_menu(Seat *seat);
char *nullseat_get_ttymode(Seat *seat, const char *mode); char *nullseat_get_ttymode(Seat *seat, const char *mode);
void nullseat_set_busy_status(Seat *seat, BusyStatus status); void nullseat_set_busy_status(Seat *seat, BusyStatus status);
int nullseat_verify_ssh_host_key( int nullseat_confirm_ssh_host_key(
Seat *seat, const char *host, int port, const char *keytype, Seat *seat, const char *host, int port, const char *keytype,
char *keystr, const char *keydisp, char **key_fingerprints, char *keystr, const char *keydisp, char **key_fingerprints, bool mismatch,
void (*callback)(void *ctx, int result), void *ctx); void (*callback)(void *ctx, int result), void *ctx);
int nullseat_confirm_weak_crypto_primitive( int nullseat_confirm_weak_crypto_primitive(
Seat *seat, const char *algtype, const char *algname, Seat *seat, const char *algtype, const char *algname,
@ -1341,9 +1354,9 @@ bool nullseat_get_cursor_position(Seat *seat, int *x, int *y);
*/ */
void console_connection_fatal(Seat *seat, const char *message); void console_connection_fatal(Seat *seat, const char *message);
int console_verify_ssh_host_key( int console_confirm_ssh_host_key(
Seat *seat, const char *host, int port, const char *keytype, Seat *seat, const char *host, int port, const char *keytype,
char *keystr, const char *keydisp, char **key_fingerprints, char *keystr, const char *keydisp, char **key_fingerprints, bool mismatch,
void (*callback)(void *ctx, int result), void *ctx); void (*callback)(void *ctx, int result), void *ctx);
int console_confirm_weak_crypto_primitive( int console_confirm_weak_crypto_primitive(
Seat *seat, const char *algtype, const char *algname, Seat *seat, const char *algtype, const char *algname,

5
ssh.h
View File

@ -1707,7 +1707,10 @@ unsigned alloc_channel_id_general(tree234 *channels, size_t localid_offset);
void add_to_commasep(strbuf *buf, const char *data); void add_to_commasep(strbuf *buf, const char *data);
bool get_commasep_word(ptrlen *list, ptrlen *word); bool get_commasep_word(ptrlen *list, ptrlen *word);
int verify_ssh_manual_host_key(Conf *conf, char **fingerprints, ssh_key *key); int verify_ssh_host_key(
Seat *seat, Conf *conf, const char *host, int port, ssh_key *key,
const char *keytype, char *keystr, const char *keydisp,
char **fingerprints, void (*callback)(void *ctx, int result), void *ctx);
typedef struct ssh_transient_hostkey_cache ssh_transient_hostkey_cache; typedef struct ssh_transient_hostkey_cache ssh_transient_hostkey_cache;
ssh_transient_hostkey_cache *ssh_transient_hostkey_cache_new(void); ssh_transient_hostkey_cache *ssh_transient_hostkey_cache_new(void);

View File

@ -9,6 +9,7 @@
#include "putty.h" #include "putty.h"
#include "mpint.h" #include "mpint.h"
#include "ssh.h" #include "ssh.h"
#include "storage.h"
#include "bpp.h" #include "bpp.h"
#include "ppl.h" #include "ppl.h"
#include "channel.h" #include "channel.h"
@ -837,58 +838,93 @@ bool ssh2_bpp_check_unimplemented(BinaryPacketProtocol *bpp, PktIn *pktin)
#undef SSH1_BITMAP_WORD #undef SSH1_BITMAP_WORD
/* ---------------------------------------------------------------------- /* ----------------------------------------------------------------------
* Function to check a host key against any manually configured in Conf. * Centralised component of SSH host key verification.
*
* verify_ssh_host_key is called from both the SSH-1 and SSH-2
* transport layers, and does the initial work of checking whether the
* host key is already known. If so, it returns success on its own
* account; otherwise, it calls out to the Seat to give an interactive
* prompt (the nature of which varies depending on the Seat itself).
*
* Return values are 0 for 'abort connection', 1 for 'ok, carry on',
* and negative for 'answer not received yet, wait for a callback'.
*/ */
int verify_ssh_manual_host_key(Conf *conf, char **fingerprints, ssh_key *key) int verify_ssh_host_key(
Seat *seat, Conf *conf, const char *host, int port, ssh_key *key,
const char *keytype, char *keystr, const char *keydisp,
char **fingerprints, void (*callback)(void *ctx, int result), void *ctx)
{ {
if (!conf_get_str_nthstrkey(conf, CONF_ssh_manual_hostkeys, 0)) /*
return -1; /* no manual keys configured */ * First, check if the Conf includes a manual specification of the
* expected host key. If so, that completely supersedes everything
* else, including the normal host key cache _and_ including
* manual overrides: we return success or failure immediately,
* entirely based on whether the key matches the Conf.
*/
if (conf_get_str_nthstrkey(conf, CONF_ssh_manual_hostkeys, 0)) {
if (fingerprints) {
for (size_t i = 0; i < SSH_N_FPTYPES; i++) {
/*
* Each fingerprint string we've been given will have
* things like 'ssh-rsa 2048' at the front of it. Strip
* those off and narrow down to just the hash at the end
* of the string.
*/
const char *fingerprint = fingerprints[i];
if (!fingerprint)
continue;
const char *p = strrchr(fingerprint, ' ');
fingerprint = p ? p+1 : fingerprint;
if (conf_get_str_str_opt(conf, CONF_ssh_manual_hostkeys,
fingerprint))
return 1; /* success */
}
}
if (fingerprints) { if (key) {
for (size_t i = 0; i < SSH_N_FPTYPES; i++) {
/* /*
* Each fingerprint string we've been given will have * Construct the base64-encoded public key blob and see if
* things like 'ssh-rsa 2048' at the front of it. Strip * that's listed.
* those off and narrow down to just the hash at the end
* of the string.
*/ */
const char *fingerprint = fingerprints[i]; strbuf *binblob;
if (!fingerprint) char *base64blob;
continue; int atoms, i;
const char *p = strrchr(fingerprint, ' '); binblob = strbuf_new();
fingerprint = p ? p+1 : fingerprint; ssh_key_public_blob(key, BinarySink_UPCAST(binblob));
atoms = (binblob->len + 2) / 3;
base64blob = snewn(atoms * 4 + 1, char);
for (i = 0; i < atoms; i++)
base64_encode_atom(binblob->u + 3*i,
binblob->len - 3*i, base64blob + 4*i);
base64blob[atoms * 4] = '\0';
strbuf_free(binblob);
if (conf_get_str_str_opt(conf, CONF_ssh_manual_hostkeys, if (conf_get_str_str_opt(conf, CONF_ssh_manual_hostkeys,
fingerprint)) base64blob)) {
sfree(base64blob);
return 1; /* success */ return 1; /* success */
} }
}
if (key) {
/*
* Construct the base64-encoded public key blob and see if
* that's listed.
*/
strbuf *binblob;
char *base64blob;
int atoms, i;
binblob = strbuf_new();
ssh_key_public_blob(key, BinarySink_UPCAST(binblob));
atoms = (binblob->len + 2) / 3;
base64blob = snewn(atoms * 4 + 1, char);
for (i = 0; i < atoms; i++)
base64_encode_atom(binblob->u + 3*i,
binblob->len - 3*i, base64blob + 4*i);
base64blob[atoms * 4] = '\0';
strbuf_free(binblob);
if (conf_get_str_str_opt(conf, CONF_ssh_manual_hostkeys, base64blob)) {
sfree(base64blob); sfree(base64blob);
return 1; /* success */
} }
sfree(base64blob);
return 0;
} }
return 0; /*
* Next, check the host key cache.
*/
int storage_status = check_stored_host_key(host, port, keytype, keystr);
if (storage_status == 0) /* matching key was found in the cache */
return 1; /* success */
/*
* The key is either missing from the cache, or does not match.
* Either way, fall back to an interactive prompt from the Seat.
*/
bool mismatch = (storage_status != 1);
return seat_confirm_ssh_host_key(
seat, host, port, keytype, keystr, keydisp, fingerprints, mismatch,
callback, ctx);
} }
/* ---------------------------------------------------------------------- /* ----------------------------------------------------------------------

View File

@ -843,39 +843,33 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s, bool *aborted)
* Authenticate remote host: verify host key. (We've already * Authenticate remote host: verify host key. (We've already
* checked the signature of the exchange hash.) * checked the signature of the exchange hash.)
*/ */
char **fingerprints = ssh2_all_fingerprints(s->hkey); {
FingerprintType fptype_default =
ssh2_pick_default_fingerprint(fingerprints);
ppl_logevent("Host key fingerprint is:");
ppl_logevent("%s", fingerprints[fptype_default]);
/* First check against manually configured host keys. */
s->dlgret = verify_ssh_manual_host_key(
s->conf, fingerprints, s->hkey);
if (s->dlgret == 0) { /* did not match */
ssh2_free_all_fingerprints(fingerprints);
ssh_sw_abort(s->ppl.ssh, "Host key did not appear in manually "
"configured list");
*aborted = true;
return;
} else if (s->dlgret < 0) { /* none configured; use standard handling */
ssh2_userkey uk = { .key = s->hkey, .comment = NULL }; ssh2_userkey uk = { .key = s->hkey, .comment = NULL };
char *keydisp = ssh2_pubkey_openssh_str(&uk); char *keydisp = ssh2_pubkey_openssh_str(&uk);
s->dlgret = seat_verify_ssh_host_key( char **fingerprints = ssh2_all_fingerprints(s->hkey);
s->ppl.seat, s->savedhost, s->savedport,
FingerprintType fptype_default =
ssh2_pick_default_fingerprint(fingerprints);
ppl_logevent("Host key fingerprint is:");
ppl_logevent("%s", fingerprints[fptype_default]);
s->dlgret = verify_ssh_host_key(
s->ppl.seat, s->conf, s->savedhost, s->savedport, s->hkey,
ssh_key_cache_id(s->hkey), s->keystr, keydisp, ssh_key_cache_id(s->hkey), s->keystr, keydisp,
fingerprints, ssh2_transport_dialog_callback, s); fingerprints, ssh2_transport_dialog_callback, s);
sfree(keydisp);
ssh2_free_all_fingerprints(fingerprints); ssh2_free_all_fingerprints(fingerprints);
sfree(keydisp);
}
#ifdef FUZZING #ifdef FUZZING
s->dlgret = 1; s->dlgret = 1;
#endif #endif
crMaybeWaitUntilV(s->dlgret >= 0); crMaybeWaitUntilV(s->dlgret >= 0);
if (s->dlgret == 0) { if (s->dlgret == 0) {
ssh_user_close(s->ppl.ssh, ssh_user_close(s->ppl.ssh,
"User aborted at host key verification"); "User aborted at host key verification");
*aborted = true; *aborted = true;
return; return;
}
} }
/* /*

View File

@ -238,42 +238,29 @@ static void ssh1_login_process_queue(PacketProtocolLayer *ppl)
* Verify the host key. * Verify the host key.
*/ */
{ {
/*
* First format the key into a string.
*/
char *keystr = rsastr_fmt(&s->hostkey); char *keystr = rsastr_fmt(&s->hostkey);
char *keydisp = ssh1_pubkey_str(&s->hostkey);
char **fingerprints = rsa_ssh1_fake_all_fingerprints(&s->hostkey); char **fingerprints = rsa_ssh1_fake_all_fingerprints(&s->hostkey);
/* First check against manually configured host keys. */ s->dlgret = verify_ssh_host_key(
s->dlgret = verify_ssh_manual_host_key(s->conf, fingerprints, NULL); s->ppl.seat, s->conf, s->savedhost, s->savedport, NULL,
if (s->dlgret == 0) { /* did not match */ "rsa", keystr, keydisp, fingerprints,
ssh2_free_all_fingerprints(fingerprints); ssh1_login_dialog_callback, s);
sfree(keystr);
ssh_proto_error(s->ppl.ssh, "Host key did not appear in manually "
"configured list");
return;
} else if (s->dlgret < 0) { /* none configured; use standard handling */
char *keydisp = ssh1_pubkey_str(&s->hostkey);
s->dlgret = seat_verify_ssh_host_key(
s->ppl.seat, s->savedhost, s->savedport, "rsa", keystr,
keydisp, fingerprints, ssh1_login_dialog_callback, s);
sfree(keydisp);
ssh2_free_all_fingerprints(fingerprints);
sfree(keystr);
#ifdef FUZZING
s->dlgret = 1;
#endif
crMaybeWaitUntilV(s->dlgret >= 0);
if (s->dlgret == 0) { ssh2_free_all_fingerprints(fingerprints);
ssh_user_close(s->ppl.ssh, sfree(keydisp);
"User aborted at host key verification"); sfree(keystr);
return; }
}
} else { #ifdef FUZZING
ssh2_free_all_fingerprints(fingerprints); s->dlgret = 1;
sfree(keystr); #endif
} crMaybeWaitUntilV(s->dlgret >= 0);
if (s->dlgret == 0) {
ssh_user_close(s->ppl.ssh,
"User aborted at host key verification");
return;
} }
for (i = 0; i < 32; i++) { for (i = 0; i < 32; i++) {

View File

@ -116,7 +116,7 @@ static const SeatVtable server_seat_vt = {
.update_specials_menu = nullseat_update_specials_menu, .update_specials_menu = nullseat_update_specials_menu,
.get_ttymode = nullseat_get_ttymode, .get_ttymode = nullseat_get_ttymode,
.set_busy_status = nullseat_set_busy_status, .set_busy_status = nullseat_set_busy_status,
.verify_ssh_host_key = nullseat_verify_ssh_host_key, .confirm_ssh_host_key = nullseat_confirm_ssh_host_key,
.confirm_weak_crypto_primitive = server_confirm_weak_crypto_primitive, .confirm_weak_crypto_primitive = server_confirm_weak_crypto_primitive,
.confirm_weak_cached_hostkey = server_confirm_weak_cached_hostkey, .confirm_weak_cached_hostkey = server_confirm_weak_cached_hostkey,
.is_utf8 = nullseat_is_never_utf8, .is_utf8 = nullseat_is_never_utf8,

View File

@ -195,7 +195,7 @@ static const SeatVtable sesschan_seat_vt = {
.update_specials_menu = nullseat_update_specials_menu, .update_specials_menu = nullseat_update_specials_menu,
.get_ttymode = nullseat_get_ttymode, .get_ttymode = nullseat_get_ttymode,
.set_busy_status = nullseat_set_busy_status, .set_busy_status = nullseat_set_busy_status,
.verify_ssh_host_key = nullseat_verify_ssh_host_key, .confirm_ssh_host_key = nullseat_confirm_ssh_host_key,
.confirm_weak_crypto_primitive = nullseat_confirm_weak_crypto_primitive, .confirm_weak_crypto_primitive = nullseat_confirm_weak_crypto_primitive,
.confirm_weak_cached_hostkey = nullseat_confirm_weak_cached_hostkey, .confirm_weak_cached_hostkey = nullseat_confirm_weak_cached_hostkey,
.is_utf8 = nullseat_is_never_utf8, .is_utf8 = nullseat_is_never_utf8,

View File

@ -338,10 +338,10 @@ static void sshproxy_connection_fatal(Seat *seat, const char *message)
} }
} }
static int sshproxy_verify_ssh_host_key( static int sshproxy_confirm_ssh_host_key(
Seat *seat, const char *host, int port, const char *keytype, Seat *seat, const char *host, int port, const char *keytype,
char *keystr, const char *keydisp, char **key_fingerprints, char *keystr, const char *keydisp, char **key_fingerprints, bool mismatch,
void (*callback)(void *ctx, int result), void *ctx) void (*callback)(void *ctx, int result), void *ctx)
{ {
SshProxy *sp = container_of(seat, SshProxy, seat); SshProxy *sp = container_of(seat, SshProxy, seat);
@ -350,39 +350,17 @@ static int sshproxy_verify_ssh_host_key(
* If we have access to the outer Seat, pass this prompt * If we have access to the outer Seat, pass this prompt
* request on to it. FIXME: appropriately adjusted * request on to it. FIXME: appropriately adjusted
*/ */
return seat_verify_ssh_host_key( return seat_confirm_ssh_host_key(
sp->clientseat, host, port, keytype, keystr, keydisp, sp->clientseat, host, port, keytype, keystr, keydisp,
key_fingerprints, callback, ctx); key_fingerprints, mismatch, callback, ctx);
} }
/* /*
* Otherwise, behave as if we're in batch mode: directly verify * Otherwise, behave as if we're in batch mode, i.e. take the safe
* the host key against the cache, and if that fails, take the * option in the absence of interactive confirmation, i.e. abort
* safe option in the absence of interactive confirmation, and * the connection.
* abort the connection.
*/ */
int hkstatus = verify_host_key(host, port, keytype, keystr); return 0;
FingerprintType fptype = ssh2_pick_default_fingerprint(key_fingerprints);
switch (hkstatus) {
case 0: /* host key matched */
return 1;
case 1: /* host key not in cache at all */
sshproxy_error(sp, "Host key not in cache for %s:%d (fingerprint %s). "
"Abandoning proxy SSH connection.", host, port,
key_fingerprints[fptype]);
return 0;
case 2:
sshproxy_error(sp, "HOST KEY DOES NOT MATCH CACHE for %s:%d "
"(fingerprint %s). Abandoning proxy SSH connection.",
host, port, key_fingerprints[fptype]);
return 0;
default:
unreachable("bad return value from verify_host_key");
}
} }
static int sshproxy_confirm_weak_crypto_primitive( static int sshproxy_confirm_weak_crypto_primitive(
@ -482,7 +460,7 @@ static const SeatVtable SshProxy_seat_vt = {
.update_specials_menu = nullseat_update_specials_menu, .update_specials_menu = nullseat_update_specials_menu,
.get_ttymode = nullseat_get_ttymode, .get_ttymode = nullseat_get_ttymode,
.set_busy_status = nullseat_set_busy_status, .set_busy_status = nullseat_set_busy_status,
.verify_ssh_host_key = sshproxy_verify_ssh_host_key, .confirm_ssh_host_key = sshproxy_confirm_ssh_host_key,
.confirm_weak_crypto_primitive = sshproxy_confirm_weak_crypto_primitive, .confirm_weak_crypto_primitive = sshproxy_confirm_weak_crypto_primitive,
.confirm_weak_cached_hostkey = sshproxy_confirm_weak_cached_hostkey, .confirm_weak_cached_hostkey = sshproxy_confirm_weak_cached_hostkey,
.is_utf8 = nullseat_is_never_utf8, .is_utf8 = nullseat_is_never_utf8,

View File

@ -81,8 +81,8 @@ void enum_settings_finish(settings_e *handle);
* be 0 (entry matches database), 1 (entry is absent in database), * be 0 (entry matches database), 1 (entry is absent in database),
* or 2 (entry exists in database and is different). * or 2 (entry exists in database and is different).
*/ */
int verify_host_key(const char *hostname, int port, int check_stored_host_key(const char *hostname, int port,
const char *keytype, const char *key); const char *keytype, const char *key);
/* /*
* Write a host key into the database, overwriting any previous * Write a host key into the database, overwriting any previous

View File

@ -102,30 +102,20 @@ static int block_and_read(int fd, void *buf, size_t len)
return ret; return ret;
} }
int console_verify_ssh_host_key( int console_confirm_ssh_host_key(
Seat *seat, const char *host, int port, const char *keytype, Seat *seat, const char *host, int port, const char *keytype,
char *keystr, const char *keydisp, char **fingerprints, char *keystr, const char *keydisp, char **fingerprints, bool mismatch,
void (*callback)(void *ctx, int result), void *ctx) void (*callback)(void *ctx, int result), void *ctx)
{ {
int ret;
char line[32]; char line[32];
struct termios cf; struct termios cf;
char *common; char *common;
const char *intro, *prompt; const char *intro, *prompt;
/*
* Verify the key.
*/
ret = verify_host_key(host, port, keytype, keystr);
if (ret == 0) /* success - key matched OK */
return 1;
FingerprintType fptype_default = FingerprintType fptype_default =
ssh2_pick_default_fingerprint(fingerprints); ssh2_pick_default_fingerprint(fingerprints);
if (ret == 2) { /* key was different */ if (mismatch) { /* key was different */
common = hk_wrongmsg_common(host, port, keytype, common = hk_wrongmsg_common(host, port, keytype,
fingerprints[fptype_default]); fingerprints[fptype_default]);
intro = hk_wrongmsg_interactive_intro; intro = hk_wrongmsg_interactive_intro;

View File

@ -3448,7 +3448,7 @@ GtkWidget *create_message_box(
NULL /* action_postproc */, NULL /* postproc_ctx */); NULL /* action_postproc */, NULL /* postproc_ctx */);
} }
struct verify_ssh_host_key_dialog_ctx { struct confirm_ssh_host_key_dialog_ctx {
char *host; char *host;
int port; int port;
char *keytype; char *keytype;
@ -3462,10 +3462,10 @@ struct verify_ssh_host_key_dialog_ctx {
GtkWidget *more_info_dialog; GtkWidget *more_info_dialog;
}; };
static void verify_ssh_host_key_result_callback(void *vctx, int result) static void confirm_ssh_host_key_result_callback(void *vctx, int result)
{ {
struct verify_ssh_host_key_dialog_ctx *ctx = struct confirm_ssh_host_key_dialog_ctx *ctx =
(struct verify_ssh_host_key_dialog_ctx *)vctx; (struct confirm_ssh_host_key_dialog_ctx *)vctx;
if (result >= 0) { if (result >= 0) {
int logical_result; int logical_result;
@ -3518,16 +3518,16 @@ static GtkWidget *add_more_info_button(GtkWidget *w, void *vctx)
static void more_info_closed(void *vctx, int result) static void more_info_closed(void *vctx, int result)
{ {
struct verify_ssh_host_key_dialog_ctx *ctx = struct confirm_ssh_host_key_dialog_ctx *ctx =
(struct verify_ssh_host_key_dialog_ctx *)vctx; (struct confirm_ssh_host_key_dialog_ctx *)vctx;
ctx->more_info_dialog = NULL; ctx->more_info_dialog = NULL;
} }
static void more_info_button_clicked(GtkButton *button, gpointer vctx) static void more_info_button_clicked(GtkButton *button, gpointer vctx)
{ {
struct verify_ssh_host_key_dialog_ctx *ctx = struct confirm_ssh_host_key_dialog_ctx *ctx =
(struct verify_ssh_host_key_dialog_ctx *)vctx; (struct confirm_ssh_host_key_dialog_ctx *)vctx;
if (ctx->more_info_dialog) if (ctx->more_info_dialog)
return; return;
@ -3539,9 +3539,9 @@ static void more_info_button_clicked(GtkButton *button, gpointer vctx)
&buttons_ok, more_info_closed, ctx); &buttons_ok, more_info_closed, ctx);
} }
int gtk_seat_verify_ssh_host_key( int gtk_seat_confirm_ssh_host_key(
Seat *seat, const char *host, int port, const char *keytype, Seat *seat, const char *host, int port, const char *keytype,
char *keystr, const char *keydisp, char **fingerprints, char *keystr, const char *keydisp, char **fingerprints, bool mismatch,
void (*callback)(void *ctx, int result), void *ctx) void (*callback)(void *ctx, int result), void *ctx)
{ {
static const char absenttxt[] = static const char absenttxt[] =
@ -3584,25 +3584,16 @@ int gtk_seat_verify_ssh_host_key(
}; };
char *text; char *text;
int ret; struct confirm_ssh_host_key_dialog_ctx *result_ctx;
struct verify_ssh_host_key_dialog_ctx *result_ctx;
GtkWidget *mainwin, *msgbox; GtkWidget *mainwin, *msgbox;
/*
* Verify the key.
*/
ret = verify_host_key(host, port, keytype, keystr);
if (ret == 0) /* success - key matched OK */
return 1;
FingerprintType fptype_default = FingerprintType fptype_default =
ssh2_pick_default_fingerprint(fingerprints); ssh2_pick_default_fingerprint(fingerprints);
text = dupprintf((ret == 2 ? wrongtxt : absenttxt), host, port, text = dupprintf((mismatch ? wrongtxt : absenttxt), host, port,
keytype, fingerprints[fptype_default]); keytype, fingerprints[fptype_default]);
result_ctx = snew(struct verify_ssh_host_key_dialog_ctx); result_ctx = snew(struct confirm_ssh_host_key_dialog_ctx);
result_ctx->callback = callback; result_ctx->callback = callback;
result_ctx->callback_ctx = ctx; result_ctx->callback_ctx = ctx;
result_ctx->host = dupstr(host); result_ctx->host = dupstr(host);
@ -3616,7 +3607,7 @@ int gtk_seat_verify_ssh_host_key(
msgbox = create_message_box_general( msgbox = create_message_box_general(
mainwin, "PuTTY Security Alert", text, mainwin, "PuTTY Security Alert", text,
string_width(fingerprints[fptype_default]), true, string_width(fingerprints[fptype_default]), true,
&buttons_hostkey, verify_ssh_host_key_result_callback, result_ctx, &buttons_hostkey, confirm_ssh_host_key_result_callback, result_ctx,
add_more_info_button, &more_info_button); add_more_info_button, &more_info_button);
result_ctx->main_dialog = msgbox; result_ctx->main_dialog = msgbox;

View File

@ -217,9 +217,9 @@ void showeventlog(eventlog_stuff *estuff, void *parentwin);
void logevent_dlg(eventlog_stuff *estuff, const char *string); void logevent_dlg(eventlog_stuff *estuff, const char *string);
int gtkdlg_askappend(Seat *seat, Filename *filename, int gtkdlg_askappend(Seat *seat, Filename *filename,
void (*callback)(void *ctx, int result), void *ctx); void (*callback)(void *ctx, int result), void *ctx);
int gtk_seat_verify_ssh_host_key( int gtk_seat_confirm_ssh_host_key(
Seat *seat, const char *host, int port, const char *keytype, Seat *seat, const char *host, int port, const char *keytype,
char *keystr, const char *keydisp, char **fingerprints, char *keystr, const char *keydisp, char **fingerprints, bool mismatch,
void (*callback)(void *ctx, int result), void *ctx); void (*callback)(void *ctx, int result), void *ctx);
int gtk_seat_confirm_weak_crypto_primitive( int gtk_seat_confirm_weak_crypto_primitive(
Seat *seat, const char *algtype, const char *algname, Seat *seat, const char *algtype, const char *algname,

View File

@ -399,7 +399,7 @@ static const SeatVtable plink_seat_vt = {
.update_specials_menu = nullseat_update_specials_menu, .update_specials_menu = nullseat_update_specials_menu,
.get_ttymode = plink_get_ttymode, .get_ttymode = plink_get_ttymode,
.set_busy_status = nullseat_set_busy_status, .set_busy_status = nullseat_set_busy_status,
.verify_ssh_host_key = console_verify_ssh_host_key, .confirm_ssh_host_key = console_confirm_ssh_host_key,
.confirm_weak_crypto_primitive = console_confirm_weak_crypto_primitive, .confirm_weak_crypto_primitive = console_confirm_weak_crypto_primitive,
.confirm_weak_cached_hostkey = console_confirm_weak_cached_hostkey, .confirm_weak_cached_hostkey = console_confirm_weak_cached_hostkey,
.is_utf8 = nullseat_is_never_utf8, .is_utf8 = nullseat_is_never_utf8,

View File

@ -595,8 +595,8 @@ void enum_settings_finish(settings_e *handle)
* *
* rsa@22:foovax.example.org 0x23,0x293487364395345345....2343 * rsa@22:foovax.example.org 0x23,0x293487364395345345....2343
*/ */
int verify_host_key(const char *hostname, int port, int check_stored_host_key(const char *hostname, int port,
const char *keytype, const char *key) const char *keytype, const char *key)
{ {
FILE *fp; FILE *fp;
char *filename; char *filename;
@ -668,10 +668,10 @@ bool have_ssh_host_key(const char *hostname, int port,
const char *keytype) const char *keytype)
{ {
/* /*
* If we have a host key, verify_host_key will return 0 or 2. * If we have a host key, check_stored_host_key will return 0 or 2.
* If we don't have one, it'll return 1. * If we don't have one, it'll return 1.
*/ */
return verify_host_key(hostname, port, keytype, "") != 1; return check_stored_host_key(hostname, port, keytype, "") != 1;
} }
void store_host_key(const char *hostname, int port, void store_host_key(const char *hostname, int port,

View File

@ -398,7 +398,7 @@ static const SeatVtable gtk_seat_vt = {
.update_specials_menu = gtk_seat_update_specials_menu, .update_specials_menu = gtk_seat_update_specials_menu,
.get_ttymode = gtk_seat_get_ttymode, .get_ttymode = gtk_seat_get_ttymode,
.set_busy_status = gtk_seat_set_busy_status, .set_busy_status = gtk_seat_set_busy_status,
.verify_ssh_host_key = gtk_seat_verify_ssh_host_key, .confirm_ssh_host_key = gtk_seat_confirm_ssh_host_key,
.confirm_weak_crypto_primitive = gtk_seat_confirm_weak_crypto_primitive, .confirm_weak_crypto_primitive = gtk_seat_confirm_weak_crypto_primitive,
.confirm_weak_cached_hostkey = gtk_seat_confirm_weak_cached_hostkey, .confirm_weak_cached_hostkey = gtk_seat_confirm_weak_cached_hostkey,
.is_utf8 = gtk_seat_is_utf8, .is_utf8 = gtk_seat_is_utf8,

View File

@ -16,9 +16,9 @@ void nullseat_connection_fatal(Seat *seat, const char *message) {}
void nullseat_update_specials_menu(Seat *seat) {} void nullseat_update_specials_menu(Seat *seat) {}
char *nullseat_get_ttymode(Seat *seat, const char *mode) { return NULL; } char *nullseat_get_ttymode(Seat *seat, const char *mode) { return NULL; }
void nullseat_set_busy_status(Seat *seat, BusyStatus status) {} void nullseat_set_busy_status(Seat *seat, BusyStatus status) {}
int nullseat_verify_ssh_host_key( int nullseat_confirm_ssh_host_key(
Seat *seat, const char *host, int port, const char *keytype, Seat *seat, const char *host, int port, const char *keytype,
char *keystr, const char *keydisp, char **key_fingerprints, char *keystr, const char *keydisp, char **key_fingerprints, bool mismatch,
void (*callback)(void *ctx, int result), void *ctx) { return 0; } void (*callback)(void *ctx, int result), void *ctx) { return 0; }
int nullseat_confirm_weak_crypto_primitive( int nullseat_confirm_weak_crypto_primitive(
Seat *seat, const char *algtype, const char *algname, Seat *seat, const char *algtype, const char *algname,

View File

@ -224,12 +224,12 @@ static int tempseat_get_userpass_input(Seat *seat, prompts_t *p)
unreachable("get_userpass_input should never be called on TempSeat"); unreachable("get_userpass_input should never be called on TempSeat");
} }
static int tempseat_verify_ssh_host_key( static int tempseat_confirm_ssh_host_key(
Seat *seat, const char *host, int port, const char *keytype, Seat *seat, const char *host, int port, const char *keytype,
char *keystr, const char *keydisp, char **key_fingerprints, char *keystr, const char *keydisp, char **key_fingerprints, bool mismatch,
void (*callback)(void *ctx, int result), void *ctx) void (*callback)(void *ctx, int result), void *ctx)
{ {
unreachable("verify_ssh_host_key should never be called on TempSeat"); unreachable("confirm_ssh_host_key should never be called on TempSeat");
} }
static int tempseat_confirm_weak_crypto_primitive( static int tempseat_confirm_weak_crypto_primitive(
@ -307,7 +307,7 @@ static const struct SeatVtable tempseat_vt = {
.update_specials_menu = tempseat_update_specials_menu, .update_specials_menu = tempseat_update_specials_menu,
.get_ttymode = tempseat_get_ttymode, .get_ttymode = tempseat_get_ttymode,
.set_busy_status = tempseat_set_busy_status, .set_busy_status = tempseat_set_busy_status,
.verify_ssh_host_key = tempseat_verify_ssh_host_key, .confirm_ssh_host_key = tempseat_confirm_ssh_host_key,
.confirm_weak_crypto_primitive = tempseat_confirm_weak_crypto_primitive, .confirm_weak_crypto_primitive = tempseat_confirm_weak_crypto_primitive,
.confirm_weak_cached_hostkey = tempseat_confirm_weak_cached_hostkey, .confirm_weak_cached_hostkey = tempseat_confirm_weak_cached_hostkey,
.is_utf8 = tempseat_is_utf8, .is_utf8 = tempseat_is_utf8,

View File

@ -32,12 +32,11 @@ void console_print_error_msg(const char *prefix, const char *msg)
fflush(stderr); fflush(stderr);
} }
int console_verify_ssh_host_key( int console_confirm_ssh_host_key(
Seat *seat, const char *host, int port, const char *keytype, Seat *seat, const char *host, int port, const char *keytype,
char *keystr, const char *keydisp, char **fingerprints, char *keystr, const char *keydisp, char **fingerprints, bool mismatch,
void (*callback)(void *ctx, int result), void *ctx) void (*callback)(void *ctx, int result), void *ctx)
{ {
int ret;
HANDLE hin; HANDLE hin;
DWORD savemode, i; DWORD savemode, i;
char *common; char *common;
@ -45,18 +44,10 @@ int console_verify_ssh_host_key(
char line[32]; char line[32];
/*
* Verify the key against the registry.
*/
ret = verify_host_key(host, port, keytype, keystr);
if (ret == 0) /* success - key matched OK */
return 1;
FingerprintType fptype_default = FingerprintType fptype_default =
ssh2_pick_default_fingerprint(fingerprints); ssh2_pick_default_fingerprint(fingerprints);
if (ret == 2) { /* key was different */ if (mismatch) { /* key was different */
common = hk_wrongmsg_common(host, port, keytype, common = hk_wrongmsg_common(host, port, keytype,
fingerprints[fptype_default]); fingerprints[fptype_default]);
intro = hk_wrongmsg_interactive_intro; intro = hk_wrongmsg_interactive_intro;

View File

@ -976,52 +976,43 @@ static INT_PTR CALLBACK HostKeyDialogProc(HWND hwnd, UINT msg,
return 0; return 0;
} }
int win_seat_verify_ssh_host_key( int win_seat_confirm_ssh_host_key(
Seat *seat, const char *host, int port, const char *keytype, Seat *seat, const char *host, int port, const char *keytype,
char *keystr, const char *keydisp, char **fingerprints, char *keystr, const char *keydisp, char **fingerprints, bool mismatch,
void (*callback)(void *ctx, int result), void *ctx) void (*callback)(void *ctx, int result), void *vctx)
{ {
int ret;
WinGuiSeat *wgs = container_of(seat, WinGuiSeat, seat); WinGuiSeat *wgs = container_of(seat, WinGuiSeat, seat);
/* static const char *const keywords[] =
* Verify the key against the registry. { "{KEYTYPE}", "{APPNAME}", NULL };
*/
ret = verify_host_key(host, port, keytype, keystr);
if (ret == 0) /* success - key matched OK */ const char *values[2];
values[0] = keytype;
values[1] = appname;
struct hostkey_dialog_ctx ctx[1];
ctx->keywords = keywords;
ctx->values = values;
ctx->fingerprints = fingerprints;
ctx->fptype_default = ssh2_pick_default_fingerprint(fingerprints);
ctx->keydisp = keydisp;
ctx->iconid = (mismatch ? IDI_WARNING : IDI_QUESTION);
ctx->helpctx = (mismatch ? WINHELP_CTX_errors_hostkey_changed :
WINHELP_CTX_errors_hostkey_absent);
ctx->host = host;
ctx->port = port;
int dlgid = (mismatch ? IDD_HK_WRONG : IDD_HK_ABSENT);
int mbret = DialogBoxParam(
hinst, MAKEINTRESOURCE(dlgid), wgs->term_hwnd,
HostKeyDialogProc, (LPARAM)ctx);
assert(mbret==IDC_HK_ACCEPT || mbret==IDC_HK_ONCE || mbret==IDCANCEL);
if (mbret == IDC_HK_ACCEPT) {
store_host_key(host, port, keytype, keystr);
return 1;
} else if (mbret == IDC_HK_ONCE) {
return 1; return 1;
else {
static const char *const keywords[] =
{ "{KEYTYPE}", "{APPNAME}", NULL };
const char *values[2];
values[0] = keytype;
values[1] = appname;
struct hostkey_dialog_ctx ctx[1];
ctx->keywords = keywords;
ctx->values = values;
ctx->fingerprints = fingerprints;
ctx->fptype_default = ssh2_pick_default_fingerprint(fingerprints);
ctx->keydisp = keydisp;
ctx->iconid = (ret == 2 ? IDI_WARNING : IDI_QUESTION);
ctx->helpctx = (ret == 2 ? WINHELP_CTX_errors_hostkey_changed :
WINHELP_CTX_errors_hostkey_absent);
ctx->host = host;
ctx->port = port;
int dlgid = (ret == 2 ? IDD_HK_WRONG : IDD_HK_ABSENT);
int mbret = DialogBoxParam(
hinst, MAKEINTRESOURCE(dlgid), wgs->term_hwnd,
HostKeyDialogProc, (LPARAM)ctx);
assert(mbret==IDC_HK_ACCEPT || mbret==IDC_HK_ONCE || mbret==IDCANCEL);
if (mbret == IDC_HK_ACCEPT) {
store_host_key(host, port, keytype, keystr);
return 1;
} else if (mbret == IDC_HK_ONCE)
return 1;
} }
return 0; /* abandon the connection */ return 0; /* abandon the connection */
} }

View File

@ -220,9 +220,9 @@ int has_embedded_chm(void); /* 1 = yes, 0 = no, -1 = N/A */
* GUI seat methods in windlg.c, so that the vtable definition in * GUI seat methods in windlg.c, so that the vtable definition in
* window.c can refer to them. * window.c can refer to them.
*/ */
int win_seat_verify_ssh_host_key( int win_seat_confirm_ssh_host_key(
Seat *seat, const char *host, int port, const char *keytype, Seat *seat, const char *host, int port, const char *keytype,
char *keystr, const char *keydisp, char **key_fingerprints, char *keystr, const char *keydisp, char **key_fingerprints, bool mismatch,
void (*callback)(void *ctx, int result), void *ctx); void (*callback)(void *ctx, int result), void *ctx);
int win_seat_confirm_weak_crypto_primitive( int win_seat_confirm_weak_crypto_primitive(
Seat *seat, const char *algtype, const char *algname, Seat *seat, const char *algtype, const char *algname,

View File

@ -93,7 +93,7 @@ static const SeatVtable plink_seat_vt = {
.update_specials_menu = nullseat_update_specials_menu, .update_specials_menu = nullseat_update_specials_menu,
.get_ttymode = nullseat_get_ttymode, .get_ttymode = nullseat_get_ttymode,
.set_busy_status = nullseat_set_busy_status, .set_busy_status = nullseat_set_busy_status,
.verify_ssh_host_key = console_verify_ssh_host_key, .confirm_ssh_host_key = console_confirm_ssh_host_key,
.confirm_weak_crypto_primitive = console_confirm_weak_crypto_primitive, .confirm_weak_crypto_primitive = console_confirm_weak_crypto_primitive,
.confirm_weak_cached_hostkey = console_confirm_weak_cached_hostkey, .confirm_weak_cached_hostkey = console_confirm_weak_cached_hostkey,
.is_utf8 = nullseat_is_never_utf8, .is_utf8 = nullseat_is_never_utf8,

View File

@ -322,8 +322,8 @@ static void hostkey_regname(strbuf *sb, const char *hostname,
escape_registry_key(hostname, sb); escape_registry_key(hostname, sb);
} }
int verify_host_key(const char *hostname, int port, int check_stored_host_key(const char *hostname, int port,
const char *keytype, const char *key) const char *keytype, const char *key)
{ {
char *otherstr; char *otherstr;
strbuf *regname; strbuf *regname;
@ -437,10 +437,10 @@ bool have_ssh_host_key(const char *hostname, int port,
const char *keytype) const char *keytype)
{ {
/* /*
* If we have a host key, verify_host_key will return 0 or 2. * If we have a host key, check_stored_host_key will return 0 or 2.
* If we don't have one, it'll return 1. * If we don't have one, it'll return 1.
*/ */
return verify_host_key(hostname, port, keytype, "") != 1; return check_stored_host_key(hostname, port, keytype, "") != 1;
} }
void store_host_key(const char *hostname, int port, void store_host_key(const char *hostname, int port,

View File

@ -340,7 +340,7 @@ static const SeatVtable win_seat_vt = {
.update_specials_menu = win_seat_update_specials_menu, .update_specials_menu = win_seat_update_specials_menu,
.get_ttymode = win_seat_get_ttymode, .get_ttymode = win_seat_get_ttymode,
.set_busy_status = win_seat_set_busy_status, .set_busy_status = win_seat_set_busy_status,
.verify_ssh_host_key = win_seat_verify_ssh_host_key, .confirm_ssh_host_key = win_seat_confirm_ssh_host_key,
.confirm_weak_crypto_primitive = win_seat_confirm_weak_crypto_primitive, .confirm_weak_crypto_primitive = win_seat_confirm_weak_crypto_primitive,
.confirm_weak_cached_hostkey = win_seat_confirm_weak_cached_hostkey, .confirm_weak_cached_hostkey = win_seat_confirm_weak_cached_hostkey,
.is_utf8 = win_seat_is_utf8, .is_utf8 = win_seat_is_utf8,