[tor-commits] [tor/master] Merge remote-tracking branch 'public/ticket9969'
nickm at torproject.org
nickm at torproject.org
Fri Jan 23 15:03:15 UTC 2015
commit 5d4bb6f61f360ddcbee83476ff2f93a13f3a19c0
Merge: bd22ad1 ac9b0a3
Author: Nick Mathewson <nickm at torproject.org>
Date: Fri Jan 23 09:36:00 2015 -0500
Merge remote-tracking branch 'public/ticket9969'
Conflicts:
src/or/directory.c
src/or/routerlist.c
src/or/routerlist.h
src/test/include.am
src/test/test.c
src/or/directory.c | 104 ++++++++--------
src/or/directory.h | 11 +-
src/or/or.h | 11 +-
src/or/routerlist.c | 288 ++++++++++++++++++++++++++------------------
src/or/routerlist.h | 4 +
src/test/include.am | 21 ++--
src/test/test.c | 98 +++++++--------
src/test/test_dir.c | 26 ++++
src/test/test_nodelist.c | 2 +-
src/test/test_routerlist.c | 103 ++++++++++++++++
10 files changed, 437 insertions(+), 231 deletions(-)
diff --cc src/or/directory.c
index 4f24f84,7f272ed..348d971
--- a/src/or/directory.c
+++ b/src/or/directory.c
@@@ -198,9 -197,49 +196,50 @@@ dir_conn_purpose_to_string(int purpose
return "(unknown)";
}
+ /** Return the requisite directory information types. */
+ STATIC dirinfo_type_t
+ dir_fetch_type(int dir_purpose, int router_purpose, const char *resource)
+ {
+ dirinfo_type_t type;
+ switch (dir_purpose) {
+ case DIR_PURPOSE_FETCH_EXTRAINFO:
+ type = EXTRAINFO_DIRINFO;
+ if (router_purpose == ROUTER_PURPOSE_BRIDGE)
+ type |= BRIDGE_DIRINFO;
+ else
+ type |= V3_DIRINFO;
+ break;
+ case DIR_PURPOSE_FETCH_SERVERDESC:
+ if (router_purpose == ROUTER_PURPOSE_BRIDGE)
+ type = BRIDGE_DIRINFO;
+ else
+ type = V3_DIRINFO;
+ break;
+ case DIR_PURPOSE_FETCH_STATUS_VOTE:
+ case DIR_PURPOSE_FETCH_DETACHED_SIGNATURES:
+ case DIR_PURPOSE_FETCH_CERTIFICATE:
+ type = V3_DIRINFO;
+ break;
+ case DIR_PURPOSE_FETCH_CONSENSUS:
+ type = V3_DIRINFO;
+ if (resource && !strcmp(resource, "microdesc"))
+ type |= MICRODESC_DIRINFO;
+ break;
+ case DIR_PURPOSE_FETCH_MICRODESC:
+ type = MICRODESC_DIRINFO;
+ break;
+ default:
+ log_warn(LD_BUG, "Unexpected purpose %d", (int)dir_purpose);
+ type = NO_DIRINFO;
+ break;
+ }
+ return type;
+ }
+
-/** Return true iff <b>identity_digest</b> is the digest of a router we
- * believe to support extrainfo downloads. (If <b>is_authority</b> we do
- * additional checking that's only valid for authorities.) */
++
+/** Return true iff <b>identity_digest</b> is the digest of a router which
+ * says that it caches extrainfos. (If <b>is_authority</b> we always
+ * believe that to be true.) */
int
router_supports_extrainfo(const char *identity_digest, int is_authority)
{
@@@ -526,25 -518,21 +539,21 @@@ MOCK_IMPL(void, directory_get_from_dirs
/* */
rs = directory_pick_generic_dirserver(type, pds_flags,
dir_purpose);
- if (!rs) {
- /*XXXX024 I'm pretty sure this can never do any good, since
- * rs isn't set. */
+ if (!rs)
get_via_tor = 1; /* last resort: try routing it via Tor */
- }
}
}
- } else { /* get_via_tor */
+ }
+
+ if (get_via_tor) {
/* Never use fascistfirewall; we're going via Tor. */
- if (1) {
- /* anybody with a non-zero dirport will do. Disregard firewalls. */
- pds_flags |= PDS_IGNORE_FASCISTFIREWALL;
- rs = router_pick_directory_server(type, pds_flags);
- }
+ pds_flags |= PDS_IGNORE_FASCISTFIREWALL;
+ rs = router_pick_directory_server(type, pds_flags);
- /* If we have any hope of building an indirect conn, we know some router
- * descriptors. If (rs==NULL), we can't build circuits anyway, so
- * there's no point in falling back to the authorities in this case. */
}
+ /* If we have any hope of building an indirect conn, we know some router
+ * descriptors. If (rs==NULL), we can't build circuits anyway, so
+ * there's no point in falling back to the authorities in this case. */
if (rs) {
const dir_indirection_t indirection =
get_via_tor ? DIRIND_ANONYMOUS : DIRIND_ONEHOP;
diff --cc src/or/routerlist.c
index b5e58a4,6b1baca..9f389a4
--- a/src/or/routerlist.c
+++ b/src/or/routerlist.c
@@@ -1288,7 -1239,10 +1288,8 @@@ router_get_fallback_dir_servers(void
const routerstatus_t *
router_pick_directory_server(dirinfo_type_t type, int flags)
{
+ int busy = 0;
const routerstatus_t *choice;
- if (get_options()->PreferTunneledDirConns)
- flags |= PDS_PREFER_TUNNELED_DIR_CONNS_;
if (!routerlist)
return NULL;
@@@ -1413,11 -1378,19 +1423,15 @@@ router_pick_dirserver_generic(smartlist
#define DIR_503_TIMEOUT (60*60)
/** Pick a random running valid directory server/mirror from our
-- * routerlist. Arguments are as for router_pick_directory_server(), except
- * that RETRY_IF_NO_SERVERS is ignored.
- * that RETRY_IF_NO_SERVERS is ignored, and:
- *
- * If the PDS_PREFER_TUNNELED_DIR_CONNS_ flag is set, prefer directory servers
- * that we can use with BEGINDIR.
++ * routerlist. Arguments are as for router_pick_directory_server(), except:
+ *
+ * If <b>n_busy_out</b> is provided, set *<b>n_busy_out</b> to the number of
+ * directories that we excluded for no other reason than
+ * PDS_NO_EXISTING_SERVERDESC_FETCH or PDS_NO_EXISTING_MICRODESC_FETCH.
*/
static const routerstatus_t *
- router_pick_directory_server_impl(dirinfo_type_t type, int flags)
+ router_pick_directory_server_impl(dirinfo_type_t type, int flags,
+ int *n_busy_out)
{
const or_options_t *options = get_options();
const node_t *result;
@@@ -1426,10 -1399,13 +1440,12 @@@
smartlist_t *overloaded_direct, *overloaded_tunnel;
time_t now = time(NULL);
const networkstatus_t *consensus = networkstatus_get_latest_consensus();
- int requireother = ! (flags & PDS_ALLOW_SELF);
- int fascistfirewall = ! (flags & PDS_IGNORE_FASCISTFIREWALL);
- int for_guard = (flags & PDS_FOR_GUARD);
- int try_excluding = 1, n_excluded = 0;
+ const int requireother = ! (flags & PDS_ALLOW_SELF);
+ const int fascistfirewall = ! (flags & PDS_IGNORE_FASCISTFIREWALL);
- const int prefer_tunnel = (flags & PDS_PREFER_TUNNELED_DIR_CONNS_);
+ const int no_serverdesc_fetching = (flags & PDS_NO_EXISTING_SERVERDESC_FETCH);
+ const int no_microdesc_fetching = (flags & PDS_NO_EXISTING_MICRODESC_FETCH);
+ const int for_guard = (flags & PDS_FOR_GUARD);
+ int try_excluding = 1, n_excluded = 0, n_busy = 0;
if (!consensus)
return NULL;
@@@ -4247,14 -4358,29 +4289,28 @@@ MOCK_IMPL(STATIC void, initiate_descrip
tor_free(resource);
}
- /** Max amount of hashes to download per request.
- * Since squid does not like URLs >= 4096 bytes we limit it to 96.
- * 4096 - strlen(http://255.255.255.255/tor/server/d/.z) == 4058
- * 4058/41 (40 for the hash and 1 for the + that separates them) => 98
- * So use 96 because it's a nice number.
+ /** Return the max number of hashes to put in a URL for a given request.
*/
- #define MAX_DL_PER_REQUEST 96
- #define MAX_MICRODESC_DL_PER_REQUEST 92
+ static int
+ max_dl_per_request(const or_options_t *options, int purpose)
+ {
+ /* Since squid does not like URLs >= 4096 bytes we limit it to 96.
+ * 4096 - strlen(http://255.255.255.255/tor/server/d/.z) == 4058
+ * 4058/41 (40 for the hash and 1 for the + that separates them) => 98
+ * So use 96 because it's a nice number.
+ */
+ int max = 96;
+ if (purpose == DIR_PURPOSE_FETCH_MICRODESC) {
+ max = 92;
+ }
+ /* If we're going to tunnel our connections, we can ask for a lot more
+ * in a request. */
- if (options->TunnelDirConns &&
- !directory_fetches_from_authorities(options)) {
++ if (!directory_fetches_from_authorities(options)) {
+ max = 500;
+ }
+ return max;
+ }
+
/** Don't split our requests so finely that we are requesting fewer than
* this number per server. */
#define MIN_DL_PER_REQUEST 4
diff --cc src/or/routerlist.h
index 0d22543,1e8b7c9..f106ca2
--- a/src/or/routerlist.h
+++ b/src/or/routerlist.h
@@@ -223,12 -211,10 +223,16 @@@ STATIC int choose_array_element_by_weig
int n_entries);
STATIC void scale_array_elements_to_u64(u64_dbl_t *entries, int n_entries,
uint64_t *total_out);
+
+MOCK_DECL(int, router_descriptor_is_older_than, (const routerinfo_t *router,
+ int seconds));
+MOCK_DECL(STATIC was_router_added_t, extrainfo_insert,
+ (routerlist_t *rl, extrainfo_t *ei, int warn_if_incompatible));
+
+ MOCK_DECL(STATIC void, initiate_descriptor_downloads,
+ (const routerstatus_t *source, int purpose, smartlist_t *digests,
+ int lo, int hi, int pds_flags));
+
#endif
#endif
diff --cc src/test/include.am
index 3c59a8b,74ac526..f07c78b
--- a/src/test/include.am
+++ b/src/test/include.am
@@@ -17,44 -17,32 +17,45 @@@ src_test_AM_CPPFLAGS = -DSHARE_DATADIR=
src_test_test_SOURCES = \
src/test/test.c \
++ src/test/test_accounting.c \
src/test/test_addr.c \
src/test/test_buffers.c \
src/test/test_cell_formats.c \
++ src/test/test_cell_queue.c \
+ src/test/test_channel.c \
+ src/test/test_channeltls.c \
++ src/test/test_checkdir.c \
src/test/test_circuitlist.c \
src/test/test_circuitmux.c \
+ src/test/test_config.c \
src/test/test_containers.c \
src/test/test_controller_events.c \
src/test/test_crypto.c \
-- src/test/test_cell_queue.c \
src/test/test_data.c \
src/test/test_dir.c \
- src/test/test_checkdir.c \
+ src/test/test_entryconn.c \
+ src/test/test_entrynodes.c \
src/test/test_extorport.c \
+ src/test/test_hs.c \
src/test/test_introduce.c \
src/test/test_logging.c \
src/test/test_microdesc.c \
+ src/test/test_nodelist.c \
src/test/test_oom.c \
- src/test/test_accounting.c \
src/test/test_options.c \
++ src/test/test_policy.c \
src/test/test_pt.c \
- src/test/test_relaycell.c \
+ src/test/test_relay.c \
++ src/test/test_relaycell.c \
src/test/test_replay.c \
src/test/test_routerkeys.c \
+ src/test/test_routerlist.c \
++ src/test/test_routerset.c \
+ src/test/test_scheduler.c \
src/test/test_socks.c \
++ src/test/test_status.c \
+ src/test/test_threads.c \
src/test/test_util.c \
- src/test/test_config.c \
- src/test/test_hs.c \
- src/test/test_nodelist.c \
- src/test/test_policy.c \
- src/test/test_status.c \
- src/test/test_routerset.c \
src/ext/tinytest.c
src_test_test_CFLAGS = $(AM_CFLAGS) $(TEST_CFLAGS)
diff --cc src/test/test.c
index fc5290f,8b53754..debee0a
--- a/src/test/test.c
+++ b/src/test/test.c
@@@ -1294,84 -1612,60 +1294,86 @@@ static struct testcase_t test_array[]
END_OF_TESTCASES
};
++extern struct testcase_t accounting_tests[];
extern struct testcase_t addr_tests[];
extern struct testcase_t buffer_tests[];
--extern struct testcase_t crypto_tests[];
--extern struct testcase_t container_tests[];
--extern struct testcase_t util_tests[];
--extern struct testcase_t dir_tests[];
- extern struct testcase_t checkdir_tests[];
--extern struct testcase_t microdesc_tests[];
--extern struct testcase_t pt_tests[];
--extern struct testcase_t config_tests[];
--extern struct testcase_t introduce_tests[];
--extern struct testcase_t replaycache_tests[];
- extern struct testcase_t relaycell_tests[];
extern struct testcase_t cell_format_tests[];
++extern struct testcase_t cell_queue_tests[];
++extern struct testcase_t channel_tests[];
++extern struct testcase_t channeltls_tests[];
++extern struct testcase_t checkdir_tests[];
extern struct testcase_t circuitlist_tests[];
extern struct testcase_t circuitmux_tests[];
--extern struct testcase_t cell_queue_tests[];
--extern struct testcase_t options_tests[];
--extern struct testcase_t socks_tests[];
-extern struct testcase_t extorport_tests[];
++extern struct testcase_t config_tests[];
++extern struct testcase_t container_tests[];
+ extern struct testcase_t controller_event_tests[];
-extern struct testcase_t logging_tests[];
-extern struct testcase_t backtrace_tests[];
++extern struct testcase_t crypto_tests[];
++extern struct testcase_t dir_tests[];
++extern struct testcase_t entryconn_tests[];
+extern struct testcase_t entrynodes_tests[];
- extern struct testcase_t thread_tests[];
+extern struct testcase_t extorport_tests[];
- extern struct testcase_t controller_event_tests[];
- extern struct testcase_t logging_tests[];
extern struct testcase_t hs_tests[];
++extern struct testcase_t introduce_tests[];
++extern struct testcase_t logging_tests[];
++extern struct testcase_t microdesc_tests[];
extern struct testcase_t nodelist_tests[];
--extern struct testcase_t routerkeys_tests[];
extern struct testcase_t oom_tests[];
- extern struct testcase_t accounting_tests[];
++extern struct testcase_t options_tests[];
+extern struct testcase_t policy_tests[];
- extern struct testcase_t status_tests[];
- extern struct testcase_t routerset_tests[];
- extern struct testcase_t router_tests[];
- extern struct testcase_t channel_tests[];
- extern struct testcase_t channeltls_tests[];
++extern struct testcase_t pt_tests[];
+extern struct testcase_t relay_tests[];
++extern struct testcase_t relaycell_tests[];
++extern struct testcase_t replaycache_tests[];
++extern struct testcase_t router_tests[];
++extern struct testcase_t routerkeys_tests[];
+ extern struct testcase_t routerlist_tests[];
++extern struct testcase_t routerset_tests[];
+extern struct testcase_t scheduler_tests[];
- extern struct testcase_t entryconn_tests[];
++extern struct testcase_t socks_tests[];
++extern struct testcase_t status_tests[];
++extern struct testcase_t thread_tests[];
++extern struct testcase_t util_tests[];
static struct testgroup_t testgroups[] = {
{ "", test_array },
- { "buffer/", buffer_tests },
- { "socks/", socks_tests },
++ { "accounting/", accounting_tests },
{ "addr/", addr_tests },
- { "crypto/", crypto_tests },
- { "container/", container_tests },
- { "util/", util_tests },
- { "util/logging/", logging_tests },
- { "util/thread/", thread_tests },
+ { "buffer/", buffer_tests },
{ "cellfmt/", cell_format_tests },
{ "cellqueue/", cell_queue_tests },
- { "dir/", dir_tests },
++ { "channel/", channel_tests },
++ { "channeltls/", channeltls_tests },
+ { "checkdir/", checkdir_tests },
- { "dir/md/", microdesc_tests },
- { "pt/", pt_tests },
- { "config/", config_tests },
- { "replaycache/", replaycache_tests },
- { "relaycell/", relaycell_tests },
- { "introduce/", introduce_tests },
{ "circuitlist/", circuitlist_tests },
{ "circuitmux/", circuitmux_tests },
- { "options/", options_tests },
- { "entrynodes/", entrynodes_tests },
+ { "config/", config_tests },
+ { "container/", container_tests },
+ { "control/", controller_event_tests },
+ { "crypto/", crypto_tests },
+ { "dir/", dir_tests },
+ { "dir/md/", microdesc_tests },
+ { "entryconn/", entryconn_tests },
++ { "entrynodes/", entrynodes_tests },
{ "extorport/", extorport_tests },
- { "control/", controller_event_tests },
{ "hs/", hs_tests },
+ { "introduce/", introduce_tests },
{ "nodelist/", nodelist_tests },
- { "routerkeys/", routerkeys_tests },
{ "oom/", oom_tests },
- { "accounting/", accounting_tests },
+ { "options/", options_tests },
+ { "policy/" , policy_tests },
- { "status/" , status_tests },
- { "routerset/" , routerset_tests },
- { "channel/", channel_tests },
- { "channeltls/", channeltls_tests },
+ { "pt/", pt_tests },
+ { "relay/" , relay_tests },
++ { "relaycell/", relaycell_tests },
++ { "replaycache/", replaycache_tests },
+ { "routerkeys/", routerkeys_tests },
+ { "routerlist/", routerlist_tests },
- { "replaycache/", replaycache_tests },
++ { "routerset/" , routerset_tests },
+ { "scheduler/", scheduler_tests },
+ { "socks/", socks_tests },
++ { "status/" , status_tests },
+ { "util/", util_tests },
+ { "util/logging/", logging_tests },
++ { "util/thread/", thread_tests },
END_OF_GROUPS
};
diff --cc src/test/test_dir.c
index a8c6c6f,c7f2480..991e613
--- a/src/test/test_dir.c
+++ b/src/test/test_dir.c
@@@ -2930,8 -2357,32 +2930,32 @@@ test_dir_http_handling(void *args
tor_free(url);
}
+ static void
+ test_dir_purpose_needs_anonymity(void *arg)
+ {
+ (void)arg;
+ tt_int_op(1, ==, purpose_needs_anonymity(0, ROUTER_PURPOSE_BRIDGE));
+ tt_int_op(1, ==, purpose_needs_anonymity(0, ROUTER_PURPOSE_GENERAL));
+ tt_int_op(0, ==, purpose_needs_anonymity(DIR_PURPOSE_FETCH_MICRODESC,
+ ROUTER_PURPOSE_GENERAL));
+ done: ;
+ }
+
+ static void
+ test_dir_fetch_type(void *arg)
+ {
+ (void)arg;
+ tt_assert(dir_fetch_type(DIR_PURPOSE_FETCH_MICRODESC, ROUTER_PURPOSE_GENERAL,
+ NULL) == MICRODESC_DIRINFO);
+ tt_assert(dir_fetch_type(DIR_PURPOSE_FETCH_SERVERDESC, ROUTER_PURPOSE_BRIDGE,
+ NULL) == BRIDGE_DIRINFO);
+ tt_assert(dir_fetch_type(DIR_PURPOSE_FETCH_CONSENSUS, ROUTER_PURPOSE_GENERAL,
+ "microdesc") == (V3_DIRINFO | MICRODESC_DIRINFO));
+ done: ;
+ }
+
#define DIR_LEGACY(name) \
- { #name, legacy_test_helper, TT_FORK, &legacy_setup, test_dir_ ## name }
+ { #name, test_dir_ ## name , TT_FORK, NULL, NULL }
#define DIR(name,flags) \
{ #name, test_dir_##name, (flags), NULL, NULL }
diff --cc src/test/test_routerlist.c
index 0000000,1c2b1f8..390b691
mode 000000,100644..100644
--- a/src/test/test_routerlist.c
+++ b/src/test/test_routerlist.c
@@@ -1,0 -1,104 +1,103 @@@
+ /* Copyright (c) 2014, The Tor Project, Inc. */
+ /* See LICENSE for licensing information */
+
+ #define ROUTERLIST_PRIVATE
+ #include "or.h"
+ #include "routerlist.h"
+ #include "directory.h"
+ #include "test.h"
+
+
+ /* 4 digests + 3 sep + pre + post + NULL */
+ static char output[4*BASE64_DIGEST256_LEN+3+2+2+1];
+
+ static void
+ mock_get_from_dirserver(uint8_t dir_purpose, uint8_t router_purpose,
+ const char *resource, int pds_flags)
+ {
+ (void)dir_purpose;
+ (void)router_purpose;
+ (void)pds_flags;
- test_assert(resource);
++ tt_assert(resource);
+ strlcpy(output, resource, sizeof(output));
+ done:
+ ;
+ }
+
+ static void
+ test_routerlist_initiate_descriptor_downloads(void *arg)
+ {
+ const char *prose = "unhurried and wise, we perceive.";
+ smartlist_t *digests = smartlist_new();
+ (void)arg;
+
+ for (int i = 0; i < 20; i++) {
+ smartlist_add(digests, (char*)prose);
+ }
+
+ MOCK(directory_get_from_dirserver, mock_get_from_dirserver);
+ initiate_descriptor_downloads(NULL, DIR_PURPOSE_FETCH_MICRODESC,
+ digests, 3, 7, 0);
+ UNMOCK(directory_get_from_dirserver);
+
- test_streq(output, "d/" \
- "dW5odXJyaWVkIGFuZCB3aXNlLCB3ZSBwZXJjZWl2ZS4-" \
- "dW5odXJyaWVkIGFuZCB3aXNlLCB3ZSBwZXJjZWl2ZS4-" \
- "dW5odXJyaWVkIGFuZCB3aXNlLCB3ZSBwZXJjZWl2ZS4-" \
- "dW5odXJyaWVkIGFuZCB3aXNlLCB3ZSBwZXJjZWl2ZS4" \
- ".z");
++ tt_str_op(output, OP_EQ, "d/"
++ "dW5odXJyaWVkIGFuZCB3aXNlLCB3ZSBwZXJjZWl2ZS4-"
++ "dW5odXJyaWVkIGFuZCB3aXNlLCB3ZSBwZXJjZWl2ZS4-"
++ "dW5odXJyaWVkIGFuZCB3aXNlLCB3ZSBwZXJjZWl2ZS4-"
++ "dW5odXJyaWVkIGFuZCB3aXNlLCB3ZSBwZXJjZWl2ZS4"
++ ".z");
+
+ done:
+ smartlist_free(digests);
+ }
+
+ static int count = 0;
+
+ static void
+ mock_initiate_descriptor_downloads(const routerstatus_t *source,
+ int purpose, smartlist_t *digests,
+ int lo, int hi, int pds_flags)
+ {
+ (void)source;
+ (void)purpose;
+ (void)digests;
+ (void)pds_flags;
+ (void)hi;
+ (void)lo;
+ count += 1;
+ }
+
+ static void
+ test_routerlist_launch_descriptor_downloads(void *arg)
+ {
+ smartlist_t *downloadable = smartlist_new();
+ time_t now = time(NULL);
+ char *cp;
+ (void)arg;
+
+ for (int i = 0; i < 100; i++) {
+ cp = tor_malloc(DIGEST256_LEN);
- test_assert(cp);
++ tt_assert(cp);
+ crypto_rand(cp, DIGEST256_LEN);
+ smartlist_add(downloadable, cp);
+ }
+
+ MOCK(initiate_descriptor_downloads, mock_initiate_descriptor_downloads);
+ launch_descriptor_downloads(DIR_PURPOSE_FETCH_MICRODESC, downloadable,
+ NULL, now);
+ tt_int_op(3, ==, count);
+ UNMOCK(initiate_descriptor_downloads);
+
+ done:
+ SMARTLIST_FOREACH(downloadable, char *, cp1, tor_free(cp1));
+ smartlist_free(downloadable);
+ }
+
+ #define NODE(name, flags) \
+ { #name, test_routerlist_##name, (flags), NULL, NULL }
+
+ struct testcase_t routerlist_tests[] = {
+ NODE(initiate_descriptor_downloads, 0),
+ NODE(launch_descriptor_downloads, 0),
+ END_OF_TESTCASES
+ };
-
More information about the tor-commits
mailing list