mirror of
				https://github.com/SeriousBug/gandi-live-dns-rust
				synced 2025-10-26 02:27:17 -05:00 
			
		
		
		
	wip implement test
This commit is contained in:
		
							parent
							
								
									9e9940dc5d
								
							
						
					
					
						commit
						6d44c3d0d0
					
				
							
								
								
									
										3
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							|  | @ -3,5 +3,6 @@ | ||||||
|         "gandi", |         "gandi", | ||||||
|         "rrset", |         "rrset", | ||||||
|         "structopt" |         "structopt" | ||||||
|     ] |     ], | ||||||
|  |     "editor.formatOnSave": true | ||||||
| } | } | ||||||
							
								
								
									
										789
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										789
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							|  | @ -9,7 +9,6 @@ strip = "symbols" | ||||||
| lto = true | lto = true | ||||||
| 
 | 
 | ||||||
| [dependencies] | [dependencies] | ||||||
| 
 |  | ||||||
| reqwest = { version = "0.11", default-features= false, features = ["json", "rustls-tls"] } | reqwest = { version = "0.11", default-features= false, features = ["json", "rustls-tls"] } | ||||||
| toml = "0.5" | toml = "0.5" | ||||||
| json = "0.12" | json = "0.12" | ||||||
|  | @ -20,6 +19,7 @@ tokio = { version = "1.20", features = ["full"] } | ||||||
| futures = "0.3" | futures = "0.3" | ||||||
| anyhow = "1.0" | anyhow = "1.0" | ||||||
| governor = "0.4" | governor = "0.4" | ||||||
|  | async-trait = "0.1" | ||||||
| # TODO: Relies on a yet-unreleased interface. Switch to an actual crate release once available | # TODO: Relies on a yet-unreleased interface. Switch to an actual crate release once available | ||||||
| die-exit = { git = "https://github.com/Xavientois/die.git", rev = "31d3801f4e21654b0b28430987b1e21fc7728676" } | die-exit = { git = "https://github.com/Xavientois/die.git", rev = "31d3801f4e21654b0b28430987b1e21fc7728676" } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										15
									
								
								src/gandi/mod.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								src/gandi/mod.rs
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,15 @@ | ||||||
|  | pub(crate) struct GandiAPI<'t> { | ||||||
|  |     pub(crate) base_url: &'t str, | ||||||
|  |     pub(crate) fqdn: &'t str, | ||||||
|  |     pub(crate) rrset_name: &'t str, | ||||||
|  |     pub(crate) rrset_type: &'t str, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl<'t> GandiAPI<'t> { | ||||||
|  |     pub(crate) fn url(&self) -> String { | ||||||
|  |         format!( | ||||||
|  |             "{}/v5/livedns/domains/{}/records/{}/{}", | ||||||
|  |             self.base_url, self.fqdn, self.rrset_name, self.rrset_type | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										8
									
								
								src/ip_source/ip_source.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								src/ip_source/ip_source.rs
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,8 @@ | ||||||
|  | use anyhow; | ||||||
|  | use async_trait::async_trait; | ||||||
|  | 
 | ||||||
|  | #[async_trait] | ||||||
|  | pub trait IPSource { | ||||||
|  |   async fn get_ipv4() -> anyhow::Result<String>; | ||||||
|  |   async fn get_ipv6() -> anyhow::Result<String>; | ||||||
|  | } | ||||||
							
								
								
									
										22
									
								
								src/ip_source/ipify.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								src/ip_source/ipify.rs
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,22 @@ | ||||||
|  | use anyhow; | ||||||
|  | use async_trait::async_trait; | ||||||
|  | 
 | ||||||
|  | use super::ip_source::IPSource; | ||||||
|  | 
 | ||||||
|  | pub(crate) struct IPSourceIpify {} | ||||||
|  | 
 | ||||||
|  | async fn get_ip(api_url: &str) -> anyhow::Result<String> { | ||||||
|  |   let response = reqwest::get(api_url).await?; | ||||||
|  |   let text = response.text().await?; | ||||||
|  |   Ok(text) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[async_trait] | ||||||
|  | impl IPSource for IPSourceIpify { | ||||||
|  |   async fn get_ipv4() -> anyhow::Result<String> { | ||||||
|  |     get_ip("https://api.ipify.org").await | ||||||
|  |   } | ||||||
|  |   async fn get_ipv6() -> anyhow::Result<String> { | ||||||
|  |     get_ip("https://api6.ipify.org").await | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								src/ip_source/mod.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								src/ip_source/mod.rs
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | ||||||
|  | pub(crate) mod ip_source; | ||||||
|  | pub(crate) mod ipify; | ||||||
							
								
								
									
										113
									
								
								src/main.rs
									
									
									
									
									
								
							
							
						
						
									
										113
									
								
								src/main.rs
									
									
									
									
									
								
							|  | @ -1,29 +1,26 @@ | ||||||
| use crate::config::Config; | use crate::config::Config; | ||||||
|  | use crate::gandi::GandiAPI; | ||||||
|  | use crate::ip_source::{ip_source::IPSource, ipify::IPSourceIpify}; | ||||||
| use anyhow; | use anyhow; | ||||||
| use clap::Parser; | use clap::Parser; | ||||||
| use futures; | use futures; | ||||||
|  | use opts::Opts; | ||||||
| use reqwest::{header, Client, ClientBuilder, StatusCode}; | use reqwest::{header, Client, ClientBuilder, StatusCode}; | ||||||
| use serde::Serialize; | use serde::Serialize; | ||||||
| use std::{num::NonZeroU32, sync::Arc, time::Duration}; | use std::{num::NonZeroU32, sync::Arc, time::Duration}; | ||||||
| use tokio::{self, task::JoinHandle}; | use tokio::{self, task::JoinHandle}; | ||||||
| mod config; | mod config; | ||||||
|  | mod gandi; | ||||||
|  | mod ip_source; | ||||||
| mod opts; | mod opts; | ||||||
| use die_exit::*; | use die_exit::*; | ||||||
| use governor; | use governor; | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| /// 30 requests per minute, see https://api.gandi.net/docs/reference/
 | /// 30 requests per minute, see https://api.gandi.net/docs/reference/
 | ||||||
| const GANDI_RATE_LIMIT: u32 = 30; | const GANDI_RATE_LIMIT: u32 = 30; | ||||||
| /// If we hit the rate limit, wait up to this many seconds before next attempt
 | /// If we hit the rate limit, wait up to this many seconds before next attempt
 | ||||||
| const GANDI_DELAY_JITTER: u64 = 20; | const GANDI_DELAY_JITTER: u64 = 20; | ||||||
| 
 | 
 | ||||||
| fn gandi_api_url(fqdn: &str, rrset_name: &str, rrset_type: &str) -> String { |  | ||||||
|     return format!( |  | ||||||
|         " https://api.gandi.net/v5/livedns/domains/{}/records/{}/{}", |  | ||||||
|         fqdn, rrset_name, rrset_type |  | ||||||
|     ); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| fn api_client(api_key: &str) -> anyhow::Result<Client> { | fn api_client(api_key: &str) -> anyhow::Result<Client> { | ||||||
|     let client_builder = ClientBuilder::new(); |     let client_builder = ClientBuilder::new(); | ||||||
| 
 | 
 | ||||||
|  | @ -38,27 +35,19 @@ fn api_client(api_key: &str) -> anyhow::Result<Client> { | ||||||
|     return Ok(client); |     return Ok(client); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| async fn get_ip(api_url: &str) -> anyhow::Result<String> { |  | ||||||
|     let response = reqwest::get(api_url).await?; |  | ||||||
|     let text = response.text().await?; |  | ||||||
|     Ok(text) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| #[derive(Serialize)] | #[derive(Serialize)] | ||||||
| pub struct APIPayload { | pub struct APIPayload { | ||||||
|     pub rrset_values: Vec<String>, |     pub rrset_values: Vec<String>, | ||||||
|     pub rrset_ttl: u32, |     pub rrset_ttl: u32, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[tokio::main(flavor = "current_thread")] | async fn run(base_url: &str, opts: Opts) -> anyhow::Result<()> { | ||||||
| async fn main() -> anyhow::Result<()> { |  | ||||||
|     let opts = opts::Opts::parse(); |  | ||||||
|     let conf = config::load_config(&opts) |     let conf = config::load_config(&opts) | ||||||
|         .die_with(|error| format!("Failed to read config file: {}", error)); |         .die_with(|error| format!("Failed to read config file: {}", error)); | ||||||
|     config::validate_config(&conf).die_with(|error| format!("Invalid config: {}", error)); |     config::validate_config(&conf).die_with(|error| format!("Invalid config: {}", error)); | ||||||
|     println!("Finding out the IP address..."); |     println!("Finding out the IP address..."); | ||||||
|     let ipv4_result = get_ip("https://api.ipify.org").await; |     let ipv4_result = IPSourceIpify::get_ipv4().await; | ||||||
|     let ipv6_result = get_ip("https://api6.ipify.org").await; |     let ipv6_result = IPSourceIpify::get_ipv6().await; | ||||||
|     let ipv4 = ipv4_result.as_ref(); |     let ipv4 = ipv4_result.as_ref(); | ||||||
|     let ipv6 = ipv6_result.as_ref(); |     let ipv6 = ipv6_result.as_ref(); | ||||||
|     println!("Found these:"); |     println!("Found these:"); | ||||||
|  | @ -84,10 +73,16 @@ async fn main() -> anyhow::Result<()> { | ||||||
|     for entry in &conf.entry { |     for entry in &conf.entry { | ||||||
|         for entry_type in Config::types(entry) { |         for entry_type in Config::types(entry) { | ||||||
|             let fqdn = Config::fqdn(&entry, &conf).to_string(); |             let fqdn = Config::fqdn(&entry, &conf).to_string(); | ||||||
|             let url = gandi_api_url(&fqdn, entry.name.as_str(), entry_type); |             let url = GandiAPI { | ||||||
|  |                 fqdn: &fqdn, | ||||||
|  |                 rrset_name: &entry.name, | ||||||
|  |                 rrset_type: &entry_type, | ||||||
|  |                 base_url, | ||||||
|  |             } | ||||||
|  |             .url(); | ||||||
|             let ip = match entry_type { |             let ip = match entry_type { | ||||||
|                 "A" => ipv4.die_with(|error| format!("Needed IPv4 for {}: {}", fqdn, error)), |                 "A" => ipv4.die_with(|error| format!("Needed IPv4 for {fqdn}: {error}")), | ||||||
|                 "AAAA" => ipv6.die_with(|error| format!("Needed IPv6 for {}: {}", fqdn, error)), |                 "AAAA" => ipv6.die_with(|error| format!("Needed IPv6 for {fqdn}: {error}")), | ||||||
|                 bad_entry_type => die!("Unexpected type in config: {}", bad_entry_type), |                 bad_entry_type => die!("Unexpected type in config: {}", bad_entry_type), | ||||||
|             }; |             }; | ||||||
|             let payload = APIPayload { |             let payload = APIPayload { | ||||||
|  | @ -121,34 +116,72 @@ async fn main() -> anyhow::Result<()> { | ||||||
|         println!("{} - {}", status, body); |         println!("{} - {}", status, body); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     return Ok(()); |     Ok(()) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[tokio::main(flavor = "current_thread")] | ||||||
|  | async fn main() -> anyhow::Result<()> { | ||||||
|  |     let opts = opts::Opts::parse(); | ||||||
|  |     run("https://api.gandi.net", opts).await | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[cfg(test)] | #[cfg(test)] | ||||||
| mod tests { | mod tests { | ||||||
|     use httpmock::MockServer; |     use std::env::temp_dir; | ||||||
|     use serde_json::json; |  | ||||||
| 
 | 
 | ||||||
|     #[test] |     use crate::{ip_source::ip_source::IPSource, opts::Opts, run}; | ||||||
|     fn create_repo_success_test() { |     use async_trait::async_trait; | ||||||
|         // Arrange
 |     use httpmock::MockServer; | ||||||
|  |     use tokio::fs; | ||||||
|  | 
 | ||||||
|  |     struct IPSourceMock {} | ||||||
|  | 
 | ||||||
|  |     #[async_trait] | ||||||
|  |     impl IPSource for IPSourceMock { | ||||||
|  |         async fn get_ipv4() -> anyhow::Result<String> { | ||||||
|  |             Ok("192.168.0.0".to_string()) | ||||||
|  |         } | ||||||
|  |         async fn get_ipv6() -> anyhow::Result<String> { | ||||||
|  |             Ok("fe80:0000:0208:74ff:feda:625c".to_string()) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #[tokio::test] | ||||||
|  |     async fn create_repo_success_test() { | ||||||
|  |         let mut temp = temp_dir().join("gandi-live-dns-test"); | ||||||
|  |         fs::create_dir_all(&temp) | ||||||
|  |             .await | ||||||
|  |             .expect("Failed to create test dir"); | ||||||
|  |         temp.push("test.toml"); | ||||||
|  |         fs::write( | ||||||
|  |             &temp, | ||||||
|  |             "fqdn = \"example.com\"\napi_key = \"xxx\"\nttl = 300\n[[entry]]\nname =\"@\"\n", | ||||||
|  |         ) | ||||||
|  |         .await | ||||||
|  |         .expect("Failed to write test config file"); | ||||||
|  |         let fqdn = "example.com"; | ||||||
|  |         let rname = "@"; | ||||||
|  |         let rtype = "A"; | ||||||
|         let server = MockServer::start(); |         let server = MockServer::start(); | ||||||
|         let mock = server.mock(|when, then| { |         let mock = server.mock(|when, then| { | ||||||
|             when.method("POST") |             when.method("PUT") | ||||||
|                 .path("/user/repos") |                 .path(format!( | ||||||
|                 .header("Authorization", "token TOKEN") |                     "/v5/livedns/domains/{fqdn}/records/{rname}/{rtype}" | ||||||
|                 .header("Content-Type", "application/json"); |                 )) | ||||||
|             then.status(201) |                 .body_contains("192.168.0.0"); | ||||||
|                 .json_body(json!({ "html_url": "http://example.com" })); |             then.status(200); | ||||||
|         }); |         }); | ||||||
|         let client = GithubClient::new("TOKEN", &server.base_url()); |  | ||||||
| 
 | 
 | ||||||
|         // Act
 |         run( | ||||||
|         let result = client.create_repo("myRepo"); |             server.base_url().as_str(), | ||||||
|  |             Opts { | ||||||
|  |                 config: Some(temp.to_string_lossy().to_string()), | ||||||
|  |             }, | ||||||
|  |         ) | ||||||
|  |         .await | ||||||
|  |         .expect("Failed when running the update"); | ||||||
| 
 | 
 | ||||||
|         // Assert
 |         // Assert
 | ||||||
|         mock.assert(); |         mock.assert(); | ||||||
|         assert_eq!(result.is_ok(), true); |  | ||||||
|         assert_eq!(result.unwrap(), "http://example.com"); |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue