diff options
author | Roger Dingledine <arma@torproject.org> | 2005-12-28 09:07:31 +0000 |
---|---|---|
committer | Roger Dingledine <arma@torproject.org> | 2005-12-28 09:07:31 +0000 |
commit | dbd7b97eff3e3f23219dac29d5e27ed81dd3e69f (patch) | |
tree | f29ebc52d013ff92805394fe1ace7dd958a7af60 /src/or/circuitbuild.c | |
parent | 5d9256a367a9dccb1efa914b95a00622635ff640 (diff) | |
download | tor-dbd7b97eff3e3f23219dac29d5e27ed81dd3e69f.tar.gz tor-dbd7b97eff3e3f23219dac29d5e27ed81dd3e69f.zip |
helper nodes are dead. long live entry nodes.
(config options EntryNodes and StrictEntryNodes still not
implemented.)
svn:r5673
Diffstat (limited to 'src/or/circuitbuild.c')
-rw-r--r-- | src/or/circuitbuild.c | 487 |
1 files changed, 251 insertions, 236 deletions
diff --git a/src/or/circuitbuild.c b/src/or/circuitbuild.c index f19f1d7708..8926bcf475 100644 --- a/src/or/circuitbuild.c +++ b/src/or/circuitbuild.c @@ -18,9 +18,10 @@ const char circuitbuild_c_id[] = /** A global list of all circuits at this hop. */ extern circuit_t *global_circuitlist; -/** A helper_node_t represents our information about a chosen fixed entry, or - * "helper" node. We can't just use a routerinfo_t, since we want to remember - * these even when we don't have a directory. */ +/** An entry_node_t represents our information about a chosen long-term + * first hop, known as a "helper" node in the literature. We can't just + * use a routerinfo_t, since we want to remember these even when we + * don't have a directory. */ typedef struct { char nickname[MAX_NICKNAME_LEN+1]; char identity[DIGEST_LEN]; @@ -30,13 +31,13 @@ typedef struct { * which it was observed to go down. */ time_t unlisted_since; /**< 0 if this router is currently listed, or the * time at which it became unlisted */ -} helper_node_t; +} entry_node_t; -/** A list of our chosen helper nodes. */ -static smartlist_t *helper_nodes = NULL; -/** A value of 1 means that the helper_nodes list has changed +/** A list of our chosen entry nodes. */ +static smartlist_t *entry_nodes = NULL; +/** A value of 1 means that the entry_nodes list has changed * and those changes need to be flushed to disk. */ -static int helper_nodes_dirty = 0; +static int entry_nodes_dirty = 0; /********* END VARIABLES ************/ @@ -48,8 +49,9 @@ static int onion_extend_cpath(uint8_t purpose, crypt_path_t **head_ptr, cpath_build_state_t *state); static int count_acceptable_routers(smartlist_t *routers); static int onion_append_hop(crypt_path_t **head_ptr, extend_info_t *choice); -static routerinfo_t *choose_random_helper(cpath_build_state_t *state); -static void helper_nodes_changed(void); + +static routerinfo_t *choose_random_entry(cpath_build_state_t *state); +static void entry_nodes_changed(void); /** Iterate over values of circ_id, starting from conn-\>next_circ_id, * and with the high bit specified by circ_id_type (see @@ -1117,6 +1119,7 @@ choose_good_exit_server_general(routerlist_t *dir, int need_uptime, routerinfo_t *router; or_options_t *options = get_options(); +//XXX preferredentries = smartlist_create(); add_nickname_list_to_smartlist(preferredentries,options->EntryNodes,1,1); @@ -1491,10 +1494,10 @@ choose_good_middle_server(uint8_t purpose, /** Pick a good entry server for the circuit to be built according to * <b>state</b>. Don't reuse a chosen exit (if any), don't use this * router (if we're an OR), and respect firewall settings; if we're - * using helper nodes, return one. + * using entry_nodes, return one. * - * If <b>state</b> is NULL, we're choosing entries to serve as helper nodes, - * not for any particular circuit. + * If <b>state</b> is NULL, we're choosing routers to serve as entry + * nodes, not for any particular circuit. */ static routerinfo_t * choose_good_entry_server(uint8_t purpose, cpath_build_state_t *state) @@ -1503,9 +1506,9 @@ choose_good_entry_server(uint8_t purpose, cpath_build_state_t *state) smartlist_t *excluded = smartlist_create(); or_options_t *options = get_options(); - if (state && options->UseHelperNodes && + if (state && options->UseEntryNodes && purpose != CIRCUIT_PURPOSE_TESTING) { - return choose_random_helper(state); + return choose_random_entry(state); } if (state && (r = build_state_get_exit_router(state))) { @@ -1530,11 +1533,10 @@ choose_good_entry_server(uint8_t purpose, cpath_build_state_t *state) // XXX we should exclude busy exit nodes here, too, // but only if there are enough other nodes available. choice = router_choose_random_node( - options->EntryNodes, options->ExcludeNodes, + NULL, options->ExcludeNodes, excluded, state ? state->need_uptime : 1, state ? state->need_capacity : 1, - options->_AllowUnverified & ALLOW_UNVERIFIED_ENTRY, - options->StrictEntryNodes); + options->_AllowUnverified & ALLOW_UNVERIFIED_ENTRY, 0); smartlist_free(excluded); return choice; } @@ -1695,8 +1697,8 @@ build_state_get_exit_nickname(cpath_build_state_t *state) return state->chosen_exit->nickname; } -/** Return the router corresponding to <b>h</b>, if <b>h</b> is - * working well enough that we are willing to use it as a helper +/** Return the router corresponding to <b>e</b>, if <b>e</b> is + * working well enough that we are willing to use it as an entry * right now. (Else return NULL.) In particular, it must be * - Listed as either up or never yet contacted; * - Present in the routerlist; @@ -1705,12 +1707,12 @@ build_state_get_exit_nickname(cpath_build_state_t *state) * - Allowed by our current ReachableAddresses config option. */ static INLINE routerinfo_t * -helper_is_live(helper_node_t *h, int need_uptime, int need_capacity) +entry_is_live(entry_node_t *e, int need_uptime, int need_capacity) { routerinfo_t *r; - if (h->down_since && h->made_contact) + if (e->down_since && e->made_contact) return NULL; - r = router_get_by_digest(h->identity); + r = router_get_by_digest(e->identity); if (!r) return NULL; if (router_is_unreliable(r, need_uptime, need_capacity)) @@ -1721,108 +1723,118 @@ helper_is_live(helper_node_t *h, int need_uptime, int need_capacity) return r; } -/** Return the number of helper nodes that we think are usable. */ +/** Return the number of entry nodes that we think are usable. */ static int -num_live_helpers(void) +num_live_entry_nodes(void) { int n = 0; - if (! helper_nodes) + if (! entry_nodes) return 0; - SMARTLIST_FOREACH(helper_nodes, helper_node_t *, helper, + SMARTLIST_FOREACH(entry_nodes, entry_node_t *, entry, { - if (helper_is_live(helper, 0, 1)) + if (entry_is_live(entry, 0, 1)) ++n; }); return n; } -/** Return 1 if <b>digest</b> matches the identity of any entry - * in the helper_nodes list. Else return 0. */ +/** Return 1 if <b>digest</b> matches the identity of any node + * in the entry_nodes list. Else return 0. */ static INLINE int -is_a_helper(char *digest) +is_an_entry(char *digest) { - SMARTLIST_FOREACH(helper_nodes, helper_node_t *, helper, - if(!memcmp(digest, helper->identity, DIGEST_LEN)) + SMARTLIST_FOREACH(entry_nodes, entry_node_t *, entry, + if(!memcmp(digest, entry->identity, DIGEST_LEN)) return 1; ); return 0; } -#define NUM_HELPER_PICK_TRIES 100 +#define NUM_ENTRY_PICK_TRIES 100 -/** Add a new (preferably stable and fast) helper to the end of our - * helper_nodes list. Return a pointer to the router if we succeed, - * or NULL if we can't find any more suitable helpers. If - * <b>tries_left</b> is <= 1, that means you should fail. */ +/** Add a new (preferably stable and fast) entry to our + * entry_nodes list. Return a pointer to the router if we succeed, + * or NULL if we can't find any more suitable entries. If + * <b>tries_left</b> is <= 1, that means you should fail. + * + * [not implemented yet] + * If <b>chosen</b> is defined, use that one, and put it at + * the *beginning* of our entry_nodes list. Else, put the one + * we pick at the end of the list. */ static routerinfo_t * -add_a_helper(int tries_left) +add_an_entry(routerinfo_t *chosen, int tries_left) { - routerinfo_t *entry; - helper_node_t *helper; + routerinfo_t *router; + entry_node_t *entry; if (--tries_left <= 0) { - warn(LD_CIRC, "Tried finding a new helper, but failed. Bad news. XXX."); + warn(LD_CIRC, "Tried finding a new entry, but failed. Bad news. XXX."); return NULL; } - entry = choose_good_entry_server(CIRCUIT_PURPOSE_C_GENERAL, NULL); - if (!entry) + if (chosen) + router = chosen; + else + router = choose_good_entry_server(CIRCUIT_PURPOSE_C_GENERAL, NULL); + if (!router) return NULL; - /* make sure it's not already a helper */ - if (is_a_helper(entry->cache_info.identity_digest)) - return add_a_helper(tries_left); /* recurse */ - helper = tor_malloc_zero(sizeof(helper_node_t)); + /* make sure it's not already an entry */ + if (is_an_entry(router->cache_info.identity_digest)) + return chosen ? NULL : add_an_entry(NULL, tries_left); /* recurse */ + entry = tor_malloc_zero(sizeof(entry_node_t)); /* XXXX Downgrade this to info before release. NM */ - notice(LD_CIRC, "Chose '%s' as new helper node.", entry->nickname); - strlcpy(helper->nickname, entry->nickname, sizeof(helper->nickname)); - memcpy(helper->identity, entry->cache_info.identity_digest, DIGEST_LEN); - smartlist_add(helper_nodes, helper); - return entry; + notice(LD_CIRC, "Chose '%s' as new entry node.", router->nickname); + strlcpy(entry->nickname, router->nickname, sizeof(entry->nickname)); + memcpy(entry->identity, router->cache_info.identity_digest, DIGEST_LEN); + smartlist_add(entry_nodes, entry); + return router; } -/** If the use of helper nodes is configured, choose more helper nodes +/** If the use of entry nodes is configured, choose more entry nodes * until we have enough in the list. */ static void -pick_helper_nodes(void) +pick_entry_nodes(void) { or_options_t *options = get_options(); int changed = 0; - if (!helper_nodes) - helper_nodes = smartlist_create(); + if (!entry_nodes) + entry_nodes = smartlist_create(); + + /* XXX this is where we prepend options->EntryNodes? */ - while (num_live_helpers() < options->NumHelperNodes) { - if (!add_a_helper(NUM_HELPER_PICK_TRIES)) + while (num_live_entry_nodes() < options->NumEntryNodes) { + if (!add_an_entry(NULL, NUM_ENTRY_PICK_TRIES)) break; changed = 1; } if (changed) - helper_nodes_changed(); + entry_nodes_changed(); } -/** Release all storage held by the list of helper nodes. */ +/** Release all storage held by the list of entry nodes. */ void -helper_nodes_free_all(void) +entry_nodes_free_all(void) { - if (helper_nodes) { - SMARTLIST_FOREACH(helper_nodes, helper_node_t *, h, tor_free(h)); - smartlist_free(helper_nodes); - helper_nodes = NULL; + if (entry_nodes) { + SMARTLIST_FOREACH(entry_nodes, entry_node_t *, e, tor_free(e)); + smartlist_free(entry_nodes); + entry_nodes = NULL; } } /* XXX These are 12 hours for now, but I'd like to make them 30 days */ -/** How long (in seconds) do we allow a helper node to be nonfunctional +/** How long (in seconds) do we allow an entry node to be nonfunctional * before we give up on it? */ -#define HELPER_ALLOW_DOWNTIME (1*12*60*60) -/** How long (in seconds) do we allow a helper node to be unlisted in the +#define ENTRY_ALLOW_DOWNTIME (1*12*60*60) +/** How long (in seconds) do we allow an entry node to be unlisted in the * directory before we give up on it? */ -#define HELPER_ALLOW_UNLISTED (1*12*60*60) +#define ENTRY_ALLOW_UNLISTED (1*12*60*60) -/** Remove all helper nodes that have been down or unlisted for so +/** Remove all entry nodes that have been down or unlisted for so * long that we don't think they'll come up again. Return 1 if we * removed any, or 0 if we did nothing. */ static int -remove_dead_helpers(void) +remove_dead_entries(void) { char dbuf[HEX_DIGEST_LEN+1]; char tbuf[ISO_TIME_LEN+1]; @@ -1830,26 +1842,26 @@ remove_dead_helpers(void) int i; int changed = 0; - for (i = 0; i < smartlist_len(helper_nodes); ) { - helper_node_t *helper = smartlist_get(helper_nodes, i); + for (i = 0; i < smartlist_len(entry_nodes); ) { + entry_node_t *entry = smartlist_get(entry_nodes, i); const char *why = NULL; time_t since = 0; - if (helper->unlisted_since && - helper->unlisted_since + HELPER_ALLOW_UNLISTED < now) { + if (entry->unlisted_since && + entry->unlisted_since + ENTRY_ALLOW_UNLISTED < now) { why = "unlisted"; - since = helper->unlisted_since; - } else if (helper->down_since && - helper->down_since + HELPER_ALLOW_DOWNTIME < now) { + since = entry->unlisted_since; + } else if (entry->down_since && + entry->down_since + ENTRY_ALLOW_DOWNTIME < now) { why = "down"; - since = helper->down_since; + since = entry->down_since; } if (why) { - base16_encode(dbuf, sizeof(dbuf), helper->identity, DIGEST_LEN); + base16_encode(dbuf, sizeof(dbuf), entry->identity, DIGEST_LEN); format_local_iso_time(tbuf, since); - warn(LD_CIRC, "Helper node '%s' (%s) has been %s since %s; removing.", - helper->nickname, dbuf, why, tbuf); - tor_free(helper); - smartlist_del_keeporder(helper_nodes, i); + warn(LD_CIRC, "Entry node '%s' (%s) has been %s since %s; removing.", + entry->nickname, dbuf, why, tbuf); + tor_free(entry); + smartlist_del_keeporder(entry_nodes, i); changed = 1; } else ++i; @@ -1858,13 +1870,13 @@ remove_dead_helpers(void) } /** A new directory or router-status has arrived; update the down/listed - * status of the helper nodes. + * status of the entry nodes. * - * A helper is 'down' if the directory lists it as nonrunning. - * A helper is 'unlisted' if the directory doesn't include it. + * An entry is 'down' if the directory lists it as nonrunning. + * An entry is 'unlisted' if the directory doesn't include it. */ void -helper_nodes_set_status_from_directory(void) +entry_nodes_set_status_from_directory(void) { /* Don't call this on startup; only on a fresh download. Otherwise we'll * think that things are unlisted. */ @@ -1872,7 +1884,7 @@ helper_nodes_set_status_from_directory(void) time_t now; int changed = 0; int severity = LOG_NOTICE; - if (! helper_nodes) + if (! entry_nodes) return; routers = router_get_routerlist(); @@ -1881,127 +1893,127 @@ helper_nodes_set_status_from_directory(void) /*XXXX Most of these warns should be non-warns. */ - SMARTLIST_FOREACH(helper_nodes, helper_node_t *, helper, + SMARTLIST_FOREACH(entry_nodes, entry_node_t *, entry, { - routerinfo_t *r = router_get_by_digest(helper->identity); + routerinfo_t *r = router_get_by_digest(entry->identity); if (! r) { - if (! helper->unlisted_since) { - helper->unlisted_since = time(NULL); + if (! entry->unlisted_since) { + entry->unlisted_since = time(NULL); changed = 1; - warn(LD_CIRC,"Helper node '%s' is not listed by directories", - helper->nickname); + warn(LD_CIRC,"Entry node '%s' is not listed by directories.", + entry->nickname); severity = LOG_WARN; } } else { - if (helper->unlisted_since) { - warn(LD_CIRC,"Helper node '%s' is listed again by directories", - helper->nickname); + if (entry->unlisted_since) { + warn(LD_CIRC,"Entry node '%s' is listed again by directories.", + entry->nickname); changed = 1; severity = LOG_WARN; } - helper->unlisted_since = 0; + entry->unlisted_since = 0; if (! r->is_running) { - if (! helper->down_since) { - helper->down_since = now; - warn(LD_CIRC, "Helper node '%s' is now down.", helper->nickname); + if (! entry->down_since) { + entry->down_since = now; + warn(LD_CIRC, "Entry node '%s' is now down.", entry->nickname); changed = 1; severity = LOG_WARN; } } else { - if (helper->down_since) { - notice(LD_CIRC,"Helper node '%s' is up in latest directories", - helper->nickname); + if (entry->down_since) { + notice(LD_CIRC,"Entry node '%s' is up in latest directories", + entry->nickname); changed = 1; } - helper->down_since = 0; + entry->down_since = 0; } } - info(LD_CIRC, "Summary: Helper '%s' is %s, %s, and %s.", - helper->nickname, - helper->down_since ? "down" : "up", - helper->unlisted_since ? "unlisted" : "listed", - helper_is_live(helper, 0, 1) ? "live" : "not live"); + info(LD_CIRC, "Summary: Entry '%s' is %s, %s, and %s.", + entry->nickname, + entry->down_since ? "down" : "up", + entry->unlisted_since ? "unlisted" : "listed", + entry_is_live(entry, 0, 1) ? "live" : "not live"); }); - if (remove_dead_helpers()) + if (remove_dead_entries()) changed = 1; if (changed) { - log_fn(severity, LD_CIRC, " (%d/%d helpers are usable/new)", - num_live_helpers(), smartlist_len(helper_nodes)); - helper_nodes_changed(); + log_fn(severity, LD_CIRC, " (%d/%d entries are usable/new)", + num_live_entry_nodes(), smartlist_len(entry_nodes)); + entry_nodes_changed(); } } /** Called when a connection to an OR with the identity digest <b>digest</b> * is established (<b>succeeded</b>==1) or has failed (<b>succeeded</b>==0). - * If the OR is a helper, change that helper's up/down status. + * If the OR is an entry, change that entry's up/down status. * Return 0 normally, or -1 if we want to tear down the new connection. */ int -helper_node_set_status(const char *digest, int succeeded) +entry_node_set_status(const char *digest, int succeeded) { int changed = 0; int refuse_conn = 0; - if (! helper_nodes) + if (! entry_nodes) return 0; - SMARTLIST_FOREACH(helper_nodes, helper_node_t *, helper, + SMARTLIST_FOREACH(entry_nodes, entry_node_t *, entry, { - if (!memcmp(helper->identity, digest, DIGEST_LEN)) { + if (!memcmp(entry->identity, digest, DIGEST_LEN)) { if (succeeded) { - if (!helper->made_contact) { - /* We've just added a new long-term helper node. Perhaps + if (!entry->made_contact) { + /* We've just added a new long-term entry node. Perhaps * the network just came back? We should give our earlier - * helpers another try too, and close this connection so + * entries another try too, and close this connection so * we don't use it before we've given the others a shot. */ - helper->made_contact = 1; - SMARTLIST_FOREACH(helper_nodes, helper_node_t *, h, + entry->made_contact = 1; + SMARTLIST_FOREACH(entry_nodes, entry_node_t *, e, { routerinfo_t *r; - if (h->made_contact) { - h->down_since = 0; - r = helper_is_live(h, 0, 1); + if (e->made_contact) { + e->down_since = 0; + r = entry_is_live(e, 0, 1); if (r) { refuse_conn = 1; r->is_running = 1; } } - if (h == helper) + if (e == entry) break; }); notice(LD_CIRC, - "Connected to new helper node '%s'. Marking earlier " - "helpers up. %d/%d helpers usable/new.", helper->nickname, - num_live_helpers(), smartlist_len(helper_nodes)); + "Connected to new entry node '%s'. Marking earlier " + "entries up. %d/%d entries usable/new.", entry->nickname, + num_live_entry_nodes(), smartlist_len(entry_nodes)); changed = 1; } - if (helper->down_since) { + if (entry->down_since) { /*XXXX shouldn't be so loud. NM */ notice(LD_CIRC, - "Connection to formerly down helper node '%s' succeeded. " - "%d/%d helpers usable/new.", helper->nickname, - num_live_helpers(), smartlist_len(helper_nodes)); - helper->down_since = 0; + "Connection to formerly down entry node '%s' succeeded. " + "%d/%d entry nodes usable/new.", entry->nickname, + num_live_entry_nodes(), smartlist_len(entry_nodes)); + entry->down_since = 0; changed = 1; } } else { - if (!helper->made_contact) { /* dump him */ + if (!entry->made_contact) { /* dump him */ notice(LD_CIRC, - "Connection to never-contacted helper node '%s' failed. " - "Removing from the list. %d/%d helpers usable/new.", - helper->nickname, - num_live_helpers()-1, smartlist_len(helper_nodes)-1); - tor_free(helper); - smartlist_del_keeporder(helper_nodes, helper_sl_idx); + "Connection to never-contacted entry node '%s' failed. " + "Removing from the list. %d/%d entry nodes usable/new.", + entry->nickname, + num_live_entry_nodes()-1, smartlist_len(entry_nodes)-1); + tor_free(entry); + smartlist_del_keeporder(entry_nodes, entry_sl_idx); changed = 1; - } else if (!helper->down_since) { - helper->down_since = time(NULL); - warn(LD_CIRC, "Connection to helper node '%s' failed." - " %d/%d helpers usable/new.", - helper->nickname, - num_live_helpers(), smartlist_len(helper_nodes)); + } else if (!entry->down_since) { + entry->down_since = time(NULL); + warn(LD_CIRC, "Connection to entry node '%s' failed." + " %d/%d entry nodes usable/new.", + entry->nickname, + num_live_entry_nodes(), smartlist_len(entry_nodes)); changed = 1; } } @@ -2009,97 +2021,97 @@ helper_node_set_status(const char *digest, int succeeded) }); if (changed) - helper_nodes_changed(); + entry_nodes_changed(); return refuse_conn ? -1 : 0; } -/** Pick a live (up and listed) helper node from the list of helpers, and +/** Pick a live (up and listed) entry node from entry_nodes, and * make sure not to pick this circuit's exit. */ static routerinfo_t * -choose_random_helper(cpath_build_state_t *state) +choose_random_entry(cpath_build_state_t *state) { - smartlist_t *live_helpers = smartlist_create(); + smartlist_t *live_entry_nodes = smartlist_create(); routerinfo_t *chosen_exit = build_state_get_exit_router(state); routerinfo_t *r; int need_uptime = state->need_uptime; int need_capacity = state->need_capacity; - if (! helper_nodes || - smartlist_len(helper_nodes) < get_options()->NumHelperNodes) - pick_helper_nodes(); + if (! entry_nodes || + smartlist_len(entry_nodes) < get_options()->NumEntryNodes) + pick_entry_nodes(); retry: - smartlist_clear(live_helpers); - SMARTLIST_FOREACH(helper_nodes, helper_node_t *, helper, + smartlist_clear(live_entry_nodes); + SMARTLIST_FOREACH(entry_nodes, entry_node_t *, entry, { - r = helper_is_live(helper, need_uptime, need_capacity); + r = entry_is_live(entry, need_uptime, need_capacity); if (r && r != chosen_exit) { - smartlist_add(live_helpers, r); - if (smartlist_len(live_helpers) >= get_options()->NumHelperNodes) + smartlist_add(live_entry_nodes, r); + if (smartlist_len(live_entry_nodes) >= get_options()->NumEntryNodes) break; /* we have enough */ } }); /* Try to have at least 2 choices available. This way we don't - * get stuck with a single live-but-crummy helper and just keep + * get stuck with a single live-but-crummy entry and just keep * using him. - * (We might get 2 live-but-crummy helpers, but so be it.) */ - if (smartlist_len(live_helpers) < 2) { + * (We might get 2 live-but-crummy entry nodes, but so be it.) */ + if (smartlist_len(live_entry_nodes) < 2) { if (need_uptime) { need_uptime = 0; /* try without that requirement */ goto retry; } - /* still no? try adding a new helper then */ - r = add_a_helper(NUM_HELPER_PICK_TRIES); + /* still no? try adding a new entry then */ + r = add_an_entry(NULL, NUM_ENTRY_PICK_TRIES); if (r) { - smartlist_add(live_helpers, r); - helper_nodes_changed(); + smartlist_add(live_entry_nodes, r); + entry_nodes_changed(); } else { if (need_capacity) { /* still no? last attempt, try without requiring capacity */ need_capacity = 0; goto retry; } - /* live_helpers will be empty below. Oh well, we tried. */ + /* live_entry_nodes will be empty below. Oh well, we tried. */ } } - r = smartlist_choose(live_helpers); - smartlist_free(live_helpers); + r = smartlist_choose(live_entry_nodes); + smartlist_free(live_entry_nodes); return r; } -/** Parse <b>state</b> and learn about the helper nodes it describes. +/** Parse <b>state</b> and learn about the entry nodes it describes. * If <b>set</b> is true, and there are no errors, replace the global - * helper list with what we find. + * entry_list with what we find. * On success, return 0. On failure, set *<b>err</b> to a string * describing the error, and return -1. */ int -helper_nodes_parse_state(or_state_t *state, int set, const char **err) +entry_nodes_parse_state(or_state_t *state, int set, const char **err) { - helper_node_t *node = NULL; - smartlist_t *new_helpers = smartlist_create(); + entry_node_t *node = NULL; + smartlist_t *new_entry_nodes = smartlist_create(); config_line_t *line; *err = NULL; - for (line = state->HelperNodes; line; line = line->next) { - if (!strcasecmp(line->key, "HelperNode")) { + for (line = state->EntryNodes; line; line = line->next) { + if (!strcasecmp(line->key, "EntryNode")) { smartlist_t *args = smartlist_create(); - node = tor_malloc_zero(sizeof(helper_node_t)); - node->made_contact = 1; /* all helpers on disk have been contacted */ - smartlist_add(new_helpers, node); + node = tor_malloc_zero(sizeof(entry_node_t)); + node->made_contact = 1; /* all entry nodes on disk have been contacted */ + smartlist_add(new_entry_nodes, node); smartlist_split_string(args, line->value, " ", SPLIT_SKIP_SPACE|SPLIT_IGNORE_BLANK, 0); if (smartlist_len(args)<2) { - *err = "Too few arguments to HelperNode"; + *err = "Too few arguments to EntryNode"; } else if (!is_legal_nickname(smartlist_get(args,0))) { - *err = "Bad nickname for HelperNode"; + *err = "Bad nickname for EntryNode"; } else { strlcpy(node->nickname, smartlist_get(args,0), MAX_NICKNAME_LEN+1); if (base16_decode(node->identity, DIGEST_LEN, smartlist_get(args,1), strlen(smartlist_get(args,1)))<0) { - *err = "Bad hex digest for HelperNode"; + *err = "Bad hex digest for EntryNode"; } } SMARTLIST_FOREACH(args, char*, cp, tor_free(cp)); @@ -2109,14 +2121,14 @@ helper_nodes_parse_state(or_state_t *state, int set, const char **err) } else { time_t when; if (!node) { - *err = "HelperNodeDownSince/UnlistedSince without HelperNode"; + *err = "EntryNodeDownSince/UnlistedSince without EntryNode"; break; } if (parse_iso_time(line->value, &when)<0) { - *err = "Bad time in HelperNodeDownSince/UnlistedSince"; + *err = "Bad time in EntryNodeDownSince/UnlistedSince"; break; } - if (!strcasecmp(line->key, "HelperNodeDownSince")) + if (!strcasecmp(line->key, "EntryNodeDownSince")) node->down_since = when; else node->unlisted_since = when; @@ -2124,108 +2136,111 @@ helper_nodes_parse_state(or_state_t *state, int set, const char **err) } if (*err || !set) { - SMARTLIST_FOREACH(new_helpers, helper_node_t *, h, tor_free(h)); - smartlist_free(new_helpers); + SMARTLIST_FOREACH(new_entry_nodes, entry_node_t *, e, tor_free(e)); + smartlist_free(new_entry_nodes); } else { /* !*err && set */ - if (helper_nodes) { - SMARTLIST_FOREACH(helper_nodes, helper_node_t *, h, tor_free(h)); - smartlist_free(helper_nodes); + if (entry_nodes) { + SMARTLIST_FOREACH(entry_nodes, entry_node_t *, e, tor_free(e)); + smartlist_free(entry_nodes); } - helper_nodes = new_helpers; - helper_nodes_dirty = 0; + entry_nodes = new_entry_nodes; + entry_nodes_dirty = 0; } return *err ? -1 : 0; } -/** Our list of helper nodes has changed, or some element of one - * of our helper nodes has changed. Write the changes to disk. */ +/** Our list of entry nodes has changed, or some element of one + * of our entry nodes has changed. Write the changes to disk. */ static void -helper_nodes_changed(void) +entry_nodes_changed(void) { - helper_nodes_dirty = 1; + entry_nodes_dirty = 1; or_state_save(); } -/** If the helper node info has not changed, do nothing and return. - * Otherwise, free the HelperNodes piece of <b>state</b> and create - * a new one out of the global helper_nodes list, and then mark - * <b>state</b> dirty so it will know to get saved to disk. +/** If the entry node info has not changed, do nothing and return. + * Otherwise, free the EntryNodes piece of <b>state</b> and create + * a new one out of the global entry_nodes list, and then mark + * <b>state</b> dirty so it will get saved to disk. */ void -helper_nodes_update_state(or_state_t *state) +entry_nodes_update_state(or_state_t *state) { config_line_t **next, *line; - if (! helper_nodes_dirty) + if (! entry_nodes_dirty) return; - config_free_lines(state->HelperNodes); - next = &state->HelperNodes; + config_free_lines(state->EntryNodes); + next = &state->EntryNodes; *next = NULL; - if (!helper_nodes) - helper_nodes = smartlist_create(); - SMARTLIST_FOREACH(helper_nodes, helper_node_t *, h, + if (!entry_nodes) + entry_nodes = smartlist_create(); + SMARTLIST_FOREACH(entry_nodes, entry_node_t *, e, { char dbuf[HEX_DIGEST_LEN+1]; - if (!h->made_contact) + if (!e->made_contact) continue; /* don't write this one to disk */ *next = line = tor_malloc_zero(sizeof(config_line_t)); - line->key = tor_strdup("HelperNode"); + line->key = tor_strdup("EntryNode"); line->value = tor_malloc(HEX_DIGEST_LEN+MAX_NICKNAME_LEN+2); - base16_encode(dbuf, sizeof(dbuf), h->identity, DIGEST_LEN); + base16_encode(dbuf, sizeof(dbuf), e->identity, DIGEST_LEN); tor_snprintf(line->value,HEX_DIGEST_LEN+MAX_NICKNAME_LEN+2, - "%s %s", h->nickname, dbuf); + "%s %s", e->nickname, dbuf); next = &(line->next); - if (h->down_since) { + if (e->down_since) { *next = line = tor_malloc_zero(sizeof(config_line_t)); - line->key = tor_strdup("HelperNodeDownSince"); + line->key = tor_strdup("EntryNodeDownSince"); line->value = tor_malloc(ISO_TIME_LEN+1); - format_iso_time(line->value, h->down_since); + format_iso_time(line->value, e->down_since); next = &(line->next); } - if (h->unlisted_since) { + if (e->unlisted_since) { *next = line = tor_malloc_zero(sizeof(config_line_t)); - line->key = tor_strdup("HelperNodeUnlistedSince"); + line->key = tor_strdup("EntryNodeUnlistedSince"); line->value = tor_malloc(ISO_TIME_LEN+1); - format_iso_time(line->value, h->unlisted_since); + format_iso_time(line->value, e->unlisted_since); next = &(line->next); } }); state->dirty = 1; - helper_nodes_dirty = 0; + entry_nodes_dirty = 0; } -/** If <b>question</b> is the string "helper-nodes", then dump +/** If <b>question</b> is the string "entry-nodes", then dump * to *<b>answer</b> a newly allocated string describing all of - * the nodes in the global helper_nodes list. See control-spec.txt - * for details. */ + * the nodes in the global entry_nodes list. See control-spec.txt + * for details. + * For backward compatibility, we also handle the string "helper-nodes". + * */ int -helper_nodes_getinfo_helper(const char *question, char **answer) +entry_nodes_getinfo(const char *question, char **answer) { - if (!strcmp(question,"helper-nodes")) { + if (!strcmp(question,"entry-nodes") || + !strcmp(question,"helper-nodes")) { smartlist_t *sl = smartlist_create(); char tbuf[ISO_TIME_LEN+1]; char dbuf[HEX_DIGEST_LEN+1]; - if (!helper_nodes) - helper_nodes = smartlist_create(); - SMARTLIST_FOREACH(helper_nodes, helper_node_t *, h, + if (!entry_nodes) + entry_nodes = smartlist_create(); + SMARTLIST_FOREACH(entry_nodes, entry_node_t *, e, { size_t len = HEX_DIGEST_LEN+ISO_TIME_LEN+32; char *c = tor_malloc(len); const char *status = NULL; time_t when = 0; - if (!h->made_contact) { + if (!e->made_contact) { status = "never-connected"; - } else if (h->unlisted_since) { - when = h->unlisted_since; + } else if (e->unlisted_since) { + when = e->unlisted_since; status = "unlisted"; - } else if (h->down_since) { - when = h->down_since; + } else if (e->down_since) { + when = e->down_since; status = "down"; } else { status = "up"; } - base16_encode(dbuf, sizeof(dbuf), h->identity, DIGEST_LEN); + base16_encode(dbuf, sizeof(dbuf), e->identity, DIGEST_LEN); if (when) { format_iso_time(tbuf, when); tor_snprintf(c, len, "$%s %s %s\n", dbuf, status, tbuf); |