1
0
mirror of https://git.tartarus.org/simon/putty.git synced 2025-01-10 01:48:00 +00:00

Allow manually confirming and caching certified keys.

In the case where a server presents a host key signed by a different
certificate from the one you've configured, it need not _always_ be
evidence of wrongdoing. I can imagine situations in which two CAs
cover overlapping sets of things, and you don't want to blanket-trust
one of them, but you do want to connect to a specific host signed by
that one.

Accordingly, PuTTY's previous policy of unconditionally aborting the
connection if certificate validation fails (which was always intended
as a stopgap until I thought through what I wanted to replace it with)
is now replaced by fallback handling: we present the host key
fingerprint to the user and give them the option to accept and/or
cache it based on the public key itself.

This means that the certified key types have to have a representation
in the host key cache. So I've assigned each one a type id, and
generate the cache string itself by simply falling back to the base
key.

(Rationale for the latter: re-signing a public key with a different
certificate doesn't change the _private_ key, or the set of valid
signatures generated with it. So if you've been convinced for reasons
other than the certificate that a particular private key is in the
possession of $host, then proof of ownership of that private key
should be enough to convince you you're talking to $host no matter
what CA has signed the public half this week.)

We now offer to receive a given certified host key type if _either_ we
have at least one CA configured to trust that host, _or_ we have a
certified key of that type cached. (So once you've decided manually
that you trust a particular key, we can still receive that key and
authenticate the host with it, even if you later delete the CA record
that it didn't match anyway.)

One change from normal (uncertified) host key handling is that for
certified key types _all_ the host key prompts use the stronger
language, with "WARNING - POTENTIAL SECURITY BREACH!" rather than the
mild 'hmm, we haven't seen this host before'. Rationale: if you
expected this CA key and got that one, it _could_ be a bold-as-brass
MITM attempt in which someone hoped you'd accept their entire CA key.
The mild wording is only for the case where we had no previous
expectations _at all_ for the host to violate: not a CA _or_ a cached
key.
This commit is contained in:
Simon Tatham 2022-07-16 11:23:13 +01:00
parent a50178eba7
commit 42740a5455
6 changed files with 82 additions and 25 deletions

View File

@ -268,7 +268,7 @@ static const ssh_keyalg *opensshcert_related_alg(const ssh_keyalg *self,
.alternate_ssh_id = opensshcert_alternate_ssh_id, \ .alternate_ssh_id = opensshcert_alternate_ssh_id, \
.related_alg = opensshcert_related_alg, \ .related_alg = opensshcert_related_alg, \
.ssh_id = ssh_alg_id_prefix "-cert-v01@openssh.com", \ .ssh_id = ssh_alg_id_prefix "-cert-v01@openssh.com", \
.cache_id = NULL, \ .cache_id = "opensshcert-" ssh_key_id_prefix, \
.extra = &opensshcert_##name##_extra, \ .extra = &opensshcert_##name##_extra, \
.is_certificate = true, \ .is_certificate = true, \
.base_alg = &name, \ .base_alg = &name, \
@ -559,8 +559,8 @@ static bool opensshcert_has_private(ssh_key *key)
static char *opensshcert_cache_str(ssh_key *key) static char *opensshcert_cache_str(ssh_key *key)
{ {
unreachable( opensshcert_key *ck = container_of(key, opensshcert_key, sshk);
"Certificates are not expected to be stored in the host key cache"); return ssh_key_cache_str(ck->basekey);
} }
static void opensshcert_time_to_iso8601(BinarySink *bs, uint64_t time) static void opensshcert_time_to_iso8601(BinarySink *bs, uint64_t time)

4
ssh.h
View File

@ -1801,8 +1801,8 @@ bool get_commasep_word(ptrlen *list, ptrlen *word);
SeatPromptResult verify_ssh_host_key( SeatPromptResult verify_ssh_host_key(
InteractionReadySeat iseat, Conf *conf, const char *host, int port, InteractionReadySeat iseat, Conf *conf, const char *host, int port,
ssh_key *key, const char *keytype, char *keystr, const char *keydisp, ssh_key *key, const char *keytype, char *keystr, const char *keydisp,
char **fingerprints, void (*callback)(void *ctx, SeatPromptResult result), char **fingerprints, int ca_count,
void *ctx); void (*callback)(void *ctx, SeatPromptResult 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

@ -855,8 +855,8 @@ bool ssh2_bpp_check_unimplemented(BinaryPacketProtocol *bpp, PktIn *pktin)
SeatPromptResult verify_ssh_host_key( SeatPromptResult verify_ssh_host_key(
InteractionReadySeat iseat, Conf *conf, const char *host, int port, InteractionReadySeat iseat, Conf *conf, const char *host, int port,
ssh_key *key, const char *keytype, char *keystr, const char *keydisp, ssh_key *key, const char *keytype, char *keystr, const char *keydisp,
char **fingerprints, void (*callback)(void *ctx, SeatPromptResult result), char **fingerprints, int ca_count,
void *ctx) void (*callback)(void *ctx, SeatPromptResult result), void *ctx)
{ {
/* /*
* First, check if the Conf includes a manual specification of the * First, check if the Conf includes a manual specification of the
@ -934,7 +934,51 @@ SeatPromptResult verify_ssh_host_key(
seat_dialog_text_append( seat_dialog_text_append(
text, SDT_TITLE, "%s Security Alert", appname); text, SDT_TITLE, "%s Security Alert", appname);
if (storage_status == 1) { if (key && ssh_key_alg(key)->is_certificate) {
seat_dialog_text_append(
text, SDT_SCARY_HEADING, "WARNING - POTENTIAL SECURITY BREACH!");
seat_dialog_text_append(
text, SDT_PARA, "This server presented a certified host key:");
seat_dialog_text_append(
text, SDT_DISPLAY, "%s (port %d)", host, port);
if (ca_count) {
seat_dialog_text_append(
text, SDT_PARA, "which was signed by a different "
"certification authority from the %s %s is configured to "
"trust for this server.", ca_count > 1 ? "ones" : "one",
appname);
if (storage_status == 2) {
seat_dialog_text_append(
text, SDT_PARA, "ALSO, that key does not match the key "
"%s had previously cached for this server.", appname);
seat_dialog_text_append(
text, SDT_PARA, "This means that either another "
"certification authority is operating in this realm AND "
"the server administrator has changed the host key, or "
"you have actually connected to another computer "
"pretending to be the server.");
} else {
seat_dialog_text_append(
text, SDT_PARA, "This means that either another "
"certification authority is operating in this realm, or "
"you have actually connected to another computer "
"pretending to be the server.");
}
} else {
assert(storage_status == 2);
seat_dialog_text_append(
text, SDT_PARA, "which does not match the certified key %s "
"had previously cached for this server.", appname);
seat_dialog_text_append(
text, SDT_PARA, "This means that either the server "
"administrator has changed the host key, or you have actually "
"connected to another computer pretending to be the server.");
}
seat_dialog_text_append(
text, SDT_PARA, "The new %s key fingerprint is:", keytype);
seat_dialog_text_append(
text, SDT_DISPLAY, "%s", fingerprints[fptype_default]);
} else if (storage_status == 1) {
seat_dialog_text_append( seat_dialog_text_append(
text, SDT_PARA, "The host key is not cached for this server:"); text, SDT_PARA, "The host key is not cached for this server:");
seat_dialog_text_append( seat_dialog_text_append(

View File

@ -718,8 +718,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s, bool *aborted)
} }
} }
s->keystr = (s->hkey && !ssh_key_alg(s->hkey)->is_certificate ? s->keystr = ssh_key_cache_str(s->hkey);
ssh_key_cache_str(s->hkey) : NULL);
#ifndef NO_GSSAPI #ifndef NO_GSSAPI
if (s->gss_kex_used) { if (s->gss_kex_used) {
/* /*
@ -868,8 +867,6 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s, bool *aborted)
* hash.) * hash.)
*/ */
if (ssh_key_alg(s->hkey)->is_certificate) { if (ssh_key_alg(s->hkey)->is_certificate) {
ssh2_free_all_fingerprints(fingerprints);
char *base_fp = ssh2_fingerprint(ssh_key_base_key(s->hkey), char *base_fp = ssh2_fingerprint(ssh_key_base_key(s->hkey),
fptype_default); fptype_default);
ppl_logevent("Host key is a certificate, whose base key has " ppl_logevent("Host key is a certificate, whose base key has "
@ -917,24 +914,26 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s, bool *aborted)
} }
if (cert_ok) { if (cert_ok) {
strbuf_free(error); strbuf_free(error);
ssh2_free_all_fingerprints(fingerprints);
ppl_logevent("Accepted certificate"); ppl_logevent("Accepted certificate");
goto host_key_ok;
} else { } else {
ppl_logevent("Rejected host key certificate: %s", ppl_logevent("Rejected host key certificate: %s",
error->s); error->s);
ssh_sw_abort(s->ppl.ssh, /* now fall through into normal host key checking */
"Rejected host key certificate: %s",
error->s);
*aborted = true;
strbuf_free(error);
return;
} }
} else { }
{
char *keydisp = ssh2_pubkey_openssh_str(&uk); char *keydisp = ssh2_pubkey_openssh_str(&uk);
int ca_count = ssh_key_alg(s->hkey)->is_certificate ?
count234(s->host_cas) : 0;
s->spr = verify_ssh_host_key( s->spr = verify_ssh_host_key(
ppl_get_iseat(&s->ppl), s->conf, s->savedhost, s->savedport, ppl_get_iseat(&s->ppl), s->conf, s->savedhost, s->savedport,
s->hkey, ssh_key_cache_id(s->hkey), s->keystr, keydisp, s->hkey, ssh_key_cache_id(s->hkey), s->keystr, keydisp,
fingerprints, ssh2_transport_dialog_callback, s); fingerprints, ca_count, ssh2_transport_dialog_callback, s);
ssh2_free_all_fingerprints(fingerprints); ssh2_free_all_fingerprints(fingerprints);
sfree(keydisp); sfree(keydisp);
@ -947,8 +946,22 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s, bool *aborted)
ssh_spr_close(s->ppl.ssh, s->spr, "host key verification"); ssh_spr_close(s->ppl.ssh, s->spr, "host key verification");
return; return;
} }
if (ssh_key_alg(s->hkey)->is_certificate) {
/*
* Explain what's going on in the Event Log: if we
* got here by way of a certified key whose
* certificate we didn't like, then we should
* explain why we chose to continue with the
* connection anyway!
*/
ppl_logevent("Accepting certified host key anyway based "
"on cache");
}
} }
host_key_ok:
/* /*
* Save this host key, to check against the one presented in * Save this host key, to check against the one presented in
* subsequent rekeys. * subsequent rekeys.

View File

@ -243,7 +243,7 @@ static void ssh1_login_process_queue(PacketProtocolLayer *ppl)
s->spr = verify_ssh_host_key( s->spr = verify_ssh_host_key(
ppl_get_iseat(&s->ppl), s->conf, s->savedhost, s->savedport, NULL, ppl_get_iseat(&s->ppl), s->conf, s->savedhost, s->savedport, NULL,
"rsa", keystr, keydisp, fingerprints, "rsa", keystr, keydisp, fingerprints, 0,
ssh1_login_dialog_callback, s); ssh1_login_dialog_callback, s);
ssh2_free_all_fingerprints(fingerprints); ssh2_free_all_fingerprints(fingerprints);

View File

@ -725,8 +725,8 @@ static void ssh2_write_kexinit_lists(
} }
} }
/* Next, add uncertified algorithms we already know a key for /* Next, add algorithms we already know a key for (unless
* (unless configured not to do that) */ * configured not to do that) */
warn = false; warn = false;
for (i = 0; i < n_preferred_hk; i++) { for (i = 0; i < n_preferred_hk; i++) {
if (preferred_hk[i] == HK_WARN) if (preferred_hk[i] == HK_WARN)
@ -734,8 +734,8 @@ static void ssh2_write_kexinit_lists(
for (j = 0; j < lenof(ssh2_hostkey_algs); j++) { for (j = 0; j < lenof(ssh2_hostkey_algs); j++) {
const struct ssh_signkey_with_user_pref_id *a = const struct ssh_signkey_with_user_pref_id *a =
&ssh2_hostkey_algs[j]; &ssh2_hostkey_algs[j];
if (a->alg->is_certificate || !a->alg->cache_id) if (a->alg->is_certificate && accept_certs)
continue; continue; /* already added this one */
if (a->id != preferred_hk[i]) if (a->id != preferred_hk[i])
continue; continue;
if (conf_get_bool(conf, CONF_ssh_prefer_known_hostkeys) && if (conf_get_bool(conf, CONF_ssh_prefer_known_hostkeys) &&