diff options
author | Nick Mathewson <nickm@torproject.org> | 2005-07-28 19:01:48 +0000 |
---|---|---|
committer | Nick Mathewson <nickm@torproject.org> | 2005-07-28 19:01:48 +0000 |
commit | 299af3d3936be134bb1965eb8bc52916fd90d76c (patch) | |
tree | d2547c2c3d897ab830046791fd9a21d7ff59b5a6 /src/or | |
parent | c22bcecd839a86b9e1c07d942d47ee56d45a35d3 (diff) | |
download | tor-299af3d3936be134bb1965eb8bc52916fd90d76c.tar.gz tor-299af3d3936be134bb1965eb8bc52916fd90d76c.zip |
Possibly broken implementation of persistant state; handles helper nodes; does not handle accounting info yet.
svn:r4680
Diffstat (limited to 'src/or')
-rw-r--r-- | src/or/circuitbuild.c | 123 | ||||
-rw-r--r-- | src/or/config.c | 307 | ||||
-rw-r--r-- | src/or/hibernate.c | 3 | ||||
-rw-r--r-- | src/or/main.c | 1 | ||||
-rw-r--r-- | src/or/or.h | 22 |
5 files changed, 429 insertions, 27 deletions
diff --git a/src/or/circuitbuild.c b/src/or/circuitbuild.c index 8a92ea6125..6c20ac8a57 100644 --- a/src/or/circuitbuild.c +++ b/src/or/circuitbuild.c @@ -31,6 +31,7 @@ typedef struct { /** A list of our chosen helper nodes. */ static smartlist_t *helper_nodes = NULL; +static int helper_nodes_dirty = 0; /********* END VARIABLES ************/ @@ -46,6 +47,7 @@ static void pick_helper_nodes(void); static routerinfo_t *choose_random_helper(void); static void clear_helper_nodes(void); static void remove_dead_helpers(void); +static void helper_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 @@ -1635,6 +1637,7 @@ pick_helper_nodes(void) strlcpy(helper->nickname, entry->nickname, sizeof(helper->nickname)); memcpy(helper->identity, entry->identity_digest, DIGEST_LEN); smartlist_add(helper_nodes, helper); + helper_nodes_changed(); } } @@ -1644,6 +1647,7 @@ clear_helper_nodes(void) { SMARTLIST_FOREACH(helper_nodes, helper_node_t *, h, tor_free(h)); smartlist_clear(helper_nodes); + helper_nodes_changed(); } /** How long (in seconds) do we allow a helper node to be nonfunctional before @@ -1681,6 +1685,7 @@ remove_dead_helpers(void) helper->nickname, dbuf, why, tbuf); tor_free(helper); smartlist_del(helper_nodes, i); + helper_nodes_changed(); } else ++i; } @@ -1747,9 +1752,11 @@ helper_nodes_set_status_from_directory(void) } }); - if (changed) + if (changed) { log_fn(LOG_WARN, " (%d/%d helpers are usable)", num_live_helpers(), smartlist_len(helper_nodes)); + helper_nodes_changed(); + } remove_dead_helpers(); pick_helper_nodes(); @@ -1775,6 +1782,7 @@ helper_node_set_status(const char *digest, int succeeded) "Connection to formerly down helper node '%s' succeeded. " "%d/%d helpers usable.", helper->nickname, num_live_helpers(), smartlist_len(helper_nodes)); + helper_nodes_changed(); } helper->down_since = 0; } else if (!helper->down_since) { @@ -1782,6 +1790,7 @@ helper_node_set_status(const char *digest, int succeeded) log_fn(LOG_WARN, "Connection to helper node '%s' failed. %d/%d helpers usable.", helper->nickname, num_live_helpers(), smartlist_len(helper_nodes)); + helper_nodes_changed(); } } }); @@ -1818,3 +1827,115 @@ choose_random_helper(void) return r; } +/** DOCDOC */ +int +helper_nodes_parse_state(or_state_t *state, int set, const char **err) +{ + helper_node_t *node = NULL; + smartlist_t *helpers = smartlist_create(); + config_line_t *line; + + *err = NULL; + for (line = state->HelperNodes; line; line = line->next) { + if (!strcasecmp(line->key, "HelperNode")) { + smartlist_t *args = smartlist_create(); + node = tor_malloc_zero(sizeof(helper_node_t)); + smartlist_add(helpers, 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"; + break; + } + if (!is_legal_nickname(smartlist_get(args,0))) { + *err = "Bad nickname for HelperNode"; + break; + } + 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,11)))<0) { + *err = "Bad hex digest for HelperNode"; + break; + } + } else { + time_t when; + if (!node) { + *err = "HelperNodeDownSince/UnlistedSince without HelperNode"; + break; + } + if (parse_iso_time(line->value, &when)<0) { + *err = "Bad time in HelperNodeDownSince/UnlistedSince"; + break; + } + if (!strcasecmp(line->key, "HelperNodeDownSince")) + node->down_since = when; + else + node->unlisted_since = when; + } + } + + if (*err || !set) { + SMARTLIST_FOREACH(helpers, helper_node_t *, h, tor_free(h)); + smartlist_free(helpers); + } + if (!*err && set) { + if (helper_nodes) { + SMARTLIST_FOREACH(helper_nodes, helper_node_t *, h, tor_free(h)); + smartlist_free(helper_nodes); + } + helper_nodes = helpers; + helper_nodes_dirty = 0; + } + return *err ? -1 : 0; +} + +/** DOCDOC */ +static void +helper_nodes_changed(void) +{ + helper_nodes_dirty = 1; + + or_state_save(); +} + +/** DOCDOC */ +int +helper_nodes_update_state(or_state_t *state) +{ + config_line_t **next, *line; + if (! helper_nodes_dirty) + return 0; + + config_free_lines(state->HelperNodes); + next = &state->HelperNodes; + *next = NULL; + SMARTLIST_FOREACH(helper_nodes, helper_node_t *, h, + { + char dbuf[HEX_DIGEST_LEN+1]; + *next = line = tor_malloc_zero(sizeof(config_line_t)); + line->key = tor_strdup("HelperNode"); + line->value = tor_malloc(HEX_DIGEST_LEN+MAX_NICKNAME_LEN+2); + base16_encode(dbuf, sizeof(dbuf), h->identity, DIGEST_LEN); + tor_snprintf(line->value,HEX_DIGEST_LEN+MAX_NICKNAME_LEN+2, + "%s %s", h->nickname, dbuf); + next = &(line->next); + if (h->down_since) { + *next = line = tor_malloc_zero(sizeof(config_line_t)); + line->key = tor_strdup("HelperNodeDownSince"); + line->value = tor_malloc(ISO_TIME_LEN+1); + format_iso_time(line->value, h->down_since); + next = &(line->next); + } + if (h->unlisted_since) { + *next = line = tor_malloc_zero(sizeof(config_line_t)); + line->key = tor_strdup("HelperNodeUnlistedSince"); + line->value = tor_malloc(ISO_TIME_LEN+1); + format_iso_time(line->value, h->unlisted_since); + next = &(line->next); + } + }); + state->dirty = 1; + helper_nodes_dirty = 0; + + return 1; +} diff --git a/src/or/config.c b/src/or/config.c index 43edabb687..47c0fd1a44 100644 --- a/src/or/config.c +++ b/src/or/config.c @@ -24,6 +24,7 @@ typedef enum config_type_t { CONFIG_TYPE_MEMUNIT, /**< A number of bytes, with optional units*/ CONFIG_TYPE_DOUBLE, /**< A floating-point value */ CONFIG_TYPE_BOOL, /**< A boolean value, expressed as 0 or 1. */ + CONFIG_TYPE_ISOTIME, /**< An ISO-formated time relative to GMT. */ CONFIG_TYPE_CSV, /**< A list of strings, separated by commas and optional * whitespace. */ CONFIG_TYPE_LINELIST, /**< Uninterpreted config lines */ @@ -47,7 +48,7 @@ typedef struct config_abbrev_t { #define PLURAL(tok) { #tok, #tok "s", 0 } /* A list of command-line abbreviations. */ -static config_abbrev_t _config_abbrevs[] = { +static config_abbrev_t _option_abbrevs[] = { PLURAL(ExitNode), PLURAL(EntryNode), PLURAL(ExcludeNode), @@ -75,6 +76,7 @@ typedef struct config_var_t { config_type_t type; /**< How to interpret the type and turn it into a value. */ off_t var_offset; /**< Offset of the corresponding member of or_options_t. */ const char *initvalue; /**< String (or null) describing initial value. */ + const char *description; } config_var_t; /** Return the offset of <b>member</b> within the type <b>tp</b>, in bytes */ @@ -83,7 +85,7 @@ typedef struct config_var_t { * CONFIG_TYPE_<b>conftype</b>, and corresponds to * or_options_t.<b>member</b>" */ -#define VAR(name,conftype,member,initvalue) \ +#define VAR(name,conftype,member,initvalue) \ { name, CONFIG_TYPE_ ## conftype, STRUCT_OFFSET(or_options_t, member), initvalue } /** An entry for config_vars: "The option <b>name</b> is obsolete." */ #define OBSOLETE(name) { name, CONFIG_TYPE_OBSOLETE, 0, NULL } @@ -92,7 +94,7 @@ typedef struct config_var_t { * abbreviations, order is significant, since the first matching option will * be chosen first. */ -static config_var_t _config_vars[] = { +static config_var_t _option_vars[] = { VAR("Address", STRING, Address, NULL), VAR("AccountingStart", STRING, AccountingStart, NULL), VAR("AllowUnverifiedNodes",CSV, AllowUnverifiedNodes, "middle,rendezvous"), @@ -186,8 +188,50 @@ static config_var_t _config_vars[] = { { NULL, CONFIG_TYPE_OBSOLETE, 0, NULL } }; #undef VAR + +#define VAR(name,conftype,member,initvalue) \ + { name, CONFIG_TYPE_ ## conftype, STRUCT_OFFSET(or_state_t, member), initvalue } +static config_var_t _state_vars[] = { + VAR("LastWritten", ISOTIME, LastWritten, NULL), + + VAR("AccountingIntervalStart", ISOTIME, AccountingIntervalStart, NULL), + VAR("AccountingBytesWrittenInInterval", MEMUNIT, + AccountingBytesWrittenInInterval, NULL), + VAR("AccountingBytesReadInterval", MEMUNIT, AccountingBytesReadInInterval,NULL), + VAR("AccountingSecondsActive", INTERVAL, AccountingSecondsActive, NULL), + VAR("AccountingExpectedUsage", MEMUNIT, AccountingExpectedUsage, NULL), + + VAR("HelperNodes", LINELIST_V, HelperNodes, NULL), + VAR("HelperNode", LINELIST_S, HelperNodes, NULL), + VAR("HelperNodeDownSince", LINELIST_S, HelperNodes, NULL), + VAR("HelperNodeUnlistedSince", LINELIST_S, HelperNodes, NULL), + { NULL, CONFIG_TYPE_OBSOLETE, 0, NULL } +}; + +#undef VAR #undef OBSOLETE +/** DOCDOC*/ +typedef struct config_var_description_t { + const char *name; + const char *description; +} config_var_description_t; + +static config_var_description_t options_description[] = { + { "Address", "The advertised (external) address we should use" }, + // { "AccountingStart", ""}, + { NULL, NULL }, +}; + +static config_var_description_t state_description[] = { + { "HelperNode", "One of the nodes we have chosen as a fixed entry" }, + { "HelperNodeDownSince", + "The last helper node has been down since this time." }, + { "HelperNodeUnlistedSince", + "The last helper node has been unlisted since this time." }, + { NULL, NULL }, +}; + typedef int (*validate_fn_t)(void*); typedef struct { @@ -197,6 +241,7 @@ typedef struct { config_abbrev_t *abbrevs; config_var_t *vars; validate_fn_t validate_fn; + config_var_description_t *descriptions; } config_format_t; #define CHECK(fmt, cfg) do { \ @@ -211,7 +256,7 @@ static void config_line_append(config_line_t **lst, const char *key, const char *val); static void option_reset(config_format_t *fmt, or_options_t *options, config_var_t *var); -static void options_free(config_format_t *fmt, or_options_t *options); +static void config_free(config_format_t *fmt, void *options); static int option_is_same(config_format_t *fmt, or_options_t *o1, or_options_t *o2,const char *name); static or_options_t *options_dup(config_format_t *fmt, or_options_t *old); @@ -236,7 +281,8 @@ static int validate_data_directory(or_options_t *options); static int write_configuration_file(const char *fname, or_options_t *options); static config_line_t *get_assigned_option(config_format_t *fmt, or_options_t *options, const char *key); -static void config_init(config_format_t *fmt, or_options_t *options); +static void config_init(config_format_t *fmt, void *options); +static int or_state_validate(or_state_t *options); static uint64_t config_parse_memunit(const char *s, int *ok); static int config_parse_interval(const char *s, int *ok); @@ -252,9 +298,22 @@ static config_format_t config_format = { sizeof(or_options_t), OR_OPTIONS_MAGIC, STRUCT_OFFSET(or_options_t, _magic), - _config_abbrevs, - _config_vars, - (validate_fn_t)options_validate + _option_abbrevs, + _option_vars, + (validate_fn_t)options_validate, + options_description +}; + +#define OR_STATE_MAGIC 0x57A73f57 + +static config_format_t state_format = { + sizeof(or_state_t), + OR_STATE_MAGIC, + STRUCT_OFFSET(or_state_t, _magic), + NULL, + _state_vars, + (validate_fn_t)or_state_validate, + state_description }; /* @@ -262,9 +321,11 @@ static config_format_t config_format = { */ /** Command-line and config-file options. */ -static or_options_t *global_options=NULL; +static or_options_t *global_options = NULL; /** Name of most recently read torrc file. */ static char *config_fname = NULL; +/** Persistant serialized state. */ +static or_state_t *global_state = NULL; static void * config_alloc(config_format_t *fmt) @@ -290,14 +351,14 @@ void set_options(or_options_t *new_val) { if (global_options) - options_free(&config_format, global_options); + config_free(&config_format, global_options); global_options = new_val; } void config_free_all(void) { - options_free(&config_format, global_options); + config_free(&config_format, global_options); tor_free(config_fname); } @@ -388,6 +449,11 @@ options_act(void) libevent_initialized = 1; } + /* Load state */ + if (! global_state) + if (or_state_load()) + return -1; + options->_ConnLimit = set_max_file_descriptors((unsigned)options->ConnLimit, MAXCONNECTIONS); if (options->_ConnLimit < 0) @@ -695,6 +761,13 @@ config_assign_line(config_format_t *fmt, *(double *)lvalue = atof(c->value); break; + case CONFIG_TYPE_ISOTIME: + if (parse_iso_time(c->value, (time_t *)lvalue)) { + log(LOG_WARN, "Invalid time '%s' for keyword '%s'", c->value, c->key); + return -2; + } + break; + case CONFIG_TYPE_CSV: if (*(smartlist_t**)lvalue) { SMARTLIST_FOREACH(*(smartlist_t**)lvalue, char *, cp, tor_free(cp)); @@ -814,6 +887,15 @@ get_assigned_option(config_format_t *fmt, or_options_t *options, const char *key return NULL; } break; + case CONFIG_TYPE_ISOTIME: + if (*(time_t*)value) { + result->value = tor_malloc(ISO_TIME_LEN+1); + format_iso_time(result->value, *(time_t*)value); + } else { + tor_free(result->key); + tor_free(result); + } + break; case CONFIG_TYPE_INTERVAL: case CONFIG_TYPE_UINT: /* This means every or_options_t uint or bool element @@ -866,7 +948,7 @@ get_assigned_option(config_format_t *fmt, or_options_t *options, const char *key */ static int config_assign(config_format_t *fmt, - or_options_t *options, config_line_t *list, int reset) + void *options, config_line_t *list, int reset) { config_line_t *p; @@ -911,17 +993,17 @@ options_trial_assign(config_line_t *list, int reset) or_options_t *trial_options = options_dup(&config_format, get_options()); if ((r=config_assign(&config_format, trial_options, list, reset)) < 0) { - options_free(&config_format, trial_options); + config_free(&config_format, trial_options); return r; } if (options_validate(trial_options) < 0) { - options_free(&config_format, trial_options); + config_free(&config_format, trial_options); return -2; } if (options_transition_allowed(get_options(), trial_options) < 0) { - options_free(&config_format, trial_options); + config_free(&config_format, trial_options); return -3; } @@ -947,6 +1029,8 @@ option_reset(config_format_t *fmt, or_options_t *options, config_var_t *var) case CONFIG_TYPE_DOUBLE: *(double*)lvalue = 0.0; break; + case CONFIG_TYPE_ISOTIME: + *(time_t*)lvalue = 0; case CONFIG_TYPE_INTERVAL: case CONFIG_TYPE_UINT: case CONFIG_TYPE_BOOL: @@ -1132,7 +1216,7 @@ get_default_nickname(void) /** Release storage held by <b>options</b> */ static void -options_free(config_format_t *fmt,or_options_t *options) +config_free(config_format_t *fmt, void *options) { int i; void *lvalue; @@ -1147,6 +1231,7 @@ options_free(config_format_t *fmt,or_options_t *options) case CONFIG_TYPE_UINT: case CONFIG_TYPE_BOOL: case CONFIG_TYPE_DOUBLE: + case CONFIG_TYPE_ISOTIME: case CONFIG_TYPE_OBSOLETE: break; /* nothing to free for these config types */ case CONFIG_TYPE_STRING: @@ -1238,8 +1323,9 @@ options_init(or_options_t *options) config_init(&config_format, options); } +/* DOCDOC */ static void -config_init(config_format_t *fmt, or_options_t *options) +config_init(config_format_t *fmt, void *options) { int i; config_var_t *var; @@ -1253,14 +1339,15 @@ config_init(config_format_t *fmt, or_options_t *options) } } +/* DOCDOC */ static char * -config_dump(config_format_t *fmt, or_options_t *options, int minimal) +config_dump(config_format_t *fmt, void *options, int minimal) { smartlist_t *elements; or_options_t *defaults; config_line_t *line; char *result; - int i; + int i, j; defaults = config_alloc(fmt); config_init(fmt, defaults); @@ -1276,6 +1363,18 @@ config_dump(config_format_t *fmt, or_options_t *options, int minimal) continue; if (minimal && option_is_same(fmt, options, defaults, fmt->vars[i].name)) continue; + + for (j=0; fmt->descriptions[j].name; ++j) { + if (!strcasecmp(fmt->vars[i].name, fmt->descriptions[j].name)) { + const char *desc = fmt->descriptions[j].description; + size_t len = strlen(desc)+8; + char *tmp = tor_malloc(len); + tor_snprintf(tmp, len, "# %s\n",desc); + smartlist_add(elements, tmp); + break; + } + } + line = get_assigned_option(fmt, options, fmt->vars[i].name); for (; line; line = line->next) { size_t len = strlen(line->key) + strlen(line->value) + 3; @@ -1735,32 +1834,32 @@ options_transition_allowed(or_options_t *old, or_options_t *new_val) } if (old->RunAsDaemon != new_val->RunAsDaemon) { - log_fn(LOG_WARN,"During reload, changing RunAsDaemon is not allowed. Failing."); + log_fn(LOG_WARN,"While Tor is running, changing RunAsDaemon is not allowed. Failing."); return -1; } if (old->ORPort != new_val->ORPort) { - log_fn(LOG_WARN,"During reload, changing ORPort is not allowed. Failing."); + log_fn(LOG_WARN,"While Tor is running, changing ORPort is not allowed. Failing."); return -1; } if (strcmp(old->DataDirectory,new_val->DataDirectory)!=0) { - log_fn(LOG_WARN,"During reload, changing DataDirectory (%s->%s) is not allowed. Failing.", old->DataDirectory, new_val->DataDirectory); + log_fn(LOG_WARN,"While Tor is running, changing DataDirectory (%s->%s) is not allowed. Failing.", old->DataDirectory, new_val->DataDirectory); return -1; } if (!opt_streq(old->User, new_val->User)) { - log_fn(LOG_WARN,"During reload, changing User is not allowed. Failing."); + log_fn(LOG_WARN,"While Tor is running, changing User is not allowed. Failing."); return -1; } if (!opt_streq(old->Group, new_val->Group)) { - log_fn(LOG_WARN,"During reload, changing Group is not allowed. Failing."); + log_fn(LOG_WARN,"While Tor is running, changing Group is not allowed. Failing."); return -1; } if (old->HardwareAccel != new_val->HardwareAccel) { - log_fn(LOG_WARN,"During reload, changing HardwareAccel is not allowed. Failing."); + log_fn(LOG_WARN,"While Tor is running, changing HardwareAccel is not allowed. Failing."); return -1; } @@ -1984,7 +2083,7 @@ options_init_from_torrc(int argc, char **argv) return 0; err: tor_free(fname); - options_free(&config_format, newoptions); + config_free(&config_format, newoptions); return -1; } @@ -2795,6 +2894,162 @@ check_libevent_version(const char *m, const char *v, int server) } #endif +/* Versioning issues and state: we want to be able to understand old state + * files, and not choke on new ones. + * + * We could preserve all unrecognized variables across invocations, but we could + * screw up order, if their order is significant with respect to existing + * options. + * + * We could just dump unrecognized variables if you downgrade. + * + * This needs thought. XXXX NM + */ + +/** DOCDOC */ +or_state_t * +get_or_state(void) +{ + return global_state; +} + +/** DOCDOC */ +static char * +get_or_state_fname(void) +{ + char *fname = NULL; + or_options_t *options = get_options(); + size_t len = strlen(options->DataDirectory) + 16; + fname = tor_malloc(len); + tor_snprintf(fname, len, "%s/state", options->DataDirectory); + return fname; +} + +/** DOCDOC */ +static int +or_state_validate(or_state_t *state) +{ + const char *err; + if (helper_nodes_parse_state(state, 0, &err)<0) { + log_fn(LOG_WARN, "Unable to parse helper nodes: %s", err); + return -1; + } + return 0; +} + +/** DOCDOC */ +static void +or_state_set(or_state_t *new_state) +{ + const char *err; + tor_assert(new_state); + if (global_state) + config_free(&state_format, global_state); + global_state = new_state; + if (helper_nodes_parse_state(global_state, 1, &err)<0) + log_fn(LOG_WARN,"Unparseable helper nodes state: %s",err); + +} + +/* DOCDOC */ +int +or_state_load(void) +{ + or_state_t *new_state = NULL; + char *contents = NULL, *fname; + int r = -1; + + fname = get_or_state_fname(); + switch (file_status(fname)) { + case FN_FILE: + if (!(contents = read_file_to_str(fname, 0))) { + log_fn(LOG_WARN, "Unable to read state file %s", fname); + goto done; + } + break; + case FN_NOENT: + break; + default: + log_fn(LOG_WARN,"State file %s is not a file? Failing.", fname); + goto done; + } + new_state = tor_malloc_zero(sizeof(or_state_t)); + new_state->_magic = OR_STATE_MAGIC; + config_init(&state_format, new_state); + if (contents) { + config_line_t *lines=NULL; + int assign_retval; + if (config_get_lines(contents, &lines)<0) + goto done; + assign_retval = config_assign(&state_format, new_state, lines, 0); + config_free_lines(lines); + if (assign_retval<0) + goto done; + } + + if (or_state_validate(new_state) < 0) + goto done; + + if (contents) + log_fn(LOG_INFO, "Loaded state from %s", fname); + else + log_fn(LOG_INFO, "Initialized state"); + or_state_set(new_state); + new_state = NULL; + if (!contents) { + global_state->dirty = 1; + or_state_save(); + } + + r = 0; + done: + tor_free(fname); + tor_free(contents); + if (new_state) + config_free(&state_format, new_state); + + return r; +} + +/** DOCDOC */ +int +or_state_save(void) +{ + char *state, *contents; + char tbuf[ISO_TIME_LEN+1]; + size_t len; + char *fname; + + helper_nodes_update_state(global_state); + + if (!global_state->dirty) + return 0; + + global_state->LastWritten = time(NULL); + state = config_dump(&state_format, global_state, 0); + len = strlen(state)+128; + contents = tor_malloc(len); + format_local_iso_time(tbuf, time(NULL)); + tor_snprintf(contents, len, + "# Tor state file last generated on %s\n" + "# You *do not* need to edit this file.\n\n%s", + tbuf, state); + tor_free(state); + fname = get_or_state_fname(); + if (write_str_to_file(fname, contents, 0)<0) { + log_fn(LOG_WARN, "Unable to write state to file %s", fname); + tor_free(fname); + tor_free(contents); + return -1; + } + log_fn(LOG_INFO, "Saved state to %s", fname); + tor_free(fname); + tor_free(contents); + + global_state->dirty = 0; + return 0; +} + /** Dump the version of every file to the log. */ static void print_cvs_version(void) diff --git a/src/or/hibernate.c b/src/or/hibernate.c index db9c108131..43f50e96c8 100644 --- a/src/or/hibernate.c +++ b/src/or/hibernate.c @@ -539,6 +539,9 @@ accounting_record_bandwidth_usage(time_t now) tor_snprintf(fname, sizeof(fname), "%s/bw_accounting", get_options()->DataDirectory); + + + return write_str_to_file(fname, buf, 0); } diff --git a/src/or/main.c b/src/or/main.c index 242f826371..ee623045e9 100644 --- a/src/or/main.c +++ b/src/or/main.c @@ -1320,6 +1320,7 @@ tor_init(int argc, char *argv[]) log_fn(LOG_ERR, "Unable to seed random number generator. Exiting."); return -1; } + return 0; } diff --git a/src/or/or.h b/src/or/or.h index 0c788904a0..a31503b4d5 100644 --- a/src/or/or.h +++ b/src/or/or.h @@ -1176,6 +1176,22 @@ typedef struct { int RephistTrackTime; /**< How many seconds do we keep rephist info? */ } or_options_t; +/** Persistent state for an onion router, as saved to disk. */ +typedef struct { + uint32_t _magic; + int dirty; + + /* XXXX These options aren't actually attached to anything yet. */ + time_t LastWritten; + time_t AccountingIntervalStart; + uint64_t AccountingBytesReadInInterval; + uint64_t AccountingBytesWrittenInInterval; + int AccountingSecondsActive; + uint64_t AccountingExpectedUsage; + + config_line_t *HelperNodes; +} or_state_t; + #define MAX_SOCKS_REPLY_LEN 1024 #define MAX_SOCKS_ADDR_LEN 256 #define SOCKS_COMMAND_CONNECT 0x01 @@ -1261,6 +1277,8 @@ const char *build_state_get_exit_nickname(cpath_build_state_t *state); void helper_node_set_status(const char *digest, int succeeded); void helper_nodes_set_status_from_directory(void); +int helper_nodes_update_state(or_state_t *state); +int helper_nodes_parse_state(or_state_t *state, int set, const char **err); /********************************* circuitlist.c ***********************/ @@ -1350,6 +1368,10 @@ config_line_t *option_get_assignment(or_options_t *options, char *options_dump(or_options_t *options, int minimal); int options_save_current(void); +or_state_t *get_or_state(void); +int or_state_load(void); +int or_state_save(void); + /********************************* connection.c ***************************/ const char *conn_type_to_string(int type); |