mirror of
https://gitlab.torproject.org/tpo/core/tor.git
synced 2024-12-12 05:33:35 +01:00
d194f6bedf
Closes ticket 29357, and comes with appropriate notions of caution.
370 lines
11 KiB
C
370 lines
11 KiB
C
/* Copyright (c) 2018-2019, The Tor Project, Inc. */
|
|
/* See LICENSE for licensing information */
|
|
|
|
/**
|
|
* \file test_mainloop.c
|
|
* \brief Tests for functions closely related to the Tor main loop
|
|
*/
|
|
|
|
#define CONFIG_PRIVATE
|
|
#define MAINLOOP_PRIVATE
|
|
#define STATEFILE_PRIVATE
|
|
|
|
#include "test/test.h"
|
|
#include "test/log_test_helpers.h"
|
|
|
|
#include "core/or/or.h"
|
|
#include "core/mainloop/connection.h"
|
|
#include "core/mainloop/mainloop.h"
|
|
#include "core/mainloop/netstatus.h"
|
|
|
|
#include "feature/hs/hs_service.h"
|
|
|
|
#include "app/config/config.h"
|
|
#include "app/config/statefile.h"
|
|
#include "app/config/or_state_st.h"
|
|
|
|
static const uint64_t BILLION = 1000000000;
|
|
|
|
static void
|
|
test_mainloop_update_time_normal(void *arg)
|
|
{
|
|
(void)arg;
|
|
|
|
monotime_enable_test_mocking();
|
|
/* This is arbitrary */
|
|
uint64_t mt_now = UINT64_C(7493289274986);
|
|
/* This time is in the past as of when this test was written. */
|
|
time_t now = 1525272090;
|
|
monotime_coarse_set_mock_time_nsec(mt_now);
|
|
reset_uptime();
|
|
update_current_time(now);
|
|
tt_int_op(approx_time(), OP_EQ, now);
|
|
tt_int_op(get_uptime(), OP_EQ, 0);
|
|
|
|
update_current_time(now); // Same time as before is a no-op.
|
|
tt_int_op(get_uptime(), OP_EQ, 0);
|
|
|
|
now += 1;
|
|
mt_now += BILLION;
|
|
monotime_coarse_set_mock_time_nsec(mt_now);
|
|
update_current_time(now);
|
|
tt_int_op(approx_time(), OP_EQ, now);
|
|
tt_int_op(get_uptime(), OP_EQ, 1);
|
|
|
|
now += 2; // two-second jump is unremarkable.
|
|
mt_now += 2*BILLION;
|
|
update_current_time(now);
|
|
monotime_coarse_set_mock_time_nsec(mt_now);
|
|
tt_int_op(approx_time(), OP_EQ, now);
|
|
tt_int_op(get_uptime(), OP_EQ, 3);
|
|
|
|
now -= 1; // a one-second hop backwards is also unremarkable.
|
|
update_current_time(now);
|
|
tt_int_op(approx_time(), OP_EQ, now); // it changes the approx time...
|
|
tt_int_op(get_uptime(), OP_EQ, 3); // but it doesn't roll back our uptime
|
|
|
|
done:
|
|
monotime_disable_test_mocking();
|
|
}
|
|
|
|
static void
|
|
test_mainloop_update_time_jumps(void *arg)
|
|
{
|
|
(void)arg;
|
|
|
|
monotime_enable_test_mocking();
|
|
/* This is arbitrary */
|
|
uint64_t mt_now = UINT64_C(7493289274986);
|
|
/* This time is in the past as of when this test was written. */
|
|
time_t now = 220897152;
|
|
monotime_coarse_set_mock_time_nsec(mt_now);
|
|
reset_uptime();
|
|
update_current_time(now);
|
|
tt_int_op(approx_time(), OP_EQ, now);
|
|
tt_int_op(get_uptime(), OP_EQ, 0);
|
|
|
|
/* Put some uptime on the clock.. */
|
|
now += 3;
|
|
mt_now += 3*BILLION;
|
|
monotime_coarse_set_mock_time_nsec(mt_now);
|
|
update_current_time(now);
|
|
tt_int_op(approx_time(), OP_EQ, now);
|
|
tt_int_op(get_uptime(), OP_EQ, 3);
|
|
|
|
/* Now try jumping forward and backward, without updating the monotonic
|
|
* clock. */
|
|
setup_capture_of_logs(LOG_NOTICE);
|
|
now += 1800;
|
|
update_current_time(now);
|
|
expect_single_log_msg_containing(
|
|
"Your system clock just jumped 1800 seconds forward");
|
|
tt_int_op(approx_time(), OP_EQ, now);
|
|
tt_int_op(get_uptime(), OP_EQ, 3); // no uptime change.
|
|
mock_clean_saved_logs();
|
|
|
|
now -= 600;
|
|
update_current_time(now);
|
|
expect_single_log_msg_containing(
|
|
"Your system clock just jumped 600 seconds backward");
|
|
tt_int_op(approx_time(), OP_EQ, now);
|
|
tt_int_op(get_uptime(), OP_EQ, 3); // no uptime change.
|
|
mock_clean_saved_logs();
|
|
|
|
/* uptime tracking should go normally now if the clock moves sensibly. */
|
|
now += 2;
|
|
mt_now += 2*BILLION;
|
|
update_current_time(now);
|
|
tt_int_op(approx_time(), OP_EQ, now);
|
|
tt_int_op(get_uptime(), OP_EQ, 5);
|
|
|
|
/* If we skip forward by a few minutes but the monotonic clock agrees,
|
|
* we've just been idle: that counts as not worth warning about. */
|
|
now += 1800;
|
|
mt_now += 1800*BILLION;
|
|
monotime_coarse_set_mock_time_nsec(mt_now);
|
|
update_current_time(now);
|
|
expect_no_log_entry();
|
|
tt_int_op(approx_time(), OP_EQ, now);
|
|
tt_int_op(get_uptime(), OP_EQ, 5); // this doesn't count to uptime, though.
|
|
|
|
/* If we skip forward by a long time, even if the clock agrees, it's
|
|
* idnless that counts. */
|
|
now += 4000;
|
|
mt_now += 4000*BILLION;
|
|
monotime_coarse_set_mock_time_nsec(mt_now);
|
|
update_current_time(now);
|
|
expect_single_log_msg_containing("Tor has been idle for 4000 seconds");
|
|
tt_int_op(approx_time(), OP_EQ, now);
|
|
tt_int_op(get_uptime(), OP_EQ, 5);
|
|
|
|
done:
|
|
teardown_capture_of_logs();
|
|
monotime_disable_test_mocking();
|
|
}
|
|
|
|
static int schedule_rescan_called = 0;
|
|
static void
|
|
mock_schedule_rescan_periodic_events(void)
|
|
{
|
|
++schedule_rescan_called;
|
|
}
|
|
|
|
static void
|
|
test_mainloop_user_activity(void *arg)
|
|
{
|
|
(void)arg;
|
|
const time_t start = 1542658829;
|
|
update_approx_time(start);
|
|
|
|
MOCK(schedule_rescan_periodic_events, mock_schedule_rescan_periodic_events);
|
|
|
|
reset_user_activity(start);
|
|
tt_i64_op(get_last_user_activity_time(), OP_EQ, start);
|
|
|
|
set_network_participation(false);
|
|
|
|
// reset can move backwards and forwards, but does not change network
|
|
// participation.
|
|
reset_user_activity(start-10);
|
|
tt_i64_op(get_last_user_activity_time(), OP_EQ, start-10);
|
|
reset_user_activity(start+10);
|
|
tt_i64_op(get_last_user_activity_time(), OP_EQ, start+10);
|
|
|
|
tt_int_op(schedule_rescan_called, OP_EQ, 0);
|
|
tt_int_op(false, OP_EQ, is_participating_on_network());
|
|
|
|
// "note" can only move forward. Calling it from a non-participating
|
|
// state makes us rescan the periodic callbacks and set participation.
|
|
note_user_activity(start+20);
|
|
tt_i64_op(get_last_user_activity_time(), OP_EQ, start+20);
|
|
tt_int_op(true, OP_EQ, is_participating_on_network());
|
|
tt_int_op(schedule_rescan_called, OP_EQ, 1);
|
|
|
|
// Calling it again will move us forward, but not call rescan again.
|
|
note_user_activity(start+25);
|
|
tt_i64_op(get_last_user_activity_time(), OP_EQ, start+25);
|
|
tt_int_op(true, OP_EQ, is_participating_on_network());
|
|
tt_int_op(schedule_rescan_called, OP_EQ, 1);
|
|
|
|
// We won't move backwards.
|
|
note_user_activity(start+20);
|
|
tt_i64_op(get_last_user_activity_time(), OP_EQ, start+25);
|
|
tt_int_op(true, OP_EQ, is_participating_on_network());
|
|
tt_int_op(schedule_rescan_called, OP_EQ, 1);
|
|
|
|
// We _will_ adjust if the clock jumps though.
|
|
netstatus_note_clock_jumped(500);
|
|
tt_i64_op(get_last_user_activity_time(), OP_EQ, start+525);
|
|
|
|
netstatus_note_clock_jumped(-400);
|
|
tt_i64_op(get_last_user_activity_time(), OP_EQ, start+125);
|
|
|
|
done:
|
|
UNMOCK(schedule_rescan_periodic_events);
|
|
}
|
|
|
|
static unsigned int
|
|
mock_get_num_services(void)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
static connection_t *
|
|
mock_connection_gbtu(int type)
|
|
{
|
|
(void) type;
|
|
return (void *)"hello fellow connections";
|
|
}
|
|
|
|
static void
|
|
test_mainloop_check_participation(void *arg)
|
|
{
|
|
(void)arg;
|
|
or_options_t *options = options_new();
|
|
const time_t start = 1542658829;
|
|
const time_t ONE_DAY = 24*60*60;
|
|
|
|
// Suppose we've been idle for a day or two
|
|
reset_user_activity(start - 2*ONE_DAY);
|
|
set_network_participation(true);
|
|
check_network_participation_callback(start, options);
|
|
tt_int_op(is_participating_on_network(), OP_EQ, false);
|
|
tt_i64_op(get_last_user_activity_time(), OP_EQ, start-2*ONE_DAY);
|
|
|
|
// suppose we've been idle for 2 days... but we are a server.
|
|
reset_user_activity(start - 2*ONE_DAY);
|
|
options->ORPort_set = 1;
|
|
set_network_participation(true);
|
|
check_network_participation_callback(start+2, options);
|
|
tt_int_op(is_participating_on_network(), OP_EQ, true);
|
|
tt_i64_op(get_last_user_activity_time(), OP_EQ, start+2);
|
|
options->ORPort_set = 0;
|
|
|
|
// idle for 2 days, but we have a hidden service.
|
|
reset_user_activity(start - 2*ONE_DAY);
|
|
set_network_participation(true);
|
|
MOCK(hs_service_get_num_services, mock_get_num_services);
|
|
check_network_participation_callback(start+3, options);
|
|
tt_int_op(is_participating_on_network(), OP_EQ, true);
|
|
tt_i64_op(get_last_user_activity_time(), OP_EQ, start+3);
|
|
UNMOCK(hs_service_get_num_services);
|
|
|
|
// idle for 2 days but we have at least one user connection
|
|
MOCK(connection_get_by_type_nonlinked, mock_connection_gbtu);
|
|
reset_user_activity(start - 2*ONE_DAY);
|
|
set_network_participation(true);
|
|
options->DormantTimeoutDisabledByIdleStreams = 1;
|
|
check_network_participation_callback(start+10, options);
|
|
tt_int_op(is_participating_on_network(), OP_EQ, true);
|
|
tt_i64_op(get_last_user_activity_time(), OP_EQ, start+10);
|
|
|
|
// as above, but DormantTimeoutDisabledByIdleStreams is not set
|
|
reset_user_activity(start - 2*ONE_DAY);
|
|
set_network_participation(true);
|
|
options->DormantTimeoutDisabledByIdleStreams = 0;
|
|
check_network_participation_callback(start+13, options);
|
|
tt_int_op(is_participating_on_network(), OP_EQ, false);
|
|
tt_i64_op(get_last_user_activity_time(), OP_EQ, start-2*ONE_DAY);
|
|
UNMOCK(connection_get_by_type_nonlinked);
|
|
options->DormantTimeoutDisabledByIdleStreams = 1;
|
|
|
|
// idle for 2 days but DormantClientTimeout is 3 days
|
|
reset_user_activity(start - 2*ONE_DAY);
|
|
set_network_participation(true);
|
|
options->DormantClientTimeout = ONE_DAY * 3;
|
|
check_network_participation_callback(start+30, options);
|
|
tt_int_op(is_participating_on_network(), OP_EQ, true);
|
|
tt_i64_op(get_last_user_activity_time(), OP_EQ, start-2*ONE_DAY);
|
|
|
|
done:
|
|
or_options_free(options);
|
|
UNMOCK(hs_service_get_num_services);
|
|
UNMOCK(connection_get_by_type_nonlinked);
|
|
}
|
|
|
|
static void
|
|
test_mainloop_dormant_load_state(void *arg)
|
|
{
|
|
(void)arg;
|
|
or_state_t *state = or_state_new();
|
|
const time_t start = 1543956575;
|
|
|
|
reset_user_activity(0);
|
|
set_network_participation(false);
|
|
|
|
// When we construct a new state, it starts out in "auto" mode.
|
|
tt_int_op(state->Dormant, OP_EQ, -1);
|
|
|
|
// Initializing from "auto" makes us start out (by default) non-Dormant,
|
|
// with activity right now.
|
|
netstatus_load_from_state(state, start);
|
|
tt_assert(is_participating_on_network());
|
|
tt_i64_op(get_last_user_activity_time(), OP_EQ, start);
|
|
|
|
// Initializing from dormant clears the last user activity time, and
|
|
// makes us dormant.
|
|
state->Dormant = 1;
|
|
netstatus_load_from_state(state, start);
|
|
tt_assert(! is_participating_on_network());
|
|
tt_i64_op(get_last_user_activity_time(), OP_EQ, 0);
|
|
|
|
// Initializing from non-dormant sets the last user activity time, and
|
|
// makes us non-dormant.
|
|
state->Dormant = 0;
|
|
state->MinutesSinceUserActivity = 123;
|
|
netstatus_load_from_state(state, start);
|
|
tt_assert(is_participating_on_network());
|
|
tt_i64_op(get_last_user_activity_time(), OP_EQ, start - 123*60);
|
|
|
|
// If we would start dormant, but DormantCanceledByStartup is set, then
|
|
// we start up non-dormant.
|
|
state->Dormant = 1;
|
|
get_options_mutable()->DormantCanceledByStartup = 1;
|
|
netstatus_load_from_state(state, start);
|
|
tt_assert(is_participating_on_network());
|
|
tt_i64_op(get_last_user_activity_time(), OP_EQ, start);
|
|
|
|
done:
|
|
or_state_free(state);
|
|
}
|
|
|
|
static void
|
|
test_mainloop_dormant_save_state(void *arg)
|
|
{
|
|
(void)arg;
|
|
or_state_t *state = or_state_new();
|
|
const time_t start = 1543956575;
|
|
|
|
// Can we save a non-dormant state correctly?
|
|
reset_user_activity(start - 1000);
|
|
set_network_participation(true);
|
|
netstatus_flush_to_state(state, start);
|
|
|
|
tt_int_op(state->Dormant, OP_EQ, 0);
|
|
tt_int_op(state->MinutesSinceUserActivity, OP_EQ, 1000 / 60);
|
|
|
|
// Can we save a dormant state correctly?
|
|
set_network_participation(false);
|
|
netstatus_flush_to_state(state, start);
|
|
|
|
tt_int_op(state->Dormant, OP_EQ, 1);
|
|
tt_int_op(state->MinutesSinceUserActivity, OP_EQ, 0);
|
|
|
|
done:
|
|
or_state_free(state);
|
|
}
|
|
|
|
#define MAINLOOP_TEST(name) \
|
|
{ #name, test_mainloop_## name , TT_FORK, NULL, NULL }
|
|
|
|
struct testcase_t mainloop_tests[] = {
|
|
MAINLOOP_TEST(update_time_normal),
|
|
MAINLOOP_TEST(update_time_jumps),
|
|
MAINLOOP_TEST(user_activity),
|
|
MAINLOOP_TEST(check_participation),
|
|
MAINLOOP_TEST(dormant_load_state),
|
|
MAINLOOP_TEST(dormant_save_state),
|
|
END_OF_TESTCASES
|
|
};
|