Skip to content

Commit 59340a6

Browse files
authored
Merge pull request #18 from stek29/feature-ipv6
IPv6 support
2 parents ce2d709 + 0a5cfc6 commit 59340a6

File tree

4 files changed

+98
-23
lines changed

4 files changed

+98
-23
lines changed

lib/providers/cloudflare/api.go

+16-8
Original file line numberDiff line numberDiff line change
@@ -18,13 +18,20 @@ type CloudflareAPI struct {
1818
}
1919

2020
type Record struct {
21-
ID string `json:"id"`
22-
Type string `json:"type"`
23-
Content string `json:"content"`
24-
Name string `json:"name"`
25-
Proxied bool `json:"proxied"`
21+
ID string `json:"id"`
22+
Type RecordType `json:"type"`
23+
Content string `json:"content"`
24+
Name string `json:"name"`
25+
Proxied bool `json:"proxied"`
2626
}
2727

28+
type RecordType = string
29+
30+
const (
31+
RecordTypeA = RecordType("A")
32+
RecordTypeAAAA = RecordType("AAAA")
33+
)
34+
2835
type RecordResponse struct {
2936
Result []Record `json:"result"`
3037
}
@@ -44,8 +51,8 @@ func NewCloudflareClient(token string, zoneID string, host string) (*CloudflareA
4451
return &api, nil
4552
}
4653

47-
func (api *CloudflareAPI) ListDNSRecords() ([]Record, error) {
48-
uri := fmt.Sprintf("/zones/%s/dns_records?type=A&name=%s", api.ZoneID, api.Host)
54+
func (api *CloudflareAPI) ListDNSRecords(recType RecordType) ([]Record, error) {
55+
uri := fmt.Sprintf("/zones/%s/dns_records?type=%s&name=%s", api.ZoneID, recType, api.Host)
4956
resp, err := api.request("GET", uri, nil)
5057
if err != nil {
5158
return nil, err
@@ -90,7 +97,8 @@ func (api *CloudflareAPI) request(method string, uri string, body io.Reader) ([]
9097
defer resp.Body.Close()
9198

9299
if resp.StatusCode != http.StatusOK {
93-
panic("Status code not 200")
100+
respBody, _ := ioutil.ReadAll(resp.Body)
101+
return nil, fmt.Errorf("Status code not 200 but %v, body: %v", err, string(respBody))
94102
}
95103

96104
respBody, err := ioutil.ReadAll(resp.Body)

lib/providers/cloudflare/cloudflare.go

+35-3
Original file line numberDiff line numberDiff line change
@@ -2,9 +2,10 @@ package cloudflare
22

33
import (
44
"errors"
5-
"github.com/hugomd/cloudflare-ddns/lib/providers"
65
"log"
76
"os"
7+
8+
"github.com/hugomd/cloudflare-ddns/lib/providers"
89
)
910

1011
type Cloudflare struct {
@@ -45,7 +46,7 @@ func NewProvider() (providers.Provider, error) {
4546
}
4647

4748
api, err := NewCloudflareClient(APITOKEN, ZONEID, HOST)
48-
49+
4950
if err != nil {
5051
return nil, err
5152
}
@@ -58,7 +59,38 @@ func NewProvider() (providers.Provider, error) {
5859
}
5960

6061
func (api *Cloudflare) UpdateRecord(ip string) error {
61-
records, err := api.client.ListDNSRecords()
62+
records, err := api.client.ListDNSRecords(RecordTypeA)
63+
if err != nil {
64+
return err
65+
}
66+
67+
var record Record
68+
for i := range records {
69+
if records[i].Name == HOST {
70+
record = records[i]
71+
}
72+
}
73+
74+
if record == (Record{}) {
75+
return errors.New("Host not found")
76+
}
77+
78+
if ip != record.Content {
79+
record.Content = ip
80+
err = api.client.UpdateDNSRecord(record)
81+
if err != nil {
82+
return err
83+
}
84+
log.Printf("IP changed, updated to %s", ip)
85+
} else {
86+
log.Print("No change in IP, not updating record")
87+
}
88+
89+
return nil
90+
}
91+
92+
func (api *Cloudflare) UpdateRecord6(ip string) error {
93+
records, err := api.client.ListDNSRecords(RecordTypeAAAA)
6294
if err != nil {
6395
return err
6496
}

lib/providers/providers.go

+1
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ package providers
22

33
type Provider interface {
44
UpdateRecord(ip string) error
5+
UpdateRecord6(ip string) error
56
}
67

78
type ProviderInit func() (Provider, error)

main.go

+46-12
Original file line numberDiff line numberDiff line change
@@ -3,14 +3,15 @@ package main
33
import (
44
"bytes"
55
"flag"
6-
"github.com/hugomd/cloudflare-ddns/lib/providers"
7-
_ "github.com/hugomd/cloudflare-ddns/lib/providers/_all"
86
"io/ioutil"
97
"log"
108
"net/http"
119
"os"
1210
"strings"
1311
"time"
12+
13+
"github.com/hugomd/cloudflare-ddns/lib/providers"
14+
_ "github.com/hugomd/cloudflare-ddns/lib/providers/_all"
1415
)
1516

1617
func checkIP() (string, error) {
@@ -28,6 +29,22 @@ func checkIP() (string, error) {
2829
return string(bytes.TrimSpace(buf)), nil
2930
}
3031

32+
func checkIP6() (string, error) {
33+
// has to be a service without A records (for ex api6.ipify.org cant be used)
34+
rsp, err := http.Get("https://v6.ident.me")
35+
if err != nil {
36+
return "", err
37+
}
38+
defer rsp.Body.Close()
39+
40+
buf, err := ioutil.ReadAll(rsp.Body)
41+
if err != nil {
42+
return "", err
43+
}
44+
45+
return string(bytes.TrimSpace(buf)), nil
46+
}
47+
3148
func setEnvVarsFromConfig(filename *string) error {
3249
contents, err := ioutil.ReadFile(*filename)
3350
if err != nil {
@@ -51,6 +68,8 @@ func main() {
5168

5269
CheckDuration := flag.Duration("duration", 0, "update interval (ex. 15s, 1m, 6h); if not specified or set to 0s, run only once and exit")
5370
ConfigFile := flag.String("config", "", "location of an (optional) config file to load environment variables from")
71+
run4 := flag.Bool("ipv4", true, "update A record")
72+
run6 := flag.Bool("ipv6", false, "update AAAA record")
5473
flag.Parse()
5574

5675
if *CheckDuration == time.Duration(0) {
@@ -66,20 +85,20 @@ func main() {
6685
}
6786
}
6887

69-
runddns()
88+
runddns(*run4, *run6)
7089

7190
if runonce {
7291
os.Exit(0)
7392
}
7493

7594
for range ticker.C {
76-
runddns()
95+
runddns(*run4, *run6)
7796
}
7897

7998
return
8099
}
81100

82-
func runddns() {
101+
func runddns(run4, run6 bool) {
83102
PROVIDER := os.Getenv("PROVIDER")
84103
if PROVIDER == "" {
85104
log.Fatal("PROVIDER env. variable is required")
@@ -90,14 +109,29 @@ func runddns() {
90109
panic(err)
91110
}
92111

93-
ip, err := checkIP()
94-
if err != nil {
95-
panic(err)
112+
if run4 {
113+
ip, err := checkIP()
114+
if err != nil {
115+
panic(err)
116+
}
117+
log.Printf("IP is %s", ip)
118+
119+
err = provider.UpdateRecord(ip)
120+
if err != nil {
121+
panic(err)
122+
}
96123
}
97-
log.Printf("IP is %s", ip)
98124

99-
err = provider.UpdateRecord(ip)
100-
if err != nil {
101-
panic(err)
125+
if run6 {
126+
ip, err := checkIP6()
127+
if err != nil {
128+
panic(err)
129+
}
130+
log.Printf("IPv6 is %s", ip)
131+
132+
err = provider.UpdateRecord6(ip)
133+
if err != nil {
134+
panic(err)
135+
}
102136
}
103137
}

0 commit comments

Comments
 (0)