summaryrefslogtreecommitdiff
path: root/tools/josef_auditor.py
blob: 4cb0d04ea3f604e609d60c10e34d582935912a4a (plain)
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
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
#!/usr/bin/python
# -*- coding: utf-8 -*-		

import time
import base64
import argparse
from pympler.asizeof import asizeof
from certtools import *


base_urls = ["https://plausible.ct.nordu.net/",
			"https://ct1.digicert-ct.com/log/", 
			"https://ct.izenpe.com/",
			"https://log.certly.io/", 
			"https://ct.googleapis.com/aviator/", 
			"https://ct.googleapis.com/pilot/",
			"https://ct.googleapis.com/rocketeer/",
			]

logkeys = {}
logkeys["https://plausible.ct.nordu.net/"] = get_public_key_from_file("../../plausible-logkey.pem")
logkeys["https://ct.googleapis.com/rocketeer/"] = get_public_key_from_file("../../rocketeer-logkey.pem")
logkeys["https://ct.googleapis.com/aviator/"] = get_public_key_from_file("../../aviator-logkey.pem")
logkeys["https://ct.googleapis.com/pilot/"] = get_public_key_from_file("../../pilot-logkey.pem")
logkeys["https://log.certly.io/"] = get_public_key_from_file("../../certly-logkey.pem")
logkeys["https://ct.izenpe.com/"] = get_public_key_from_file("../../izenpe-logkey.pem")
logkeys["https://ct1.digicert-ct.com/log/"] = get_public_key_from_file("../../digicert-logkey.pem")

parser = argparse.ArgumentParser(description="")
parser.add_argument('--audit', action='store_true', help="run lightweight auditor ensuring consistency in STH")
parser.add_argument('--build-sth', action='store_true', help="get all entries and construct STH")


def reduce_leafs_to_root(layer0):
    if len(layer0) == 0:
        return [[hashlib.sha256().digest()]]
    current_layer = layer0
    while len(current_layer) > 1:
        current_layer = next_merkle_layer(current_layer)
    return current_layer

# Get STH and verify signature
def fetch_all_sth():
	sths = {}
	for base_url in base_urls:
		try:
				sths[base_url] =  get_sth(base_url)
		except:
			print "Failed to retrieve STH from " + base_url
			continue

		try:
			check_sth_signature(base_url, sths[base_url], logkeys[base_url])
		except:
			print "Could not verify signature from " + base_url + "!!!"
			continue
	return sths


def verify_consistency(old, new):
	for url in old:
		# try:
			if old[url]["tree_size"]!= new[url]["tree_size"]:
				consistency_proof = get_consistency_proof(url, old[url]["tree_size"], new[url]["tree_size"] )
				decoded_consistency_proof = []
				for item in consistency_proof:
					decoded_consistency_proof.append(base64.b64decode(item))
				res = verify_consistency_proof(decoded_consistency_proof, old[url]["tree_size"], new[url]["tree_size"], old[url]["sha256_root_hash"])
				
				if old[url]["sha256_root_hash"] != str(base64.b64encode(res[0])):
					print "Verification of old hash failed!!!"
					print old[url]["sha256_root_hash"], str(base64.b64encode(res[0]))
				elif new[url]["sha256_root_hash"] != str(base64.b64encode(res[1])):
					print "Verification of new hash failed!!!"
					print new[url]["sha256_root_hash"], str(base64.b64encode(res[1]))
				else:
					print time.strftime("%H:%M:%S", time.gmtime()) + " New STH from " + url + ", timestamp: " + str(new[url]["timestamp"]) + ", size: " + str(new[url]["tree_size"]) + "...OK."

		# except:
		# 	print "ERROR: Could not verify consistency for " + url


def fetch_and_build_tree(old_sth, base_url):
	print "Getting all entries from " + base_url

	sth = old_sth[base_url]

	entries = []
	leafs = []

	while len(leafs) < sth["tree_size"]:
		pre_size = len(leafs)
		entries = get_entries(base_url, len(leafs), sth["tree_size"])["entries"]
		
		for item in entries:
			leafs.append(get_leaf_hash(base64.b64decode(item["leaf_input"])))
		print "Got entries " + str(pre_size) + " to " + str(len(leafs)) + " (total leaf size: " + str(asizeof(leafs)/1024/1024) + "MB)"


	# tree = build_merkle_tree(leafs)
	root = base64.b64encode(reduce_leafs_to_root(leafs)[0])
	# print "Tree size: " + str(asizeof(tree)/1024/1024) + "MB"
	# print len(leafs)

	if root == sth["sha256_root_hash"]:
		print "Verifying root hashes...OK."
	else:
		print "ERROR: Failed to verify root hashes!"
		print "STH root: " + sth["sha256_root_hash"]
		print "Tree root: " + root


def main(args):
	print "Started " + time.strftime("%H:%M:%S", time.gmtime())
	old_sth = fetch_all_sth()

	if args.build_sth:
		fetch_and_build_tree(old_sth, base_urls[2])

	if args.audit:
		print "Running auditor for " +str(len(base_urls)) + " logs..."

		while True:
			time.sleep(1*60-4)
			# print time.strftime("%H:%M:%S", time.gmtime()) + " checking for updates..."
			new_sth = fetch_all_sth()
			verify_consistency(old_sth, new_sth)
			old_sth = new_sth





if __name__ == '__main__':
    main(parser.parse_args())