mirror of
https://gitlab.torproject.org/tpo/core/tor.git
synced 2024-11-27 22:03:31 +01:00
Merge branch 'maint-0.4.0'
This commit is contained in:
commit
0642650865
4
changes/ticket21377
Normal file
4
changes/ticket21377
Normal file
@ -0,0 +1,4 @@
|
||||
o Minor features (dircache):
|
||||
- When a directory authority is using a bandwidth file to obtain the
|
||||
bandwidth values that will be included in the next vote, serve this
|
||||
bandwidth file at /tor/status-vote/next/bandwidth. Closes ticket 21377.
|
@ -49,7 +49,8 @@
|
||||
#define ROUTERDESC_BY_DIGEST_CACHE_LIFETIME (48*60*60)
|
||||
#define ROBOTS_CACHE_LIFETIME (24*60*60)
|
||||
#define MICRODESC_CACHE_LIFETIME (48*60*60)
|
||||
|
||||
/* Bandwidth files change every hour. */
|
||||
#define BANDWIDTH_CACHE_LIFETIME (30*60)
|
||||
/** Parse an HTTP request string <b>headers</b> of the form
|
||||
* \verbatim
|
||||
* "\%s [http[s]://]\%s HTTP/1..."
|
||||
@ -351,12 +352,15 @@ static int handle_get_robots(dir_connection_t *conn,
|
||||
const get_handler_args_t *args);
|
||||
static int handle_get_networkstatus_bridges(dir_connection_t *conn,
|
||||
const get_handler_args_t *args);
|
||||
static int handle_get_next_bandwidth(dir_connection_t *conn,
|
||||
const get_handler_args_t *args);
|
||||
|
||||
/** Table for handling GET requests. */
|
||||
static const url_table_ent_t url_table[] = {
|
||||
{ "/tor/", 0, handle_get_frontpage },
|
||||
{ "/tor/status-vote/current/consensus", 1, handle_get_current_consensus },
|
||||
{ "/tor/status-vote/current/", 1, handle_get_status_vote },
|
||||
{ "/tor/status-vote/next/bandwidth", 0, handle_get_next_bandwidth },
|
||||
{ "/tor/status-vote/next/", 1, handle_get_status_vote },
|
||||
{ "/tor/micro/d/", 1, handle_get_microdesc },
|
||||
{ "/tor/server/", 1, handle_get_descriptor },
|
||||
@ -1453,6 +1457,42 @@ handle_get_networkstatus_bridges(dir_connection_t *conn,
|
||||
return 0;
|
||||
}
|
||||
|
||||
/** Helper function for GET the bandwidth file used for the next vote */
|
||||
static int
|
||||
handle_get_next_bandwidth(dir_connection_t *conn,
|
||||
const get_handler_args_t *args)
|
||||
{
|
||||
log_debug(LD_DIR, "Getting next bandwidth.");
|
||||
const or_options_t *options = get_options();
|
||||
const compress_method_t compress_method =
|
||||
find_best_compression_method(args->compression_supported, 1);
|
||||
|
||||
if (options->V3BandwidthsFile) {
|
||||
char *bandwidth = read_file_to_str(options->V3BandwidthsFile,
|
||||
RFTS_IGNORE_MISSING, NULL);
|
||||
if (bandwidth != NULL) {
|
||||
ssize_t len = strlen(bandwidth);
|
||||
write_http_response_header(conn, compress_method != NO_METHOD ? -1 : len,
|
||||
compress_method, BANDWIDTH_CACHE_LIFETIME);
|
||||
if (compress_method != NO_METHOD) {
|
||||
conn->compress_state = tor_compress_new(1, compress_method,
|
||||
choose_compression_level(len/2));
|
||||
log_debug(LD_DIR, "Compressing bandwidth file.");
|
||||
connection_buf_add_compress(bandwidth, len, conn, 0);
|
||||
/* Flush the compression state. */
|
||||
connection_buf_add_compress("", 0, conn, 1);
|
||||
} else {
|
||||
log_debug(LD_DIR, "Not compressing bandwidth file.");
|
||||
connection_buf_add(bandwidth, len, TO_CONN(conn));
|
||||
}
|
||||
tor_free(bandwidth);
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
write_short_http_response(conn, 404, "Not found");
|
||||
return 0;
|
||||
}
|
||||
|
||||
/** Helper function for GET robots.txt or /tor/robots.txt */
|
||||
static int
|
||||
handle_get_robots(dir_connection_t *conn, const get_handler_args_t *args)
|
||||
|
@ -2223,6 +2223,31 @@ test_dir_handle_get_status_vote_next_authority_not_found(void* data)
|
||||
tor_free(header);
|
||||
}
|
||||
|
||||
static void
|
||||
test_dir_handle_get_status_vote_next_bandwidth_not_found(void* data)
|
||||
{
|
||||
dir_connection_t *conn = NULL;
|
||||
char *header = NULL;
|
||||
(void) data;
|
||||
|
||||
MOCK(connection_write_to_buf_impl_, connection_write_to_buf_mock);
|
||||
|
||||
conn = new_dir_conn();
|
||||
|
||||
tt_int_op(0, OP_EQ, directory_handle_command_get(conn,
|
||||
GET("/tor/status-vote/next/bandwdith"), NULL, 0));
|
||||
|
||||
fetch_from_buf_http(TO_CONN(conn)->outbuf, &header, MAX_HEADERS_SIZE,
|
||||
NULL, NULL, 1, 0);
|
||||
tt_assert(header);
|
||||
tt_str_op(NOT_FOUND, OP_EQ, header);
|
||||
|
||||
done:
|
||||
UNMOCK(connection_write_to_buf_impl_);
|
||||
connection_free_minimal(TO_CONN(conn));
|
||||
tor_free(header);
|
||||
}
|
||||
|
||||
NS_DECL(const char*,
|
||||
dirvote_get_pending_consensus, (consensus_flavor_t flav));
|
||||
|
||||
@ -2462,6 +2487,85 @@ test_dir_handle_get_status_vote_next_authority(void* data)
|
||||
dirvote_free_all();
|
||||
}
|
||||
|
||||
static void
|
||||
test_dir_handle_get_status_vote_next_bandwidth(void* data)
|
||||
{
|
||||
dir_connection_t *conn = NULL;
|
||||
char *header = NULL, *body = NULL;
|
||||
size_t body_used = 0;
|
||||
(void) data;
|
||||
|
||||
const char *content =
|
||||
"1541171221\n"
|
||||
"node_id=$68A483E05A2ABDCA6DA5A3EF8DB5177638A27F80 "
|
||||
"master_key_ed25519=YaqV4vbvPYKucElk297eVdNArDz9HtIwUoIeo0+cVIpQ "
|
||||
"bw=760 nick=Test time=2018-05-08T16:13:26\n";
|
||||
|
||||
init_mock_options();
|
||||
MOCK(get_options, mock_get_options);
|
||||
mock_options->V3BandwidthsFile = tor_strdup(
|
||||
get_fname_rnd("V3BandwidthsFile")
|
||||
);
|
||||
|
||||
write_str_to_file(mock_options->V3BandwidthsFile, content, 0);
|
||||
|
||||
MOCK(connection_write_to_buf_impl_, connection_write_to_buf_mock);
|
||||
|
||||
conn = new_dir_conn();
|
||||
tt_int_op(0, OP_EQ, directory_handle_command_get(conn,
|
||||
GET("/tor/status-vote/next/bandwidth"), NULL, 0));
|
||||
|
||||
fetch_from_buf_http(TO_CONN(conn)->outbuf, &header, MAX_HEADERS_SIZE,
|
||||
&body, &body_used, strlen(content)+1, 0);
|
||||
|
||||
tt_assert(header);
|
||||
tt_ptr_op(strstr(header, "HTTP/1.0 200 OK\r\n"), OP_EQ, header);
|
||||
tt_assert(strstr(header, "Content-Type: text/plain\r\n"));
|
||||
tt_assert(strstr(header, "Content-Encoding: identity\r\n"));
|
||||
tt_assert(strstr(header, "Content-Length: 167\r\n"));
|
||||
|
||||
/* Check cache lifetime */
|
||||
char expbuf[RFC1123_TIME_LEN+1];
|
||||
time_t now = time(NULL);
|
||||
/* BANDWIDTH_CACHE_LIFETIME is defined in dircache.c. */
|
||||
format_rfc1123_time(expbuf, (time_t)(now + 30*60));
|
||||
char *expires = NULL;
|
||||
/* Change to 'Cache-control: max-age=%d' if using http/1.1. */
|
||||
tor_asprintf(&expires, "Expires: %s\r\n", expbuf);
|
||||
tt_assert(strstr(header, expires));
|
||||
|
||||
tt_int_op(body_used, OP_EQ, strlen(body));
|
||||
tt_str_op(content, OP_EQ, body);
|
||||
|
||||
tor_free(header);
|
||||
tor_free(body);
|
||||
|
||||
/* Request the file using compression, the result should be the same. */
|
||||
tt_int_op(0, OP_EQ, directory_handle_command_get(conn,
|
||||
GET("/tor/status-vote/next/bandwidth.z"), NULL, 0));
|
||||
|
||||
fetch_from_buf_http(TO_CONN(conn)->outbuf, &header, MAX_HEADERS_SIZE,
|
||||
&body, &body_used, strlen(content)+1, 0);
|
||||
|
||||
tt_assert(header);
|
||||
tt_ptr_op(strstr(header, "HTTP/1.0 200 OK\r\n"), OP_EQ, header);
|
||||
tt_assert(strstr(header, "Content-Encoding: deflate\r\n"));
|
||||
|
||||
/* Since using connection_write_to_buf_mock instead of mocking
|
||||
* connection_buf_add_compress, the content is not actually compressed.
|
||||
* If it would, the size and content would be different than the original.
|
||||
*/
|
||||
|
||||
done:
|
||||
UNMOCK(get_options);
|
||||
UNMOCK(connection_write_to_buf_impl_);
|
||||
connection_free_minimal(TO_CONN(conn));
|
||||
tor_free(header);
|
||||
tor_free(body);
|
||||
tor_free(expires);
|
||||
or_options_free(mock_options);
|
||||
}
|
||||
|
||||
static void
|
||||
test_dir_handle_get_status_vote_current_authority(void* data)
|
||||
{
|
||||
@ -2637,6 +2741,8 @@ struct testcase_t dir_handle_get_tests[] = {
|
||||
DIR_HANDLE_CMD(status_vote_current_authority, 0),
|
||||
DIR_HANDLE_CMD(status_vote_next_authority_not_found, 0),
|
||||
DIR_HANDLE_CMD(status_vote_next_authority, 0),
|
||||
DIR_HANDLE_CMD(status_vote_next_bandwidth_not_found, 0),
|
||||
DIR_HANDLE_CMD(status_vote_next_bandwidth, 0),
|
||||
DIR_HANDLE_CMD(status_vote_current_consensus_ns_not_enough_sigs, TT_FORK),
|
||||
DIR_HANDLE_CMD(status_vote_current_consensus_ns_not_found, TT_FORK),
|
||||
DIR_HANDLE_CMD(status_vote_current_consensus_too_old, TT_FORK),
|
||||
|
Loading…
Reference in New Issue
Block a user