package duck import ( "fmt" "io/ioutil" "log" "runtime/debug" "strings" "time" "github.com/TimothyYe/godns" ) var ( // DuckUrl the API address for Duck DNS DuckUrl = "https://www.duckdns.org/update?domains=%s&token=%s&%s" ) // Handler struct type Handler struct { Configuration *godns.Settings } // SetConfiguration pass dns settings and store it to handler instance func (handler *Handler) SetConfiguration(conf *godns.Settings) { handler.Configuration = conf } // DomainLoop the main logic loop func (handler *Handler) DomainLoop(domain *godns.Domain, panicChan chan<- godns.Domain) { defer func() { if err := recover(); err != nil { log.Printf("Recovered in %v: %v\n", err, debug.Stack()) panicChan <- *domain } }() var lastIP string for { currentIP, err := godns.GetCurrentIP(handler.Configuration) if err != nil { log.Println("get_currentIP:", err) continue } log.Println("currentIP is:", currentIP) //check against locally cached IP, if no change, skip update if currentIP == lastIP { log.Printf("IP is the same as cached one. Skip update.\n") } else { lastIP = currentIP client := godns.GetHttpClient(handler.Configuration) var ip string if strings.ToUpper(handler.Configuration.IPType) == godns.IPV4 { ip = fmt.Sprintf("ip=%s", currentIP) } else if strings.ToUpper(handler.Configuration.IPType) == godns.IPV6 { ip = fmt.Sprintf("ipv6=%s", currentIP) } for _, subDomain := range domain.SubDomains { // update IP with HTTP GET request resp, err := client.Get(fmt.Sprintf(DuckUrl, subDomain, handler.Configuration.LoginToken, ip)) if err != nil { // handle error log.Print("Failed to update sub domain:", subDomain) continue } defer resp.Body.Close() body, err := ioutil.ReadAll(resp.Body) if err != nil || string(body) != "OK" { log.Println("Failed to update the IP") continue } else { log.Print("IP updated to:", currentIP) } // Send mail notification if notify is enabled if handler.Configuration.Notify.Enabled { log.Print("Sending notification to:", handler.Configuration.Notify.SendTo) if err := godns.SendNotify(handler.Configuration, fmt.Sprintf("%s.%s", subDomain, domain.DomainName), currentIP); err != nil { log.Println("Failed to send notification") } } } } // Sleep with interval log.Printf("Going to sleep, will start next checking in %d seconds...\r\n", handler.Configuration.Interval) time.Sleep(time.Second * time.Duration(handler.Configuration.Interval)) } }