This modified version of the Heikki Toivonen's certdata2pem script automatically downloads the freshest certdata.txt file and converts it to PEM format. Requires M2Crypto.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 | #!/usr/bin/env python
"""
Small utility to download the Mozilla-format certificates
(/mozilla/security/nss/lib/ckfw/builtins/certdata.txt in the Mozilla CVS)
and convert them into PEM.
Original version:
http://svn.osafoundation.org/m2crypto/trunk/demo/x509/certdata2pem.py
Based on the idea from:
http://curl.haxx.se/docs/parse-certs.txt.
Copyright (c) 2007 Open Source Applications Foundation.
"""
import array
import urllib
from M2Crypto import X509
certdata_url = ("http://mxr.mozilla.org/seamonkey/source/security/nss" +
"/lib/ckfw/builtins/certdata.txt?raw=1")
def download_and_convert():
certdata = urllib.urlopen(certdata_url)
try:
with open('cacert.pem', 'wb') as out:
for block in certdata_to_pem(certdata):
out.write(block)
finally:
certdata.close()
def certdata_to_pem(certdata):
counter = 0
value = None
name = None
for line in certdata:
line = line.strip()
if line.startswith('CKA_LABEL'):
assert value is None
label_encoding, name, dummy = line.split('"')
label, encoding = label_encoding.split()
assert encoding == 'UTF8'
elif line == 'CKA_VALUE MULTILINE_OCTAL':
assert name is not None
value = array.array('c')
elif value is not None and line == 'END':
assert name is not None
print 'Writing ' + name
x509 = X509.load_cert_string(value.tostring(), X509.FORMAT_DER)
if not x509.verify():
print ' Skipping ' + name + ' since it does not verify'
name = None
value = None
continue
counter += 1
yield (name + '\n' + '=' * len(name) + '\n\n' +
'SHA1 Fingerprint=' + x509.get_fingerprint('sha1') + '\n' +
x509.as_text() +
x509.as_pem() +
'\n')
name = None
value = None
elif value is not None:
assert name is not None
for number in line.split('\\'):
if not number:
continue
value.append(chr(int(number, 8)))
print 'Wrote %d certificates' % counter
if __name__ == '__main__':
download_and_convert()
|