wip: designing acme
This commit is contained in:
		
					parent
					
						
							
								9b9622edc5
							
						
					
				
			
			
				commit
				
					
						63f9d1dabc
					
				
			
		
					 3 changed files with 87 additions and 51 deletions
				
			
		|  | @ -15,14 +15,14 @@ enum FileType { | ||||||
|   Cert, |   Cert, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[derive(Debug)] | #[derive(Debug, PartialEq, Eq)] | ||||||
| pub struct DirCache { | pub struct DirCache { | ||||||
|   account_dir: PathBuf, |   pub(super) account_dir: PathBuf, | ||||||
|   cert_dir: PathBuf, |   pub(super) cert_dir: PathBuf, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl DirCache { | impl DirCache { | ||||||
|   pub fn new<P>(dir: P, server_name: impl AsRef<Path>) -> Self |   pub fn new<P>(dir: P, server_name: &str) -> Self | ||||||
|   where |   where | ||||||
|     P: AsRef<Path>, |     P: AsRef<Path>, | ||||||
|   { |   { | ||||||
|  |  | ||||||
|  | @ -1,32 +1,28 @@ | ||||||
| use crate::dir_cache::DirCache; |  | ||||||
| use crate::{ | use crate::{ | ||||||
|   constants::{ACME_DIR_URL, ACME_REGISTRY_PATH}, |   constants::{ACME_DIR_URL, ACME_REGISTRY_PATH}, | ||||||
|  |   dir_cache::DirCache, | ||||||
|   error::RpxyAcmeError, |   error::RpxyAcmeError, | ||||||
|   log::*, |   log::*, | ||||||
| }; | }; | ||||||
| use rustc_hash::FxHashMap as HashMap; | use rustc_hash::FxHashMap as HashMap; | ||||||
| use rustls_acme::AcmeConfig; | // use rustls_acme::AcmeConfig;
 | ||||||
| use std::{fmt::Debug, path::PathBuf, sync::Arc}; | use std::path::PathBuf; | ||||||
| use url::Url; | use url::Url; | ||||||
| 
 | 
 | ||||||
| #[derive(Debug)] | #[derive(Debug)] | ||||||
| /// ACME settings
 | /// ACME settings
 | ||||||
| pub struct AcmeContexts<EC, EA = EC> | pub struct AcmeContexts { | ||||||
| where |  | ||||||
|   EC: Debug + 'static, |  | ||||||
|   EA: Debug + 'static, |  | ||||||
| { |  | ||||||
|   /// ACME directory url
 |   /// ACME directory url
 | ||||||
|   acme_dir_url: Url, |   acme_dir_url: Url, | ||||||
|   /// ACME registry directory
 |   /// ACME registry directory
 | ||||||
|   acme_registry_dir: PathBuf, |   acme_registry_dir: PathBuf, | ||||||
|   /// ACME contacts
 |   /// ACME contacts
 | ||||||
|   contacts: Vec<String>, |   contacts: Vec<String>, | ||||||
|   /// ACME config
 |   /// ACME directly cache information
 | ||||||
|   inner: HashMap<String, Box<AcmeConfig<EC, EA>>>, |   inner: HashMap<String, DirCache>, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl AcmeContexts<std::io::Error> { | impl AcmeContexts { | ||||||
|   /// Create a new instance. Note that for each domain, a new AcmeConfig is created.
 |   /// Create a new instance. Note that for each domain, a new AcmeConfig is created.
 | ||||||
|   /// This means that for each domain, a distinct operation will be dispatched and separated certificates will be generated.
 |   /// This means that for each domain, a distinct operation will be dispatched and separated certificates will be generated.
 | ||||||
|   pub fn try_new( |   pub fn try_new( | ||||||
|  | @ -35,6 +31,9 @@ impl AcmeContexts<std::io::Error> { | ||||||
|     contacts: &[String], |     contacts: &[String], | ||||||
|     domains: &[String], |     domains: &[String], | ||||||
|   ) -> Result<Self, RpxyAcmeError> { |   ) -> Result<Self, RpxyAcmeError> { | ||||||
|  |     // Install aws_lc_rs as default crypto provider for rustls
 | ||||||
|  |     let _ = rustls::crypto::CryptoProvider::install_default(rustls::crypto::aws_lc_rs::default_provider()); | ||||||
|  | 
 | ||||||
|     let acme_registry_dir = acme_registry_dir |     let acme_registry_dir = acme_registry_dir | ||||||
|       .map(|v| v.to_ascii_lowercase()) |       .map(|v| v.to_ascii_lowercase()) | ||||||
|       .map_or_else(|| PathBuf::from(ACME_REGISTRY_PATH), PathBuf::from); |       .map_or_else(|| PathBuf::from(ACME_REGISTRY_PATH), PathBuf::from); | ||||||
|  | @ -46,25 +45,33 @@ impl AcmeContexts<std::io::Error> { | ||||||
|       .as_deref() |       .as_deref() | ||||||
|       .map_or_else(|| Url::parse(ACME_DIR_URL), Url::parse)?; |       .map_or_else(|| Url::parse(ACME_DIR_URL), Url::parse)?; | ||||||
|     let contacts = contacts.iter().map(|email| format!("mailto:{email}")).collect::<Vec<_>>(); |     let contacts = contacts.iter().map(|email| format!("mailto:{email}")).collect::<Vec<_>>(); | ||||||
|     let rustls_client_config = rustls::ClientConfig::builder() |     // let rustls_client_config = rustls::ClientConfig::builder()
 | ||||||
|       .dangerous() // The `Verifier` we're using is actually safe
 |     //   .dangerous() // The `Verifier` we're using is actually safe
 | ||||||
|       .with_custom_certificate_verifier(std::sync::Arc::new(rustls_platform_verifier::Verifier::new())) |     //   .with_custom_certificate_verifier(std::sync::Arc::new(rustls_platform_verifier::Verifier::new()))
 | ||||||
|       .with_no_client_auth(); |     //   .with_no_client_auth();
 | ||||||
|     let rustls_client_config = Arc::new(rustls_client_config); |     // let rustls_client_config = Arc::new(rustls_client_config);
 | ||||||
| 
 | 
 | ||||||
|     let inner = domains |     let inner = domains | ||||||
|       .iter() |       .iter() | ||||||
|       .map(|domain| { |       .map(|domain| { | ||||||
|         let dir_cache = DirCache::new(&acme_registry_dir, domain); |         let domain = domain.to_ascii_lowercase(); | ||||||
|         let config = AcmeConfig::new([domain]) |         let dir_cache = DirCache::new(&acme_registry_dir, &domain); | ||||||
|           .contact(&contacts) |         (domain, dir_cache) | ||||||
|           .cache(dir_cache) |  | ||||||
|           .directory(acme_dir_url.as_str()) |  | ||||||
|           .client_tls_config(rustls_client_config.clone()); |  | ||||||
|         let config = Box::new(config); |  | ||||||
|         (domain.to_ascii_lowercase(), config) |  | ||||||
|       }) |       }) | ||||||
|       .collect::<HashMap<_, _>>(); |       .collect::<HashMap<_, _>>(); | ||||||
|  |     // let inner = domains
 | ||||||
|  |     //   .iter()
 | ||||||
|  |     //   .map(|domain| {
 | ||||||
|  |     //     let dir_cache = DirCache::new(&acme_registry_dir, domain);
 | ||||||
|  |     //     let config = AcmeConfig::new([domain])
 | ||||||
|  |     //       .contact(&contacts)
 | ||||||
|  |     //       .cache(dir_cache)
 | ||||||
|  |     //       .directory(acme_dir_url.as_str())
 | ||||||
|  |     //       .client_tls_config(rustls_client_config.clone());
 | ||||||
|  |     //     let config = Box::new(config);
 | ||||||
|  |     //     (domain.to_ascii_lowercase(), config)
 | ||||||
|  |     //   })
 | ||||||
|  |     //   .collect::<HashMap<_, _>>();
 | ||||||
| 
 | 
 | ||||||
|     Ok(Self { |     Ok(Self { | ||||||
|       acme_dir_url, |       acme_dir_url, | ||||||
|  | @ -77,6 +84,8 @@ impl AcmeContexts<std::io::Error> { | ||||||
| 
 | 
 | ||||||
| #[cfg(test)] | #[cfg(test)] | ||||||
| mod tests { | mod tests { | ||||||
|  |   use crate::constants::ACME_ACCOUNT_SUBDIR; | ||||||
|  | 
 | ||||||
|   use super::*; |   use super::*; | ||||||
| 
 | 
 | ||||||
|   #[test] |   #[test] | ||||||
|  | @ -84,13 +93,30 @@ mod tests { | ||||||
|     let acme_dir_url = "https://acme.example.com/directory"; |     let acme_dir_url = "https://acme.example.com/directory"; | ||||||
|     let acme_registry_dir = "/tmp/acme"; |     let acme_registry_dir = "/tmp/acme"; | ||||||
|     let contacts = vec!["test@example.com".to_string()]; |     let contacts = vec!["test@example.com".to_string()]; | ||||||
|     let acme_contexts: AcmeContexts<std::io::Error> = AcmeContexts::try_new( |     let acme_contexts: AcmeContexts = AcmeContexts::try_new( | ||||||
|       Some(acme_dir_url), |       Some(acme_dir_url), | ||||||
|       Some(acme_registry_dir), |       Some(acme_registry_dir), | ||||||
|       &contacts, |       &contacts, | ||||||
|       &["example.com".to_string(), "example.org".to_string()], |       &["example.com".to_string(), "example.org".to_string()], | ||||||
|     ) |     ) | ||||||
|     .unwrap(); |     .unwrap(); | ||||||
|     println!("{:#?}", acme_contexts); |     assert_eq!(acme_contexts.inner.len(), 2); | ||||||
|  |     assert_eq!(acme_contexts.contacts, vec!["mailto:test@example.com".to_string()]); | ||||||
|  |     assert_eq!(acme_contexts.acme_dir_url.as_str(), acme_dir_url); | ||||||
|  |     assert_eq!(acme_contexts.acme_registry_dir, PathBuf::from(acme_registry_dir)); | ||||||
|  |     assert_eq!( | ||||||
|  |       acme_contexts.inner["example.com"], | ||||||
|  |       DirCache { | ||||||
|  |         account_dir: PathBuf::from(acme_registry_dir).join(ACME_ACCOUNT_SUBDIR), | ||||||
|  |         cert_dir: PathBuf::from(acme_registry_dir).join("example.com"), | ||||||
|  |       } | ||||||
|  |     ); | ||||||
|  |     assert_eq!( | ||||||
|  |       acme_contexts.inner["example.org"], | ||||||
|  |       DirCache { | ||||||
|  |         account_dir: PathBuf::from(acme_registry_dir).join(ACME_ACCOUNT_SUBDIR), | ||||||
|  |         cert_dir: PathBuf::from(acme_registry_dir).join("example.org"), | ||||||
|  |       } | ||||||
|  |     ); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -7,7 +7,7 @@ use rpxy_lib::{AppConfig, AppConfigList, ProxyConfig}; | ||||||
| use rustc_hash::FxHashMap as HashMap; | use rustc_hash::FxHashMap as HashMap; | ||||||
| 
 | 
 | ||||||
| #[cfg(feature = "acme")] | #[cfg(feature = "acme")] | ||||||
| use rpxy_acme::{ACME_DIR_URL, ACME_REGISTRY_PATH}; | use rpxy_acme::{AcmeContexts, ACME_DIR_URL, ACME_REGISTRY_PATH}; | ||||||
| 
 | 
 | ||||||
| /// Parsed options
 | /// Parsed options
 | ||||||
| pub struct Opts { | pub struct Opts { | ||||||
|  | @ -160,27 +160,37 @@ pub async fn build_cert_manager( | ||||||
| /// Build acme manager and dummy cert and key as initial states if not exists
 | /// Build acme manager and dummy cert and key as initial states if not exists
 | ||||||
| /// TODO: CURRENTLY NOT IMPLEMENTED, UNDER DESIGNING
 | /// TODO: CURRENTLY NOT IMPLEMENTED, UNDER DESIGNING
 | ||||||
| pub async fn build_acme_manager(config: &ConfigToml) -> Result<(), anyhow::Error> { | pub async fn build_acme_manager(config: &ConfigToml) -> Result<(), anyhow::Error> { | ||||||
|   // let acme_option = config.experimental.as_ref().and_then(|v| v.acme.clone());
 |   let acme_option = config.experimental.as_ref().and_then(|v| v.acme.clone()); | ||||||
|   // if acme_option.is_none() {
 |   if acme_option.is_none() { | ||||||
|   //   return Ok(());
 |     return Ok(()); | ||||||
|   // }
 |   } | ||||||
|   // let acme_option = acme_option.unwrap();
 |   let acme_option = acme_option.unwrap(); | ||||||
|   // let mut acme_targets = AcmeTargets::try_new(
 | 
 | ||||||
|   //   acme_option.email.as_ref(),
 |   let domains = config | ||||||
|   //   acme_option.dir_url.as_deref(),
 |     .apps | ||||||
|   //   acme_option.registry_path.as_deref(),
 |     .as_ref() | ||||||
|   // )
 |     .unwrap() | ||||||
|   // .map_err(|e| anyhow!("Invalid acme configuration: {e}"))?;
 |     .0 | ||||||
|  |     .values() | ||||||
|  |     .filter_map(|app| { | ||||||
|  |       //
 | ||||||
|  |       if let Some(tls) = app.tls.as_ref() { | ||||||
|  |         if let Some(true) = tls.acme { | ||||||
|  |           return Some(app.server_name.as_ref().unwrap().to_owned()); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |       None | ||||||
|  |     }) | ||||||
|  |     .collect::<Vec<_>>(); | ||||||
|  | 
 | ||||||
|  |   let acme_contexts = AcmeContexts::try_new( | ||||||
|  |     acme_option.dir_url.as_deref(), | ||||||
|  |     acme_option.registry_path.as_deref(), | ||||||
|  |     &[acme_option.email], | ||||||
|  |     domains.as_slice(), | ||||||
|  |   )?; | ||||||
| 
 | 
 | ||||||
|   // let apps = config.apps.as_ref().unwrap();
 |  | ||||||
|   // for app in apps.0.values() {
 |  | ||||||
|   //   if let Some(tls) = app.tls.as_ref() {
 |  | ||||||
|   //     if tls.acme.unwrap_or(false) {
 |  | ||||||
|   //       acme_targets.add_target(app.server_name.as_ref().unwrap())?;
 |  | ||||||
|   //     }
 |  | ||||||
|   //   }
 |  | ||||||
|   // }
 |  | ||||||
|   // TODO: remove later
 |   // TODO: remove later
 | ||||||
|   // println!("ACME targets: {:#?}", acme_targets);
 |   println!("ACME contexts: {:#?}", acme_contexts); | ||||||
|   Ok(()) |   Ok(()) | ||||||
| } | } | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Jun Kurihara
				Jun Kurihara