mirror of
https://git.tartarus.org/simon/putty.git
synced 2025-01-25 01:02:24 +00:00
Family of key types for OpenSSH certificates.
This commit is groundwork for full certificate support, but doesn't complete the job by itself. It introduces the new key types, and adds a test in cryptsuite ensuring they work as expected, but nothing else. If you manually construct a PPK file for one of the new key types, so that it has a certificate in the public key field, then this commit enables PuTTY to present that key to a server for user authentication, either directly or via Pageant storing and using it. But I haven't yet provided any mechanism for making such a PPK, so by itself, this isn't much use. Also, these new key types are not yet included in the KEXINIT host keys list, because if they were, they'd just be treated as normal host keys, in that you'd be asked to manually confirm the SSH fingerprint of the certificate. I'll enable them for host keys once I add the missing pieces.
This commit is contained in:
parent
6f7c52dcce
commit
34d01e1b65
@ -21,6 +21,7 @@ add_sources_from_current_dir(crypto
|
||||
md5.c
|
||||
mpint.c
|
||||
ntru.c
|
||||
openssh-certs.c
|
||||
prng.c
|
||||
pubkey-pem.c
|
||||
pubkey-ppk.c
|
||||
|
660
crypto/openssh-certs.c
Normal file
660
crypto/openssh-certs.c
Normal file
@ -0,0 +1,660 @@
|
||||
/*
|
||||
* Public key type for OpenSSH certificates.
|
||||
*/
|
||||
|
||||
#include "ssh.h"
|
||||
|
||||
enum {
|
||||
SSH_CERT_TYPE_USER = 1,
|
||||
SSH_CERT_TYPE_HOST = 2,
|
||||
};
|
||||
|
||||
typedef struct opensshcert_key {
|
||||
strbuf *nonce;
|
||||
uint64_t serial;
|
||||
uint32_t type;
|
||||
strbuf *key_id;
|
||||
strbuf *valid_principals;
|
||||
uint64_t valid_after, valid_before;
|
||||
strbuf *critical_options;
|
||||
strbuf *extensions;
|
||||
strbuf *reserved;
|
||||
strbuf *signature_key;
|
||||
strbuf *signature;
|
||||
|
||||
ssh_key *basekey;
|
||||
|
||||
ssh_key sshk;
|
||||
} opensshcert_key;
|
||||
|
||||
typedef struct blob_fmt {
|
||||
const unsigned *fmt;
|
||||
size_t len;
|
||||
} blob_fmt;
|
||||
|
||||
typedef struct opensshcert_extra {
|
||||
/*
|
||||
* OpenSSH certificate formats aren't completely consistent about
|
||||
* the relationship between the public+private blob uploaded to
|
||||
* the agent for the certified key type, and the one for the base
|
||||
* key type. Here we specify the mapping.
|
||||
*
|
||||
* Each of these foo_fmt strings indicates the layout of a
|
||||
* particular version of the key, in the form of an array of
|
||||
* integers together with a length, with each integer describing
|
||||
* one of the components of the key. The integers are defined by
|
||||
* enums, so that they're tightly packed; the general idea is that
|
||||
* if you're converting from one form to another, then you use the
|
||||
* format list for the source format to read out a succession of
|
||||
* SSH strings from the source data and put them in an array
|
||||
* indexed by the integer ids, and then use the list for the
|
||||
* destination format to write the strings out to the destination
|
||||
* in the right (maybe different) order.
|
||||
*
|
||||
* pub_fmt describes the format of the public-key blob for the
|
||||
* base key type, not counting the initial string giving the key
|
||||
* type identifier itself. As far as I know, this always matches
|
||||
* the format of the public-key data appearing in the middle of
|
||||
* the certificate.
|
||||
*
|
||||
* base_ossh_fmt describes the format of the full OpenSSH blob
|
||||
* appearing in the ssh-agent protocol for the base key,
|
||||
* containing the public and private key data.
|
||||
*
|
||||
* cert_ossh_fmt describes the format of the OpenSSH blob for the
|
||||
* certificate key format, beginning just after the certificate
|
||||
* string itself.
|
||||
*/
|
||||
blob_fmt pub_fmt, base_ossh_fmt, cert_ossh_fmt;
|
||||
} opensshcert_extra;
|
||||
|
||||
/*
|
||||
* The actual integer arrays defining the per-key blob formats.
|
||||
*/
|
||||
|
||||
/* DSA is the most orthodox: only the obviously necessary public key
|
||||
* info appears at all, it's in the same order everywhere, and none of
|
||||
* it is repeated unnecessarily */
|
||||
enum { DSA_p, DSA_q, DSA_g, DSA_y, DSA_x };
|
||||
const unsigned dsa_pub_fmt[] = { DSA_p, DSA_q, DSA_g, DSA_y };
|
||||
const unsigned dsa_base_ossh_fmt[] = { DSA_p, DSA_q, DSA_g, DSA_y, DSA_x };
|
||||
const unsigned dsa_cert_ossh_fmt[] = { DSA_x };
|
||||
|
||||
/* ECDSA is almost as nice, except that it pointlessly mentions the
|
||||
* curve name in the public data, which shouldn't be necessary given
|
||||
* that the SSH key id has already implied it. But at least that's
|
||||
* consistent everywhere. */
|
||||
enum { ECDSA_curve, ECDSA_point, ECDSA_exp };
|
||||
const unsigned ecdsa_pub_fmt[] = { ECDSA_curve, ECDSA_point };
|
||||
const unsigned ecdsa_base_ossh_fmt[] = { ECDSA_curve, ECDSA_point, ECDSA_exp };
|
||||
const unsigned ecdsa_cert_ossh_fmt[] = { ECDSA_exp };
|
||||
|
||||
/* Ed25519 has the oddity that the private data following the
|
||||
* certificate in the OpenSSH blob is preceded by an extra copy of the
|
||||
* public data, for no obviously necessary reason since that doesn't
|
||||
* happen in any of the rest of these formats */
|
||||
enum { EDDSA_point, EDDSA_exp };
|
||||
const unsigned eddsa_pub_fmt[] = { EDDSA_point };
|
||||
const unsigned eddsa_base_ossh_fmt[] = { EDDSA_point, EDDSA_exp };
|
||||
const unsigned eddsa_cert_ossh_fmt[] = { EDDSA_point, EDDSA_exp };
|
||||
|
||||
/* And RSA has the quirk that the modulus and exponent are reversed in
|
||||
* the base key type's OpenSSH blob! */
|
||||
enum { RSA_e, RSA_n, RSA_d, RSA_p, RSA_q, RSA_iqmp };
|
||||
const unsigned rsa_pub_fmt[] = { RSA_e, RSA_n };
|
||||
const unsigned rsa_base_ossh_fmt[] = {
|
||||
RSA_n, RSA_e, RSA_d, RSA_p, RSA_q, RSA_iqmp };
|
||||
const unsigned rsa_cert_ossh_fmt[] = { RSA_d, RSA_p, RSA_q, RSA_iqmp };
|
||||
|
||||
/*
|
||||
* Routines to transform one kind of blob into another based on those
|
||||
* foo_fmt integer arrays.
|
||||
*/
|
||||
typedef struct BlobTransformer {
|
||||
ptrlen *parts;
|
||||
size_t nparts;
|
||||
} BlobTransformer;
|
||||
|
||||
#define BLOBTRANS_DECLARE(bt) BlobTransformer bt[1] = { { NULL, 0 } }
|
||||
|
||||
static inline void blobtrans_clear(BlobTransformer *bt)
|
||||
{
|
||||
sfree(bt->parts);
|
||||
bt->parts = NULL;
|
||||
bt->nparts = 0;
|
||||
}
|
||||
|
||||
static inline bool blobtrans_read(BlobTransformer *bt, BinarySource *src,
|
||||
blob_fmt blob)
|
||||
{
|
||||
size_t nparts = bt->nparts;
|
||||
for (size_t i = 0; i < blob.len; i++)
|
||||
if (nparts < blob.fmt[i]+1)
|
||||
nparts = blob.fmt[i]+1;
|
||||
|
||||
if (nparts > bt->nparts) {
|
||||
bt->parts = sresize(bt->parts, nparts, ptrlen);
|
||||
while (bt->nparts < nparts)
|
||||
bt->parts[bt->nparts++] = make_ptrlen(NULL, 0);
|
||||
}
|
||||
|
||||
for (size_t i = 0; i < blob.len; i++) {
|
||||
size_t j = blob.fmt[i];
|
||||
ptrlen part = get_string(src);
|
||||
if (bt->parts[j].ptr) {
|
||||
/*
|
||||
* If the same string appears in both the public blob and
|
||||
* the private data, check they match. (This happens in
|
||||
* Ed25519: an extra copy of the public point string
|
||||
* appears in the certified OpenSSH data after the
|
||||
* certificate and before the private key.)
|
||||
*/
|
||||
if (!ptrlen_eq_ptrlen(bt->parts[j], part))
|
||||
return false;
|
||||
}
|
||||
bt->parts[j] = part;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
static inline void blobtrans_write(BlobTransformer *bt, BinarySink *bs,
|
||||
blob_fmt blob)
|
||||
{
|
||||
for (size_t i = 0; i < blob.len; i++) {
|
||||
assert(i < bt->nparts);
|
||||
ptrlen part = bt->parts[blob.fmt[i]];
|
||||
assert(part.ptr);
|
||||
put_stringpl(bs, part);
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* Forward declarations.
|
||||
*/
|
||||
static ssh_key *opensshcert_new_pub(const ssh_keyalg *self, ptrlen pub);
|
||||
static ssh_key *opensshcert_new_priv(
|
||||
const ssh_keyalg *self, ptrlen pub, ptrlen priv);
|
||||
static ssh_key *opensshcert_new_priv_openssh(
|
||||
const ssh_keyalg *self, BinarySource *src);
|
||||
static void opensshcert_freekey(ssh_key *key);
|
||||
static char *opensshcert_invalid(ssh_key *key, unsigned flags);
|
||||
static void opensshcert_sign(ssh_key *key, ptrlen data, unsigned flags,
|
||||
BinarySink *bs);
|
||||
static bool opensshcert_verify(ssh_key *key, ptrlen sig, ptrlen data);
|
||||
static void opensshcert_public_blob(ssh_key *key, BinarySink *bs);
|
||||
static void opensshcert_private_blob(ssh_key *key, BinarySink *bs);
|
||||
static void opensshcert_openssh_blob(ssh_key *key, BinarySink *bs);
|
||||
static bool opensshcert_has_private(ssh_key *key);
|
||||
static char *opensshcert_cache_str(ssh_key *key);
|
||||
static key_components *opensshcert_components(ssh_key *key);
|
||||
static int opensshcert_pubkey_bits(const ssh_keyalg *self, ptrlen blob);
|
||||
static unsigned opensshcert_supported_flags(const ssh_keyalg *self);
|
||||
static const char *opensshcert_alternate_ssh_id(const ssh_keyalg *self,
|
||||
unsigned flags);
|
||||
|
||||
/*
|
||||
* Top-level vtables for the certified key formats, defined via a list
|
||||
* macro so I can also make an array of them all.
|
||||
*/
|
||||
|
||||
#define KEYALG_LIST(X) \
|
||||
X(ssh_dsa, "ssh-dss", dsa) \
|
||||
X(ssh_rsa, "ssh-rsa", rsa) \
|
||||
X(ssh_rsa_sha256, "rsa-sha2-256", rsa) \
|
||||
X(ssh_rsa_sha512, "rsa-sha2-512", rsa) \
|
||||
X(ssh_ecdsa_ed25519, "ssh-ed25519", eddsa) \
|
||||
X(ssh_ecdsa_nistp256, "ecdsa-sha2-nistp256", ecdsa) \
|
||||
X(ssh_ecdsa_nistp384, "ecdsa-sha2-nistp384", ecdsa) \
|
||||
X(ssh_ecdsa_nistp521, "ecdsa-sha2-nistp521", ecdsa) \
|
||||
/* end of list */
|
||||
|
||||
#define KEYALG_DEF(name, ssh_id_prefix, fmt_prefix) \
|
||||
const struct opensshcert_extra opensshcert_##name##_extra = { \
|
||||
.pub_fmt = { .fmt = fmt_prefix ## _pub_fmt, \
|
||||
.len = lenof(fmt_prefix ## _pub_fmt) }, \
|
||||
.base_ossh_fmt = { .fmt = fmt_prefix ## _base_ossh_fmt, \
|
||||
.len = lenof(fmt_prefix ## _base_ossh_fmt) }, \
|
||||
.cert_ossh_fmt = { .fmt = fmt_prefix ## _cert_ossh_fmt, \
|
||||
.len = lenof(fmt_prefix ## _cert_ossh_fmt) }, \
|
||||
}; \
|
||||
\
|
||||
const ssh_keyalg opensshcert_##name = { \
|
||||
.new_pub = opensshcert_new_pub, \
|
||||
.new_priv = opensshcert_new_priv, \
|
||||
.new_priv_openssh = opensshcert_new_priv_openssh, \
|
||||
.freekey = opensshcert_freekey, \
|
||||
.invalid = opensshcert_invalid, \
|
||||
.sign = opensshcert_sign, \
|
||||
.verify = opensshcert_verify, \
|
||||
.public_blob = opensshcert_public_blob, \
|
||||
.private_blob = opensshcert_private_blob, \
|
||||
.openssh_blob = opensshcert_openssh_blob, \
|
||||
.has_private = opensshcert_has_private, \
|
||||
.cache_str = opensshcert_cache_str, \
|
||||
.components = opensshcert_components, \
|
||||
.pubkey_bits = opensshcert_pubkey_bits, \
|
||||
.supported_flags = opensshcert_supported_flags, \
|
||||
.alternate_ssh_id = opensshcert_alternate_ssh_id, \
|
||||
.ssh_id = ssh_id_prefix "-cert-v01@openssh.com", \
|
||||
.cache_id = NULL, \
|
||||
.extra = &opensshcert_##name##_extra, \
|
||||
.is_certificate = true, \
|
||||
.base_alg = &name, \
|
||||
};
|
||||
KEYALG_LIST(KEYALG_DEF)
|
||||
#undef KEYALG_DEF
|
||||
|
||||
#define KEYALG_LIST_ENTRY(name, ssh_id_prefix, fmt_prefix) &opensshcert_##name,
|
||||
static const ssh_keyalg *const opensshcert_all_keyalgs[] = {
|
||||
KEYALG_LIST(KEYALG_LIST_ENTRY)
|
||||
};
|
||||
#undef KEYALG_LIST_ENTRY
|
||||
|
||||
static opensshcert_key *opensshcert_new_shared(
|
||||
const ssh_keyalg *self, ptrlen blob, strbuf **basepub_out)
|
||||
{
|
||||
const opensshcert_extra *extra = self->extra;
|
||||
|
||||
BinarySource src[1];
|
||||
BinarySource_BARE_INIT_PL(src, blob);
|
||||
|
||||
/* Check the initial key-type string */
|
||||
if (!ptrlen_eq_string(get_string(src), self->ssh_id))
|
||||
return NULL;
|
||||
|
||||
opensshcert_key *ck = snew(opensshcert_key);
|
||||
memset(ck, 0, sizeof(*ck));
|
||||
ck->sshk.vt = self;
|
||||
|
||||
ck->nonce = strbuf_dup(get_string(src));
|
||||
strbuf *basepub = strbuf_new();
|
||||
{
|
||||
put_stringz(basepub, self->base_alg->ssh_id);
|
||||
|
||||
/* Make the base public key blob out of the public key
|
||||
* material in the certificate. This invocation of the
|
||||
* blobtrans system doesn't do any format translation, but it
|
||||
* does ensure that the right amount of data is copied so that
|
||||
* src ends up in the right position to read the remaining
|
||||
* certificate fields. */
|
||||
BLOBTRANS_DECLARE(bt);
|
||||
blobtrans_read(bt, src, extra->pub_fmt);
|
||||
blobtrans_write(bt, BinarySink_UPCAST(basepub), extra->pub_fmt);
|
||||
blobtrans_clear(bt);
|
||||
}
|
||||
ck->serial = get_uint64(src);
|
||||
ck->type = get_uint32(src);
|
||||
ck->key_id = strbuf_dup(get_string(src));
|
||||
ck->valid_principals = strbuf_dup(get_string(src));
|
||||
ck->valid_after = get_uint64(src);
|
||||
ck->valid_before = get_uint64(src);
|
||||
ck->critical_options = strbuf_dup(get_string(src));
|
||||
ck->extensions = strbuf_dup(get_string(src));
|
||||
ck->reserved = strbuf_dup(get_string(src));
|
||||
ck->signature_key = strbuf_dup(get_string(src));
|
||||
ck->signature = strbuf_dup(get_string(src));
|
||||
|
||||
|
||||
if (get_err(src)) {
|
||||
ssh_key_free(&ck->sshk);
|
||||
strbuf_free(basepub);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
*basepub_out = basepub;
|
||||
return ck;
|
||||
}
|
||||
|
||||
static ssh_key *opensshcert_new_pub(const ssh_keyalg *self, ptrlen pub)
|
||||
{
|
||||
strbuf *basepub;
|
||||
opensshcert_key *ck = opensshcert_new_shared(self, pub, &basepub);
|
||||
if (!ck)
|
||||
return NULL;
|
||||
|
||||
ck->basekey = ssh_key_new_pub(self->base_alg, ptrlen_from_strbuf(basepub));
|
||||
strbuf_free(basepub);
|
||||
|
||||
if (!ck->basekey) {
|
||||
ssh_key_free(&ck->sshk);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
return &ck->sshk;
|
||||
}
|
||||
|
||||
static ssh_key *opensshcert_new_priv(
|
||||
const ssh_keyalg *self, ptrlen pub, ptrlen priv)
|
||||
{
|
||||
strbuf *basepub;
|
||||
opensshcert_key *ck = opensshcert_new_shared(self, pub, &basepub);
|
||||
if (!ck)
|
||||
return NULL;
|
||||
|
||||
ck->basekey = ssh_key_new_priv(self->base_alg,
|
||||
ptrlen_from_strbuf(basepub), priv);
|
||||
strbuf_free(basepub);
|
||||
|
||||
if (!ck->basekey) {
|
||||
ssh_key_free(&ck->sshk);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
return &ck->sshk;
|
||||
}
|
||||
|
||||
static ssh_key *opensshcert_new_priv_openssh(
|
||||
const ssh_keyalg *self, BinarySource *src)
|
||||
{
|
||||
const opensshcert_extra *extra = self->extra;
|
||||
|
||||
ptrlen cert = get_string(src);
|
||||
|
||||
strbuf *basepub;
|
||||
opensshcert_key *ck = opensshcert_new_shared(self, cert, &basepub);
|
||||
if (!ck)
|
||||
return NULL;
|
||||
|
||||
strbuf *baseossh = strbuf_new();
|
||||
|
||||
/* Make the base OpenSSH key blob out of the public key blob
|
||||
* returned from opensshcert_new_shared, and the trailing
|
||||
* private data following the certificate */
|
||||
BLOBTRANS_DECLARE(bt);
|
||||
|
||||
BinarySource pubsrc[1];
|
||||
BinarySource_BARE_INIT_PL(pubsrc, ptrlen_from_strbuf(basepub));
|
||||
get_string(pubsrc); /* skip key type id */
|
||||
|
||||
/* blobtrans_read might fail in this case, because we're reading
|
||||
* from two sources and they might fail to match */
|
||||
bool success = blobtrans_read(bt, pubsrc, extra->pub_fmt) &&
|
||||
blobtrans_read(bt, src, extra->cert_ossh_fmt);
|
||||
|
||||
blobtrans_write(bt, BinarySink_UPCAST(baseossh), extra->base_ossh_fmt);
|
||||
blobtrans_clear(bt);
|
||||
|
||||
if (!success) {
|
||||
ssh_key_free(&ck->sshk);
|
||||
strbuf_free(basepub);
|
||||
strbuf_free(baseossh);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
strbuf_free(basepub);
|
||||
|
||||
BinarySource osshsrc[1];
|
||||
BinarySource_BARE_INIT_PL(osshsrc, ptrlen_from_strbuf(baseossh));
|
||||
ck->basekey = ssh_key_new_priv_openssh(self->base_alg, osshsrc);
|
||||
strbuf_free(baseossh);
|
||||
|
||||
if (!ck->basekey) {
|
||||
ssh_key_free(&ck->sshk);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
return &ck->sshk;
|
||||
}
|
||||
|
||||
static void opensshcert_freekey(ssh_key *key)
|
||||
{
|
||||
opensshcert_key *ck = container_of(key, opensshcert_key, sshk);
|
||||
|
||||
/* If this function is called from one of the above constructors
|
||||
* because it failed part way through, we might not have managed
|
||||
* to construct ck->basekey, so it might be NULL. */
|
||||
if (ck->basekey)
|
||||
ssh_key_free(ck->basekey);
|
||||
|
||||
strbuf_free(ck->nonce);
|
||||
strbuf_free(ck->key_id);
|
||||
strbuf_free(ck->valid_principals);
|
||||
strbuf_free(ck->critical_options);
|
||||
strbuf_free(ck->extensions);
|
||||
strbuf_free(ck->reserved);
|
||||
strbuf_free(ck->signature_key);
|
||||
strbuf_free(ck->signature);
|
||||
|
||||
sfree(ck);
|
||||
}
|
||||
|
||||
static ssh_key *opensshcert_ca_pub_key(opensshcert_key *ck, ptrlen *algname)
|
||||
{
|
||||
ptrlen ca_keyblob = ptrlen_from_strbuf(ck->signature_key);
|
||||
|
||||
if (algname)
|
||||
*algname = pubkey_blob_to_alg_name(ca_keyblob);
|
||||
|
||||
const ssh_keyalg *ca_alg = pubkey_blob_to_alg(ca_keyblob);
|
||||
if (!ca_alg)
|
||||
return NULL; /* don't even recognise the certifying key type */
|
||||
|
||||
return ssh_key_new_pub(ca_alg, ca_keyblob);
|
||||
}
|
||||
|
||||
static void opensshcert_signature_preimage(opensshcert_key *ck, BinarySink *bs)
|
||||
{
|
||||
put_stringz(bs, ck->sshk.vt->ssh_id);
|
||||
put_stringpl(bs, ptrlen_from_strbuf(ck->nonce));
|
||||
|
||||
strbuf *basepub = strbuf_new();
|
||||
ssh_key_public_blob(ck->basekey, BinarySink_UPCAST(basepub));
|
||||
BinarySource src[1];
|
||||
BinarySource_BARE_INIT_PL(src, ptrlen_from_strbuf(basepub));
|
||||
get_string(src); /* skip initial key type string */
|
||||
put_data(bs, get_ptr(src), get_avail(src));
|
||||
strbuf_free(basepub);
|
||||
|
||||
put_uint64(bs, ck->serial);
|
||||
put_uint32(bs, ck->type);
|
||||
put_stringpl(bs, ptrlen_from_strbuf(ck->key_id));
|
||||
put_stringpl(bs, ptrlen_from_strbuf(ck->valid_principals));
|
||||
put_uint64(bs, ck->valid_after);
|
||||
put_uint64(bs, ck->valid_before);
|
||||
put_stringpl(bs, ptrlen_from_strbuf(ck->critical_options));
|
||||
put_stringpl(bs, ptrlen_from_strbuf(ck->extensions));
|
||||
put_stringpl(bs, ptrlen_from_strbuf(ck->reserved));
|
||||
put_stringpl(bs, ptrlen_from_strbuf(ck->signature_key));
|
||||
}
|
||||
|
||||
static void opensshcert_public_blob(ssh_key *key, BinarySink *bs)
|
||||
{
|
||||
opensshcert_key *ck = container_of(key, opensshcert_key, sshk);
|
||||
|
||||
opensshcert_signature_preimage(ck, bs);
|
||||
put_stringpl(bs, ptrlen_from_strbuf(ck->signature));
|
||||
}
|
||||
|
||||
static void opensshcert_private_blob(ssh_key *key, BinarySink *bs)
|
||||
{
|
||||
opensshcert_key *ck = container_of(key, opensshcert_key, sshk);
|
||||
ssh_key_private_blob(ck->basekey, bs);
|
||||
}
|
||||
|
||||
static void opensshcert_openssh_blob(ssh_key *key, BinarySink *bs)
|
||||
{
|
||||
opensshcert_key *ck = container_of(key, opensshcert_key, sshk);
|
||||
const opensshcert_extra *extra = key->vt->extra;
|
||||
|
||||
strbuf *cert = strbuf_new();
|
||||
ssh_key_public_blob(key, BinarySink_UPCAST(cert));
|
||||
put_stringsb(bs, cert);
|
||||
|
||||
strbuf *baseossh = strbuf_new_nm();
|
||||
ssh_key_openssh_blob(ck->basekey, BinarySink_UPCAST(baseossh));
|
||||
BinarySource basesrc[1];
|
||||
BinarySource_BARE_INIT_PL(basesrc, ptrlen_from_strbuf(baseossh));
|
||||
|
||||
BLOBTRANS_DECLARE(bt);
|
||||
blobtrans_read(bt, basesrc, extra->base_ossh_fmt);
|
||||
blobtrans_write(bt, bs, extra->cert_ossh_fmt);
|
||||
blobtrans_clear(bt);
|
||||
|
||||
strbuf_free(baseossh);
|
||||
}
|
||||
|
||||
static bool opensshcert_has_private(ssh_key *key)
|
||||
{
|
||||
opensshcert_key *ck = container_of(key, opensshcert_key, sshk);
|
||||
return ssh_key_has_private(ck->basekey);
|
||||
}
|
||||
|
||||
static char *opensshcert_cache_str(ssh_key *key)
|
||||
{
|
||||
unreachable(
|
||||
"Certificates are not expected to be stored in the host key cache");
|
||||
}
|
||||
|
||||
static void opensshcert_time_to_iso8601(BinarySink *bs, uint64_t time)
|
||||
{
|
||||
time_t t = time;
|
||||
char buf[256];
|
||||
put_data(bs, buf, strftime(buf, sizeof(buf),
|
||||
"%a %F %T %Z", localtime(&t)));
|
||||
}
|
||||
|
||||
static void opensshcert_string_list_key_components(
|
||||
key_components *kc, strbuf *input, const char *title, const char *title2)
|
||||
{
|
||||
BinarySource src[1];
|
||||
BinarySource_BARE_INIT_PL(src, ptrlen_from_strbuf(input));
|
||||
|
||||
const char *titles[2] = { title, title2 };
|
||||
size_t ntitles = (title2 ? 2 : 1);
|
||||
|
||||
unsigned index = 0;
|
||||
while (get_avail(src)) {
|
||||
for (size_t ti = 0; ti < ntitles; ti++) {
|
||||
ptrlen value = get_string(src);
|
||||
if (get_err(src))
|
||||
break;
|
||||
char *name = dupprintf("%s_%u", titles[ti], index);
|
||||
key_components_add_text_pl(kc, name, value);
|
||||
sfree(name);
|
||||
}
|
||||
index++;
|
||||
}
|
||||
}
|
||||
|
||||
static key_components *opensshcert_components(ssh_key *key)
|
||||
{
|
||||
opensshcert_key *ck = container_of(key, opensshcert_key, sshk);
|
||||
key_components *kc = ssh_key_components(ck->basekey);
|
||||
key_components_add_binary(kc, "cert_nonce", ptrlen_from_strbuf(ck->nonce));
|
||||
key_components_add_uint(kc, "cert_serial", ck->serial);
|
||||
switch (ck->type) {
|
||||
case SSH_CERT_TYPE_HOST:
|
||||
key_components_add_text(kc, "cert_type", "host");
|
||||
break;
|
||||
case SSH_CERT_TYPE_USER:
|
||||
key_components_add_text(kc, "cert_type", "user");
|
||||
break;
|
||||
default:
|
||||
key_components_add_uint(kc, "cert_type", ck->type);
|
||||
break;
|
||||
}
|
||||
key_components_add_text(kc, "cert_key_id", ck->key_id->s);
|
||||
opensshcert_string_list_key_components(kc, ck->valid_principals,
|
||||
"cert_valid_principal", NULL);
|
||||
key_components_add_uint(kc, "cert_valid_after", ck->valid_after);
|
||||
key_components_add_uint(kc, "cert_valid_before", ck->valid_before);
|
||||
/* Translate the validity period into human-legible dates, but
|
||||
* only if they're not the min/max integer. Rationale: if you see
|
||||
* "584554051223-11-09 07:00:15 UTC" as the expiry time you'll be
|
||||
* as likely to think it's a weird buffer overflow as half a
|
||||
* trillion years in the future! */
|
||||
if (ck->valid_after != 0) {
|
||||
strbuf *date = strbuf_new();
|
||||
opensshcert_time_to_iso8601(BinarySink_UPCAST(date), ck->valid_after);
|
||||
key_components_add_text_pl(kc, "cert_valid_after_date",
|
||||
ptrlen_from_strbuf(date));
|
||||
strbuf_free(date);
|
||||
}
|
||||
if (ck->valid_before != 0xFFFFFFFFFFFFFFFF) {
|
||||
strbuf *date = strbuf_new();
|
||||
opensshcert_time_to_iso8601(BinarySink_UPCAST(date), ck->valid_before);
|
||||
key_components_add_text_pl(kc, "cert_valid_before_date",
|
||||
ptrlen_from_strbuf(date));
|
||||
strbuf_free(date);
|
||||
}
|
||||
opensshcert_string_list_key_components(kc, ck->critical_options,
|
||||
"cert_critical_option",
|
||||
"cert_critical_option_data");
|
||||
opensshcert_string_list_key_components(kc, ck->extensions,
|
||||
"cert_extension",
|
||||
"cert_extension_data");
|
||||
key_components_add_binary(kc, "cert_ca_key", ptrlen_from_strbuf(
|
||||
ck->signature_key));
|
||||
|
||||
ptrlen ca_algname;
|
||||
ssh_key *ca_key = opensshcert_ca_pub_key(ck, &ca_algname);
|
||||
key_components_add_text_pl(kc, "cert_ca_key_algorithm_id", ca_algname);
|
||||
|
||||
if (ca_key) {
|
||||
key_components *kc_ca_key = ssh_key_components(ca_key);
|
||||
for (size_t i = 0; i < kc_ca_key->ncomponents; i++) {
|
||||
key_component *comp = &kc_ca_key->components[i];
|
||||
char *subname = dupcat("cert_ca_key_", comp->name);
|
||||
key_components_add_copy(kc, subname, comp);
|
||||
sfree(subname);
|
||||
}
|
||||
key_components_free(kc_ca_key);
|
||||
ssh_key_free(ca_key);
|
||||
}
|
||||
|
||||
key_components_add_binary(kc, "cert_ca_sig", ptrlen_from_strbuf(
|
||||
ck->signature));
|
||||
return kc;
|
||||
}
|
||||
|
||||
static int opensshcert_pubkey_bits(const ssh_keyalg *self, ptrlen blob)
|
||||
{
|
||||
BinarySource src[1];
|
||||
BinarySource_BARE_INIT_PL(src, blob);
|
||||
|
||||
get_string(src); /* key type */
|
||||
get_string(src); /* nonce */
|
||||
return ssh_key_public_bits(
|
||||
self->base_alg, make_ptrlen(get_ptr(src), get_avail(src)));
|
||||
}
|
||||
|
||||
static unsigned opensshcert_supported_flags(const ssh_keyalg *self)
|
||||
{
|
||||
return ssh_keyalg_supported_flags(self->base_alg);
|
||||
}
|
||||
|
||||
static const char *opensshcert_alternate_ssh_id(const ssh_keyalg *self,
|
||||
unsigned flags)
|
||||
{
|
||||
const char *base_id = ssh_keyalg_alternate_ssh_id(self->base_alg, flags);
|
||||
|
||||
for (size_t i = 0; i < lenof(opensshcert_all_keyalgs); i++) {
|
||||
const ssh_keyalg *alg_i = opensshcert_all_keyalgs[i];
|
||||
if (!strcmp(base_id, alg_i->base_alg->ssh_id))
|
||||
return alg_i->ssh_id;
|
||||
}
|
||||
|
||||
return self->ssh_id;
|
||||
}
|
||||
|
||||
static char *opensshcert_invalid(ssh_key *key, unsigned flags)
|
||||
{
|
||||
opensshcert_key *ck = container_of(key, opensshcert_key, sshk);
|
||||
return ssh_key_invalid(ck->basekey, flags);
|
||||
}
|
||||
|
||||
static bool opensshcert_verify(ssh_key *key, ptrlen sig, ptrlen data)
|
||||
{
|
||||
/* This method is pure *signature* verification; checking the
|
||||
* certificate is done elsewhere. */
|
||||
opensshcert_key *ck = container_of(key, opensshcert_key, sshk);
|
||||
return ssh_key_verify(ck->basekey, sig, data);
|
||||
}
|
||||
|
||||
static void opensshcert_sign(ssh_key *key, ptrlen data, unsigned flags,
|
||||
BinarySink *bs)
|
||||
{
|
||||
opensshcert_key *ck = container_of(key, opensshcert_key, sshk);
|
||||
ssh_key_sign(ck->basekey, data, flags, bs);
|
||||
}
|
10
ssh.h
10
ssh.h
@ -854,6 +854,8 @@ struct ssh_keyalg {
|
||||
const char *ssh_id; /* string identifier in the SSH protocol */
|
||||
const char *cache_id; /* identifier used in PuTTY's host key cache */
|
||||
const void *extra; /* private to the public key methods */
|
||||
bool is_certificate; /* is this a certified key type? */
|
||||
const ssh_keyalg *base_alg; /* if so, for what underlying key alg? */
|
||||
};
|
||||
|
||||
static inline ssh_key *ssh_key_new_pub(const ssh_keyalg *self, ptrlen pub)
|
||||
@ -1098,6 +1100,14 @@ extern const ssh_keyalg ssh_ecdsa_ed448;
|
||||
extern const ssh_keyalg ssh_ecdsa_nistp256;
|
||||
extern const ssh_keyalg ssh_ecdsa_nistp384;
|
||||
extern const ssh_keyalg ssh_ecdsa_nistp521;
|
||||
extern const ssh_keyalg opensshcert_ssh_dsa;
|
||||
extern const ssh_keyalg opensshcert_ssh_rsa;
|
||||
extern const ssh_keyalg opensshcert_ssh_rsa_sha256;
|
||||
extern const ssh_keyalg opensshcert_ssh_rsa_sha512;
|
||||
extern const ssh_keyalg opensshcert_ssh_ecdsa_ed25519;
|
||||
extern const ssh_keyalg opensshcert_ssh_ecdsa_nistp256;
|
||||
extern const ssh_keyalg opensshcert_ssh_ecdsa_nistp384;
|
||||
extern const ssh_keyalg opensshcert_ssh_ecdsa_nistp521;
|
||||
extern const ssh2_macalg ssh_hmac_md5;
|
||||
extern const ssh2_macalg ssh_hmac_sha1;
|
||||
extern const ssh2_macalg ssh_hmac_sha1_buggy;
|
||||
|
@ -568,6 +568,14 @@ const ssh_keyalg *const all_keyalgs[] = {
|
||||
&ssh_ecdsa_nistp521,
|
||||
&ssh_ecdsa_ed25519,
|
||||
&ssh_ecdsa_ed448,
|
||||
&opensshcert_ssh_dsa,
|
||||
&opensshcert_ssh_rsa,
|
||||
&opensshcert_ssh_rsa_sha256,
|
||||
&opensshcert_ssh_rsa_sha512,
|
||||
&opensshcert_ssh_ecdsa_ed25519,
|
||||
&opensshcert_ssh_ecdsa_nistp256,
|
||||
&opensshcert_ssh_ecdsa_nistp384,
|
||||
&opensshcert_ssh_ecdsa_nistp521,
|
||||
};
|
||||
const size_t n_keyalgs = lenof(all_keyalgs);
|
||||
|
||||
|
@ -46,6 +46,12 @@ BEGIN_ENUM_TYPE(keyalg)
|
||||
ENUM_VALUE("p256", &ssh_ecdsa_nistp256)
|
||||
ENUM_VALUE("p384", &ssh_ecdsa_nistp384)
|
||||
ENUM_VALUE("p521", &ssh_ecdsa_nistp521)
|
||||
ENUM_VALUE("dsa-cert", &opensshcert_ssh_dsa)
|
||||
ENUM_VALUE("rsa-cert", &opensshcert_ssh_rsa)
|
||||
ENUM_VALUE("ed25519-cert", &opensshcert_ssh_ecdsa_ed25519)
|
||||
ENUM_VALUE("p256-cert", &opensshcert_ssh_ecdsa_nistp256)
|
||||
ENUM_VALUE("p384-cert", &opensshcert_ssh_ecdsa_nistp384)
|
||||
ENUM_VALUE("p521-cert", &opensshcert_ssh_ecdsa_nistp521)
|
||||
END_ENUM_TYPE(keyalg)
|
||||
|
||||
BEGIN_ENUM_TYPE(cipheralg)
|
||||
|
Loading…
Reference in New Issue
Block a user