#define CONFIG_PRIVATE
#define MAINLOOP_PRIVATE
#define STATEFILE_PRIVATE
#include "test/test.h"
#include "test/log_test_helpers.h"
#include "lib/confmgt/confmgt.h"
#include "core/or/or.h"
#include "core/mainloop/connection.h"
#include "core/mainloop/mainloop.h"
#include "core/mainloop/mainloop_state_st.h"
#include "core/mainloop/mainloop_sys.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"
#include "app/main/subsysmgr.h"
static const uint64_t BILLION = 1000000000;
static void
test_mainloop_update_time_normal(void *arg)
{
(void)arg;
monotime_enable_test_mocking();
uint64_t mt_now = UINT64_C(7493289274986);
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); 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; 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; update_current_time(now);
tt_int_op(approx_time(), OP_EQ, now); tt_int_op(get_uptime(), OP_EQ, 3);
done:
monotime_disable_test_mocking();
}
static void
test_mainloop_update_time_jumps(void *arg)
{
(void)arg;
monotime_enable_test_mocking();
uint64_t mt_now = UINT64_C(7493289274986);
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);
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);
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); 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); mock_clean_saved_logs();
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);
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);
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_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_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);
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);
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);
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;
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);
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;
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);
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);
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;
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 *or_state = or_state_new();
mainloop_state_t *state;
{
int idx = subsystems_get_state_idx(&sys_mainloop);
tor_assert(idx >= 0);
state = config_mgr_get_obj_mutable(get_state_mgr(), or_state, idx);
}
const time_t start = 1543956575;
reset_user_activity(0);
set_network_participation(false);
tt_int_op(state->Dormant, OP_EQ, -1);
netstatus_load_from_state(state, start);
tt_assert(is_participating_on_network());
tt_i64_op(get_last_user_activity_time(), OP_EQ, start);
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);
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);
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(or_state);
}
static void
test_mainloop_dormant_save_state(void *arg)
{
(void)arg;
mainloop_state_t *state = tor_malloc_zero(sizeof(mainloop_state_t));
const time_t start = 1543956575;
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);
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:
tor_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
};