summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--doc/TODO5
-rw-r--r--src/or/dirvote.c9
-rw-r--r--src/or/routerlist.c48
3 files changed, 57 insertions, 5 deletions
diff --git a/doc/TODO b/doc/TODO
index a6a82f8ce4..ccc327637d 100644
--- a/doc/TODO
+++ b/doc/TODO
@@ -90,9 +90,10 @@ Things we'd like to do in 0.2.0.x:
o Generate certificates
o Authorities load certificates
o Clients cache certificates on disk
- * Learn new ones when they show up in votes.
- * Forget ones that are very old.
+ o Learn new ones when they show up in votes.
+ o Forget ones that are very old.
- Download as needed.
+ o Actually invoke trusted_dirs_flush_certs_to_disk()
* Serve list as needed.
* Detect whether votes are really all for the same period.
o Avoid double-checking signatures every time we get a vote.
diff --git a/src/or/dirvote.c b/src/or/dirvote.c
index e46e18e8f4..23ef147a99 100644
--- a/src/or/dirvote.c
+++ b/src/or/dirvote.c
@@ -1140,6 +1140,15 @@ dirvote_add_vote(const char *vote_body, const char **msg_out)
*msg_out = "Vote not from a recognized v3 authority";
goto err;
}
+ tor_assert(vote->cert);
+ if (!authority_cert_get_by_digests(vote->cert->cache_info.identity_digest,
+ vote->cert->signing_key_digest)) {
+ /* Hey, it's a new cert! */
+ trusted_dirs_load_certs_from_string(
+ vote->cert->cache_info.signed_descriptor_body,
+ 0 /* from_store */);
+ }
+
/* XXXX020 check times; make sure epochs match. */
SMARTLIST_FOREACH(pending_vote_list, pending_vote_t *, v, {
diff --git a/src/or/routerlist.c b/src/or/routerlist.c
index ac74617bda..729141f1da 100644
--- a/src/or/routerlist.c
+++ b/src/or/routerlist.c
@@ -243,6 +243,9 @@ trusted_dirs_load_certs_from_string(const char *contents, int from_store)
if (!from_store)
trusted_dir_servers_certs_changed = 1;
}
+
+ trusted_dirs_flush_certs_to_disk();
+
return 0;
}
@@ -251,7 +254,12 @@ void
trusted_dirs_flush_certs_to_disk(void)
{
char filename[512];
- smartlist_t *chunks = smartlist_create();
+ smartlist_t *chunks;
+
+ if (!trusted_dir_servers_certs_changed)
+ return;
+
+ chunks = smartlist_create();
tor_snprintf(filename,sizeof(filename),"%s"PATH_SEPARATOR"cached-certs",
get_options()->DataDirectory);
@@ -277,6 +285,35 @@ trusted_dirs_flush_certs_to_disk(void)
}
/** DOCDOC */
+static void
+trusted_dirs_remove_old_certs(void)
+{
+ /* Any certificate that has been superseded for more than 48 hours is
+ * irrelevant. */
+#define OLD_CERT_LIFETIME (48*60*60)
+ SMARTLIST_FOREACH(trusted_dir_servers, trusted_dir_server_t *, ds,
+ {
+ authority_cert_t *newest = NULL;
+ if (!ds->v3_certs)
+ continue;
+ SMARTLIST_FOREACH(ds->v3_certs, authority_cert_t *, cert,
+ if (!newest || (cert->cache_info.published_on >
+ newest->cache_info.published_on))
+ newest = cert);
+ SMARTLIST_FOREACH(ds->v3_certs, authority_cert_t *, cert,
+ if (newest && (newest->cache_info.published_on >
+ cert->cache_info.published_on + OLD_CERT_LIFETIME)) {
+ SMARTLIST_DEL_CURRENT(ds->v3_certs, cert);
+ authority_cert_free(cert);
+ trusted_dir_servers_certs_changed = 1;
+ });
+ });
+#undef OLD_CERT_LIFETIME
+
+ trusted_dirs_flush_certs_to_disk();
+}
+
+/** DOCDOC */
authority_cert_t *
authority_cert_get_by_digests(const char *id_digest,
const char *sk_digest)
@@ -622,9 +659,11 @@ int
router_reload_router_list(void)
{
if (router_reload_router_list_impl(0))
- return 1;
+ return -1;
if (router_reload_router_list_impl(1))
- return 1;
+ return -1;
+ if (trusted_dirs_reload_certs())
+ return -1;
return 0;
}
@@ -2697,6 +2736,9 @@ routerlist_remove_old_routers(void)
routerinfo_t *router;
signed_descriptor_t *sd;
digestmap_t *retain;
+
+ trusted_dirs_remove_old_certs();
+
if (!routerlist || !networkstatus_list)
return;