2022-01-06 01:09:46 +08:00
|
|
|
package packetframe
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/StackExchange/dnscontrol/v3/models"
|
|
|
|
"github.com/StackExchange/dnscontrol/v3/pkg/diff"
|
2022-12-12 04:02:58 +08:00
|
|
|
"github.com/StackExchange/dnscontrol/v3/pkg/diff2"
|
2022-01-06 01:09:46 +08:00
|
|
|
"github.com/StackExchange/dnscontrol/v3/providers"
|
|
|
|
)
|
|
|
|
|
|
|
|
// packetframeProvider is the handle for this provider.
|
|
|
|
type packetframeProvider struct {
|
|
|
|
client *http.Client
|
|
|
|
baseURL *url.URL
|
|
|
|
token string
|
|
|
|
domainIndex map[string]zone
|
|
|
|
}
|
|
|
|
|
|
|
|
// newPacketframe creates the provider.
|
|
|
|
func newPacketframe(m map[string]string, metadata json.RawMessage) (providers.DNSServiceProvider, error) {
|
|
|
|
if m["token"] == "" {
|
|
|
|
return nil, fmt.Errorf("missing Packetframe token")
|
|
|
|
}
|
|
|
|
|
|
|
|
baseURL, err := url.Parse(defaultBaseURL)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("invalid base URL for Packetframe")
|
|
|
|
}
|
|
|
|
client := http.Client{}
|
|
|
|
|
|
|
|
api := &packetframeProvider{client: &client, baseURL: baseURL, token: m["token"]}
|
|
|
|
|
|
|
|
return api, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var features = providers.DocumentationNotes{
|
2022-03-03 00:19:15 +08:00
|
|
|
providers.CanGetZones: providers.Unimplemented(),
|
|
|
|
providers.CanUsePTR: providers.Can(),
|
|
|
|
providers.CanUseSRV: providers.Can(),
|
2022-01-06 01:09:46 +08:00
|
|
|
providers.DocDualHost: providers.Cannot(),
|
|
|
|
providers.DocOfficiallySupported: providers.Cannot(),
|
|
|
|
}
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
fns := providers.DspFuncs{
|
|
|
|
Initializer: newPacketframe,
|
|
|
|
RecordAuditor: AuditRecords,
|
|
|
|
}
|
|
|
|
providers.RegisterDomainServiceProviderType("PACKETFRAME", fns, features)
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetNameservers returns the nameservers for a domain.
|
|
|
|
func (api *packetframeProvider) GetNameservers(domain string) ([]*models.Nameserver, error) {
|
|
|
|
return models.ToNameservers(defaultNameServerNames)
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetZoneRecords gets the records of a zone and returns them in RecordConfig format.
|
|
|
|
func (api *packetframeProvider) GetZoneRecords(domain string) (models.Records, error) {
|
|
|
|
|
|
|
|
if api.domainIndex == nil {
|
|
|
|
if err := api.fetchDomainList(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
zone, ok := api.domainIndex[domain+"."]
|
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("%q not a zone in Packetframe account", domain)
|
|
|
|
}
|
|
|
|
|
|
|
|
records, err := api.getRecords(zone.ID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not load records for domain %q", domain)
|
|
|
|
}
|
|
|
|
|
|
|
|
existingRecords := make([]*models.RecordConfig, len(records))
|
|
|
|
|
|
|
|
dc := models.DomainConfig{
|
|
|
|
Name: domain,
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := range records {
|
|
|
|
existingRecords[i] = toRc(&dc, &records[i])
|
|
|
|
}
|
|
|
|
|
|
|
|
return existingRecords, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetDomainCorrections returns the corrections for a domain.
|
|
|
|
func (api *packetframeProvider) GetDomainCorrections(dc *models.DomainConfig) ([]*models.Correction, error) {
|
|
|
|
dc, err := dc.Copy()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
dc.Punycode()
|
|
|
|
|
|
|
|
if api.domainIndex == nil {
|
|
|
|
if err := api.fetchDomainList(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
zone, ok := api.domainIndex[dc.Name+"."]
|
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("no such zone %q in Packetframe account", dc.Name)
|
|
|
|
}
|
|
|
|
|
|
|
|
records, err := api.getRecords(zone.ID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not load records for domain %q", dc.Name)
|
|
|
|
}
|
|
|
|
|
|
|
|
existingRecords := make([]*models.RecordConfig, len(records))
|
|
|
|
|
|
|
|
for i := range records {
|
|
|
|
existingRecords[i] = toRc(dc, &records[i])
|
|
|
|
}
|
|
|
|
|
|
|
|
// Normalize
|
|
|
|
models.PostProcessRecords(existingRecords)
|
|
|
|
|
|
|
|
var corrections []*models.Correction
|
2023-02-28 08:44:23 +08:00
|
|
|
var create, dels, modify diff.Changeset
|
2023-03-14 02:19:19 +08:00
|
|
|
var differ diff.Differ
|
2023-02-28 08:44:23 +08:00
|
|
|
if !diff2.EnableDiff2 {
|
2023-03-14 02:19:19 +08:00
|
|
|
differ = diff.New(dc)
|
2023-02-28 08:44:23 +08:00
|
|
|
} else {
|
2023-03-14 02:19:19 +08:00
|
|
|
differ = diff.NewCompat(dc)
|
2023-02-28 08:44:23 +08:00
|
|
|
}
|
2023-03-14 02:19:19 +08:00
|
|
|
_, create, dels, modify, err = differ.IncrementalDiff(existingRecords)
|
2023-02-28 08:44:23 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, m := range create {
|
|
|
|
req, err := toReq(zone.ID, dc, m.Desired)
|
2022-01-06 01:09:46 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-02-28 08:44:23 +08:00
|
|
|
corr := &models.Correction{
|
|
|
|
Msg: m.String(),
|
|
|
|
F: func() error {
|
|
|
|
_, err := api.createRecord(req)
|
|
|
|
return err
|
|
|
|
},
|
|
|
|
}
|
|
|
|
corrections = append(corrections, corr)
|
|
|
|
}
|
2022-01-06 01:09:46 +08:00
|
|
|
|
2023-02-28 08:44:23 +08:00
|
|
|
for _, m := range dels {
|
|
|
|
original := m.Existing.Original.(*domainRecord)
|
|
|
|
if original.ID == "0" { // Skip the default nameservers
|
|
|
|
continue
|
2022-01-07 22:24:24 +08:00
|
|
|
}
|
|
|
|
|
2023-02-28 08:44:23 +08:00
|
|
|
corr := &models.Correction{
|
|
|
|
Msg: m.String(),
|
|
|
|
F: func() error {
|
|
|
|
err := api.deleteRecord(zone.ID, original.ID)
|
|
|
|
return err
|
|
|
|
},
|
2022-01-06 01:09:46 +08:00
|
|
|
}
|
2023-02-28 08:44:23 +08:00
|
|
|
corrections = append(corrections, corr)
|
|
|
|
}
|
2022-01-06 01:09:46 +08:00
|
|
|
|
2023-02-28 08:44:23 +08:00
|
|
|
for _, m := range modify {
|
|
|
|
original := m.Existing.Original.(*domainRecord)
|
|
|
|
if original.ID == "0" { // Skip the default nameservers
|
|
|
|
continue
|
2022-01-07 22:24:24 +08:00
|
|
|
}
|
|
|
|
|
2023-02-28 08:44:23 +08:00
|
|
|
req, _ := toReq(zone.ID, dc, m.Desired)
|
|
|
|
req.ID = original.ID
|
|
|
|
corr := &models.Correction{
|
|
|
|
Msg: m.String(),
|
|
|
|
F: func() error {
|
|
|
|
err := api.modifyRecord(req)
|
|
|
|
return err
|
|
|
|
},
|
|
|
|
}
|
|
|
|
corrections = append(corrections, corr)
|
2022-01-06 01:09:46 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return corrections, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func toReq(zoneID string, dc *models.DomainConfig, rc *models.RecordConfig) (*domainRecord, error) {
|
|
|
|
req := &domainRecord{
|
|
|
|
Type: rc.Type,
|
|
|
|
TTL: int(rc.TTL),
|
|
|
|
Label: rc.GetLabel(),
|
|
|
|
Zone: zoneID,
|
|
|
|
}
|
|
|
|
|
|
|
|
switch rc.Type { // #rtype_variations
|
|
|
|
case "A", "AAAA", "PTR", "TXT", "CNAME", "NS":
|
|
|
|
req.Value = rc.GetTargetField()
|
|
|
|
case "MX":
|
|
|
|
req.Value = fmt.Sprintf("%d %s", rc.MxPreference, rc.GetTargetField())
|
|
|
|
case "SRV":
|
|
|
|
req.Value = fmt.Sprintf("%d %d %d %s", rc.SrvPriority, rc.SrvWeight, rc.SrvPort, rc.GetTargetField())
|
|
|
|
default:
|
|
|
|
return nil, fmt.Errorf("packetframe.toReq rtype %q unimplemented", rc.Type)
|
|
|
|
}
|
|
|
|
|
|
|
|
return req, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func toRc(dc *models.DomainConfig, r *domainRecord) *models.RecordConfig {
|
|
|
|
rc := &models.RecordConfig{
|
|
|
|
Type: r.Type,
|
|
|
|
TTL: uint32(r.TTL),
|
|
|
|
Original: r,
|
|
|
|
}
|
|
|
|
|
|
|
|
label := strings.TrimSuffix(r.Label, dc.Name+".")
|
|
|
|
label = strings.TrimSuffix(label, ".")
|
|
|
|
if label == "" {
|
|
|
|
label = "@"
|
|
|
|
}
|
|
|
|
rc.SetLabel(label, dc.Name)
|
|
|
|
|
|
|
|
switch rtype := r.Type; rtype { // #rtype_variations
|
|
|
|
case "TXT":
|
2023-01-14 02:00:49 +08:00
|
|
|
rc.SetTargetTXT(r.Value)
|
2022-01-06 01:09:46 +08:00
|
|
|
case "SRV":
|
|
|
|
spl := strings.Split(r.Value, " ")
|
|
|
|
prio, _ := strconv.ParseUint(spl[0], 10, 16)
|
|
|
|
weight, _ := strconv.ParseUint(spl[1], 10, 16)
|
|
|
|
port, _ := strconv.ParseUint(spl[2], 10, 16)
|
|
|
|
rc.SetTargetSRV(uint16(prio), uint16(weight), uint16(port), spl[3])
|
|
|
|
case "MX":
|
|
|
|
spl := strings.Split(r.Value, " ")
|
|
|
|
prio, _ := strconv.ParseUint(spl[0], 10, 16)
|
|
|
|
rc.SetTargetMX(uint16(prio), spl[1])
|
|
|
|
default:
|
|
|
|
rc.SetTarget(r.Value)
|
|
|
|
}
|
|
|
|
|
|
|
|
return rc
|
|
|
|
}
|