1
0
digitalocean-dynamic-dns-ip/digitalocean-dynamic-ip.go
John Aylward 3fdd43dfa8 update to support paging and custom config location.
expand config to support a few more options
2019-07-22 11:23:47 -04:00

240 lines
7.1 KiB
Go
Executable File

package main
import (
"bytes"
"encoding/json"
"flag"
"fmt"
"io/ioutil"
"log"
"net"
"net/http"
"net/url"
"os"
"strconv"
homedir "github.com/mitchellh/go-homedir"
)
func checkError(err error) {
if err != nil {
log.Fatal(err)
}
}
var config ClientConfig
// ClientConfig : configuration json
type ClientConfig struct {
APIKey string `json:"apiKey"`
DOPageSize int `json:"doPageSize"`
Domains []Domain `json:"domains"`
}
// Domain : domains to be changed
type Domain struct {
Domain string `json:"domain"`
Records []DNSRecord `json:"records"`
}
// DNSRecord : Modifyiable DNS record
type DNSRecord struct {
ID int64 `json:"id"`
Type string `json:"type"`
Name string `json:"name"`
Priority *int `json:"priority"`
Port *int `json:"port"`
Weight *int `json:"weight"`
TTL int `json:"ttl"`
Flags *uint8 `json:"flags"`
Tag *string `json:"tag"`
Data string `json:"data"`
}
// DOResponse : DigitalOcean DNS Records response.
type DOResponse struct {
DomainRecords []DNSRecord `json:"domain_records"`
Meta struct {
Total int `json:"total"`
} `json:"meta"`
Links struct {
Pages struct {
First string `json:"first"`
Previous string `json:"prev"`
Next string `json:"next"`
Last string `json:"last"`
} `json:"pages"`
} `json:"links"`
}
//GetConfig : get configuration file ~/.digitalocean-dynamic-ip.json
func GetConfig() ClientConfig {
cmdHelp := flag.Bool("h", false, "Show the help message")
cmdHelp2 := flag.Bool("help", false, "Show the help message")
flag.Parse()
if *cmdHelp || *cmdHelp2 {
usage()
os.Exit(1)
}
configFile := ""
if len(flag.Args()) == 0 {
var err error
configFile, err = homedir.Dir()
checkError(err)
configFile += "/.digitalocean-dynamic-ip.json"
} else {
configFile = flag.Args()[0]
}
getfile, err := ioutil.ReadFile(configFile)
checkError(err)
var config ClientConfig
json.Unmarshal(getfile, &config)
checkError(err)
return config
}
func usage() {
os.Stdout.WriteString(fmt.Sprintf("To use this program you can specify the following command options:\n"+
"-h | -help\n\tShow this help message\n"+
"[config_file]\n\tlocation of the configuration file\n\n"+
"If the [config_file] parameter is not passed, then the default\n"+
"config location of ~/.digitalocean-dynamic-ip.json will be used.\n\n"+
"example usages:\n\t%[1]s -help\n"+
"\t%[1]s\n"+
"\t%[1]s %[2]s\n"+
"",
os.Args[0],
"/path/to/my/config.json",
))
}
//CheckLocalIP : get current IP of server.
func CheckLocalIP() string {
currentIPRequest, err := http.Get("https://diagnostic.opendns.com/myip")
checkError(err)
defer currentIPRequest.Body.Close()
currentIPRequestParse, err := ioutil.ReadAll(currentIPRequest.Body)
checkError(err)
return string(currentIPRequestParse)
}
//GetDomainRecords : Get DNS records of current domain.
func GetDomainRecords(domain string) []DNSRecord {
ret := make([]DNSRecord, 0)
var page DOResponse
pageParam := ""
// 20 is the default page size
if config.DOPageSize > 0 && config.DOPageSize != 20 {
pageParam = "?per_page=" + strconv.Itoa(config.DOPageSize)
}
for url := "https://api.digitalocean.com/v2/domains/" + url.PathEscape(domain) + "/records" + pageParam; url != ""; url = page.Links.Pages.Next {
page = getPage(url)
ret = append(ret, page.DomainRecords...)
}
return ret
}
func getPage(url string) DOResponse {
log.Println(url)
client := &http.Client{}
request, err := http.NewRequest("GET", url, nil)
checkError(err)
request.Header.Add("Content-type", "Application/json")
request.Header.Add("Authorization", "Bearer "+config.APIKey)
response, err := client.Do(request)
checkError(err)
defer response.Body.Close()
body, err := ioutil.ReadAll(response.Body)
// log.Print(string(body))
var jsonDOResponse DOResponse
e := json.Unmarshal(body, &jsonDOResponse)
checkError(e)
return jsonDOResponse
}
// UpdateRecords : Update DNS records of domain
func UpdateRecords(domain string, ip net.IP, toUpdateRecords []DNSRecord) {
currentIP := ip.String()
isIpv6 := ip.To4() == nil
if !isIpv6 {
// make sure we are using a v4 format
currentIP = ip.To4().String()
}
log.Printf("%s: %d to update\n", domain, len(toUpdateRecords))
updated := 0
doRecords := GetDomainRecords(domain)
// look for the item to update
if len(doRecords) < 1 {
log.Printf("%s: No DNS records found", domain)
return
}
log.Printf("%s: %d DNS records found", domain, len(doRecords))
for _, toUpdateRecord := range toUpdateRecords {
if toUpdateRecord.Type != "A" && toUpdateRecord.Type != "AAAA" {
log.Fatalf("%s: Unsupported type (Only A and AAAA records supported) for updates %+v", domain, toUpdateRecord)
}
if isIpv6 && toUpdateRecord.Type == "A" {
log.Fatalf("%s: You are trying to update an IPV4 A record with an IPV6 address: new ip: %s, config: %+v", domain, currentIP, toUpdateRecord)
}
if !isIpv6 && toUpdateRecord.Type == "AAAA" {
log.Fatalf("%s: You are trying to update an IPV6 A record with an IPV4 address: new ip: %s, config: %+v", domain, currentIP, toUpdateRecord)
}
if toUpdateRecord.ID > 0 {
// update the record directly. skip the extra search
log.Fatalf("%s: Unable to directly update records yet. Record: %+v", domain, toUpdateRecord)
}
log.Printf("%s: trying to update `%s` : `%s`", domain, toUpdateRecord.Type, toUpdateRecord.Name)
for _, doRecord := range doRecords {
//log.Printf("%s: checking `%s` : `%s`", domain, doRecord.Type, doRecord.Name)
if doRecord.Name == toUpdateRecord.Name && doRecord.Type == toUpdateRecord.Type {
if doRecord.Data == currentIP && (toUpdateRecord.TTL < 30 || doRecord.TTL == toUpdateRecord.TTL) {
log.Printf("%s: IP/TTL did not change %+v", domain, doRecord)
continue
}
log.Printf("%s: updating %+v", domain, doRecord)
// set the IP address
doRecord.Data = currentIP
if toUpdateRecord.TTL >= 30 && doRecord.TTL != toUpdateRecord.TTL {
doRecord.TTL = toUpdateRecord.TTL
}
update, err := json.Marshal(doRecord)
checkError(err)
client := &http.Client{}
request, err := http.NewRequest("PUT",
"https://api.digitalocean.com/v2/domains/"+url.PathEscape(domain)+"/records/"+strconv.FormatInt(int64(doRecord.ID), 10),
bytes.NewBuffer(update))
checkError(err)
request.Header.Set("Content-Type", "application/json")
request.Header.Add("Authorization", "Bearer "+config.APIKey)
response, err := client.Do(request)
checkError(err)
defer response.Body.Close()
body, err := ioutil.ReadAll(response.Body)
log.Printf("%s: DO update response for %s: %s\n", domain, doRecord.Name, string(body))
updated++
}
}
}
log.Printf("%s: %d of %d records updated\n", domain, updated, len(toUpdateRecords))
}
func main() {
config = GetConfig()
currentIP := CheckLocalIP()
ip := net.ParseIP(currentIP)
if ip == nil {
log.Fatalf("current IP address `%s` is not a valid IP address", currentIP)
}
for _, domain := range config.Domains {
domainName := domain.Domain
log.Printf("%s: START\n", domainName)
UpdateRecords(domainName, ip, domain.Records)
log.Printf("%s: END\n", domainName)
}
}