diff options
-rw-r--r-- | doc/spec/rend-spec.txt | 6 | ||||
-rw-r--r-- | src/or/directory.c | 35 | ||||
-rw-r--r-- | src/or/or.h | 15 | ||||
-rw-r--r-- | src/or/rendcommon.c | 65 | ||||
-rw-r--r-- | src/or/rendservice.c | 40 | ||||
-rw-r--r-- | src/or/test.c | 25 |
6 files changed, 94 insertions, 92 deletions
diff --git a/doc/spec/rend-spec.txt b/doc/spec/rend-spec.txt index dd7a499388..90277ded51 100644 --- a/doc/spec/rend-spec.txt +++ b/doc/spec/rend-spec.txt @@ -547,8 +547,10 @@ $Id$ 6 nodes. If the request is unsuccessful, Alice retries the other remaining responsible hidden service directories in a random order. Alice relies on Bob to care about a potential clock skew between the two - by possibly storing two sets of descriptors. [XXX what does this mean - Bob does in practice, if anything? -RD] + by possibly storing two sets of descriptors (see section 1.4). + [XXX what does this mean Bob does in practice, if anything? -RD] + [Cf. last sentence of 1.4. Maybe this reference would be helpful here, + so I added it. -KL] Alice's OP opens a stream via Tor to the chosen v2 hidden service directory. (She may re-use old circuits for this.) Over this stream, diff --git a/src/or/directory.c b/src/or/directory.c index ba3f675f9e..1d6f08fbb8 100644 --- a/src/or/directory.c +++ b/src/or/directory.c @@ -3104,33 +3104,26 @@ dir_split_resource_into_fingerprints(const char *resource, return 0; } -/** Determine the responsible hidden service directories for - * <b>desc_ids</b> and upload the appropriate descriptor from - * <b>desc_strs</b> to them; each smartlist must contain - * REND_NUMBER_OF_NON_CONSECUTIVE_REPLICAS entries; <b>service_id</b> and - * <b>seconds_valid</b> are only passed for logging purposes.*/ +/** Determine the responsible hidden service directories for the + * rend_encoded_v2_service_descriptor_t's in <b>descs</b> and upload them; + * <b>service_id</b> and <b>seconds_valid</b> are only passed for logging + * purposes.*/ /* XXXX020 desc_ids and desc_strs could be merged. Should they? */ /* I guess they should. -KL */ +/* And now they are. -KL */ void -directory_post_to_hs_dir(smartlist_t *desc_ids, smartlist_t *desc_strs, - const char *service_id, int seconds_valid) +directory_post_to_hs_dir(smartlist_t *descs, const char *service_id, + int seconds_valid) { int i, j; smartlist_t *responsible_dirs; routerstatus_t *hs_dir; - if (smartlist_len(desc_ids) != REND_NUMBER_OF_NON_CONSECUTIVE_REPLICAS || - smartlist_len(desc_strs) != REND_NUMBER_OF_NON_CONSECUTIVE_REPLICAS) { - log_warn(LD_BUG, "Could not post descriptors to hidden service " - "directories: Illegal number of descriptor " - "IDs/strings"); - return; - } responsible_dirs = smartlist_create(); - for (i = 0; i < REND_NUMBER_OF_NON_CONSECUTIVE_REPLICAS; i++) { - const char *desc_id = smartlist_get(desc_ids, i); - const char *desc_str = smartlist_get(desc_strs, i); + for (i = 0; i < smartlist_len(descs); i++) { + rend_encoded_v2_service_descriptor_t *desc = smartlist_get(descs, i); /* Determine responsible dirs. */ - if (hid_serv_get_responsible_directories(responsible_dirs, desc_id) < 0) { + if (hid_serv_get_responsible_directories(responsible_dirs, + desc->desc_id) < 0) { log_warn(LD_REND, "Could not determine the responsible hidden service " "directories to post descriptors to."); smartlist_free(responsible_dirs); @@ -3145,10 +3138,10 @@ directory_post_to_hs_dir(smartlist_t *desc_ids, smartlist_t *desc_strs, directory_initiate_command_routerstatus(hs_dir, DIR_PURPOSE_UPLOAD_RENDDESC_V2, ROUTER_PURPOSE_GENERAL, - 1, NULL, desc_str, - strlen(desc_str), 0); + 1, NULL, desc->desc_str, + strlen(desc->desc_str), 0); base32_encode(desc_id_base32, sizeof(desc_id_base32), - desc_id, DIGEST_LEN); + desc->desc_id, DIGEST_LEN); log_info(LD_REND, "Sending publish request for v2 descriptor for " "service '%s' with descriptor ID '%s' with validity " "of %d seconds to hidden service directory '%s' on " diff --git a/src/or/or.h b/src/or/or.h index 82a4c421c2..c9b3a6d836 100644 --- a/src/or/or.h +++ b/src/or/or.h @@ -3007,8 +3007,8 @@ int dir_split_resource_into_fingerprints(const char *resource, char *directory_dump_request_log(void); int router_supports_extrainfo(const char *identity_digest, int is_authority); -void directory_post_to_hs_dir(smartlist_t *desc_ids, smartlist_t *descs, - const char *service_id, int seconds_valid); +void directory_post_to_hs_dir(smartlist_t *descs, const char *service_id, + int seconds_valid); void directory_get_from_hs_dir(const char *desc_id, const char *query); time_t download_status_increment_failure(download_status_t *dls, @@ -3550,6 +3550,12 @@ int rend_client_send_introduction(origin_circuit_t *introcirc, /********************************* rendcommon.c ***************************/ +/** ASCII-encoded v2 hidden service descriptor. */ +typedef struct rend_encoded_v2_service_descriptor_t { + char desc_id[DIGEST_LEN]; /**< Descriptor ID. */ + char *desc_str; /**< Descriptor string. */ +} rend_encoded_v2_service_descriptor_t; + /** Information used to connect to a hidden service. */ typedef struct rend_service_descriptor_t { crypto_pk_env_t *pk; /**< This service's public key. */ @@ -3587,6 +3593,8 @@ int rend_encode_service_descriptor(rend_service_descriptor_t *desc, rend_service_descriptor_t *rend_parse_service_descriptor(const char *str, size_t len); int rend_get_service_id(crypto_pk_env_t *pk, char *out); +void rend_encoded_v2_service_descriptor_free( + rend_encoded_v2_service_descriptor_t *desc); /** A cached rendezvous descriptor. */ typedef struct rend_cache_entry_t { @@ -3611,8 +3619,7 @@ int rend_cache_store_v2_desc_as_client(const char *desc, const char *descriptor_cookie); int rend_cache_store_v2_desc_as_dir(const char *desc); int rend_cache_size(void); -int rend_encode_v2_descriptors(smartlist_t *desc_strs_out, - smartlist_t *desc_ids_out, +int rend_encode_v2_descriptors(smartlist_t *descs_out, rend_service_descriptor_t *desc, time_t now, const char *descriptor_cookie, uint8_t period); int rend_compute_v2_desc_id(char *desc_id_out, const char *service_id, diff --git a/src/or/rendcommon.c b/src/or/rendcommon.c index 651c4926ab..dd2a63bca2 100644 --- a/src/or/rendcommon.c +++ b/src/or/rendcommon.c @@ -296,7 +296,7 @@ rend_encode_v2_intro_points(char **ipos_base64, /** Attempt to parse the given <b>desc_str</b> and return true if this * succeeds, false otherwise. */ static int -rend_desc_v2_is_parsable(const char *desc_str) +rend_desc_v2_is_parsable(rend_encoded_v2_service_descriptor_t *desc) { rend_service_descriptor_t *test_parsed = NULL; char test_desc_id[DIGEST_LEN]; @@ -308,25 +308,31 @@ rend_desc_v2_is_parsable(const char *desc_str) &test_intro_content, &test_intro_size, &test_encoded_size, - &test_next, desc_str); + &test_next, desc->desc_str); if (test_parsed) rend_service_descriptor_free(test_parsed); tor_free(test_intro_content); return (res >= 0); } -/** Encode a set of new service descriptors for <b>desc</b> at time - * <b>now</b> using <b>descriptor_cookie</b> (may be <b>NULL</b> if +/** Free the storage held by an encoded v2 service descriptor. */ +void +rend_encoded_v2_service_descriptor_free( + rend_encoded_v2_service_descriptor_t *desc) +{ + if (desc->desc_str) tor_free(desc->desc_str); + tor_free(desc); +} + +/** Encode a set of rend_encoded_v2_service_descriptor_t's for <b>desc</b> + * at time <b>now</b> using <b>descriptor_cookie</b> (may be <b>NULL</b> if * introduction points shall not be encrypted) and <b>period</b> (e.g. 0 - * for the current period, 1 for the next period, etc.), write the - * ASCII-encoded outputs to newly allocated strings and add them to the - * existing <b>desc_strs</b>, and write the descriptor IDs to newly - * allocated strings and add them to the existing <b>desc_ids</b>; return - * the number of seconds that the descriptors will be found under those - * <b>desc_ids</b> by clients, or -1 if the encoding was not successful. */ + * for the current period, 1 for the next period, etc.) and add them to + * the existing list <b>descs_out</b>; return the number of seconds that + * the descriptors will be found by clients, or -1 if the encoding was not + * successful. */ int -rend_encode_v2_descriptors(smartlist_t *desc_strs_out, - smartlist_t *desc_ids_out, +rend_encode_v2_descriptors(smartlist_t *descs_out, rend_service_descriptor_t *desc, time_t now, const char *descriptor_cookie, uint8_t period) { @@ -356,7 +362,6 @@ rend_encode_v2_descriptors(smartlist_t *desc_strs_out, for (k = 0; k < REND_NUMBER_OF_NON_CONSECUTIVE_REPLICAS; k++) { char secret_id_part[DIGEST_LEN]; char secret_id_part_base32[REND_SECRET_ID_PART_LEN_BASE32 + 1]; - char *desc_id; char desc_id_base32[REND_DESC_ID_V2_LEN_BASE32 + 1]; char *permanent_key = NULL; size_t permanent_key_len; @@ -369,21 +374,22 @@ rend_encode_v2_descriptors(smartlist_t *desc_strs_out, int result = 0; size_t written = 0; char desc_digest[DIGEST_LEN]; + rend_encoded_v2_service_descriptor_t *enc = + tor_malloc_zero(sizeof(rend_encoded_v2_service_descriptor_t)); /* Calculate secret-id-part = h(time-period + cookie + replica). */ get_secret_id_part_bytes(secret_id_part, time_period, descriptor_cookie, k); base32_encode(secret_id_part_base32, sizeof(secret_id_part_base32), secret_id_part, DIGEST_LEN); /* Calculate descriptor ID. */ - desc_id = tor_malloc_zero(DIGEST_LEN); - rend_get_descriptor_id_bytes(desc_id, service_id, secret_id_part); - smartlist_add(desc_ids_out, desc_id); + rend_get_descriptor_id_bytes(enc->desc_id, service_id, secret_id_part); base32_encode(desc_id_base32, sizeof(desc_id_base32), - desc_id, DIGEST_LEN); + enc->desc_id, DIGEST_LEN); /* PEM-encode the public key */ if (crypto_pk_write_public_key_to_string(desc->pk, &permanent_key, &permanent_key_len) < 0) { log_warn(LD_BUG, "Could not write public key to string."); + rend_encoded_v2_service_descriptor_free(enc); goto err; } /* Encode timestamp. */ @@ -403,7 +409,7 @@ rend_encode_v2_descriptors(smartlist_t *desc_strs_out, protocol_versions_string[0]= '\0'; /* Assemble complete descriptor. */ desc_len = 2000 + desc->n_intro_points * 1000; /* far too long, but ok. */ - desc_str = tor_malloc_zero(desc_len); + enc->desc_str = desc_str = tor_malloc_zero(desc_len); result = tor_snprintf(desc_str, desc_len, "rendezvous-service-descriptor %s\n" "version 2\n" @@ -419,7 +425,7 @@ rend_encode_v2_descriptors(smartlist_t *desc_strs_out, tor_free(permanent_key); if (result < 0) { log_warn(LD_BUG, "Descriptor ran out of room."); - tor_free(desc_str); + rend_encoded_v2_service_descriptor_free(enc); goto err; } written = result; @@ -432,7 +438,7 @@ rend_encode_v2_descriptors(smartlist_t *desc_strs_out, ipos_base64); if (result < 0) { log_warn(LD_BUG, "could not write introduction points."); - tor_free(desc_str); + rend_encoded_v2_service_descriptor_free(enc); goto err; } written += result; @@ -442,31 +448,31 @@ rend_encode_v2_descriptors(smartlist_t *desc_strs_out, written += strlen(desc_str + written); if (crypto_digest(desc_digest, desc_str, written) < 0) { log_warn(LD_BUG, "could not create digest."); - tor_free(desc_str); + rend_encoded_v2_service_descriptor_free(enc); goto err; } if (router_append_dirobj_signature(desc_str + written, desc_len - written, desc_digest, desc->pk) < 0) { log_warn(LD_BUG, "Couldn't sign desc."); - tor_free(desc_str); + rend_encoded_v2_service_descriptor_free(enc); goto err; } written += strlen(desc_str+written); if (written+2 > desc_len) { log_warn(LD_BUG, "Could not finish desc."); - tor_free(desc_str); + rend_encoded_v2_service_descriptor_free(enc); goto err; } desc_str[written++] = '\n'; desc_str[written++] = 0; /* Check if we can parse our own descriptor. */ - if (!rend_desc_v2_is_parsable(desc_str)) { + if (!rend_desc_v2_is_parsable(enc)) { log_warn(LD_BUG, "Could not parse my own descriptor: %s", desc_str); - tor_free(desc_str); + rend_encoded_v2_service_descriptor_free(enc); goto err; } - smartlist_add(desc_strs_out, desc_str); + smartlist_add(descs_out, enc); } log_info(LD_REND, "Successfully encoded a v2 descriptor and " @@ -474,10 +480,9 @@ rend_encode_v2_descriptors(smartlist_t *desc_strs_out, goto done; err: - SMARTLIST_FOREACH(desc_ids_out, void *, id, tor_free(id)); - smartlist_clear(desc_ids_out); - SMARTLIST_FOREACH(desc_strs_out, void *, str, tor_free(str)); - smartlist_clear(desc_strs_out); + SMARTLIST_FOREACH(descs_out, rend_encoded_v2_service_descriptor_t *, d, + rend_encoded_v2_service_descriptor_free(d);); + smartlist_clear(descs_out); seconds_valid = -1; done: diff --git a/src/or/rendservice.c b/src/or/rendservice.c index 9fbd3017fe..0341d7997b 100644 --- a/src/or/rendservice.c +++ b/src/or/rendservice.c @@ -328,6 +328,11 @@ rend_config_services(or_options_t *options, int validate_only) /* XXX020 Karsten: do you really want to overwrite the * descriptor_version in the second line? Perhaps if both bits * are set you want to leave it at -1? -RD */ + /* If both bits are set, versions_bitmask will be (1<<0) + (1<<2), + * so that neither condition will be true, leaving descriptor_version + * set to -1. Does the following comment make it less confusing? -KL */ + /* If version 0 XOR 2 was set, change descriptor_version to that + * value; otherwise leave it at -1. */ if (versions_bitmask == 1 << 0) service->descriptor_version = 0; if (versions_bitmask == 1 << 2) service->descriptor_version = 2; } @@ -1101,12 +1106,11 @@ upload_service_descriptor(rend_service_t *service) get_options()->PublishHidServDescriptors) { if (hid_serv_have_enough_directories()) { int seconds_valid; - smartlist_t *desc_strs = smartlist_create(); - smartlist_t *desc_ids = smartlist_create(); + smartlist_t *descs = smartlist_create(); int i; /* Encode the current descriptor. */ - seconds_valid = rend_encode_v2_descriptors(desc_strs, desc_ids, - service->desc, now, NULL, 0); + seconds_valid = rend_encode_v2_descriptors(descs, service->desc, now, + NULL, 0); if (seconds_valid < 0) { log_warn(LD_BUG, "Internal error: couldn't encode service descriptor; " "not uploading."); @@ -1116,14 +1120,11 @@ upload_service_descriptor(rend_service_t *service) rend_get_service_id(service->desc->pk, serviceid); log_info(LD_REND, "Sending publish request for hidden service %s", serviceid); - directory_post_to_hs_dir(desc_ids, desc_strs, serviceid, seconds_valid); + directory_post_to_hs_dir(descs, serviceid, seconds_valid); /* Free memory for descriptors. */ - for (i = 0; i < REND_NUMBER_OF_NON_CONSECUTIVE_REPLICAS; i++) { - tor_free(smartlist_get(desc_strs, i)); - tor_free(smartlist_get(desc_ids, i)); - } - smartlist_clear(desc_strs); - smartlist_clear(desc_ids); + for (i = 0; i < REND_NUMBER_OF_NON_CONSECUTIVE_REPLICAS; i++) + rend_encoded_v2_service_descriptor_free(smartlist_get(descs, i)); + smartlist_clear(descs); /* Update next upload time. */ if (seconds_valid - REND_TIME_PERIOD_OVERLAPPING_V2_DESCS > rendpostperiod) @@ -1135,23 +1136,18 @@ upload_service_descriptor(rend_service_t *service) REND_TIME_PERIOD_OVERLAPPING_V2_DESCS + 1; /* Post also the next descriptors, if necessary. */ if (seconds_valid < REND_TIME_PERIOD_OVERLAPPING_V2_DESCS) { - seconds_valid = rend_encode_v2_descriptors(desc_strs, desc_ids, - service->desc, now, - NULL, 1); + seconds_valid = rend_encode_v2_descriptors(descs, service->desc, + now, NULL, 1); if (seconds_valid < 0) { log_warn(LD_BUG, "Internal error: couldn't encode service " "descriptor; not uploading."); return; } - directory_post_to_hs_dir(desc_ids, desc_strs, serviceid, - seconds_valid); + directory_post_to_hs_dir(descs, serviceid, seconds_valid); /* Free memory for descriptors. */ - for (i = 0; i < REND_NUMBER_OF_NON_CONSECUTIVE_REPLICAS; i++) { - tor_free(smartlist_get(desc_strs, i)); - tor_free(smartlist_get(desc_ids, i)); - } - smartlist_free(desc_strs); - smartlist_free(desc_ids); + for (i = 0; i < REND_NUMBER_OF_NON_CONSECUTIVE_REPLICAS; i++) + rend_encoded_v2_service_descriptor_free(smartlist_get(descs, i)); + smartlist_free(descs); } uploaded = 1; log_info(LD_REND, "Successfully uploaded v2 rend descriptors!"); diff --git a/src/or/test.c b/src/or/test.c index 7c6099e7cf..f83a4087a2 100644 --- a/src/or/test.c +++ b/src/or/test.c @@ -3276,8 +3276,7 @@ test_rend_fns_v2(void) char service_id[DIGEST_LEN]; char service_id_base32[REND_SERVICE_ID_LEN_BASE32+1]; const char *next_desc; - smartlist_t *desc_strs = smartlist_create(); - smartlist_t *desc_ids = smartlist_create(); + smartlist_t *descs = smartlist_create(); char computed_desc_id[DIGEST_LEN]; char parsed_desc_id[DIGEST_LEN]; crypto_pk_env_t *pk1, *pk2; @@ -3318,19 +3317,22 @@ test_rend_fns_v2(void) generated->intro_point_extend_info[i] = info; strmap_set(generated->intro_keys, info->nickname, pk2); } - test_assert(rend_encode_v2_descriptors(desc_strs, desc_ids, generated, now, + test_assert(rend_encode_v2_descriptors(descs, generated, now, NULL, 0) > 0); test_assert(rend_compute_v2_desc_id(computed_desc_id, service_id_base32, NULL, now, 0) == 0); - test_assert(smartlist_digest_isin(desc_ids, computed_desc_id)); + test_memeq(((rend_encoded_v2_service_descriptor_t *) + smartlist_get(descs, 0))->desc_id, computed_desc_id, DIGEST_LEN); test_assert(rend_parse_v2_service_descriptor(&parsed, parsed_desc_id, &intro_points_encrypted, &intro_points_size, &encoded_size, &next_desc, - desc_strs->list[0]) == 0); + ((rend_encoded_v2_service_descriptor_t *) + smartlist_get(descs, 0))->desc_str) == 0); test_assert(parsed); - test_assert(smartlist_digest_isin(desc_ids, parsed_desc_id)); + test_memeq(((rend_encoded_v2_service_descriptor_t *) + smartlist_get(descs, 0))->desc_id, parsed_desc_id, DIGEST_LEN); test_assert(rend_decrypt_introduction_points(parsed, NULL, intro_points_encrypted, intro_points_size) == 3); @@ -3351,13 +3353,10 @@ test_rend_fns_v2(void) test_eq(gen_info->port, par_info->port); } tor_free(intro_points_encrypted); - /*for (i = 0; i < REND_NUMBER_OF_NON_CONSECUTIVE_REPLICAS; i++) - tor_free(desc_strs[i]);*/ - smartlist_free(desc_strs); - for (i = 0; i < parsed->n_intro_points; i++) { - tor_free(parsed->intro_point_extend_info[i]->onion_key); - tor_free(generated->intro_point_extend_info[i]->onion_key); - } + for (i = 0; i < REND_NUMBER_OF_NON_CONSECUTIVE_REPLICAS; i++) + rend_encoded_v2_service_descriptor_free(smartlist_get(descs, i)); + smartlist_free(descs); + rend_service_descriptor_free(parsed); } #define ENT(x) { #x, test_ ## x, 0, 0 } |