mirror of
https://github.com/qdm12/ddns-updater.git
synced 2026-04-05 08:54:09 -04:00
- Remove `IPV6_PREFIX` environment variable (unneeded) and remove associated code - Update all documentation for each provider supporting IPv6 - Build IPv6 as prefix:suffix when getting it from a public IP source for each record IPv6 suffix parameter - Automatically disable provider_ip if public ip is IPv6 and IPv6 suffix is set (they are not compatible with each other)
152 lines
3.4 KiB
Go
152 lines
3.4 KiB
Go
package ionos
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"net/http"
|
|
"net/netip"
|
|
|
|
"github.com/qdm12/ddns-updater/internal/models"
|
|
"github.com/qdm12/ddns-updater/internal/provider/constants"
|
|
"github.com/qdm12/ddns-updater/internal/provider/errors"
|
|
"github.com/qdm12/ddns-updater/internal/provider/utils"
|
|
"github.com/qdm12/ddns-updater/pkg/publicip/ipversion"
|
|
)
|
|
|
|
type Provider struct {
|
|
domain string
|
|
host string
|
|
ipVersion ipversion.IPVersion
|
|
ipv6Suffix netip.Prefix
|
|
apiKey string
|
|
}
|
|
|
|
func New(data json.RawMessage, domain, host string,
|
|
ipVersion ipversion.IPVersion, ipv6Suffix netip.Prefix) (
|
|
p *Provider, err error) {
|
|
extraSettings := struct {
|
|
APIKey string `json:"api_key"`
|
|
}{}
|
|
if err := json.Unmarshal(data, &extraSettings); err != nil {
|
|
return nil, fmt.Errorf("decoding ionos extra settings: %w", err)
|
|
}
|
|
p = &Provider{
|
|
domain: domain,
|
|
host: host,
|
|
ipVersion: ipVersion,
|
|
ipv6Suffix: ipv6Suffix,
|
|
apiKey: extraSettings.APIKey,
|
|
}
|
|
if err := p.isValid(); err != nil {
|
|
return nil, err
|
|
}
|
|
return p, nil
|
|
}
|
|
|
|
func (p *Provider) isValid() error {
|
|
if p.apiKey == "" {
|
|
return fmt.Errorf("%w", errors.ErrTokenNotSet)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (p *Provider) String() string {
|
|
return utils.ToString(p.domain, p.host, constants.Ionos, p.ipVersion)
|
|
}
|
|
|
|
func (p *Provider) Domain() string {
|
|
return p.domain
|
|
}
|
|
|
|
func (p *Provider) Host() string {
|
|
return p.host
|
|
}
|
|
|
|
func (p *Provider) IPVersion() ipversion.IPVersion {
|
|
return p.ipVersion
|
|
}
|
|
|
|
func (p *Provider) IPv6Suffix() netip.Prefix {
|
|
return p.ipv6Suffix
|
|
}
|
|
|
|
func (p *Provider) Proxied() bool {
|
|
return false
|
|
}
|
|
|
|
func (p *Provider) BuildDomainName() string {
|
|
return utils.BuildDomainName(p.host, p.domain)
|
|
}
|
|
|
|
func (p *Provider) HTML() models.HTMLRow {
|
|
return models.HTMLRow{
|
|
Domain: fmt.Sprintf("<a href=\"http://%s\">%s</a>", p.BuildDomainName(), p.BuildDomainName()),
|
|
Host: p.Host(),
|
|
Provider: "<a href=\"https://www.ionos.com/\">Ionos</a>",
|
|
IPVersion: p.ipVersion.String(),
|
|
}
|
|
}
|
|
|
|
// See https://developer.hosting.ionos.com/docs/dns
|
|
func (p *Provider) Update(ctx context.Context, client *http.Client, ip netip.Addr) (
|
|
newIP netip.Addr, err error) {
|
|
zones, err := p.getZones(ctx, client)
|
|
if err != nil {
|
|
return netip.Addr{}, fmt.Errorf("getting zones: %w", err)
|
|
}
|
|
|
|
var zoneID string
|
|
for _, zone := range zones {
|
|
if zone.Name == p.domain {
|
|
zoneID = zone.ID
|
|
break
|
|
}
|
|
}
|
|
|
|
if zoneID == "" {
|
|
return netip.Addr{}, fmt.Errorf("%w: in %d zones for domain %s",
|
|
errors.ErrZoneNotFound, len(zones), p.domain)
|
|
}
|
|
|
|
recordType := constants.A
|
|
if ip.Is6() {
|
|
recordType = constants.AAAA
|
|
}
|
|
|
|
records, err := p.getRecords(ctx, client, zoneID, recordType)
|
|
if err != nil {
|
|
return netip.Addr{}, fmt.Errorf("getting records: %w", err)
|
|
}
|
|
|
|
const usualRecordsCount = 1
|
|
matchingRecords := make([]apiRecord, 0, usualRecordsCount)
|
|
fullDomainName := p.BuildDomainName()
|
|
for _, record := range records {
|
|
if record.Name == fullDomainName {
|
|
matchingRecords = append(matchingRecords, record)
|
|
}
|
|
}
|
|
|
|
if len(matchingRecords) == 0 {
|
|
err = p.createRecord(ctx, client, zoneID, ip)
|
|
if err != nil {
|
|
return netip.Addr{}, fmt.Errorf("creating record: %w", err)
|
|
}
|
|
return ip, nil
|
|
}
|
|
|
|
for _, matchingRecord := range matchingRecords {
|
|
if matchingRecord.Content == ip.String() {
|
|
continue // already up to date
|
|
}
|
|
|
|
err = p.updateRecord(ctx, client, zoneID, matchingRecord, ip)
|
|
if err != nil {
|
|
return netip.Addr{}, fmt.Errorf("updating record: %w", err)
|
|
}
|
|
}
|
|
|
|
return ip, nil
|
|
}
|