package ddns import ( "context" "fmt" "log" "strings" "time" "github.com/libdns/libdns" "github.com/miekg/dns" "github.com/naiba/nezha/model" "github.com/naiba/nezha/pkg/utils" ) var ( dnsTimeOut = 10 * time.Second customDNSServers []string ) type IP struct { Ipv4Addr string Ipv6Addr string } type Provider struct { ctx context.Context ipAddr string recordType string domain string prefix string zone string DDNSProfile *model.DDNSProfile IPAddrs *IP Setter libdns.RecordSetter } func InitDNSServers(s string) { if s != "" { customDNSServers = strings.Split(s, ",") } } func (provider *Provider) UpdateDomain(ctx context.Context) { provider.ctx = ctx for _, domain := range provider.DDNSProfile.Domains { for retries := 0; retries < int(provider.DDNSProfile.MaxRetries); retries++ { provider.domain = domain log.Printf("NEZHA>> 正在尝试更新域名(%s)DDNS(%d/%d)", provider.domain, retries+1, provider.DDNSProfile.MaxRetries) if err := provider.updateDomain(); err != nil { log.Printf("NEZHA>> 尝试更新域名(%s)DDNS失败: %v", provider.domain, err) } else { log.Printf("NEZHA>> 尝试更新域名(%s)DDNS成功", provider.domain) break } } } } func (provider *Provider) updateDomain() error { var err error provider.prefix, provider.zone, err = splitDomainSOA(provider.domain) if err != nil { return err } // 当IPv4和IPv6同时成功才算作成功 if *provider.DDNSProfile.EnableIPv4 { provider.recordType = getRecordString(true) provider.ipAddr = provider.IPAddrs.Ipv4Addr if err = provider.addDomainRecord(); err != nil { return err } } if *provider.DDNSProfile.EnableIPv6 { provider.recordType = getRecordString(false) provider.ipAddr = provider.IPAddrs.Ipv6Addr if err = provider.addDomainRecord(); err != nil { return err } } return nil } func (provider *Provider) addDomainRecord() error { _, err := provider.Setter.SetRecords(provider.ctx, provider.zone, []libdns.Record{ { Type: provider.recordType, Name: provider.prefix, Value: provider.ipAddr, TTL: time.Minute, }, }) return err } func splitDomainSOA(domain string) (prefix string, zone string, err error) { c := &dns.Client{Timeout: dnsTimeOut} domain += "." indexes := dns.Split(domain) servers := utils.DNSServers if len(customDNSServers) > 0 { servers = customDNSServers } var r *dns.Msg for _, idx := range indexes { m := new(dns.Msg) m.SetQuestion(domain[idx:], dns.TypeSOA) for _, server := range servers { r, _, err = c.Exchange(m, server) if err != nil { return } if len(r.Answer) > 0 { if soa, ok := r.Answer[0].(*dns.SOA); ok { zone = soa.Hdr.Name prefix = libdns.RelativeName(domain, zone) return } } } } return "", "", fmt.Errorf("SOA record not found for domain: %s", domain) } func getRecordString(isIpv4 bool) string { if isIpv4 { return "A" } return "AAAA" }