/* Copyright (c) 2001 Matej Pfajfar. * Copyright (c) 2001-2004, Roger Dingledine. * Copyright (c) 2004-2006, Roger Dingledine, Nick Mathewson. * Copyright (c) 2007-2021, The Tor Project, Inc. */ /* See LICENSE for licensing information */ /** * @file dirauth_config.c * @brief Code to interpret the user's configuration of Tor's directory * authority module. **/ #include "orconfig.h" #include "feature/dirauth/dirauth_config.h" #include "lib/encoding/confline.h" #include "lib/confmgt/confmgt.h" #include "lib/conf/confdecl.h" #include "lib/version/torversion.h" /* Required for dirinfo_type_t in or_options_t */ #include "core/or/or.h" #include "core/or/tor_version_st.h" #include "core/or/versions.h" #include "app/config/config.h" #include "app/config/resolve_addr.h" #include "feature/dirauth/voting_schedule.h" #include "feature/stats/rephist.h" #include "feature/dirauth/authmode.h" #include "feature/dirauth/bwauth.h" #include "feature/dirauth/dirauth_periodic.h" #include "feature/dirauth/dirauth_sys.h" #include "feature/dirauth/dirvote.h" #include "feature/dirauth/guardfraction.h" #include "feature/dirauth/dirauth_options_st.h" /* Copied from config.c, we will refactor later in 29211. */ #define REJECT(arg) \ STMT_BEGIN *msg = tor_strdup(arg); return -1; STMT_END #if defined(__GNUC__) && __GNUC__ <= 3 #define COMPLAIN(args...) \ STMT_BEGIN log_warn(LD_CONFIG, args); STMT_END #else #define COMPLAIN(args, ...) \ STMT_BEGIN log_warn(LD_CONFIG, args, ##__VA_ARGS__); STMT_END #endif /* defined(__GNUC__) && __GNUC__ <= 3 */ #define YES_IF_CHANGED_INT(opt) \ if (!CFG_EQ_INT(old_options, new_options, opt)) return 1; /** Return true iff we are configured to reject request under load for non * relay connections. */ bool dirauth_should_reject_requests_under_load(void) { return !!dirauth_get_options()->AuthDirRejectRequestsUnderLoad; } /** * Legacy validation/normalization function for the dirauth mode options in * options. Uses old_options as the previous options. * * Returns 0 on success, returns -1 and sets *msg to a newly allocated string * on error. */ int options_validate_dirauth_mode(const or_options_t *old_options, or_options_t *options, char **msg) { if (BUG(!options)) return -1; if (BUG(!msg)) return -1; if (!authdir_mode(options)) return 0; /* confirm that our address isn't broken, so we can complain now */ tor_addr_t tmp; if (!find_my_address(options, AF_INET, LOG_WARN, &tmp, NULL, NULL)) REJECT("Failed to resolve/guess local address. See logs for details."); if (!options->ContactInfo && !options->TestingTorNetwork) REJECT("Authoritative directory servers must set ContactInfo"); if (options->UseEntryGuards) { log_info(LD_CONFIG, "Authoritative directory servers can't set " "UseEntryGuards. Disabling."); options->UseEntryGuards = 0; } if (!options->DownloadExtraInfo && authdir_mode_v3(options)) { log_info(LD_CONFIG, "Authoritative directories always try to download " "extra-info documents. Setting DownloadExtraInfo."); options->DownloadExtraInfo = 1; } if (!(options->BridgeAuthoritativeDir || options->V3AuthoritativeDir)) REJECT("AuthoritativeDir is set, but none of " "(Bridge/V3)AuthoritativeDir is set."); /* If we have a v3bandwidthsfile and it's broken, complain on startup */ if (options->V3BandwidthsFile && !old_options) { dirserv_read_measured_bandwidths(options->V3BandwidthsFile, NULL, NULL, NULL); } /* same for guardfraction file */ if (options->GuardfractionFile && !old_options) { dirserv_read_guardfraction_file(options->GuardfractionFile, NULL); } if (!options->DirPort_set) REJECT("Running as authoritative directory, but no DirPort set."); if (!options->ORPort_set) REJECT("Running as authoritative directory, but no ORPort set."); if (options->ClientOnly) REJECT("Running as authoritative directory, but ClientOnly also set."); return 0; } /** * Legacy validation/normalization function for the dirauth schedule options * in options. Uses old_options as the previous options. * * Returns 0 on success, returns -1 and sets *msg to a newly allocated string * on error. */ int options_validate_dirauth_schedule(const or_options_t *old_options, or_options_t *options, char **msg) { (void)old_options; if (BUG(!options)) return -1; if (BUG(!msg)) return -1; if (!authdir_mode_v3(options)) return 0; if (options->V3AuthVoteDelay + options->V3AuthDistDelay >= options->V3AuthVotingInterval/2) { REJECT("V3AuthVoteDelay plus V3AuthDistDelay must be less than half " "V3AuthVotingInterval"); } if (options->V3AuthVoteDelay < MIN_VOTE_SECONDS) { if (options->TestingTorNetwork) { if (options->V3AuthVoteDelay < MIN_VOTE_SECONDS_TESTING) { REJECT("V3AuthVoteDelay is way too low."); } else { COMPLAIN("V3AuthVoteDelay is very low. " "This may lead to failure to vote for a consensus."); } } else { REJECT("V3AuthVoteDelay is way too low."); } } if (options->V3AuthDistDelay < MIN_DIST_SECONDS) { if (options->TestingTorNetwork) { if (options->V3AuthDistDelay < MIN_DIST_SECONDS_TESTING) { REJECT("V3AuthDistDelay is way too low."); } else { COMPLAIN("V3AuthDistDelay is very low. " "This may lead to missing votes in a consensus."); } } else { REJECT("V3AuthDistDelay is way too low."); } } if (options->V3AuthNIntervalsValid < 2) REJECT("V3AuthNIntervalsValid must be at least 2."); if (options->V3AuthVotingInterval < MIN_VOTE_INTERVAL) { if (options->TestingTorNetwork) { if (options->V3AuthVotingInterval < MIN_VOTE_INTERVAL_TESTING) { /* Unreachable, covered by earlier checks */ REJECT("V3AuthVotingInterval is insanely low."); /* LCOV_EXCL_LINE */ } else { COMPLAIN("V3AuthVotingInterval is very low. " "This may lead to failure to synchronise for a consensus."); } } else { REJECT("V3AuthVotingInterval is insanely low."); } } else if (options->V3AuthVotingInterval > 24*60*60) { REJECT("V3AuthVotingInterval is insanely high."); } else if (((24*60*60) % options->V3AuthVotingInterval) != 0) { COMPLAIN("V3AuthVotingInterval does not divide evenly into 24 hours."); } return 0; } /** * Legacy validation/normalization function for the dirauth testing options * in options. Uses old_options as the previous options. * * Returns 0 on success, returns -1 and sets *msg to a newly allocated string * on error. */ int options_validate_dirauth_testing(const or_options_t *old_options, or_options_t *options, char **msg) { (void)old_options; if (BUG(!options)) return -1; if (BUG(!msg)) return -1; if (!authdir_mode(options)) return 0; if (!authdir_mode_v3(options)) return 0; if (options->TestingV3AuthInitialVotingInterval < MIN_VOTE_INTERVAL_TESTING_INITIAL) { REJECT("TestingV3AuthInitialVotingInterval is insanely low."); } else if (((30*60) % options->TestingV3AuthInitialVotingInterval) != 0) { REJECT("TestingV3AuthInitialVotingInterval does not divide evenly into " "30 minutes."); } if (options->TestingV3AuthInitialVoteDelay < MIN_VOTE_SECONDS_TESTING) { REJECT("TestingV3AuthInitialVoteDelay is way too low."); } if (options->TestingV3AuthInitialDistDelay < MIN_DIST_SECONDS_TESTING) { REJECT("TestingV3AuthInitialDistDelay is way too low."); } if (options->TestingV3AuthInitialVoteDelay + options->TestingV3AuthInitialDistDelay >= options->TestingV3AuthInitialVotingInterval) { REJECT("TestingV3AuthInitialVoteDelay plus TestingV3AuthInitialDistDelay " "must be less than TestingV3AuthInitialVotingInterval"); } if (options->TestingV3AuthVotingStartOffset > MIN(options->TestingV3AuthInitialVotingInterval, options->V3AuthVotingInterval)) { REJECT("TestingV3AuthVotingStartOffset is higher than the voting " "interval."); } else if (options->TestingV3AuthVotingStartOffset < 0) { REJECT("TestingV3AuthVotingStartOffset must be non-negative."); } return 0; } /** * Return true if changing the configuration from old to new * affects the timing of the voting subsystem */ static int options_transition_affects_dirauth_timing(const or_options_t *old_options, const or_options_t *new_options) { tor_assert(old_options); tor_assert(new_options); if (authdir_mode_v3(old_options) != authdir_mode_v3(new_options)) return 1; if (! authdir_mode_v3(new_options)) return 0; YES_IF_CHANGED_INT(V3AuthVotingInterval); YES_IF_CHANGED_INT(V3AuthVoteDelay); YES_IF_CHANGED_INT(V3AuthDistDelay); YES_IF_CHANGED_INT(TestingV3AuthInitialVotingInterval); YES_IF_CHANGED_INT(TestingV3AuthInitialVoteDelay); YES_IF_CHANGED_INT(TestingV3AuthInitialDistDelay); YES_IF_CHANGED_INT(TestingV3AuthVotingStartOffset); return 0; } /** Fetch the active option list, and take dirauth actions based on it. All of * the things we do should survive being done repeatedly. If present, * old_options contains the previous value of the options. * * Return 0 if all goes well, return -1 if it's time to die. * * Note: We haven't moved all the "act on new configuration" logic * into the options_act* functions yet. Some is still in do_hup() and other * places. */ int options_act_dirauth(const or_options_t *old_options) { const or_options_t *options = get_options(); /* We may need to reschedule some dirauth stuff if our status changed. */ if (old_options) { if (options_transition_affects_dirauth_timing(old_options, options)) { dirauth_sched_recalculate_timing(options, time(NULL)); reschedule_dirvote(options); } } return 0; } /** Fetch the active option list, and take dirauth mtbf actions based on it. * All of the things we do should survive being done repeatedly. If present, * old_options contains the previous value of the options. * * Must be called immediately after a successful or_state_load(). * * Return 0 if all goes well, return -1 if it's time to die. * * Note: We haven't moved all the "act on new configuration" logic * into the options_act* functions yet. Some is still in do_hup() and other * places. */ int options_act_dirauth_mtbf(const or_options_t *old_options) { (void)old_options; const or_options_t *options = get_options(); int running_tor = options->command == CMD_RUN_TOR; if (!authdir_mode(options)) return 0; /* Load dirauth state */ if (running_tor) { rep_hist_load_mtbf_data(time(NULL)); } return 0; } /** Fetch the active option list, and take dirauth statistics actions based * on it. All of the things we do should survive being done repeatedly. If * present, old_options contains the previous value of the options. * * Sets *print_notice_out if we enabled stats, and need to print * a stats log using options_act_relay_stats_msg(). * * Return 0 if all goes well, return -1 if it's time to die. * * Note: We haven't moved all the "act on new configuration" logic * into the options_act* functions yet. Some is still in do_hup() and other * places. */ int options_act_dirauth_stats(const or_options_t *old_options, bool *print_notice_out) { if (BUG(!print_notice_out)) return -1; const or_options_t *options = get_options(); if (authdir_mode_bridge(options)) { time_t now = time(NULL); int print_notice = 0; if (!old_options || !authdir_mode_bridge(old_options)) { rep_hist_desc_stats_init(now); print_notice = 1; } if (print_notice) *print_notice_out = 1; } /* If we used to have statistics enabled but we just disabled them, stop gathering them. */ if (old_options && authdir_mode_bridge(old_options) && !authdir_mode_bridge(options)) rep_hist_desc_stats_term(); return 0; } /** * Make any necessary modifications to a dirauth_options_t that occur * before validation. On success return 0; on failure return -1 and * set *msg_out to a newly allocated error string. **/ static int dirauth_options_pre_normalize(void *arg, char **msg_out) { dirauth_options_t *options = arg; (void)msg_out; if (!options->RecommendedClientVersions) options->RecommendedClientVersions = config_lines_dup(options->RecommendedVersions); if (!options->RecommendedServerVersions) options->RecommendedServerVersions = config_lines_dup(options->RecommendedVersions); if (config_ensure_bandwidth_cap(&options->AuthDirFastGuarantee, "AuthDirFastGuarantee", msg_out) < 0) return -1; if (config_ensure_bandwidth_cap(&options->AuthDirGuardBWGuarantee, "AuthDirGuardBWGuarantee", msg_out) < 0) return -1; return 0; } /** * Check whether a dirauth_options_t is correct. * * On success return 0; on failure return -1 and set *msg_out to a * newly allocated error string. **/ static int dirauth_options_validate(const void *arg, char **msg) { const dirauth_options_t *options = arg; tor_version_t minimal_accepted_server_version, recommended_version; if (options->VersioningAuthoritativeDirectory && (!options->RecommendedClientVersions || !options->RecommendedServerVersions)) { REJECT("Versioning authoritative dir servers must set " "Recommended*Versions."); } if (options->AuthDirVoteGuardBwThresholdFraction > 1.0 || options->AuthDirVoteGuardBwThresholdFraction < 0.0) { REJECT("Guard bandwdith threshold fraction is invalid."); } if (tor_version_parse(options->MinimalAcceptedServerVersion, &minimal_accepted_server_version) != 0) { REJECT("Invalid MinimalAcceptedServerVersion"); } tor_assertf(tor_version_parse(get_short_version(), &recommended_version) == 0, "We failed to parse our own version"); if (tor_version_compare(&recommended_version, &minimal_accepted_server_version) < 0) { REJECT("MinimalAcceptedServerVersion wants to reject the version " "this node is running"); } char *recommended_versions; int found_recommended_rejected_version = 0; /* Call these functions to produce warnings only. */ recommended_versions = format_recommended_version_list( options->RecommendedClientVersions, 1); tor_free(recommended_versions); recommended_versions = format_recommended_version_list( options->RecommendedServerVersions, 1); smartlist_t *version_sl = smartlist_new(); smartlist_split_string(version_sl, recommended_versions, ",", SPLIT_SKIP_SPACE, 0); SMARTLIST_FOREACH_BEGIN(version_sl, const char *, version) { if (version[0] != '\0' && tor_version_parse(version, &recommended_version) != 0) { COMPLAIN("Found unparseable version in RecommendedServerVersions"); continue; } if (tor_version_compare(&recommended_version, &minimal_accepted_server_version) < 0) { found_recommended_rejected_version = 1; break; } } SMARTLIST_FOREACH_END(version); SMARTLIST_FOREACH(version_sl, char *, version, tor_free(version)); smartlist_free(version_sl); tor_free(recommended_versions); if (found_recommended_rejected_version) REJECT("MinimalAcceptedServerVersion wants to reject a recommended " "version"); if (options->TestingAuthDirTimeToLearnReachability > 2*60*60) { COMPLAIN("TestingAuthDirTimeToLearnReachability is insanely high."); } return 0; } /* Declare the options field table for dirauth_options */ #define CONF_CONTEXT TABLE #include "feature/dirauth/dirauth_options.inc" #undef CONF_CONTEXT /** Magic number for dirauth_options_t. */ #define DIRAUTH_OPTIONS_MAGIC 0x41757448 /** * Declare the configuration options for the dirauth module. **/ const config_format_t dirauth_options_fmt = { .size = sizeof(dirauth_options_t), .magic = { "dirauth_options_t", DIRAUTH_OPTIONS_MAGIC, offsetof(dirauth_options_t, magic) }, .vars = dirauth_options_t_vars, .pre_normalize_fn = dirauth_options_pre_normalize, .validate_fn = dirauth_options_validate };