mirror of
				https://github.com/SeriousBug/gandi-live-dns-rust
				synced 2025-10-25 02:07:01 -05:00 
			
		
		
		
	Skip updating the IP address if it did not change (#88)
* Skip updating the IP address if it did not change * Update readme
This commit is contained in:
		
							parent
							
								
									f8060fad42
								
							
						
					
					
						commit
						327b14a00a
					
				
							
								
								
									
										31
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										31
									
								
								README.md
									
									
									
									
									
								
							|  | @ -1,5 +1,7 @@ | |||
| ## gandi-live-dns-rust | ||||
| ## Gandi Live Dns Rust <!-- omit in toc --> | ||||
| 
 | ||||
| <!-- ALL-CONTRIBUTORS-BADGE:START - Do not remove or modify this section --> | ||||
| 
 | ||||
| [](#contributors) <!-- ALL-CONTRIBUTORS-BADGE:END --> | ||||
| [](https://github.com/SeriousBug/gandi-live-dns-rust/actions/workflows/test.yml) | ||||
| [](https://codecov.io/gh/SeriousBug/gandi-live-dns-rust) | ||||
|  | @ -19,6 +21,23 @@ program can update both IPv4 and IPv6 addresses for one or more domains and | |||
| subdomains. It can be used as a one-shot tool managed with a systemd timer | ||||
| or cron, or a long-running process that reschedules itself. | ||||
| 
 | ||||
| ## Table of Contents <!-- omit in toc --> | ||||
| 
 | ||||
| - [Usage](#usage) | ||||
|   - [System packages](#system-packages) | ||||
|   - [Prebuilt binaries](#prebuilt-binaries) | ||||
|   - [With docker](#with-docker) | ||||
|   - [From source](#from-source) | ||||
| - [Automation](#automation) | ||||
|   - [By running as a background process](#by-running-as-a-background-process) | ||||
|     - [Skipped updates](#skipped-updates) | ||||
|   - [With a Systemd timer](#with-a-systemd-timer) | ||||
| - [Development](#development) | ||||
|   - [Local builds](#local-builds) | ||||
|   - [Making a release](#making-a-release) | ||||
| - [Alternatives](#alternatives) | ||||
| - [Contributors](#contributors) | ||||
| 
 | ||||
| ## Usage | ||||
| 
 | ||||
| The Gandi Live DNS API is rate limited at 30 requests per minute. This program | ||||
|  | @ -90,7 +109,7 @@ docker run --rm -it -v $(pwd)/gandi.toml:/gandi.toml:ro seriousbug/gandi-live-dn | |||
| Or with a `docker-compose.yml` file, add it in the arguments: | ||||
| 
 | ||||
| ```yml | ||||
|   gandi-live-dns: | ||||
| gandi-live-dns: | ||||
|   image: seriousbug/gandi-live-dns-rust:latest | ||||
|   restart: always | ||||
|   volumes: | ||||
|  | @ -99,6 +118,14 @@ Or with a `docker-compose.yml` file, add it in the arguments: | |||
|   command: --repeat=86400 | ||||
| ``` | ||||
| 
 | ||||
| #### Skipped updates | ||||
| 
 | ||||
| In background process mode, the tool will avoid sending an update to Gandi if | ||||
| your IP address has not changed since the last update. This only works so long | ||||
| as the tool continues to run, it will send an update when restarted even if your | ||||
| IP address has not changed. You can also override this behavior by adding | ||||
| `always_update = true` to the top of your config file. | ||||
| 
 | ||||
| ### With a Systemd timer | ||||
| 
 | ||||
| The `Packaging` folder contains a Systemd service and timer, which you can use | ||||
|  |  | |||
|  | @ -44,6 +44,8 @@ pub struct Config { | |||
|     pub entry: Vec<Entry>, | ||||
|     #[serde(default = "default_ttl")] | ||||
|     pub ttl: u32, | ||||
|     #[serde(default)] | ||||
|     pub always_update: bool, | ||||
| } | ||||
| 
 | ||||
| const DEFAULT_TYPES: &[&str] = &["A"]; | ||||
|  | @ -156,6 +158,7 @@ name = "@" | |||
|         assert_eq!(conf.entry[1].types, vec!["A".to_string()]); | ||||
|         // default
 | ||||
|         assert_eq!(conf.ip_source, IPSourceName::Ipify); | ||||
|         assert_eq!(conf.always_update, false); | ||||
|     } | ||||
| 
 | ||||
|     #[test] | ||||
|  | @ -170,6 +173,7 @@ fqdn = "example.com" | |||
| api_key = "yyy" | ||||
| ttl = 1200 | ||||
| ip_source = "Icanhazip" | ||||
| always_update = true | ||||
| 
 | ||||
| [[entry]] | ||||
| name = "www" | ||||
|  | @ -193,6 +197,7 @@ name = "@" | |||
|         assert_eq!(conf.entry[0].name, "www"); | ||||
|         assert_eq!(conf.entry[1].name, "@"); | ||||
|         assert_eq!(conf.ip_source, IPSourceName::Icanhazip); | ||||
|         assert_eq!(conf.always_update, true); | ||||
|     } | ||||
| 
 | ||||
|     #[test] | ||||
|  |  | |||
							
								
								
									
										175
									
								
								src/main.rs
									
									
									
									
									
								
							
							
						
						
									
										175
									
								
								src/main.rs
									
									
									
									
									
								
							|  | @ -4,6 +4,7 @@ use crate::ip_source::{ip_source::IPSource, ipify::IPSourceIpify}; | |||
| use clap::Parser; | ||||
| use config::IPSourceName; | ||||
| use ip_source::icanhazip::IPSourceIcanhazip; | ||||
| use opts::Opts; | ||||
| use reqwest::{header, Client, ClientBuilder, StatusCode}; | ||||
| use serde::Serialize; | ||||
| use std::{num::NonZeroU32, sync::Arc, time::Duration}; | ||||
|  | @ -40,8 +41,16 @@ pub struct APIPayload { | |||
|     pub rrset_ttl: u32, | ||||
| } | ||||
| 
 | ||||
| async fn run(base_url: &str, ip_source: &Box<dyn IPSource>, conf: &Config) -> anyhow::Result<()> { | ||||
|     config::validate_config(conf).die_with(|error| format!("Invalid config: {}", error)); | ||||
| async fn run( | ||||
|     base_url: &str, | ||||
|     ip_source: &Box<dyn IPSource>, | ||||
|     conf: &Config, | ||||
|     opts: &Opts, | ||||
| ) -> anyhow::Result<()> { | ||||
|     let mut last_ipv4: Option<String> = None; | ||||
|     let mut last_ipv6: Option<String> = None; | ||||
| 
 | ||||
|     loop { | ||||
|         println!("Finding out the IP address..."); | ||||
|         let (ipv4_result, ipv6_result) = join!(ip_source.get_ipv4(), ip_source.get_ipv6()); | ||||
|         let ipv4 = ipv4_result.as_ref(); | ||||
|  | @ -56,6 +65,19 @@ async fn run(base_url: &str, ip_source: &Box<dyn IPSource>, conf: &Config) -> an | |||
|             Err(err) => eprintln!("\tIPv6 failed: {}", err), | ||||
|         } | ||||
| 
 | ||||
|         let ipv4_same = last_ipv4 | ||||
|             .as_ref() | ||||
|             .map(|p| ipv4.map(|q| p == q).unwrap_or(false)) | ||||
|             .unwrap_or(false); | ||||
|         let ipv6_same = last_ipv6 | ||||
|             .as_ref() | ||||
|             .map(|p| ipv6.map(|q| p == q).unwrap_or(false)) | ||||
|             .unwrap_or(false); | ||||
| 
 | ||||
|         last_ipv4 = ipv4.ok().map(|v| v.to_string()); | ||||
|         last_ipv6 = ipv6.ok().map(|v| v.to_string()); | ||||
| 
 | ||||
|         if !ipv4_same || !ipv6_same || conf.always_update { | ||||
|             let client = api_client(&conf.api_key)?; | ||||
|             let mut tasks: Vec<JoinHandle<(StatusCode, String)>> = Vec::new(); | ||||
|             println!("Attempting to update DNS entries now"); | ||||
|  | @ -111,6 +133,18 @@ async fn run(base_url: &str, ip_source: &Box<dyn IPSource>, conf: &Config) -> an | |||
|             for (status, body) in results { | ||||
|                 println!("{} - {}", status, body); | ||||
|             } | ||||
|         } else { | ||||
|             println!("IP address has not changed since last update"); | ||||
|         } | ||||
| 
 | ||||
|         if let Some(repeat) = opts.repeat { | ||||
|             // If configured to repeat, do so
 | ||||
|             sleep(Duration::from_secs(repeat)).await; | ||||
|             continue; | ||||
|         } | ||||
|         // Otherwise this is one-shot, we should exit now
 | ||||
|         break; | ||||
|     } | ||||
| 
 | ||||
|     Ok(()) | ||||
| } | ||||
|  | @ -121,36 +155,23 @@ async fn main() -> anyhow::Result<()> { | |||
|     let conf = config::load_config(&opts) | ||||
|         .die_with(|error| format!("Failed to read config file: {}", error)); | ||||
| 
 | ||||
|     // run indefinitely if repeat is given
 | ||||
|     if let Some(delay) = opts.repeat { | ||||
|         loop { | ||||
|             run_dispatch(&conf).await.ok(); | ||||
|             sleep(Duration::from_secs(delay)).await | ||||
|         } | ||||
|     } | ||||
|     // otherwise run just once
 | ||||
|     else { | ||||
|         run_dispatch(&conf).await?; | ||||
|         Ok(()) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| async fn run_dispatch(conf: &Config) -> anyhow::Result<()> { | ||||
|     let ip_source: Box<dyn IPSource> = match conf.ip_source { | ||||
|         IPSourceName::Ipify => Box::new(IPSourceIpify), | ||||
|         IPSourceName::Icanhazip => Box::new(IPSourceIcanhazip), | ||||
|     }; | ||||
|     run("https://api.gandi.net", &ip_source, conf).await | ||||
|     config::validate_config(&conf).die_with(|error| format!("Invalid config: {}", error)); | ||||
|     run("https://api.gandi.net", &ip_source, &conf, &opts).await?; | ||||
|     Ok(()) | ||||
| } | ||||
| 
 | ||||
| #[cfg(test)] | ||||
| mod tests { | ||||
|     use std::env::temp_dir; | ||||
|     use std::{env::temp_dir, time::Duration}; | ||||
| 
 | ||||
|     use crate::{config, ip_source::ip_source::IPSource, opts::Opts, run}; | ||||
|     use async_trait::async_trait; | ||||
|     use httpmock::MockServer; | ||||
|     use tokio::fs; | ||||
|     use tokio::{fs, task::LocalSet, time::sleep}; | ||||
| 
 | ||||
|     struct IPSourceMock; | ||||
| 
 | ||||
|  | @ -165,7 +186,7 @@ mod tests { | |||
|     } | ||||
| 
 | ||||
|     #[tokio::test] | ||||
|     async fn create_repo_success_test() { | ||||
|     async fn single_shot() { | ||||
|         let mut temp = temp_dir().join("gandi-live-dns-test"); | ||||
|         fs::create_dir_all(&temp) | ||||
|             .await | ||||
|  | @ -196,11 +217,121 @@ mod tests { | |||
|         }; | ||||
|         let conf = config::load_config(&opts).expect("Failed to load config"); | ||||
|         let ip_source: Box<dyn IPSource> = Box::new(IPSourceMock); | ||||
|         run(server.base_url().as_str(), &ip_source, &conf) | ||||
|         run(server.base_url().as_str(), &ip_source, &conf, &opts) | ||||
|             .await | ||||
|             .expect("Failed when running the update"); | ||||
| 
 | ||||
|         // Assert
 | ||||
|         mock.assert(); | ||||
|     } | ||||
| 
 | ||||
|     #[test] | ||||
|     fn repeat() { | ||||
|         let runtime = tokio::runtime::Builder::new_current_thread() | ||||
|             .enable_all() | ||||
|             .build() | ||||
|             .unwrap(); | ||||
| 
 | ||||
|         LocalSet::new().block_on(&runtime, async { | ||||
|             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 mock = server.mock(|when, then| { | ||||
|                 when.method("PUT") | ||||
|                     .path(format!( | ||||
|                         "/v5/livedns/domains/{fqdn}/records/{rname}/{rtype}" | ||||
|                     )) | ||||
|                     .body_contains("192.168.0.0"); | ||||
|                 then.status(200); | ||||
|             }); | ||||
| 
 | ||||
|             let server_url = server.base_url(); | ||||
|             let handle = tokio::task::spawn_local(async move { | ||||
|                 let opts = Opts { | ||||
|                     config: Some(temp.to_string_lossy().to_string()), | ||||
|                     repeat: Some(1), | ||||
|                     ..Opts::default() | ||||
|                 }; | ||||
|                 let conf = config::load_config(&opts).expect("Failed to load config"); | ||||
|                 let ip_source: Box<dyn IPSource> = Box::new(IPSourceMock); | ||||
|                 run(&server_url, &ip_source, &conf, &opts) | ||||
|                     .await | ||||
|                     .expect("Failed when running the update"); | ||||
|             }); | ||||
| 
 | ||||
|             sleep(Duration::from_secs(3)).await; | ||||
|             handle.abort(); | ||||
| 
 | ||||
|             // Only should update once because the IP doesn't change
 | ||||
|             mock.assert(); | ||||
|         }); | ||||
|     } | ||||
| 
 | ||||
|     #[test] | ||||
|     fn repeat_always_update() { | ||||
|         let runtime = tokio::runtime::Builder::new_current_thread() | ||||
|             .enable_all() | ||||
|             .build() | ||||
|             .unwrap(); | ||||
| 
 | ||||
|         LocalSet::new().block_on(&runtime, async { | ||||
|             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\"\nalways_update = true\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 mock = server.mock(|when, then| { | ||||
|                 when.method("PUT") | ||||
|                     .path(format!( | ||||
|                         "/v5/livedns/domains/{fqdn}/records/{rname}/{rtype}" | ||||
|                     )) | ||||
|                     .body_contains("192.168.0.0"); | ||||
|                 then.status(200); | ||||
|             }); | ||||
| 
 | ||||
|             let server_url = server.base_url(); | ||||
|             let handle = tokio::task::spawn_local(async move { | ||||
|                 let opts = Opts { | ||||
|                     config: Some(temp.to_string_lossy().to_string()), | ||||
|                     repeat: Some(1), | ||||
|                     ..Opts::default() | ||||
|                 }; | ||||
|                 let conf = config::load_config(&opts).expect("Failed to load config"); | ||||
|                 let ip_source: Box<dyn IPSource> = Box::new(IPSourceMock); | ||||
|                 run(&server_url, &ip_source, &conf, &opts) | ||||
|                     .await | ||||
|                     .expect("Failed when running the update"); | ||||
|             }); | ||||
| 
 | ||||
|             sleep(Duration::from_secs(3)).await; | ||||
|             handle.abort(); | ||||
| 
 | ||||
|             // Should update multiple times since always_update
 | ||||
|             assert!(mock.hits() > 1); | ||||
|         }); | ||||
|     } | ||||
| } | ||||
|  |  | |||
		Loading…
	
		Reference in a new issue