mirror of
https://github.com/taigrr/godns
synced 2025-01-18 04:03:25 -08:00
Add Cloudflare handler
This commit is contained in:
243
handler/cloudflare_handler.go
Normal file
243
handler/cloudflare_handler.go
Normal file
@@ -0,0 +1,243 @@
|
||||
package handler
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"encoding/json"
|
||||
"io"
|
||||
"io/ioutil"
|
||||
"log"
|
||||
"net/http"
|
||||
"runtime/debug"
|
||||
"time"
|
||||
|
||||
"github.com/TimothyYe/godns"
|
||||
"golang.org/x/net/proxy"
|
||||
)
|
||||
|
||||
// CloudflareHandler struct definition
|
||||
type CloudflareHandler struct {
|
||||
Configuration *godns.Settings
|
||||
API string
|
||||
}
|
||||
|
||||
// DNS api response
|
||||
type DNSRecordResponse struct {
|
||||
Records []DNSRecord `json:"result"`
|
||||
Success bool `json:"success"`
|
||||
}
|
||||
|
||||
// DNS update api response
|
||||
type DNSRecordUpdateResponse struct {
|
||||
Record DNSRecord `json:"result"`
|
||||
Success bool `json:"success"`
|
||||
}
|
||||
|
||||
type DNSRecord struct {
|
||||
Id string `json:"id"`
|
||||
Ip string `json:"content"`
|
||||
Name string `json:"name"`
|
||||
Proxied bool `json:"proxied"`
|
||||
Type string `json:"type"`
|
||||
ZoneId string `json:"zone_id"`
|
||||
}
|
||||
|
||||
func (r *DNSRecord) SetIp(ip string) {
|
||||
r.Ip = ip
|
||||
}
|
||||
|
||||
// response from zone api request
|
||||
type ZoneResponse struct {
|
||||
Zones []Zone `json:"result"`
|
||||
Success bool `json:"success"`
|
||||
}
|
||||
|
||||
// nested results, only care about name and id
|
||||
type Zone struct {
|
||||
Id string `json:"id"`
|
||||
Name string `json:"name"`
|
||||
}
|
||||
|
||||
// SetConfiguration pass dns settings and store it to handler instance
|
||||
func (handler *CloudflareHandler) SetConfiguration(conf *godns.Settings) {
|
||||
handler.Configuration = conf
|
||||
handler.API = "https://api.cloudflare.com/client/v4"
|
||||
}
|
||||
|
||||
// DomainLoop the main logic loop
|
||||
func (handler *CloudflareHandler) 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
|
||||
}
|
||||
}()
|
||||
|
||||
for {
|
||||
currentIp, err := godns.GetCurrentIP(handler.Configuration)
|
||||
if err != nil {
|
||||
log.Println("Error in GetCurrentIP:", err)
|
||||
continue
|
||||
}
|
||||
log.Println("Current IP is:", currentIp)
|
||||
// TODO: check against locally cached IP, if no change, skip update
|
||||
|
||||
log.Println("Checking IP for domain", domain.DomainName)
|
||||
zoneId := handler.getZone(domain.DomainName)
|
||||
if zoneId != "" {
|
||||
records := handler.getDNSRecords(zoneId)
|
||||
|
||||
// update records
|
||||
for _, rec := range records {
|
||||
if recordTracked(domain, &rec) != true {
|
||||
log.Println("Skiping record:", rec.Name)
|
||||
continue
|
||||
}
|
||||
if rec.Ip != currentIp {
|
||||
log.Printf("IP mismatch: Current(%+v) vs Cloudflare(%+v)\r\n", currentIp, rec.Ip)
|
||||
handler.updateRecord(rec, currentIp)
|
||||
} else {
|
||||
log.Printf("Record OK: %+v - %+v\r\n", rec.Name, rec.Ip)
|
||||
}
|
||||
}
|
||||
} else {
|
||||
log.Println("Failed to find zone for domain:", domain.DomainName)
|
||||
}
|
||||
|
||||
// Interval is 5 minutes
|
||||
log.Printf("Going to sleep, will start next checking in %d minutes...\r\n", godns.INTERVAL)
|
||||
time.Sleep(time.Minute * godns.INTERVAL)
|
||||
}
|
||||
}
|
||||
|
||||
// Check if record is present in domain conf
|
||||
func recordTracked(domain *godns.Domain, record *DNSRecord) bool {
|
||||
|
||||
if record.Name == domain.DomainName {
|
||||
return true
|
||||
}
|
||||
|
||||
for _, subDomain := range domain.SubDomains {
|
||||
sd := subDomain + "." + domain.DomainName
|
||||
if record.Name == sd {
|
||||
return true
|
||||
}
|
||||
}
|
||||
|
||||
return false
|
||||
}
|
||||
|
||||
// Create a new request with auth in place and optional proxy
|
||||
func (handler *CloudflareHandler) newRequest(method, url string, body io.Reader) (*http.Request, *http.Client) {
|
||||
client := &http.Client{}
|
||||
|
||||
if handler.Configuration.Socks5Proxy != "" {
|
||||
log.Println("use socks5 proxy:" + handler.Configuration.Socks5Proxy)
|
||||
dialer, err := proxy.SOCKS5("tcp", handler.Configuration.Socks5Proxy, nil, proxy.Direct)
|
||||
if err != nil {
|
||||
log.Println("can't connect to the proxy:", err)
|
||||
} else {
|
||||
httpTransport := &http.Transport{}
|
||||
client.Transport = httpTransport
|
||||
httpTransport.Dial = dialer.Dial
|
||||
}
|
||||
}
|
||||
|
||||
req, _ := http.NewRequest(method, handler.API+url, body)
|
||||
req.Header.Set("Content-Type", "application/json")
|
||||
req.Header.Set("X-Auth-Email", handler.Configuration.Email)
|
||||
req.Header.Set("X-Auth-Key", handler.Configuration.Password)
|
||||
return req, client
|
||||
}
|
||||
|
||||
// Find the correct zone via domain name
|
||||
func (handler *CloudflareHandler) getZone(domain string) string {
|
||||
|
||||
var z ZoneResponse
|
||||
|
||||
req, client := handler.newRequest("GET", "/zones", nil)
|
||||
resp, err := client.Do(req)
|
||||
if err != nil {
|
||||
log.Println("Request error:", err.Error())
|
||||
return ""
|
||||
}
|
||||
|
||||
body, _ := ioutil.ReadAll(resp.Body)
|
||||
err = json.Unmarshal(body, &z)
|
||||
if err != nil {
|
||||
log.Printf("Decoder error: %+v\n", err)
|
||||
log.Printf("Response body: %+v\n", string(body))
|
||||
return ""
|
||||
}
|
||||
if z.Success != true {
|
||||
log.Printf("Response failed: %+v\n", string(body))
|
||||
return ""
|
||||
}
|
||||
|
||||
for _, zone := range z.Zones {
|
||||
if zone.Name == domain {
|
||||
return zone.Id
|
||||
}
|
||||
}
|
||||
return ""
|
||||
}
|
||||
|
||||
// Get all DNS A records for a zone
|
||||
func (handler *CloudflareHandler) getDNSRecords(zoneId string) []DNSRecord {
|
||||
|
||||
var empty []DNSRecord
|
||||
var r DNSRecordResponse
|
||||
|
||||
req, client := handler.newRequest("GET", "/zones/"+zoneId+"/dns_records?type=A", nil)
|
||||
resp, err := client.Do(req)
|
||||
if err != nil {
|
||||
log.Println("Request error:", err.Error())
|
||||
return empty
|
||||
}
|
||||
|
||||
body, _ := ioutil.ReadAll(resp.Body)
|
||||
err = json.Unmarshal(body, &r)
|
||||
if err != nil {
|
||||
log.Printf("Decoder error: %+v\n", err)
|
||||
log.Printf("Response body: %+v\n", string(body))
|
||||
return empty
|
||||
}
|
||||
if r.Success != true {
|
||||
body, _ := ioutil.ReadAll(resp.Body)
|
||||
log.Printf("Response failed: %+v\n", string(body))
|
||||
return empty
|
||||
|
||||
}
|
||||
return r.Records
|
||||
}
|
||||
|
||||
// Update DNS A Record with new IP
|
||||
func (handler *CloudflareHandler) updateRecord(record DNSRecord, newIp string) {
|
||||
|
||||
var r DNSRecordUpdateResponse
|
||||
record.SetIp(newIp)
|
||||
|
||||
j, _ := json.Marshal(record)
|
||||
req, client := handler.newRequest("PUT",
|
||||
"/zones/"+record.ZoneId+"/dns_records/"+record.Id,
|
||||
bytes.NewBuffer(j),
|
||||
)
|
||||
resp, err := client.Do(req)
|
||||
if err != nil {
|
||||
log.Println("Request error:", err.Error())
|
||||
return
|
||||
}
|
||||
|
||||
body, _ := ioutil.ReadAll(resp.Body)
|
||||
err = json.Unmarshal(body, &r)
|
||||
if err != nil {
|
||||
log.Printf("Decoder error: %+v\n", err)
|
||||
log.Printf("Response body: %+v\n", string(body))
|
||||
return
|
||||
}
|
||||
if r.Success != true {
|
||||
body, _ := ioutil.ReadAll(resp.Body)
|
||||
log.Printf("Response failed: %+v\n", string(body))
|
||||
} else {
|
||||
log.Printf("Record updated: %+v - %+v", record.Name, record.Ip)
|
||||
}
|
||||
}
|
||||
167
handler/cloudflare_handler_test.go
Normal file
167
handler/cloudflare_handler_test.go
Normal file
@@ -0,0 +1,167 @@
|
||||
package handler
|
||||
|
||||
import (
|
||||
"encoding/json"
|
||||
"strings"
|
||||
"testing"
|
||||
|
||||
"github.com/TimothyYe/godns"
|
||||
)
|
||||
|
||||
func TestResponseToJSON(t *testing.T) {
|
||||
s := strings.NewReader(`
|
||||
{
|
||||
"errors": [],
|
||||
"messages": [],
|
||||
"result": [
|
||||
{
|
||||
"id": "mk2b6fa491c12445a4376666a32429e1",
|
||||
"name": "example.com",
|
||||
"status": "active"
|
||||
}
|
||||
],
|
||||
"result_info": {
|
||||
"count": 1,
|
||||
"page": 1,
|
||||
"per_page": 20,
|
||||
"total_count": 1,
|
||||
"total_pages": 1
|
||||
},
|
||||
"success": true
|
||||
}`)
|
||||
|
||||
var resp ZoneResponse
|
||||
err := json.NewDecoder(s).Decode(&resp)
|
||||
if err != nil {
|
||||
t.Error(err.Error())
|
||||
}
|
||||
if resp.Success != true {
|
||||
t.Errorf("Success Error: %#v != true ", resp.Success)
|
||||
}
|
||||
if resp.Zones[0].Id != "mk2b6fa491c12445a4376666a32429e1" {
|
||||
t.Errorf("ID Error: %#v != mk2b6fa491c12445a4376666a32429e1 ", resp.Zones[0].Id)
|
||||
}
|
||||
if resp.Zones[0].Name != "example.com" {
|
||||
t.Errorf("Name Error: %#v != example.com", resp.Zones[0].Name)
|
||||
}
|
||||
}
|
||||
|
||||
func TestDNSResponseToJSON(t *testing.T) {
|
||||
s := strings.NewReader(`
|
||||
{
|
||||
"errors": [],
|
||||
"messages": [],
|
||||
"result": [
|
||||
{
|
||||
"content": "127.0.0.1",
|
||||
"id": "F11cc63e02a42d38174b8e7c548a7b6f",
|
||||
"name": "example.com",
|
||||
"type": "A",
|
||||
"zone_id": "mk2b6fa491c12445a4376666a32429e1",
|
||||
"zone_name": "example.com"
|
||||
}
|
||||
],
|
||||
"success": true
|
||||
}`)
|
||||
|
||||
var resp DNSRecordResponse
|
||||
err := json.NewDecoder(s).Decode(&resp)
|
||||
if err != nil {
|
||||
t.Error(err.Error())
|
||||
}
|
||||
if resp.Success != true {
|
||||
t.Errorf("Success Error: %#v != true ", resp.Success)
|
||||
}
|
||||
if resp.Records[0].Id != "F11cc63e02a42d38174b8e7c548a7b6f" {
|
||||
t.Errorf("ID Error: %#v != F11cc63e02a42d38174b8e7c548a7b6f ", resp.Records[0].Id)
|
||||
}
|
||||
if resp.Records[0].Name != "example.com" {
|
||||
t.Errorf("Name Error: %#v != example.com", resp.Records[0].Name)
|
||||
}
|
||||
}
|
||||
func TestDNSUpdateResponseToJSON(t *testing.T) {
|
||||
s := strings.NewReader(`
|
||||
{
|
||||
"result": {
|
||||
"id": "F11cc63e02a42d38174b8e7c548a7b6f",
|
||||
"type": "A",
|
||||
"name": "example.com",
|
||||
"content": "127.0.0.1",
|
||||
"proxiable": true,
|
||||
"proxied": true,
|
||||
"ttl": 1,
|
||||
"locked": false,
|
||||
"zone_id": "mk2b6fa491c12445a4376666a32429e1",
|
||||
"zone_name": "example.com",
|
||||
"modified_on": "2018-10-12T14:29:53.205191Z",
|
||||
"created_on": "2018-10-12T14:29:53.205191Z",
|
||||
"meta": {
|
||||
"auto_added": false,
|
||||
"managed_by_apps": false,
|
||||
"managed_by_argo_tunnel": false
|
||||
}
|
||||
},
|
||||
"success": true,
|
||||
"errors": [],
|
||||
"messages": []
|
||||
}`)
|
||||
|
||||
var resp DNSRecordUpdateResponse
|
||||
err := json.NewDecoder(s).Decode(&resp)
|
||||
if err != nil {
|
||||
t.Error(err.Error())
|
||||
}
|
||||
if resp.Success != true {
|
||||
t.Errorf("Success Error: %#v != true ", resp.Success)
|
||||
}
|
||||
if resp.Record.Id != "F11cc63e02a42d38174b8e7c548a7b6f" {
|
||||
t.Errorf("ID Error: %#v != F11cc63e02a42d38174b8e7c548a7b6f ", resp.Record.Id)
|
||||
}
|
||||
if resp.Record.Name != "example.com" {
|
||||
t.Errorf("Name Error: %#v != example.com", resp.Record.Name)
|
||||
}
|
||||
}
|
||||
|
||||
func TestRecordTracked(t *testing.T) {
|
||||
s := strings.NewReader(`
|
||||
{
|
||||
"errors": [],
|
||||
"messages": [],
|
||||
"result": [
|
||||
{
|
||||
"content": "127.0.0.1",
|
||||
"id": "F11cc63e02a42d38174b8e7c548a7b6f",
|
||||
"name": "example.com",
|
||||
"type": "A",
|
||||
"zone_id": "mk2b6fa491c12445a4376666a32429e1",
|
||||
"zone_name": "example.com"
|
||||
},
|
||||
{
|
||||
"content": "127.0.0.1",
|
||||
"id": "G00cc63e02a42d38174b8e7c548a7b6f",
|
||||
"name": "www.example.com",
|
||||
"type": "A",
|
||||
"zone_id": "mk2b6fa491c12445a4376666a32429e1",
|
||||
"zone_name": "www.example.com"
|
||||
}
|
||||
],
|
||||
"success": true
|
||||
}`)
|
||||
|
||||
var resp DNSRecordResponse
|
||||
err := json.NewDecoder(s).Decode(&resp)
|
||||
if err != nil {
|
||||
t.Error(err.Error())
|
||||
}
|
||||
|
||||
domain := &godns.Domain{
|
||||
DomainName: "example.com",
|
||||
SubDomains: []string{"www"},
|
||||
}
|
||||
|
||||
for _, rec := range resp.Records {
|
||||
if recordTracked(domain, &rec) != true {
|
||||
t.Errorf("invalid record skip: %+v\r\n", rec.Name)
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -13,6 +13,8 @@ func CreateHandler(provider string) IHandler {
|
||||
var handler IHandler
|
||||
|
||||
switch provider {
|
||||
case godns.CLOUDFLARE:
|
||||
handler = IHandler(&CloudflareHandler{})
|
||||
case godns.DNSPOD:
|
||||
handler = IHandler(&DNSPodHandler{})
|
||||
case godns.HE:
|
||||
|
||||
Reference in New Issue
Block a user