3 # Debian/OpenSSL Weak Key Detector
5 # Copyright (C) 2008, Florian Weimer <fw@deneb.enyo.de>
7 # Permission to use, copy, modify, and distribute this software for
8 # any purpose with or without fee is hereby granted, provided that the
9 # above copyright notice and this permission notice appear in all
12 # THE SOFTWARE IS PROVIDED "AS IS" AND FLORIAN WEIMER AND HIS
13 # CONTRIBUTORS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS SOFTWARE
14 # INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN
15 # NO EVENT SHALL FLORIAN WEIMER OR HIS CONTRIBUTORS BE LIABLE FOR ANY
16 # SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
17 # WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN
18 # AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING
19 # OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
22 # Blacklist data has been provided by Kees Cook, Peter Palfrader and
25 # Patches and comments are welcome. Please send them to
26 # <fw@deneb.enyo.de>, and use "dowkd" in the subject line.
33 usage: $0 [OPTIONS...] COMMAND [ARGUMENTS...]
37 file: examine files on the command line for weak keys
38 host: examine the specified hosts for weak SSH keys
39 user: examine user SSH keys for weakness; examine all users if no
41 help: show this help screen
45 -c FILE: set the database cache file name (default: dowkd.db)
47 dowkd currently handles the following OpenSSH host and user keys,
48 provided they have been generated on a little-endian architecture
49 (such as i386 or amd64): RSA/1024, RSA/2048 and DSA/1024. (The
50 OpenSSH version in Debian does not support DSA key generation with)
53 OpenVPN shared also detected on little-endian architecture.
55 Note that the blacklist by dowkd may be incomplete; it is only
56 intended as a quick check.
68 my $db_file = 'dowkd.db';
74 $db = tie
%db, 'DB_File', $db_file, O_RDWR
| O_CREAT
, 0777, $DB_BTREE
75 or die "error: could not open database: $!\n";
77 $db{''} = $db_version;
78 while (my $line = <DATA
>) {
79 next if $line =~ /^\**$/;
81 $line =~ /^[0-9a-f]{32}$/ or die "error: invalid data line";
82 $line =~ s/(..)/chr(hex($1))/ge;
91 $db = tie
%db, 'DB_File', $db_file, O_RDONLY
, 0777, $DB_BTREE
92 or die "error: could not open database: $!\n";
93 my $stored_version = $db{''};
94 $stored_version && $stored_version eq $db_version or create_db
;
101 sub safe_backtick
(@
) {
104 open $fh, '-|', @args
105 or die "error: failed to spawn $args[0]: $!\n";
111 @result = scalar(<$fh>);
114 $?
== 0 or return undef;
123 my $keys_vulnerable = 0;
126 print STDERR
"summary: keys found: $keys_found, weak keys: $keys_vulnerable\n";
129 sub check_hash
($$) {
130 my ($name, $hash) = @_;
132 if (exists $db{$hash}) {
134 print "$name: weak key\n";
138 sub ssh_fprint_file
($) {
140 my $data = safe_backtick qw
/ssh-keygen -l -f/, $name;
141 defined $data or return ();
142 my @data = $data =~ /^(\d+) ([0-9a-f]{2}(?::[0-9a-f]{2}){15})/;
143 return @data if @data == 2;
147 sub ssh_fprint_check
($$$) {
148 my ($name, $length, $hash) = @_;
149 if ($length == 1024 || $length == 2048) {
151 $hash =~ s/(..)/chr(hex($1))/ge;
152 check_hash
$name, $hash;
154 warn "$name: warning: no suitable blacklist\n";
158 sub from_ssh_key_file
($) {
160 my ($length, $hash) = ssh_fprint_file
$name;
161 if ($length && $hash) {
162 ssh_fprint_check
"$name:1", $length, $hash;
164 warn "$name:1: warning: failed to parse SSH key file\n";
170 seek $tmp, 0, 0 or die "seek: $!";
171 truncate $tmp, 0 or die "truncate: $!";
174 sub from_ssh_auth_file
($) {
177 unless (open $auth, '<', $name) {
178 warn "$name:0: error: open failed: $!\n";
181 my $tmp = new File
::Temp
;
182 while (my $line = <$auth>) {
184 next if $line =~ m/^\s*(#|$)/;
187 print $tmp "$line\n" or die "print: $!";
189 my ($length, $hash) = ssh_fprint_file
"$tmp";
190 if ($length && $hash) {
191 ssh_fprint_check
"$name:$lineno", $length, $hash;
193 warn "$name:$lineno: warning: unparsable line\n";
198 sub from_openvpn_key
($) {
201 unless (open $key, '<', $name) {
202 warn "$name:0: open failed: $!\n";
207 while (my $line = <$key>) {
209 if ($line =~ /^-----BEGIN OpenVPN Static key V1-----/) {
212 if ($line =~ /^([0-9a-f]{32})/) {
214 $line =~ s/(..)/chr(hex($1))/ge;
215 check_hash
"$name:$.", $line;
218 warn "$name:$.: warning: illegal OpenVPN file format\n";
225 sub from_ssh_host
(@
) {
228 push @lines, safe_backtick qw
/ssh-keyscan -t rsa/, @names;
229 push @lines, safe_backtick qw
/ssh-keyscan -t dsa/, @names;
231 my $tmp = new File
::Temp
;
232 for my $line (@lines) {
233 next if $line =~ /^#/;
234 my ($host, $data) = $line =~ /^(\S+) (.*)$/;
236 print $tmp "$data\n" or die "print: $!";
238 my ($length, $hash) = ssh_fprint_file
"$tmp";
239 if ($length && $hash) {
240 ssh_fprint_check
"$host", $length, $hash;
242 warn "$host: warning: unparsable line\n";
249 my ($name,$passwd,$uid,$gid,
250 $quota,$comment,$gcos,$dir,$shell,$expire) = getpwnam($user);
252 warn "warning: user $user does not exist\n";
255 my $file = "$dir/.ssh/authorized_keys";
256 from_ssh_auth_file
$file if -r
$file;
257 $file = "$dir/.ssh/authorized_keys2";
258 from_ssh_auth_file
$file if -r
$file;
259 $file = "$dir/.ssh/known_hosts";
260 from_ssh_auth_file
$file if -r
$file;
261 $file = "$dir/.ssh/known_hosts2";
262 from_ssh_auth_file
$file if -r
$file;
263 $file = "$dir/.ssh/id_rsa.pub";
264 from_ssh_key_file
$file if -r
$file;
265 $file = "$dir/.ssh/id_dsa.pub";
266 from_ssh_key_file
$file if -r
$file;
269 sub from_user_all
() {
270 # This was one loop initially, but does not work with some Perl
274 while (my $name = getpwent) {
278 from_user
$_ for @names;
281 if (@ARGV && $ARGV[0] eq '-c') {
283 $db_file = shift @ARGV if @ARGV;
287 my $cmd = shift @ARGV;
288 if ($cmd eq 'file') {
289 for my $name (@ARGV) {
290 next if from_openvpn_key
$name;
291 from_ssh_auth_file
$name;
293 } elsif ($cmd eq 'host') {
295 } elsif ($cmd eq 'user') {
297 from_user
$_ for @ARGV;
301 } elsif ($cmd eq 'help') {
305 die "error: invalid command, use \"help\" to get help\n";