#pragma once #include #include #include #include #include #include "ini.hpp" #include "definition.hpp" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace llarp { using SectionValues_t = llarp::ConfigParser::SectionValues_t; using Config_impl_t = llarp::ConfigParser::Config_impl_t; // TODO: don't use these maps. they're sloppy and difficult to follow /// Small struct to gather all parameters needed for config generation to reduce the number of /// parameters that need to be passed around. struct ConfigGenParameters { bool isRelay = false; fs::path defaultDataDir; }; struct RouterConfig { size_t m_minConnectedRouters = 0; size_t m_maxConnectedRouters = 0; std::string m_netId; std::string m_nickname; fs::path m_dataDir; bool m_blockBogons = false; IpAddress m_publicAddress; int m_workerThreads = -1; int m_numNetThreads = -1; size_t m_JobQueueSize = 0; std::string m_routerContactFile; std::string m_encryptionKeyFile; std::string m_identityKeyFile; std::string m_transportKeyFile; bool m_isRelay = false; void defineConfigOptions(ConfigDefinition& conf, const ConfigGenParameters& params); }; /// config for path hop selection struct PeerSelectionConfig { /// in our hops what netmask will we use for unique ips for hops /// i.e. 32 for every hop unique ip, 24 unique /24 per hop, etc /// int m_UniqueHopsNetmaskSize; /// set of countrys to exclude from path building (2 char country code) std::unordered_set m_ExcludeCountries; void defineConfigOptions(ConfigDefinition& conf, const ConfigGenParameters& params); /// return true if this set of router contacts is acceptable against this config bool Acceptable(const std::set& hops) const; }; struct NetworkConfig { std::optional m_enableProfiling; bool m_saveProfiles; std::string m_strictConnect; std::string m_ifname; IPRange m_ifaddr; std::optional m_keyfile; std::string m_endpointType; bool m_reachable = false; std::optional m_Hops; std::optional m_Paths; bool m_AllowExit = false; std::set m_snodeBlacklist; net::IPRangeMap m_ExitMap; net::IPRangeMap m_LNSExitMap; std::unordered_map m_ExitAuths; std::unordered_map m_LNSExitAuths; std::unordered_map m_mapAddrs; service::AuthType m_AuthType = service::AuthType::eAuthTypeNone; std::optional m_AuthUrl; std::optional m_AuthMethod; std::unordered_set m_AuthWhitelist; std::vector m_SRVRecords; std::optional m_baseV6Address; // TODO: // on-up // on-down // on-ready void defineConfigOptions(ConfigDefinition& conf, const ConfigGenParameters& params); }; struct DnsConfig { IpAddress m_bind; std::vector m_upstreamDNS; void defineConfigOptions(ConfigDefinition& conf, const ConfigGenParameters& params); }; struct LinksConfig { struct LinkInfo { std::string interface; int addressFamily = -1; uint16_t port = -1; }; /// Create a LinkInfo from the given string. /// @throws if str does not represent a LinkInfo. LinkInfo LinkInfoFromINIValues(std::string_view name, std::string_view value); LinkInfo m_OutboundLink; std::vector m_InboundLinks; void defineConfigOptions(ConfigDefinition& conf, const ConfigGenParameters& params); }; struct ConnectConfig { std::vector routers; void defineConfigOptions(ConfigDefinition& conf, const ConfigGenParameters& params); }; struct ApiConfig { bool m_enableRPCServer = false; std::string m_rpcBindAddr; void defineConfigOptions(ConfigDefinition& conf, const ConfigGenParameters& params); }; struct LokidConfig { bool whitelistRouters = false; fs::path ident_keyfile; oxenmq::address lokidRPCAddr; void defineConfigOptions(ConfigDefinition& conf, const ConfigGenParameters& params); }; struct BootstrapConfig { std::vector files; std::set routers; bool seednode; void defineConfigOptions(ConfigDefinition& conf, const ConfigGenParameters& params); }; struct LoggingConfig { LogType m_logType = LogType::Unknown; LogLevel m_logLevel = eLogNone; std::string m_logFile; void defineConfigOptions(ConfigDefinition& conf, const ConfigGenParameters& params); }; struct Config { explicit Config(fs::path datadir); ~Config() = default; RouterConfig router; NetworkConfig network; PeerSelectionConfig paths; ConnectConfig connect; DnsConfig dns; LinksConfig links; ApiConfig api; LokidConfig lokid; BootstrapConfig bootstrap; LoggingConfig logging; // Initialize config definition void initializeConfig(ConfigDefinition& conf, const ConfigGenParameters& params); /// Insert config entries for backwards-compatibility (e.g. so that the config system will /// tolerate old values that are no longer accepted) /// /// @param conf is the config to modify void addBackwardsCompatibleConfigOptions(ConfigDefinition& conf); // Load a config from the given file if the config file is not provided LoadDefault is called bool Load(std::optional fname = std::nullopt, bool isRelay = false); std::string generateBaseClientConfig(); std::string generateBaseRouterConfig(); void Save(); void Override(std::string section, std::string key, std::string value); void AddDefault(std::string section, std::string key, std::string value); /// create a config with the default parameters for an embedded lokinet static std::shared_ptr EmbeddedConfig(); private: /// Load (initialize) a default config. /// /// This delegates to the ConfigDefinition to generate a default config, /// as though an empty config were specified. /// /// If using Config without the intention of loading from file (or string), this is necessary /// in order to obtain sane defaults. /// /// @param isRelay determines whether the config will reflect that of a relay or client /// @param dataDir is a path representing a directory to be used as the data dir /// @return true on success, false otherwise bool LoadDefault(bool isRelay); void LoadOverrides(); std::vector> m_Additional; ConfigParser m_Parser; const fs::path m_DataDir; }; void ensureConfig(fs::path dataDir, fs::path confFile, bool overwrite, bool asRouter); } // namespace llarp