package rwth // The documentation is hosted at https://noc-portal.rz.rwth-aachen.de/dns-admin/en/api_tokens and // https://blog.rwth-aachen.de/itc/2022/07/13/api-im-dns-admin/ import ( "encoding/json" "fmt" "io" "net/http" "net/url" "strconv" "strings" "time" "github.com/StackExchange/dnscontrol/v4/models" "github.com/StackExchange/dnscontrol/v4/pkg/printer" ) const ( baseURL = "https://noc-portal.rz.rwth-aachen.de/dns-admin/api/v1" ) // RecordReply represents a DNS Record in an API. type RecordReply struct { ID int `json:"id"` ZoneID int `json:"zone_id"` Type string `json:"type"` Content string `json:"content"` Status string `json:"status"` UpdatedAt time.Time `json:"updated_at"` Editable bool `json:"editable"` } type zone struct { ID int `json:"id"` ZoneName string `json:"zone_name"` Status string `json:"status"` UpdatedAt time.Time `json:"updated_at"` LastDeploy time.Time `json:"last_deploy"` Dnssec struct { ZoneSigningKey struct { CreatedAt time.Time `json:"created_at"` } `json:"zone_signing_key"` KeySigningKey struct { CreatedAt time.Time `json:"created_at"` } `json:"key_signing_key"` } `json:"dnssec"` } func checkIsLockedSystemAPIRecord(record RecordReply) error { if record.Type == "soa_record" { // The upload of a BIND zone file can change the SOA record. // Implementing this edge case this is too complex for now. return fmt.Errorf("SOA records are locked in RWTH zones. They are hence not available for updating") } return nil } func checkIsLockedSystemRecord(record *models.RecordConfig) error { if record.Type == "SOA" { // The upload of a BIND zone file can change the SOA record. // Implementing this edge case this is too complex for now. return fmt.Errorf("SOA records are locked in RWTH zones. They are hence not available for updating") } return nil } func (api *rwthProvider) createRecord(domain string, record *models.RecordConfig) error { if err := checkIsLockedSystemRecord(record); err != nil { return err } req := url.Values{} req.Set("record_content", api.printRecConfig(*record)) return api.request("/create_record", "POST", req, nil) } func (api *rwthProvider) destroyRecord(record RecordReply) error { if err := checkIsLockedSystemAPIRecord(record); err != nil { return err } req := url.Values{} req.Set("record_id", strconv.Itoa(record.ID)) return api.request("/destroy_record", "DELETE", req, nil) } func (api *rwthProvider) updateRecord(id int, record models.RecordConfig) error { if err := checkIsLockedSystemRecord(&record); err != nil { return err } req := url.Values{} req.Set("record_id", strconv.Itoa(id)) req.Set("record_content", api.printRecConfig(record)) return api.request("/update_record", "POST", req, nil) } func (api *rwthProvider) getAllRecords(domain string) ([]models.RecordConfig, error) { zone, err := api.getZone(domain) if err != nil { return nil, err } records := make([]models.RecordConfig, 0) response := []RecordReply{} request := url.Values{} request.Set("zone_id", strconv.Itoa(zone.ID)) if err := api.request("/list_records", "GET", request, &response); err != nil { return nil, fmt.Errorf("failed fetching zone records for %q: %w", domain, err) } for _, apiRecord := range response { if checkIsLockedSystemAPIRecord(apiRecord) != nil { continue } dnsRec, err := NewRR(apiRecord.Content) // Parse content as DNS record if err != nil { return nil, err } recConfig, err := models.RRtoRC(dnsRec, domain) // and make it a RC if err != nil { return nil, err } recConfig.Original = apiRecord // but keep our ApiRecord as the original records = append(records, recConfig) } return records, nil } func (api *rwthProvider) getAllZones() error { if api.zones != nil { return nil } zones := map[string]zone{} response := &[]zone{} if err := api.request("/list_zones", "GET", url.Values{}, response); err != nil { return fmt.Errorf("failed fetching zones: %w", err) } for _, zone := range *response { zones[zone.ZoneName] = zone } api.zones = zones return nil } func (api *rwthProvider) getZone(name string) (*zone, error) { if err := api.getAllZones(); err != nil { return nil, err } zone, ok := api.zones[name] if !ok { return nil, fmt.Errorf("%q is not a zone in this RWTH account", name) } return &zone, nil } // Deploy the zone func (api *rwthProvider) deployZone(domain string) error { zone, err := api.getZone(domain) if err != nil { return err } req := url.Values{} req.Set("zone_id", strconv.Itoa(zone.ID)) return api.request("/deploy_zone", "POST", req, nil) } // Send a request func (api *rwthProvider) request(endpoint string, method string, request url.Values, target interface{}) error { requestBody := strings.NewReader(request.Encode()) req, err := http.NewRequest(method, baseURL+endpoint, requestBody) if err != nil { return err } req.Header.Add("PRIVATE-TOKEN", api.apiToken) req.Header.Add("Content-Type", "application/x-www-form-urlencoded") resp, err := http.DefaultClient.Do(req) if err != nil { return err } cleanupResponseBody := func() { err := resp.Body.Close() if err != nil { printer.Printf("failed closing response body: %q\n", err) } } defer cleanupResponseBody() if resp.StatusCode != http.StatusOK { data, _ := io.ReadAll(resp.Body) printer.Printf(string(data)) return fmt.Errorf("bad status code from RWTH: %d not 200", resp.StatusCode) } if target == nil { return nil } decoder := json.NewDecoder(resp.Body) return decoder.Decode(target) }