127 lines
4.3 KiB
Python
Raw Normal View History

2017-06-27 14:09:51 +02:00
import sys
import os
import errno
import time
import json
from base64 import b64decode
from watchdog.observers import Observer
from watchdog.events import FileSystemEventHandler
class Handler(FileSystemEventHandler):
def on_created(self, event):
self.handle(event)
def on_modified(self, event):
self.handle(event)
def handle(self, event):
# Check if it's a JSON file
if not event.is_directory and event.src_path.endswith('.json'):
print('Certificates changed')
# Read JSON file
data = json.loads(open(event.src_path).read())
# Determine ACME version
acme_version = 2 if 'acme-v02' in data['Account']['Registration']['uri'] else 1
# Find certificates
if acme_version == 1:
certs = data['DomainsCertificate']['Certs']
elif acme_version == 2:
certs = data['Certificates']
2017-06-27 14:09:51 +02:00
# Loop over all certificates
for c in certs:
if acme_version == 1:
name = c['Certificate']['Domain']
privatekey = c['Certificate']['PrivateKey']
fullchain = c['Certificate']['Certificate']
sans = c['Domains']['SANs']
elif acme_version == 2:
name = c['Domain']['Main']
privatekey = c['Key']
fullchain = c['Certificate']
sans = c['Domain']['SANs']
2018-05-22 02:56:26 +02:00
2017-06-27 14:09:51 +02:00
# Decode private key, certificate and chain
privatekey = b64decode(privatekey).decode('utf-8')
fullchain = b64decode(fullchain).decode('utf-8')
2017-06-27 14:09:51 +02:00
start = fullchain.find('-----BEGIN CERTIFICATE-----', 1)
cert = fullchain[0:start]
chain = fullchain[start:]
# Create domain directory if it doesn't exist
2018-05-22 02:56:26 +02:00
directory = 'certs/' + name + '/'
2017-06-27 14:09:51 +02:00
try:
os.makedirs(directory)
except OSError as error:
if error.errno != errno.EEXIST:
raise
# Write private key, certificate and chain to file
with open(directory + 'privkey.pem', 'w') as f:
f.write(privatekey)
with open(directory + 'cert.pem', 'w') as f:
f.write(cert)
with open(directory + 'chain.pem', 'w') as f:
f.write(chain)
with open(directory + 'fullchain.pem', 'w') as f:
f.write(fullchain)
2017-06-27 15:56:54 +02:00
# Write private key, certificate and chain to flat files
directory = 'certs_flat/'
2018-05-22 02:56:26 +02:00
with open(directory + name + '.key', 'w') as f:
2017-06-27 15:56:54 +02:00
f.write(privatekey)
2018-05-22 02:56:26 +02:00
with open(directory + name + '.crt', 'w') as f:
2017-06-27 15:56:54 +02:00
f.write(fullchain)
2018-05-22 02:56:26 +02:00
with open(directory + name + '.chain.pem', 'w') as f:
2017-06-27 15:56:54 +02:00
f.write(chain)
if sans:
for name in sans:
2017-06-27 15:56:54 +02:00
with open(directory + name + '.key', 'w') as f:
f.write(privatekey)
with open(directory + name + '.crt', 'w') as f:
f.write(fullchain)
with open(directory + name + '.chain.pem', 'w') as f:
f.write(chain)
print('Extracted certificate for: ' + name + (', ' + ', '.join(sans) if sans else ''))
2017-06-27 14:09:51 +02:00
if __name__ == "__main__":
# Determine path to watch
path = sys.argv[1] if len(sys.argv) > 1 else './data'
2017-06-27 15:56:54 +02:00
# Create output directories if it doesn't exist
2017-06-27 14:09:51 +02:00
try:
os.makedirs('certs')
except OSError as error:
if error.errno != errno.EEXIST:
raise
2017-06-27 15:56:54 +02:00
try:
os.makedirs('certs_flat')
except OSError as error:
if error.errno != errno.EEXIST:
raise
2017-06-27 14:09:51 +02:00
# Create event handler and observer
event_handler = Handler()
observer = Observer()
# Register the directory to watch
observer.schedule(event_handler, path)
# Main loop to watch the directory
observer.start()
try:
while True:
time.sleep(1)
except KeyboardInterrupt:
observer.stop()
observer.join()