2016-08-23 08:31:50 +08:00
|
|
|
package cloudflare
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2019-07-03 00:32:54 +08:00
|
|
|
"io/ioutil"
|
2016-08-23 08:31:50 +08:00
|
|
|
"net/http"
|
2017-05-20 02:15:57 +08:00
|
|
|
"strconv"
|
2018-02-16 01:02:50 +08:00
|
|
|
"strings"
|
|
|
|
"time"
|
2016-08-23 08:31:50 +08:00
|
|
|
|
2020-04-15 04:47:30 +08:00
|
|
|
"github.com/StackExchange/dnscontrol/v3/models"
|
2016-08-23 08:31:50 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2017-05-20 02:15:57 +08:00
|
|
|
baseURL = "https://api.cloudflare.com/client/v4/"
|
|
|
|
zonesURL = baseURL + "zones/"
|
|
|
|
recordsURL = zonesURL + "%s/dns_records/"
|
|
|
|
pageRulesURL = zonesURL + "%s/pagerules/"
|
|
|
|
singlePageRuleURL = pageRulesURL + "%s"
|
|
|
|
singleRecordURL = recordsURL + "%s"
|
2016-08-23 08:31:50 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
// get list of domains for account. Cache so the ids can be looked up from domain name
|
2020-10-26 21:25:30 +08:00
|
|
|
func (c *cloudflareProvider) fetchDomainList() error {
|
2016-08-23 08:31:50 +08:00
|
|
|
c.domainIndex = map[string]string{}
|
2016-12-17 04:10:27 +08:00
|
|
|
c.nameservers = map[string][]string{}
|
2016-08-23 08:31:50 +08:00
|
|
|
page := 1
|
|
|
|
for {
|
|
|
|
zr := &zoneResponse{}
|
|
|
|
url := fmt.Sprintf("%s?page=%d&per_page=50", zonesURL, page)
|
|
|
|
if err := c.get(url, zr); err != nil {
|
2020-08-31 07:52:37 +08:00
|
|
|
return fmt.Errorf("failed fetching domain list from cloudflare: %s", err)
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
if !zr.Success {
|
2020-08-31 07:52:37 +08:00
|
|
|
return fmt.Errorf("failed fetching domain list from cloudflare: %s", stringifyErrors(zr.Errors))
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
for _, zone := range zr.Result {
|
|
|
|
c.domainIndex[zone.Name] = zone.ID
|
2020-08-31 08:38:08 +08:00
|
|
|
c.nameservers[zone.Name] = append(c.nameservers[zone.Name], zone.Nameservers...)
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
ri := zr.ResultInfo
|
|
|
|
if len(zr.Result) == 0 || ri.Page*ri.PerPage >= ri.TotalCount {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
page++
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// get all records for a domain
|
2020-10-26 21:25:30 +08:00
|
|
|
func (c *cloudflareProvider) getRecordsForDomain(id string, domain string) ([]*models.RecordConfig, error) {
|
2016-08-23 08:31:50 +08:00
|
|
|
url := fmt.Sprintf(recordsURL, id)
|
|
|
|
page := 1
|
2017-01-12 03:38:07 +08:00
|
|
|
records := []*models.RecordConfig{}
|
2016-08-23 08:31:50 +08:00
|
|
|
for {
|
|
|
|
reqURL := fmt.Sprintf("%s?page=%d&per_page=100", url, page)
|
|
|
|
var data recordsResponse
|
|
|
|
if err := c.get(reqURL, &data); err != nil {
|
2020-08-31 07:52:37 +08:00
|
|
|
return nil, fmt.Errorf("failed fetching record list from cloudflare: %s", err)
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
if !data.Success {
|
2020-08-31 07:52:37 +08:00
|
|
|
return nil, fmt.Errorf("failed fetching record list cloudflare: %s", stringifyErrors(data.Errors))
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
for _, rec := range data.Result {
|
2018-01-10 01:53:16 +08:00
|
|
|
// fmt.Printf("REC: %+v\n", rec)
|
2018-02-16 01:02:50 +08:00
|
|
|
records = append(records, rec.nativeToRecord(domain))
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
ri := data.ResultInfo
|
|
|
|
if len(data.Result) == 0 || ri.Page*ri.PerPage >= ri.TotalCount {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
page++
|
|
|
|
}
|
|
|
|
return records, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// create a correction to delete a record
|
2020-10-26 21:25:30 +08:00
|
|
|
func (c *cloudflareProvider) deleteRec(rec *cfRecord, domainID string) *models.Correction {
|
2016-08-23 08:31:50 +08:00
|
|
|
return &models.Correction{
|
|
|
|
Msg: fmt.Sprintf("DELETE record: %s %s %d %s (id=%s)", rec.Name, rec.Type, rec.TTL, rec.Content, rec.ID),
|
|
|
|
F: func() error {
|
|
|
|
endpoint := fmt.Sprintf(singleRecordURL, domainID, rec.ID)
|
|
|
|
req, err := http.NewRequest("DELETE", endpoint, nil)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
c.setHeaders(req)
|
|
|
|
_, err = handleActionResponse(http.DefaultClient.Do(req))
|
|
|
|
return err
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
func (c *cloudflareProvider) createZone(domainName string) (string, error) {
|
2017-05-06 05:20:43 +08:00
|
|
|
type createZone struct {
|
|
|
|
Name string `json:"name"`
|
2018-12-19 22:48:27 +08:00
|
|
|
|
|
|
|
Account struct {
|
|
|
|
ID string `json:"id"`
|
|
|
|
Name string `json:"name"`
|
|
|
|
} `json:"account"`
|
2017-05-06 05:20:43 +08:00
|
|
|
}
|
|
|
|
var id string
|
|
|
|
cz := &createZone{
|
|
|
|
Name: domainName}
|
2018-12-19 22:48:27 +08:00
|
|
|
|
|
|
|
if c.AccountID != "" || c.AccountName != "" {
|
|
|
|
cz.Account.ID = c.AccountID
|
|
|
|
cz.Account.Name = c.AccountName
|
|
|
|
}
|
|
|
|
|
2017-05-06 05:20:43 +08:00
|
|
|
buf := &bytes.Buffer{}
|
|
|
|
encoder := json.NewEncoder(buf)
|
|
|
|
if err := encoder.Encode(cz); err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
req, err := http.NewRequest("POST", zonesURL, buf)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
c.setHeaders(req)
|
|
|
|
id, err = handleActionResponse(http.DefaultClient.Do(req))
|
|
|
|
return id, err
|
|
|
|
}
|
|
|
|
|
2020-05-30 22:40:21 +08:00
|
|
|
func cfDSData(rec *models.RecordConfig) *cfRecData {
|
|
|
|
return &cfRecData{
|
|
|
|
KeyTag: rec.DsKeyTag,
|
|
|
|
Algorithm: rec.DsAlgorithm,
|
|
|
|
DigestType: rec.DsDigestType,
|
|
|
|
Digest: rec.DsDigest,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-11 04:02:06 +08:00
|
|
|
func cfSrvData(rec *models.RecordConfig) *cfRecData {
|
2018-02-16 01:02:50 +08:00
|
|
|
serverParts := strings.Split(rec.GetLabelFQDN(), ".")
|
2019-11-15 00:25:20 +08:00
|
|
|
c := &cfRecData{
|
2017-08-11 04:02:06 +08:00
|
|
|
Service: serverParts[0],
|
|
|
|
Proto: serverParts[1],
|
|
|
|
Name: strings.Join(serverParts[2:], "."),
|
|
|
|
Port: rec.SrvPort,
|
|
|
|
Priority: rec.SrvPriority,
|
|
|
|
Weight: rec.SrvWeight,
|
|
|
|
}
|
2019-11-15 00:25:20 +08:00
|
|
|
c.Target = cfTarget(rec.GetTargetField())
|
|
|
|
return c
|
2017-08-11 04:02:06 +08:00
|
|
|
}
|
|
|
|
|
2017-12-20 23:25:23 +08:00
|
|
|
func cfCaaData(rec *models.RecordConfig) *cfRecData {
|
|
|
|
return &cfRecData{
|
|
|
|
Tag: rec.CaaTag,
|
|
|
|
Flags: rec.CaaFlag,
|
2018-02-16 01:02:50 +08:00
|
|
|
Value: rec.GetTargetField(),
|
2017-12-20 23:25:23 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-21 10:32:39 +08:00
|
|
|
func cfTlsaData(rec *models.RecordConfig) *cfRecData {
|
|
|
|
return &cfRecData{
|
2020-06-18 21:37:57 +08:00
|
|
|
Usage: rec.TlsaUsage,
|
|
|
|
Selector: rec.TlsaSelector,
|
|
|
|
MatchingType: rec.TlsaMatchingType,
|
|
|
|
Certificate: rec.GetTargetField(),
|
2019-05-21 10:32:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func cfSshfpData(rec *models.RecordConfig) *cfRecData {
|
|
|
|
return &cfRecData{
|
|
|
|
Algorithm: rec.SshfpAlgorithm,
|
2020-06-18 21:37:57 +08:00
|
|
|
HashType: rec.SshfpFingerprint,
|
2019-05-21 10:32:39 +08:00
|
|
|
Fingerprint: rec.GetTargetField(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
func (c *cloudflareProvider) createRec(rec *models.RecordConfig, domainID string) []*models.Correction {
|
2016-08-23 08:31:50 +08:00
|
|
|
type createRecord struct {
|
2017-08-11 04:02:06 +08:00
|
|
|
Name string `json:"name"`
|
|
|
|
Type string `json:"type"`
|
|
|
|
Content string `json:"content"`
|
|
|
|
TTL uint32 `json:"ttl"`
|
|
|
|
Priority uint16 `json:"priority"`
|
|
|
|
Data *cfRecData `json:"data"`
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
var id string
|
2018-02-16 01:02:50 +08:00
|
|
|
content := rec.GetTargetField()
|
2016-08-23 08:31:50 +08:00
|
|
|
if rec.Metadata[metaOriginalIP] != "" {
|
|
|
|
content = rec.Metadata[metaOriginalIP]
|
|
|
|
}
|
|
|
|
prio := ""
|
|
|
|
if rec.Type == "MX" {
|
2017-07-20 03:53:40 +08:00
|
|
|
prio = fmt.Sprintf(" %d ", rec.MxPreference)
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
2020-05-30 22:40:21 +08:00
|
|
|
if rec.Type == "DS" {
|
|
|
|
content = fmt.Sprintf("%d %d %d %s", rec.DsKeyTag, rec.DsAlgorithm, rec.DsDigestType, rec.DsDigest)
|
|
|
|
}
|
2016-08-23 08:31:50 +08:00
|
|
|
arr := []*models.Correction{{
|
2018-02-16 01:02:50 +08:00
|
|
|
Msg: fmt.Sprintf("CREATE record: %s %s %d%s %s", rec.GetLabel(), rec.Type, rec.TTL, prio, content),
|
2016-08-23 08:31:50 +08:00
|
|
|
F: func() error {
|
|
|
|
|
|
|
|
cf := &createRecord{
|
2018-02-16 01:02:50 +08:00
|
|
|
Name: rec.GetLabel(),
|
2016-08-23 08:31:50 +08:00
|
|
|
Type: rec.Type,
|
|
|
|
TTL: rec.TTL,
|
|
|
|
Content: content,
|
2017-07-20 03:53:40 +08:00
|
|
|
Priority: rec.MxPreference,
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
2017-08-11 04:02:06 +08:00
|
|
|
if rec.Type == "SRV" {
|
|
|
|
cf.Data = cfSrvData(rec)
|
2018-02-16 01:02:50 +08:00
|
|
|
cf.Name = rec.GetLabelFQDN()
|
2017-12-20 23:25:23 +08:00
|
|
|
} else if rec.Type == "CAA" {
|
|
|
|
cf.Data = cfCaaData(rec)
|
2018-02-16 01:02:50 +08:00
|
|
|
cf.Name = rec.GetLabelFQDN()
|
2017-12-20 23:25:23 +08:00
|
|
|
cf.Content = ""
|
2019-05-21 10:32:39 +08:00
|
|
|
} else if rec.Type == "TLSA" {
|
|
|
|
cf.Data = cfTlsaData(rec)
|
|
|
|
cf.Name = rec.GetLabelFQDN()
|
|
|
|
} else if rec.Type == "SSHFP" {
|
|
|
|
cf.Data = cfSshfpData(rec)
|
|
|
|
cf.Name = rec.GetLabelFQDN()
|
2020-05-30 22:40:21 +08:00
|
|
|
} else if rec.Type == "DS" {
|
|
|
|
cf.Data = cfDSData(rec)
|
2017-08-11 04:02:06 +08:00
|
|
|
}
|
2016-08-23 08:31:50 +08:00
|
|
|
endpoint := fmt.Sprintf(recordsURL, domainID)
|
|
|
|
buf := &bytes.Buffer{}
|
|
|
|
encoder := json.NewEncoder(buf)
|
|
|
|
if err := encoder.Encode(cf); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
req, err := http.NewRequest("POST", endpoint, buf)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
c.setHeaders(req)
|
|
|
|
id, err = handleActionResponse(http.DefaultClient.Do(req))
|
|
|
|
return err
|
|
|
|
},
|
|
|
|
}}
|
|
|
|
if rec.Metadata[metaProxy] != "off" {
|
|
|
|
arr = append(arr, &models.Correction{
|
2018-02-16 01:02:50 +08:00
|
|
|
Msg: fmt.Sprintf("ACTIVATE PROXY for new record %s %s %d %s", rec.GetLabel(), rec.Type, rec.TTL, rec.GetTargetField()),
|
2016-08-23 08:31:50 +08:00
|
|
|
F: func() error { return c.modifyRecord(domainID, id, true, rec) },
|
|
|
|
})
|
|
|
|
}
|
|
|
|
return arr
|
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
func (c *cloudflareProvider) modifyRecord(domainID, recID string, proxied bool, rec *models.RecordConfig) error {
|
2016-08-23 08:31:50 +08:00
|
|
|
if domainID == "" || recID == "" {
|
Switch to Go 1.13 error wrapping (#604)
* Replaced errors.Wrap with fmt.Errorf (#589)
* Find: errors\.Wrap\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Replaced errors.Wrapf with fmt.Errorf (#589)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])(,[^)]+)\)
* Replace: fmt.Errorf($2: %w$3$4, $1)
* Replaced errors.Errorf with fmt.Errorf (#589)
* Find: errors\.Errorf
Replace: fmt.Errorf
* Cleaned up remaining imports
* Cleanup
* Regenerate provider support matrix
This was broken by #533 ... and it's now the third time this has been missed.
2020-01-29 00:06:56 +08:00
|
|
|
return fmt.Errorf("cannot modify record if domain or record id are empty")
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
type record struct {
|
2017-08-11 04:02:06 +08:00
|
|
|
ID string `json:"id"`
|
|
|
|
Proxied bool `json:"proxied"`
|
|
|
|
Name string `json:"name"`
|
|
|
|
Type string `json:"type"`
|
|
|
|
Content string `json:"content"`
|
|
|
|
Priority uint16 `json:"priority"`
|
|
|
|
TTL uint32 `json:"ttl"`
|
|
|
|
Data *cfRecData `json:"data"`
|
|
|
|
}
|
2018-02-16 01:02:50 +08:00
|
|
|
r := record{
|
|
|
|
ID: recID,
|
|
|
|
Proxied: proxied,
|
|
|
|
Name: rec.GetLabel(),
|
|
|
|
Type: rec.Type,
|
|
|
|
Content: rec.GetTargetField(),
|
|
|
|
Priority: rec.MxPreference,
|
|
|
|
TTL: rec.TTL,
|
|
|
|
Data: nil,
|
|
|
|
}
|
2017-08-11 04:02:06 +08:00
|
|
|
if rec.Type == "SRV" {
|
|
|
|
r.Data = cfSrvData(rec)
|
2018-02-16 01:02:50 +08:00
|
|
|
r.Name = rec.GetLabelFQDN()
|
2017-12-20 23:25:23 +08:00
|
|
|
} else if rec.Type == "CAA" {
|
|
|
|
r.Data = cfCaaData(rec)
|
2018-02-16 01:02:50 +08:00
|
|
|
r.Name = rec.GetLabelFQDN()
|
2017-12-20 23:25:23 +08:00
|
|
|
r.Content = ""
|
2019-05-21 10:32:39 +08:00
|
|
|
} else if rec.Type == "TLSA" {
|
|
|
|
r.Data = cfTlsaData(rec)
|
|
|
|
r.Name = rec.GetLabelFQDN()
|
|
|
|
} else if rec.Type == "SSHFP" {
|
|
|
|
r.Data = cfSshfpData(rec)
|
|
|
|
r.Name = rec.GetLabelFQDN()
|
2020-05-30 22:40:21 +08:00
|
|
|
} else if rec.Type == "DS" {
|
|
|
|
r.Data = cfDSData(rec)
|
|
|
|
r.Content = ""
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
endpoint := fmt.Sprintf(singleRecordURL, domainID, recID)
|
|
|
|
buf := &bytes.Buffer{}
|
|
|
|
encoder := json.NewEncoder(buf)
|
|
|
|
if err := encoder.Encode(r); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
req, err := http.NewRequest("PUT", endpoint, buf)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
c.setHeaders(req)
|
|
|
|
_, err = handleActionResponse(http.DefaultClient.Do(req))
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-06-13 19:32:54 +08:00
|
|
|
// change universal ssl state
|
2020-10-26 21:25:30 +08:00
|
|
|
func (c *cloudflareProvider) changeUniversalSSL(domainID string, state bool) error {
|
2019-06-13 19:32:54 +08:00
|
|
|
type setUniversalSSL struct {
|
|
|
|
Enabled bool `json:"enabled"`
|
|
|
|
}
|
|
|
|
us := &setUniversalSSL{
|
|
|
|
Enabled: state,
|
|
|
|
}
|
|
|
|
|
|
|
|
// create json
|
|
|
|
buf := &bytes.Buffer{}
|
|
|
|
encoder := json.NewEncoder(buf)
|
|
|
|
if err := encoder.Encode(us); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// send request.
|
|
|
|
endpoint := fmt.Sprintf(zonesURL+"%s/ssl/universal/settings", domainID)
|
|
|
|
req, err := http.NewRequest("PATCH", endpoint, buf)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
c.setHeaders(req)
|
|
|
|
_, err = handleActionResponse(http.DefaultClient.Do(req))
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// change universal ssl state
|
2020-10-26 21:25:30 +08:00
|
|
|
func (c *cloudflareProvider) getUniversalSSL(domainID string) (bool, error) {
|
2019-06-13 19:32:54 +08:00
|
|
|
type universalSSLResponse struct {
|
|
|
|
Success bool `json:"success"`
|
|
|
|
Errors []interface{} `json:"errors"`
|
|
|
|
Messages []interface{} `json:"messages"`
|
|
|
|
Result struct {
|
|
|
|
Enabled bool `json:"enabled"`
|
|
|
|
} `json:"result"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// send request.
|
|
|
|
endpoint := fmt.Sprintf(zonesURL+"%s/ssl/universal/settings", domainID)
|
|
|
|
var result universalSSLResponse
|
|
|
|
err := c.get(endpoint, &result)
|
|
|
|
if err != nil {
|
|
|
|
return true, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return result.Result.Enabled, err
|
|
|
|
}
|
|
|
|
|
2016-08-23 08:31:50 +08:00
|
|
|
// common error handling for all action responses
|
|
|
|
func handleActionResponse(resp *http.Response, err error) (id string, e error) {
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
result := &basicResponse{}
|
|
|
|
decoder := json.NewDecoder(resp.Body)
|
|
|
|
if err = decoder.Decode(result); err != nil {
|
2020-08-31 07:52:37 +08:00
|
|
|
return "", fmt.Errorf("unknown error. Status code: %d", resp.StatusCode)
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
if resp.StatusCode != 200 {
|
Switch to Go 1.13 error wrapping (#604)
* Replaced errors.Wrap with fmt.Errorf (#589)
* Find: errors\.Wrap\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Replaced errors.Wrapf with fmt.Errorf (#589)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])(,[^)]+)\)
* Replace: fmt.Errorf($2: %w$3$4, $1)
* Replaced errors.Errorf with fmt.Errorf (#589)
* Find: errors\.Errorf
Replace: fmt.Errorf
* Cleaned up remaining imports
* Cleanup
* Regenerate provider support matrix
This was broken by #533 ... and it's now the third time this has been missed.
2020-01-29 00:06:56 +08:00
|
|
|
return "", fmt.Errorf(stringifyErrors(result.Errors))
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
return result.Result.ID, nil
|
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
func (c *cloudflareProvider) setHeaders(req *http.Request) {
|
2020-10-19 02:04:50 +08:00
|
|
|
if len(c.APIToken) > 0 {
|
|
|
|
req.Header.Set("Authorization", "Bearer "+c.APIToken)
|
2019-10-23 23:48:00 +08:00
|
|
|
} else {
|
2020-10-19 02:04:50 +08:00
|
|
|
req.Header.Set("X-Auth-Key", c.APIKey)
|
|
|
|
req.Header.Set("X-Auth-Email", c.APIUser)
|
2019-10-23 23:48:00 +08:00
|
|
|
}
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// generic get handler. makes request and unmarshalls response to given interface
|
2020-10-26 21:25:30 +08:00
|
|
|
func (c *cloudflareProvider) get(endpoint string, target interface{}) error {
|
2016-08-23 08:31:50 +08:00
|
|
|
req, err := http.NewRequest("GET", endpoint, nil)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
c.setHeaders(req)
|
|
|
|
resp, err := http.DefaultClient.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
if resp.StatusCode != 200 {
|
2019-07-03 00:32:54 +08:00
|
|
|
dat, _ := ioutil.ReadAll(resp.Body)
|
|
|
|
fmt.Println(string(dat))
|
Switch to Go 1.13 error wrapping (#604)
* Replaced errors.Wrap with fmt.Errorf (#589)
* Find: errors\.Wrap\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Replaced errors.Wrapf with fmt.Errorf (#589)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])(,[^)]+)\)
* Replace: fmt.Errorf($2: %w$3$4, $1)
* Replaced errors.Errorf with fmt.Errorf (#589)
* Find: errors\.Errorf
Replace: fmt.Errorf
* Cleaned up remaining imports
* Cleanup
* Regenerate provider support matrix
This was broken by #533 ... and it's now the third time this has been missed.
2020-01-29 00:06:56 +08:00
|
|
|
return fmt.Errorf("bad status code from cloudflare: %d not 200", resp.StatusCode)
|
2016-08-23 08:31:50 +08:00
|
|
|
}
|
|
|
|
decoder := json.NewDecoder(resp.Body)
|
|
|
|
return decoder.Decode(target)
|
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
func (c *cloudflareProvider) getPageRules(id string, domain string) ([]*models.RecordConfig, error) {
|
2017-05-20 02:15:57 +08:00
|
|
|
url := fmt.Sprintf(pageRulesURL, id)
|
|
|
|
data := pageRuleResponse{}
|
|
|
|
if err := c.get(url, &data); err != nil {
|
2020-08-31 07:52:37 +08:00
|
|
|
return nil, fmt.Errorf("failed fetching page rule list from cloudflare: %s", err)
|
2017-05-20 02:15:57 +08:00
|
|
|
}
|
|
|
|
if !data.Success {
|
2020-08-31 07:52:37 +08:00
|
|
|
return nil, fmt.Errorf("failed fetching page rule list cloudflare: %s", stringifyErrors(data.Errors))
|
2017-05-20 02:15:57 +08:00
|
|
|
}
|
|
|
|
recs := []*models.RecordConfig{}
|
|
|
|
for _, pr := range data.Result {
|
|
|
|
// only interested in forwarding rules. Lets be very specific, and skip anything else
|
|
|
|
if len(pr.Actions) != 1 || len(pr.Targets) != 1 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if pr.Actions[0].ID != "forwarding_url" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
err := json.Unmarshal([]byte(pr.Actions[0].Value), &pr.ForwardingInfo)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
var thisPr = pr
|
2018-02-16 01:02:50 +08:00
|
|
|
r := &models.RecordConfig{
|
2017-05-20 02:15:57 +08:00
|
|
|
Type: "PAGE_RULE",
|
|
|
|
Original: thisPr,
|
|
|
|
TTL: 1,
|
2018-02-16 01:02:50 +08:00
|
|
|
}
|
|
|
|
r.SetLabel("@", domain)
|
|
|
|
r.SetTarget(fmt.Sprintf("%s,%s,%d,%d", // $FROM,$TO,$PRIO,$CODE
|
|
|
|
pr.Targets[0].Constraint.Value,
|
|
|
|
pr.ForwardingInfo.URL,
|
|
|
|
pr.Priority,
|
|
|
|
pr.ForwardingInfo.StatusCode))
|
|
|
|
recs = append(recs, r)
|
2017-05-20 02:15:57 +08:00
|
|
|
}
|
|
|
|
return recs, nil
|
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
func (c *cloudflareProvider) deletePageRule(recordID, domainID string) error {
|
2017-05-20 02:15:57 +08:00
|
|
|
endpoint := fmt.Sprintf(singlePageRuleURL, domainID, recordID)
|
|
|
|
req, err := http.NewRequest("DELETE", endpoint, nil)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
c.setHeaders(req)
|
|
|
|
_, err = handleActionResponse(http.DefaultClient.Do(req))
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
func (c *cloudflareProvider) updatePageRule(recordID, domainID string, target string) error {
|
2017-05-20 02:15:57 +08:00
|
|
|
if err := c.deletePageRule(recordID, domainID); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return c.createPageRule(domainID, target)
|
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
func (c *cloudflareProvider) createPageRule(domainID string, target string) error {
|
2017-05-20 02:15:57 +08:00
|
|
|
endpoint := fmt.Sprintf(pageRulesURL, domainID)
|
|
|
|
return c.sendPageRule(endpoint, "POST", target)
|
|
|
|
}
|
|
|
|
|
2020-10-26 21:25:30 +08:00
|
|
|
func (c *cloudflareProvider) sendPageRule(endpoint, method string, data string) error {
|
2018-01-10 01:53:16 +08:00
|
|
|
// from to priority code
|
2017-05-20 02:15:57 +08:00
|
|
|
parts := strings.Split(data, ",")
|
|
|
|
priority, _ := strconv.Atoi(parts[2])
|
|
|
|
code, _ := strconv.Atoi(parts[3])
|
|
|
|
fwdInfo := &pageRuleFwdInfo{
|
|
|
|
StatusCode: code,
|
|
|
|
URL: parts[1],
|
|
|
|
}
|
|
|
|
dat, _ := json.Marshal(fwdInfo)
|
|
|
|
pr := &pageRule{
|
|
|
|
Status: "active",
|
|
|
|
Priority: priority,
|
|
|
|
Targets: []pageRuleTarget{
|
|
|
|
{Target: "url", Constraint: pageRuleConstraint{Operator: "matches", Value: parts[0]}},
|
|
|
|
},
|
|
|
|
Actions: []pageRuleAction{
|
|
|
|
{ID: "forwarding_url", Value: json.RawMessage(dat)},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
buf := &bytes.Buffer{}
|
|
|
|
enc := json.NewEncoder(buf)
|
|
|
|
if err := enc.Encode(pr); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
req, err := http.NewRequest(method, endpoint, buf)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
c.setHeaders(req)
|
|
|
|
_, err = handleActionResponse(http.DefaultClient.Do(req))
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2016-08-23 08:31:50 +08:00
|
|
|
func stringifyErrors(errors []interface{}) string {
|
|
|
|
dat, err := json.Marshal(errors)
|
|
|
|
if err != nil {
|
|
|
|
return "???"
|
|
|
|
}
|
|
|
|
return string(dat)
|
|
|
|
}
|
|
|
|
|
|
|
|
type recordsResponse struct {
|
|
|
|
basicResponse
|
|
|
|
Result []*cfRecord `json:"result"`
|
|
|
|
ResultInfo pagingInfo `json:"result_info"`
|
|
|
|
}
|
2017-05-20 02:15:57 +08:00
|
|
|
|
2016-08-23 08:31:50 +08:00
|
|
|
type basicResponse struct {
|
|
|
|
Success bool `json:"success"`
|
|
|
|
Errors []interface{} `json:"errors"`
|
|
|
|
Messages []interface{} `json:"messages"`
|
|
|
|
Result struct {
|
|
|
|
ID string `json:"id"`
|
|
|
|
} `json:"result"`
|
|
|
|
}
|
|
|
|
|
2017-05-20 02:15:57 +08:00
|
|
|
type pageRuleResponse struct {
|
|
|
|
basicResponse
|
|
|
|
Result []*pageRule `json:"result"`
|
|
|
|
ResultInfo pagingInfo `json:"result_info"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type pageRule struct {
|
|
|
|
ID string `json:"id,omitempty"`
|
|
|
|
Targets []pageRuleTarget `json:"targets"`
|
|
|
|
Actions []pageRuleAction `json:"actions"`
|
|
|
|
Priority int `json:"priority"`
|
|
|
|
Status string `json:"status"`
|
|
|
|
ModifiedOn time.Time `json:"modified_on,omitempty"`
|
|
|
|
CreatedOn time.Time `json:"created_on,omitempty"`
|
|
|
|
ForwardingInfo *pageRuleFwdInfo `json:"-"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type pageRuleTarget struct {
|
|
|
|
Target string `json:"target"`
|
|
|
|
Constraint pageRuleConstraint `json:"constraint"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type pageRuleConstraint struct {
|
|
|
|
Operator string `json:"operator"`
|
|
|
|
Value string `json:"value"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type pageRuleAction struct {
|
|
|
|
ID string `json:"id"`
|
|
|
|
Value json.RawMessage `json:"value"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type pageRuleFwdInfo struct {
|
|
|
|
URL string `json:"url"`
|
|
|
|
StatusCode int `json:"status_code"`
|
|
|
|
}
|
|
|
|
|
2016-08-23 08:31:50 +08:00
|
|
|
type zoneResponse struct {
|
|
|
|
basicResponse
|
|
|
|
Result []struct {
|
|
|
|
ID string `json:"id"`
|
|
|
|
Name string `json:"name"`
|
|
|
|
Nameservers []string `json:"name_servers"`
|
|
|
|
} `json:"result"`
|
|
|
|
ResultInfo pagingInfo `json:"result_info"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type pagingInfo struct {
|
|
|
|
Page int `json:"page"`
|
|
|
|
PerPage int `json:"per_page"`
|
|
|
|
Count int `json:"count"`
|
|
|
|
TotalCount int `json:"total_count"`
|
|
|
|
}
|