feat: hot-reloading of config file
This commit is contained in:
		
					parent
					
						
							
								5e76c2b055
							
						
					
				
			
			
				commit
				
					
						58e22d33af
					
				
			
		
					 16 changed files with 213 additions and 58 deletions
				
			
		|  | @ -1,4 +1,9 @@ | |||
| mod parse; | ||||
| mod service; | ||||
| mod toml; | ||||
| 
 | ||||
| pub use parse::build_settings; | ||||
| pub use { | ||||
|   self::toml::ConfigToml, | ||||
|   parse::{build_settings, parse_opts}, | ||||
|   service::ConfigTomlReloader, | ||||
| }; | ||||
|  |  | |||
|  | @ -7,28 +7,30 @@ use crate::{ | |||
| use clap::Arg; | ||||
| use rpxy_lib::{AppConfig, AppConfigList, ProxyConfig}; | ||||
| 
 | ||||
| pub fn build_settings() -> std::result::Result<(ProxyConfig, AppConfigList<CryptoFileSource>), anyhow::Error> { | ||||
| pub fn parse_opts() -> Result<String, anyhow::Error> { | ||||
|   let _ = include_str!("../../Cargo.toml"); | ||||
|   let options = clap::command!().arg( | ||||
|     Arg::new("config_file") | ||||
|       .long("config") | ||||
|       .short('c') | ||||
|       .value_name("FILE") | ||||
|       .help("Configuration file path like \"./config.toml\""), | ||||
|       .required(true) | ||||
|       .help("Configuration file path like ./config.toml"), | ||||
|   ); | ||||
|   let matches = options.get_matches(); | ||||
| 
 | ||||
|   ///////////////////////////////////
 | ||||
|   let config = if let Some(config_file_path) = matches.get_one::<String>("config_file") { | ||||
|     ConfigToml::new(config_file_path)? | ||||
|   } else { | ||||
|     // Default config Toml
 | ||||
|     ConfigToml::default() | ||||
|   }; | ||||
|   let config_file_path = matches.get_one::<String>("config_file").unwrap(); | ||||
| 
 | ||||
|   Ok(config_file_path.to_string()) | ||||
| } | ||||
| 
 | ||||
| pub fn build_settings( | ||||
|   config: &ConfigToml, | ||||
| ) -> std::result::Result<(ProxyConfig, AppConfigList<CryptoFileSource>), anyhow::Error> { | ||||
|   ///////////////////////////////////
 | ||||
|   // build proxy config
 | ||||
|   let proxy_config: ProxyConfig = (&config).try_into()?; | ||||
|   let proxy_config: ProxyConfig = config.try_into()?; | ||||
|   // For loggings
 | ||||
|   if proxy_config.listen_sockets.iter().any(|addr| addr.is_ipv6()) { | ||||
|     info!("Listen both IPv4 and IPv6") | ||||
|  | @ -50,7 +52,7 @@ pub fn build_settings() -> std::result::Result<(ProxyConfig, AppConfigList<Crypt | |||
| 
 | ||||
|   ///////////////////////////////////
 | ||||
|   // backend_apps
 | ||||
|   let apps = config.apps.ok_or(anyhow!("Missing application spec"))?; | ||||
|   let apps = config.apps.clone().ok_or(anyhow!("Missing application spec"))?; | ||||
| 
 | ||||
|   // assertions for all backend apps
 | ||||
|   ensure!(!apps.0.is_empty(), "Wrong application spec."); | ||||
|  | @ -88,9 +90,8 @@ pub fn build_settings() -> std::result::Result<(ProxyConfig, AppConfigList<Crypt | |||
| 
 | ||||
|   let app_config_list = AppConfigList { | ||||
|     inner: app_config_list_inner, | ||||
|     default_app: config.default_app.map(|v| v.to_ascii_lowercase()), // default backend application for plaintext http requests
 | ||||
|     default_app: config.default_app.clone().map(|v| v.to_ascii_lowercase()), // default backend application for plaintext http requests
 | ||||
|   }; | ||||
| 
 | ||||
|   Ok((proxy_config, app_config_list)) | ||||
|   // todo!()
 | ||||
| } | ||||
|  |  | |||
							
								
								
									
										24
									
								
								rpxy-bin/src/config/service.rs
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								rpxy-bin/src/config/service.rs
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,24 @@ | |||
| use super::toml::ConfigToml; | ||||
| use async_trait::async_trait; | ||||
| use hot_reload::{Reload, ReloaderError}; | ||||
| 
 | ||||
| #[derive(Clone)] | ||||
| pub struct ConfigTomlReloader { | ||||
|   pub config_path: String, | ||||
| } | ||||
| 
 | ||||
| #[async_trait] | ||||
| impl Reload<ConfigToml> for ConfigTomlReloader { | ||||
|   type Source = String; | ||||
|   async fn new(source: &Self::Source) -> Result<Self, ReloaderError<ConfigToml>> { | ||||
|     Ok(Self { | ||||
|       config_path: source.clone(), | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   async fn reload(&self) -> Result<Option<ConfigToml>, ReloaderError<ConfigToml>> { | ||||
|     let conf = ConfigToml::new(&self.config_path) | ||||
|       .map_err(|_e| ReloaderError::<ConfigToml>::Reload("Failed to reload config toml"))?; | ||||
|     Ok(Some(conf)) | ||||
|   } | ||||
| } | ||||
|  | @ -8,11 +8,12 @@ use rustc_hash::FxHashMap as HashMap; | |||
| use serde::Deserialize; | ||||
| use std::{fs, net::SocketAddr}; | ||||
| 
 | ||||
| #[derive(Deserialize, Debug, Default)] | ||||
| #[derive(Deserialize, Debug, Default, PartialEq, Eq, Clone)] | ||||
| pub struct ConfigToml { | ||||
|   pub listen_port: Option<u16>, | ||||
|   pub listen_port_tls: Option<u16>, | ||||
|   pub listen_ipv6: Option<bool>, | ||||
|   pub tcp_listen_backlog: Option<u32>, | ||||
|   pub max_concurrent_streams: Option<u32>, | ||||
|   pub max_clients: Option<u32>, | ||||
|   pub apps: Option<Apps>, | ||||
|  | @ -21,7 +22,7 @@ pub struct ConfigToml { | |||
| } | ||||
| 
 | ||||
| #[cfg(feature = "http3")] | ||||
| #[derive(Deserialize, Debug, Default)] | ||||
| #[derive(Deserialize, Debug, Default, PartialEq, Eq, Clone)] | ||||
| pub struct Http3Option { | ||||
|   pub alt_svc_max_age: Option<u32>, | ||||
|   pub request_max_body_size: Option<usize>, | ||||
|  | @ -31,24 +32,24 @@ pub struct Http3Option { | |||
|   pub max_idle_timeout: Option<u64>, | ||||
| } | ||||
| 
 | ||||
| #[derive(Deserialize, Debug, Default)] | ||||
| #[derive(Deserialize, Debug, Default, PartialEq, Eq, Clone)] | ||||
| pub struct Experimental { | ||||
|   #[cfg(feature = "http3")] | ||||
|   pub h3: Option<Http3Option>, | ||||
|   pub ignore_sni_consistency: Option<bool>, | ||||
| } | ||||
| 
 | ||||
| #[derive(Deserialize, Debug, Default)] | ||||
| #[derive(Deserialize, Debug, Default, PartialEq, Eq, Clone)] | ||||
| pub struct Apps(pub HashMap<String, Application>); | ||||
| 
 | ||||
| #[derive(Deserialize, Debug, Default)] | ||||
| #[derive(Deserialize, Debug, Default, PartialEq, Eq, Clone)] | ||||
| pub struct Application { | ||||
|   pub server_name: Option<String>, | ||||
|   pub reverse_proxy: Option<Vec<ReverseProxyOption>>, | ||||
|   pub tls: Option<TlsOption>, | ||||
| } | ||||
| 
 | ||||
| #[derive(Deserialize, Debug, Default)] | ||||
| #[derive(Deserialize, Debug, Default, PartialEq, Eq, Clone)] | ||||
| pub struct TlsOption { | ||||
|   pub tls_cert_path: Option<String>, | ||||
|   pub tls_cert_key_path: Option<String>, | ||||
|  | @ -56,7 +57,7 @@ pub struct TlsOption { | |||
|   pub client_ca_cert_path: Option<String>, | ||||
| } | ||||
| 
 | ||||
| #[derive(Deserialize, Debug, Default)] | ||||
| #[derive(Deserialize, Debug, Default, PartialEq, Eq, Clone)] | ||||
| pub struct ReverseProxyOption { | ||||
|   pub path: Option<String>, | ||||
|   pub replace_path: Option<String>, | ||||
|  | @ -65,7 +66,7 @@ pub struct ReverseProxyOption { | |||
|   pub load_balance: Option<String>, | ||||
| } | ||||
| 
 | ||||
| #[derive(Deserialize, Debug, Default)] | ||||
| #[derive(Deserialize, Debug, Default, PartialEq, Eq, Clone)] | ||||
| pub struct UpstreamParams { | ||||
|   pub location: String, | ||||
|   pub tls: Option<bool>, | ||||
|  | @ -112,6 +113,11 @@ impl TryInto<ProxyConfig> for &ConfigToml { | |||
|       }) | ||||
|       .collect(); | ||||
| 
 | ||||
|     // tcp backlog
 | ||||
|     if let Some(backlog) = self.tcp_listen_backlog { | ||||
|       proxy_config.tcp_listen_backlog = backlog; | ||||
|     } | ||||
| 
 | ||||
|     // max values
 | ||||
|     if let Some(c) = self.max_clients { | ||||
|       proxy_config.max_clients = c as usize; | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Jun Kurihara
				Jun Kurihara