Files
ddns-updater-qdm12-3/internal/provider/providers/ovh/provider.go
2024-10-20 13:12:33 +00:00

255 lines
6.6 KiB
Go

package ovh
import (
"context"
"encoding/json"
"fmt"
"io"
"net/http"
"net/netip"
"net/url"
"strings"
"time"
"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/headers"
"github.com/qdm12/ddns-updater/internal/provider/utils"
"github.com/qdm12/ddns-updater/pkg/publicip/ipversion"
)
type Provider struct {
domain string
owner string
ipVersion ipversion.IPVersion
ipv6Suffix netip.Prefix
username string
password string
mode string
apiURL *url.URL
appKey string
appSecret string
consumerKey string
timeNow func() time.Time
serverDelta time.Duration
}
func New(data json.RawMessage, domain, owner string,
ipVersion ipversion.IPVersion, ipv6Suffix netip.Prefix) (
p *Provider, err error,
) {
extraSettings := struct {
Username string `json:"username"`
Password string `json:"password"`
Mode string `json:"mode"`
APIEndpoint string `json:"api_endpoint"`
AppKey string `json:"app_key"`
AppSecret string `json:"app_secret"`
ConsumerKey string `json:"consumer_key"`
}{}
err = json.Unmarshal(data, &extraSettings)
if err != nil {
return nil, err
}
apiURL, err := convertShortEndpoint(extraSettings.APIEndpoint)
if err != nil {
return nil, err
}
err = validateSettings(domain, extraSettings.Mode, owner, extraSettings.AppKey,
extraSettings.ConsumerKey, extraSettings.AppSecret, extraSettings.Username, extraSettings.Password)
if err != nil {
return nil, fmt.Errorf("validating provider specific settings: %w", err)
}
return &Provider{
domain: domain,
owner: owner,
ipVersion: ipVersion,
ipv6Suffix: ipv6Suffix,
username: extraSettings.Username,
password: extraSettings.Password,
mode: extraSettings.Mode,
apiURL: apiURL,
appKey: extraSettings.AppKey,
appSecret: extraSettings.AppSecret,
consumerKey: extraSettings.ConsumerKey,
timeNow: time.Now,
}, nil
}
func validateSettings(domain, mode, owner, appKey, consumerKey,
appSecret, username, password string,
) (err error) {
err = utils.CheckDomain(domain)
if err != nil {
return fmt.Errorf("%w: %w", errors.ErrDomainNotValid, err)
}
if mode == "api" {
switch {
case appKey == "":
return fmt.Errorf("%w", errors.ErrAppKeyNotSet)
case consumerKey == "":
return fmt.Errorf("%w", errors.ErrConsumerKeyNotSet)
case appSecret == "":
return fmt.Errorf("%w", errors.ErrSecretNotSet)
}
} else {
switch {
case username == "":
return fmt.Errorf("%w", errors.ErrUsernameNotSet)
case password == "":
return fmt.Errorf("%w", errors.ErrPasswordNotSet)
case owner == "*":
return fmt.Errorf("%w", errors.ErrOwnerWildcard)
}
}
return nil
}
func (p *Provider) String() string {
return utils.ToString(p.domain, p.owner, constants.OVH, p.ipVersion)
}
func (p *Provider) Domain() string {
return p.domain
}
func (p *Provider) Owner() string {
return p.owner
}
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.owner, p.domain)
}
func (p *Provider) HTML() models.HTMLRow {
return models.HTMLRow{
Domain: fmt.Sprintf("<a href=\"http://%s\">%s</a>", p.BuildDomainName(), p.BuildDomainName()),
Owner: p.Owner(),
Provider: "<a href=\"https://www.ovh.com/\">OVH DNS</a>",
IPVersion: p.ipVersion.String(),
}
}
func (p *Provider) updateWithDynHost(ctx context.Context, client *http.Client,
ip netip.Addr,
) (newIP netip.Addr, err error) {
u := url.URL{
Scheme: "https",
User: url.UserPassword(p.username, p.password),
Host: "www.ovh.com",
Path: "/nic/update",
}
values := url.Values{}
values.Set("system", "dyndns")
values.Set("hostname", utils.BuildURLQueryHostname(p.owner, p.domain))
values.Set("myip", ip.String())
u.RawQuery = values.Encode()
request, err := http.NewRequestWithContext(ctx, http.MethodGet, u.String(), nil)
if err != nil {
return netip.Addr{}, fmt.Errorf("creating http request: %w", err)
}
headers.SetUserAgent(request)
response, err := client.Do(request)
if err != nil {
return netip.Addr{}, fmt.Errorf("doing http request: %w", err)
}
defer response.Body.Close()
b, err := io.ReadAll(response.Body)
if err != nil {
return netip.Addr{}, fmt.Errorf("reading response body: %w", err)
}
s := string(b)
if response.StatusCode != http.StatusOK {
return netip.Addr{}, fmt.Errorf("%w: %d: %s", errors.ErrHTTPStatusNotValid, response.StatusCode, s)
}
switch {
case strings.HasPrefix(s, constants.Notfqdn):
return netip.Addr{}, fmt.Errorf("%w", errors.ErrHostnameNotExists)
case strings.HasPrefix(s, "badrequest"):
return netip.Addr{}, fmt.Errorf("%w", errors.ErrBadRequest)
case strings.HasPrefix(s, "nochg"):
return ip, nil
case strings.HasPrefix(s, "good"):
return ip, nil
case strings.HasPrefix(s, "nohost"):
return netip.Addr{}, fmt.Errorf("%w", errors.ErrHostnameNotExists)
default:
return netip.Addr{}, fmt.Errorf("%w: %s", errors.ErrUnknownResponse, s)
}
}
func (p *Provider) updateWithZoneDNS(ctx context.Context, client *http.Client, ip netip.Addr) (
newIP netip.Addr, err error,
) {
ipStr := ip.Unmap().String()
recordType := constants.A
if ip.Is6() {
recordType = constants.AAAA
}
// subDomain filter of the ovh api expect an empty string to get @ record
subDomain := p.owner
if subDomain == "@" {
subDomain = ""
}
timestamp, err := p.getAdjustedUnixTimestamp(ctx, client)
if err != nil {
return netip.Addr{}, fmt.Errorf("obtain adjusted time from OVH: %w", err)
}
recordIDs, err := p.getRecords(ctx, client, recordType, subDomain, timestamp)
if err != nil {
return netip.Addr{}, fmt.Errorf("listing records: %w", err)
}
if len(recordIDs) == 0 {
err = p.createRecord(ctx, client, recordType, subDomain, ipStr, timestamp)
if err != nil {
return netip.Addr{}, fmt.Errorf("creating record: %w", err)
}
} else {
for _, recordID := range recordIDs {
err = p.updateRecord(ctx, client, recordID, ipStr, timestamp)
if err != nil {
return netip.Addr{}, fmt.Errorf("updating record: %w", err)
}
}
}
err = p.refresh(ctx, client, timestamp)
if err != nil {
return netip.Addr{}, fmt.Errorf("refreshing records: %w", err)
}
return ip, nil
}
func (p *Provider) Update(ctx context.Context, client *http.Client, ip netip.Addr) (newIP netip.Addr, err error) {
if p.mode != "api" {
return p.updateWithDynHost(ctx, client, ip)
}
return p.updateWithZoneDNS(ctx, client, ip)
}