提交 439df508 编写于 作者: D Dr. Stephen Henson

Fix c_rehash script, add -fingerprint option to crl.

上级 0d3b0afe
...@@ -4,6 +4,12 @@ ...@@ -4,6 +4,12 @@
Changes between 0.9.5a and 0.9.6 [xx XXX 2000] Changes between 0.9.5a and 0.9.6 [xx XXX 2000]
*) Enhance c_rehash script. Old version would mishandle certificates
with the same subject name hash and wouldn't handle CRLs at all.
Added -fingerprint option to crl utility, to support new c_rehash
features.
[Steve Henson]
*) Eliminate non-ANSI declarations in crypto.h and stack.h. *) Eliminate non-ANSI declarations in crypto.h and stack.h.
[Ulf Möller] [Ulf Möller]
......
...@@ -890,12 +890,13 @@ EOF ...@@ -890,12 +890,13 @@ EOF
### (system 'make depend') == 0 or exit $? if $depflags ne ""; ### (system 'make depend') == 0 or exit $? if $depflags ne "";
# Run "make depend" manually if you want to be able to delete # Run "make depend" manually if you want to be able to delete
# the source code files of ciphers you left out. # the source code files of ciphers you left out.
&dofile("tools/c_rehash",$openssldir,'^DIR=', 'DIR=%s',);
if ( $perl =~ m@^/@) { if ( $perl =~ m@^/@) {
&dofile("tools/c_rehash",$perl,'^#!/', '#!%s','^my \$dir;$', 'my $dir = "' . $openssldir . '";');
&dofile("apps/der_chop",$perl,'^#!/', '#!%s'); &dofile("apps/der_chop",$perl,'^#!/', '#!%s');
&dofile("apps/CA.pl",$perl,'^#!/', '#!%s'); &dofile("apps/CA.pl",$perl,'^#!/', '#!%s');
} else { } else {
# No path for Perl known ... # No path for Perl known ...
&dofile("tools/c_rehash",'/usr/local/bin/perl','^#!/', '#!%s','^my \$dir;$', 'my $dir = "' . $openssldir . '";');
&dofile("apps/der_chop",'/usr/local/bin/perl','^#!/', '#!%s'); &dofile("apps/der_chop",'/usr/local/bin/perl','^#!/', '#!%s');
&dofile("apps/CA.pl",'/usr/local/bin/perl','^#!/', '#!%s'); &dofile("apps/CA.pl",'/usr/local/bin/perl','^#!/', '#!%s');
} }
......
...@@ -262,7 +262,7 @@ dclean: ...@@ -262,7 +262,7 @@ dclean:
rehash: rehash.time rehash: rehash.time
rehash.time: certs rehash.time: certs
@(OPENSSL="`pwd`/apps/openssl"; export OPENSSL; sh tools/c_rehash certs) @(OPENSSL="`pwd`/apps/openssl"; export OPENSSL; $(PERL) tools/c_rehash certs)
touch rehash.time touch rehash.time
test: tests test: tests
......
...@@ -135,7 +135,7 @@ $(DLIBCRYPTO): ...@@ -135,7 +135,7 @@ $(DLIBCRYPTO):
$(PROGRAM): progs.h $(E_OBJ) $(PROGRAM).o $(DLIBCRYPTO) $(DLIBSSL) $(PROGRAM): progs.h $(E_OBJ) $(PROGRAM).o $(DLIBCRYPTO) $(DLIBSSL)
$(RM) $(PROGRAM) $(RM) $(PROGRAM)
$(CC) -o $(PROGRAM) $(CFLAGS) $(PROGRAM).o $(E_OBJ) $(PEX_LIBS) $(LIBSSL) $(LIBCRYPTO) $(EX_LIBS) $(CC) -o $(PROGRAM) $(CFLAGS) $(PROGRAM).o $(E_OBJ) $(PEX_LIBS) $(LIBSSL) $(LIBCRYPTO) $(EX_LIBS)
@(cd ..; OPENSSL="`pwd`/apps/openssl"; export OPENSSL; sh tools/c_rehash certs) @(cd ..; OPENSSL="`pwd`/apps/openssl"; export OPENSSL; $(PERL) tools/c_rehash certs)
progs.h: progs.pl progs.h: progs.pl
$(PERL) progs.pl $(E_EXE) >progs.h $(PERL) progs.pl $(E_EXE) >progs.h
......
...@@ -104,6 +104,7 @@ int MAIN(int argc, char **argv) ...@@ -104,6 +104,7 @@ int MAIN(int argc, char **argv)
int informat,outformat; int informat,outformat;
char *infile=NULL,*outfile=NULL; char *infile=NULL,*outfile=NULL;
int hash=0,issuer=0,lastupdate=0,nextupdate=0,noout=0,text=0; int hash=0,issuer=0,lastupdate=0,nextupdate=0,noout=0,text=0;
int fingerprint = 0;
char **pp,buf[256]; char **pp,buf[256];
X509_STORE *store = NULL; X509_STORE *store = NULL;
X509_STORE_CTX ctx; X509_STORE_CTX ctx;
...@@ -111,6 +112,7 @@ int MAIN(int argc, char **argv) ...@@ -111,6 +112,7 @@ int MAIN(int argc, char **argv)
X509_OBJECT xobj; X509_OBJECT xobj;
EVP_PKEY *pkey; EVP_PKEY *pkey;
int do_ver = 0; int do_ver = 0;
const EVP_MD *md_alg,*digest=EVP_md5();
apps_startup(); apps_startup();
...@@ -183,6 +185,13 @@ int MAIN(int argc, char **argv) ...@@ -183,6 +185,13 @@ int MAIN(int argc, char **argv)
nextupdate= ++num; nextupdate= ++num;
else if (strcmp(*argv,"-noout") == 0) else if (strcmp(*argv,"-noout") == 0)
noout= ++num; noout= ++num;
else if (strcmp(*argv,"-fingerprint") == 0)
fingerprint= ++num;
else if ((md_alg=EVP_get_digestbyname(*argv + 1)))
{
/* ok */
digest=md_alg;
}
else else
{ {
BIO_printf(bio_err,"unknown option %s\n",*argv); BIO_printf(bio_err,"unknown option %s\n",*argv);
...@@ -274,6 +283,26 @@ bad: ...@@ -274,6 +283,26 @@ bad:
BIO_printf(bio_out,"NONE"); BIO_printf(bio_out,"NONE");
BIO_printf(bio_out,"\n"); BIO_printf(bio_out,"\n");
} }
if (fingerprint == i)
{
int j;
unsigned int n;
unsigned char md[EVP_MAX_MD_SIZE];
if (!X509_CRL_digest(x,digest,md,&n))
{
BIO_printf(bio_err,"out of memory\n");
goto end;
}
BIO_printf(bio_out,"%s Fingerprint=",
OBJ_nid2sn(EVP_MD_type(digest)));
for (j=0; j<(int)n; j++)
{
BIO_printf(bio_out,"%02X%c",md[j],
(j+1 == (int)n)
?'\n':':');
}
}
} }
} }
......
...@@ -660,6 +660,8 @@ int X509_CRL_sign(X509_CRL *x, EVP_PKEY *pkey, const EVP_MD *md); ...@@ -660,6 +660,8 @@ int X509_CRL_sign(X509_CRL *x, EVP_PKEY *pkey, const EVP_MD *md);
int NETSCAPE_SPKI_sign(NETSCAPE_SPKI *x, EVP_PKEY *pkey, const EVP_MD *md); int NETSCAPE_SPKI_sign(NETSCAPE_SPKI *x, EVP_PKEY *pkey, const EVP_MD *md);
int X509_digest(X509 *data,const EVP_MD *type,unsigned char *md,unsigned int *len); int X509_digest(X509 *data,const EVP_MD *type,unsigned char *md,unsigned int *len);
int X509_CRL_digest(X509_CRL *data,const EVP_MD *type,unsigned char *md,unsigned int *len);
int X509_REQ_digest(X509_REQ *data,const EVP_MD *type,unsigned char *md,unsigned int *len);
int X509_NAME_digest(X509_NAME *data,const EVP_MD *type, int X509_NAME_digest(X509_NAME *data,const EVP_MD *type,
unsigned char *md,unsigned int *len); unsigned char *md,unsigned int *len);
#endif #endif
......
...@@ -417,6 +417,18 @@ int X509_digest(X509 *data, const EVP_MD *type, unsigned char *md, ...@@ -417,6 +417,18 @@ int X509_digest(X509 *data, const EVP_MD *type, unsigned char *md,
return(ASN1_digest((int (*)())i2d_X509,type,(char *)data,md,len)); return(ASN1_digest((int (*)())i2d_X509,type,(char *)data,md,len));
} }
int X509_CRL_digest(X509_CRL *data, const EVP_MD *type, unsigned char *md,
unsigned int *len)
{
return(ASN1_digest((int (*)())i2d_X509_CRL,type,(char *)data,md,len));
}
int X509_REQ_digest(X509_REQ *data, const EVP_MD *type, unsigned char *md,
unsigned int *len)
{
return(ASN1_digest((int (*)())i2d_X509_REQ,type,(char *)data,md,len));
}
int X509_NAME_digest(X509_NAME *data, const EVP_MD *type, unsigned char *md, int X509_NAME_digest(X509_NAME *data, const EVP_MD *type, unsigned char *md,
unsigned int *len) unsigned int *len)
{ {
......
#!/bin/sh #!/usr/local/bin/perl
#
# redo the hashes for the certificates in your cert path or the ones passed
# on the command line. # Perl c_rehash script, scan all files in a directory
# # and add symbolic links to their hash values.
if [ "$OPENSSL"x = "x" -o ! -x "$OPENSSL" ]; then my $openssl;
OPENSSL='openssl'
export OPENSSL my $dir;
fi
DIR=/usr/local/ssl if(defined $ENV{OPENSSL}) {
PATH=$DIR/bin:$PATH $openssl = $ENV{OPENSSL};
} else {
if [ ! -f "$OPENSSL" ]; then $openssl = "openssl";
found=0 $ENV{OPENSSL} = $openssl;
for dir in . `echo $PATH | sed -e 's/:/ /g'`; do }
if [ -f "$dir/$OPENSSL" ]; then
found=1 $ENV{PATH} .= ":$dir/bin";
break
fi if(! -f $openssl) {
done my $found = 0;
if [ $found = 0 ]; then foreach (split /:/, $ENV{PATH}) {
echo "c_rehash: rehashing skipped ('openssl' program not available)" 1>&2 if(-f "$_/$openssl") {
exit 0 $found = 1;
fi last;
fi }
}
SSL_DIR=$DIR/certs if($found == 0) {
print STDERR "c_rehash: rehashing skipped ('openssl' program not available)\n";
if [ "$*" = "" ]; then exit 0;
CERTS=${*:-${SSL_CERT_DIR:-$SSL_DIR}} }
else }
CERTS=$*
fi if(@ARGV) {
@dirlist = @ARGV;
IFS=': ' } elsif($ENV{SSL_CERT_DIR}) {
for i in $CERTS @dirlist = split /:/, $ENV{SSL_CERT_DIR};
do } else {
( $dirlist[0] = "$dir/certs";
IFS=' ' }
if [ -d $i -a -w $i ]; then
cd $i
echo "Doing $i" foreach (@dirlist) {
for i in *.pem if(-d $_ and -w $_) {
do hash_dir($_);
if [ $i != '*.pem' ]; then }
h=`$OPENSSL x509 -hash -noout -in $i` }
if [ "x$h" = "x" ]; then
echo $i does not contain a certificate sub hash_dir {
else my %hashlist;
if [ -f $h.0 ]; then print "Doing $_[0]\n";
/bin/rm -f $h.0 chdir $_[0];
fi opendir(DIR, ".");
echo "$i => $h.0" my @flist = readdir(DIR);
ln -s $i $h.0 # Delete any existing symbolic links
fi foreach (grep {/^[\da-f]+\.r{0,1}\d+$/} @flist) {
fi if(-l $_) {
done unlink $_;
fi }
) }
done closedir DIR;
FILE: foreach $fname (grep {/\.pem$/} @flist) {
# Check to see if certificates and/or CRLs present.
my ($cert, $crl) = check_file($fname);
if(!$cert && !$crl) {
print STDERR "WARNING: $fname does not contain a certificate or CRL: skipping\n";
next;
}
link_hash_cert($fname) if($cert);
link_hash_crl($fname) if($crl);
}
}
sub check_file {
my ($is_cert, $is_crl) = (0,0);
my $fname = $_[0];
open IN, $fname;
while(<IN>) {
if(/^-----BEGIN (.*)-----/) {
my $hdr = $1;
if($hdr =~ /^(X509 |TRUSTED |)CERTIFICATE$/) {
$is_cert = 1;
last if($is_crl);
} elsif($hdr eq "X509 CRL") {
$is_crl = 1;
last if($is_cert);
}
}
}
close IN;
return ($is_cert, $is_crl);
}
# Link a certificate to its subject name hash value, each hash is of
# the form <hash>.<n> where n is an integer. If the hash value already exists
# then we need to up the value of n, unless its a duplicate in which
# case we skip the link. We check for duplicates by comparing the
# certificate fingerprints
sub link_hash_cert {
my $fname = $_[0];
my ($hash, $fprint) = `$openssl x509 -hash -fingerprint -noout -in $fname`;
chomp $hash;
chomp $fprint;
$fprint =~ s/^.*=//;
$fprint =~ tr/://d;
my $suffix = 0;
# Search for an unused hash filename
while(exists $hashlist{"$hash.$suffix"}) {
# Hash matches: if fingerprint matches its a duplicate cert
if($hashlist{"$hash.$suffix"} eq $fprint) {
print STDERR "WARNING: Skipping duplicate certificate $fname\n";
return;
}
$suffix++;
}
$hash .= ".$suffix";
print "$fname => $hash\n";
symlink $fname, $hash;
$hashlist{$hash} = $fprint;
}
# Same as above except for a CRL. CRL links are of the form <hash>.r<n>
sub link_hash_crl {
my $fname = $_[0];
my ($hash, $fprint) = `$openssl crl -hash -fingerprint -noout -in $fname`;
chomp $hash;
chomp $fprint;
$fprint =~ s/^.*=//;
$fprint =~ tr/://d;
my $suffix = 0;
# Search for an unused hash filename
while(exists $hashlist{"$hash.r$suffix"}) {
# Hash matches: if fingerprint matches its a duplicate cert
if($hashlist{"$hash.r$suffix"} eq $fprint) {
print STDERR "WARNING: Skipping duplicate CRL $fname\n";
return;
}
$suffix++;
}
$hash .= ".r$suffix";
print "$fname => $hash\n";
symlink $fname, $hash;
$hashlist{$hash} = $fprint;
}
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册