1656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project#!/usr/bin/perl
2656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project#
3656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project# CA - wrapper around ca to make it easier to use ... basically ca requires
4656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project#      some setup stuff to be done before you can use it and this makes
5656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project#      things easier between now and when Eric is convinced to fix it :-)
6656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project#
7656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project# CA -newca ... will setup the right stuff
8656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project# CA -newreq[-nodes] ... will generate a certificate request
9656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project# CA -sign ... will sign the generated request and output
10656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project#
11656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project# At the end of that grab newreq.pem and newcert.pem (one has the key
12656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project# and the other the certificate) and cat them together and that is what
13656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project# you want/need ... I'll make even this a little cleaner later.
14656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project#
15656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project#
16656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project# 12-Jan-96 tjh    Added more things ... including CA -signcert which
17656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project#                  converts a certificate to a request and then signs it.
18656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project# 10-Jan-96 eay    Fixed a few more bugs and added the SSLEAY_CONFIG
19656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project#		   environment variable so this can be driven from
20656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project#		   a script.
21656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project# 25-Jul-96 eay    Cleaned up filenames some more.
22656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project# 11-Jun-96 eay    Fixed a few filename missmatches.
23656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project# 03-May-96 eay    Modified to use 'ssleay cmd' instead of 'cmd'.
24656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project# 18-Apr-96 tjh    Original hacking
25656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project#
26656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project# Tim Hudson
27656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project# tjh@cryptsoft.com
28656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project#
29656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project
30656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project# 27-Apr-98 snh    Translation into perl, fix existing CA bug.
31656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project#
32656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project#
33656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project# Steve Henson
34656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project# shenson@bigfoot.com
35656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project
36656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project# default openssl.cnf file has setup as per the following
37656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project# demoCA ... where everything is stored
38656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project
39656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Projectmy $openssl;
40656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Projectif(defined $ENV{OPENSSL}) {
41656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	$openssl = $ENV{OPENSSL};
42656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project} else {
43656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	$openssl = "openssl";
44656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	$ENV{OPENSSL} = $openssl;
45656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project}
46656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project
47656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project$SSLEAY_CONFIG=$ENV{"SSLEAY_CONFIG"};
48656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project$DAYS="-days 365";	# 1 year
49656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project$CADAYS="-days 1095";	# 3 years
50656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project$REQ="$openssl req $SSLEAY_CONFIG";
51656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project$CA="$openssl ca $SSLEAY_CONFIG";
52656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project$VERIFY="$openssl verify";
53656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project$X509="$openssl x509";
54656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project$PKCS12="$openssl pkcs12";
55656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project
56656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project$CATOP="./demoCA";
57656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project$CAKEY="cakey.pem";
58656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project$CAREQ="careq.pem";
59656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project$CACERT="cacert.pem";
60656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project
61656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project$DIRMODE = 0777;
62656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project
63656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project$RET = 0;
64656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project
65656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Projectforeach (@ARGV) {
66656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	if ( /^(-\?|-h|-help)$/ ) {
67656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    print STDERR "usage: CA -newcert|-newreq|-newreq-nodes|-newca|-sign|-verify\n";
68656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    exit 0;
69656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	} elsif (/^-newcert$/) {
70656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    # create a certificate
71656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    system ("$REQ -new -x509 -keyout newkey.pem -out newcert.pem $DAYS");
72656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    $RET=$?;
73656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    print "Certificate is in newcert.pem, private key is in newkey.pem\n"
74656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	} elsif (/^-newreq$/) {
75656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    # create a certificate request
76656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    system ("$REQ -new -keyout newkey.pem -out newreq.pem $DAYS");
77656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    $RET=$?;
78656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    print "Request is in newreq.pem, private key is in newkey.pem\n";
79656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	} elsif (/^-newreq-nodes$/) {
80656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    # create a certificate request
81656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    system ("$REQ -new -nodes -keyout newkey.pem -out newreq.pem $DAYS");
82656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    $RET=$?;
83656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    print "Request is in newreq.pem, private key is in newkey.pem\n";
84656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	} elsif (/^-newca$/) {
85656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		# if explicitly asked for or it doesn't exist then setup the
86656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		# directory structure that Eric likes to manage things
87656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    $NEW="1";
88656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    if ( "$NEW" || ! -f "${CATOP}/serial" ) {
89656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		# create the directory hierarchy
90656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		mkdir $CATOP, $DIRMODE;
91656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		mkdir "${CATOP}/certs", $DIRMODE;
92656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		mkdir "${CATOP}/crl", $DIRMODE ;
93656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		mkdir "${CATOP}/newcerts", $DIRMODE;
94656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		mkdir "${CATOP}/private", $DIRMODE;
95656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		open OUT, ">${CATOP}/index.txt";
96656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		close OUT;
97656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		open OUT, ">${CATOP}/crlnumber";
98656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		print OUT "01\n";
99656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		close OUT;
100656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    }
101656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    if ( ! -f "${CATOP}/private/$CAKEY" ) {
102656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		print "CA certificate filename (or enter to create)\n";
103656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		$FILE = <STDIN>;
104656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project
105656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		chop $FILE;
106656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project
107656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		# ask user for existing CA certificate
108656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		if ($FILE) {
109656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		    cp_pem($FILE,"${CATOP}/private/$CAKEY", "PRIVATE");
110656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		    cp_pem($FILE,"${CATOP}/$CACERT", "CERTIFICATE");
111656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		    $RET=$?;
112656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		} else {
113656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		    print "Making CA certificate ...\n";
114656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		    system ("$REQ -new -keyout " .
115656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project			"${CATOP}/private/$CAKEY -out ${CATOP}/$CAREQ");
116656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		    system ("$CA -create_serial " .
117656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project			"-out ${CATOP}/$CACERT $CADAYS -batch " .
118656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project			"-keyfile ${CATOP}/private/$CAKEY -selfsign " .
119656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project			"-extensions v3_ca " .
120656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project			"-infiles ${CATOP}/$CAREQ ");
121656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		    $RET=$?;
122656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		}
123656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    }
124656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	} elsif (/^-pkcs12$/) {
125656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    my $cname = $ARGV[1];
126656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    $cname = "My Certificate" unless defined $cname;
127656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    system ("$PKCS12 -in newcert.pem -inkey newkey.pem " .
128656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project			"-certfile ${CATOP}/$CACERT -out newcert.p12 " .
129656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project			"-export -name \"$cname\"");
130656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    $RET=$?;
131656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    print "PKCS #12 file is in newcert.p12\n";
132656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    exit $RET;
133656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	} elsif (/^-xsign$/) {
134656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    system ("$CA -policy policy_anything -infiles newreq.pem");
135656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    $RET=$?;
136656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	} elsif (/^(-sign|-signreq)$/) {
137656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    system ("$CA -policy policy_anything -out newcert.pem " .
138656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project							"-infiles newreq.pem");
139656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    $RET=$?;
140656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    print "Signed certificate is in newcert.pem\n";
141656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	} elsif (/^(-signCA)$/) {
142656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    system ("$CA -policy policy_anything -out newcert.pem " .
143656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project					"-extensions v3_ca -infiles newreq.pem");
144656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    $RET=$?;
145656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    print "Signed CA certificate is in newcert.pem\n";
146656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	} elsif (/^-signcert$/) {
147656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    system ("$X509 -x509toreq -in newreq.pem -signkey newreq.pem " .
148656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project								"-out tmp.pem");
149656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    system ("$CA -policy policy_anything -out newcert.pem " .
150656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project							"-infiles tmp.pem");
151656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    $RET = $?;
152656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    print "Signed certificate is in newcert.pem\n";
153656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	} elsif (/^-verify$/) {
154656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    if (shift) {
155656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		foreach $j (@ARGV) {
156656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		    system ("$VERIFY -CAfile $CATOP/$CACERT $j");
157656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		    $RET=$? if ($? != 0);
158656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		}
159656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		exit $RET;
160656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    } else {
161656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		    system ("$VERIFY -CAfile $CATOP/$CACERT newcert.pem");
162656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		    $RET=$?;
163656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    	    exit 0;
164656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    }
165656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	} else {
166656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    print STDERR "Unknown arg $_\n";
167656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    print STDERR "usage: CA -newcert|-newreq|-newreq-nodes|-newca|-sign|-verify\n";
168656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	    exit 1;
169656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	}
170656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project}
171656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project
172656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Projectexit $RET;
173656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project
174656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Projectsub cp_pem {
175656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Projectmy ($infile, $outfile, $bound) = @_;
176656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Projectopen IN, $infile;
177656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Projectopen OUT, ">$outfile";
178656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Projectmy $flag = 0;
179656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Projectwhile (<IN>) {
180656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	$flag = 1 if (/^-----BEGIN.*$bound/) ;
181656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	print OUT $_ if ($flag);
182656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	if (/^-----END.*$bound/) {
183656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		close IN;
184656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		close OUT;
185656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project		return;
186656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project	}
187656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project}
188656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project}
189656d9c7f52f88b3a3daccafa7655dec086c4756eThe Android Open Source Project
190