NAME
Data::SSHPubkey - utility function to parse SSH public keys with
SYNOPSIS
use Data::SSHPubkey qw(pubkeys);
# a Mojo app might accept public keys from clients, e.g.
# cat /etc/ssh/*.pub | curl ... --data-urlencode pk@- http...
# this case is supported via a scalar reference
my $keylist = pubkeys( \$c->param('pk') );
for my $ref ( @$keylist ) {
my ($type, $pubkey) = @$ref;
...
}
# a key collection host could instead wrap ssh-keyscan(1) and
# pass in a file handle
open( my $fh, '-|', qw(ssh-keyscan --), $host ) or die ...
binmode $fh;
my $keylist = pubkeys($fh);
# a string will be treated as a file to open and read
my $keylist = pubkeys( "/etc/ssh/ssh_host_ed25519_key.pub" );
# if you do not care about the key types, extract only the pub
# keys with something like
... = map { $_->[1] } @$keylist;
DESCRIPTION
Data::SSHPubkey
parses SSH public keys, or at least some of those supported by ssh-keygen(1). It may be prudent to check any uploaded data against ssh-keygen
though this module should help extract said data from a web form upload or the like to get to that step.
Currently supported public key types (the possible values that $type
above may contain):
ecdsa ed25519 rsa
PEM PKCS8 RFC4716
Neither SSH1 keys nor SSH2 DSA keys are supported.
The $pubkey
data will not include any tailing comments; those are stripped. The $pubkey
data will not end with a newline; that must be added by your software as necessary when writing out the public keys. POSIX mandates an ultimate newline, and the shell read
command is buggy by default if that ultimate newline is missing:
$ (echo data; echo -n loss) | while read line; do echo $line; done
Inner newlines for the multiline SSH public key types (PEM
, PKCS8
, and RFC4716
) will be standardized to the $/
variable. This may cause problems if ssh-keygen(1)
or equivalent on some platform demands a specific newline sequence that is not $/
.
The types PEM
, PKCS8
, and RFC4716
will need conversion for use with OpenSSH; use convert_pubkeys or these types could be excluded with something like:
my @pubkeys = grep { $_->[0] =~ m/^(?:ecdsa|ed25519|rsa)$/ }
@{ Data::SSHPubkey::pubkeys( ... ) };
or
... = map { $_->[0] =~ m/^(?:ecdsa|ed25519|rsa)$/ ? $_->[1] : () }
@{ Data::SSHPubkey::pubkeys( ... ) };
to obtain only the public key material.
SUBROUTINES
- convert_pubkeys output-from-pubkeys
-
This subroutine converts the output of pubkeys into a list of just the public keys, with the
PEM
,PKCS8
, andRFC4716
types converted into a form suitable for use with OpenSSH, using the external tool ssh-keygen(1) that is hopefully installed. - pubkeys filename-or-scalarref
-
A filename (scalar) will be opened and the public keys therein parsed; a scalar reference will be treated as an in-memory file and will likewise be opened and parsed.
This routine will croak on error as, in theory, all the errors should be due to the data passed in by the caller, or possibly the system has run out of memory, or something.
The return format is a reference to a list of
[ $type, $pubkey ]
sublists.
VARIABLES
$Data::SSHPubkey::max_keys
specifies the maximum number of keys to parse, 3
by default. An exception is thrown if more than 3
keys are seen in the input.
$Data::SSHPubkey::max_lines
specifies the maximum number of input lines this module will process before throwing an exception, 100
by default. An attacker still might supply too much data with very long lines; webserver or other configuration to limit that may be necessary.
The %Data::SSHPubkey::ssh_pubkey_types
hash contains as its keys the SSH public key types supported by this module.
BUGS
Patches might best be applied towards:
https://github.com/thrig/Data-SSHPubkey
Known Issues
Probably not enough guards or checks against hostile input.
Support for the PEM
and especially PKCS8
formats is a bit sloppy, and the base64 matching is done by a regex that may accept data that is not valid base64.
Support for various RFC 4253 formats is likely lacking (see below or the comments in the code).
More tests are necessary for more edge cases.
If the input uses fancy encodings (where fancy is anything not ASCII) lines longer than 72 8-bit bytes may be accepted. read_binary
from File::Slurper or a traditional binmode $fh
should avoid this case as the key data looked for is only a subset of ASCII (header values or comments that are ignored by this module could be UTF-8 or possibly anything else).
convert_pubkeys calls out to (modern versions of) ssh-keygen(1); ideally this might instead be done via suitable CPAN modules.
SEE ALSO
https://github.com/thrig/web_irulan
Config::OpenSSH::Authkey - older module more aimed at management of ~/.ssh/authorized_keys
data and not specifically public keys. It does have support for SSH2 DSA or SSH1 keys, though.
- RFC 822
-
Definition of white space used in various formats
[ \t]
. - RFC 1421
-
PEM format details.
- RFC 4253
-
Mentioned by RFC 4716 but it is unclear to me what the section 6.6 "Public Key Algorithms" formats exactly are.
- RFC 4716
-
Secure Shell (SSH) public key file format.
AUTHOR
thrig - Jeremy Mates (cpan:JMATES) <jmates at cpan.org>
COPYRIGHT AND LICENSE
Copyright (C) 2019 by Jeremy Mates
This program is distributed under the (Revised) BSD License: http://www.opensource.org/licenses/BSD-3-Clause