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)
171 lines
4.4 KiB
Go
171 lines
4.4 KiB
Go
package njalla
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"net/http"
|
|
"net/netip"
|
|
"net/url"
|
|
|
|
"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
|
|
host string
|
|
ipVersion ipversion.IPVersion
|
|
ipv6Suffix netip.Prefix
|
|
key string
|
|
useProviderIP bool
|
|
}
|
|
|
|
func New(data json.RawMessage, domain, host string,
|
|
ipVersion ipversion.IPVersion, ipv6Suffix netip.Prefix) (
|
|
p *Provider, err error) {
|
|
extraSettings := struct {
|
|
Key string `json:"key"`
|
|
UseProviderIP bool `json:"provider_ip"`
|
|
}{}
|
|
err = json.Unmarshal(data, &extraSettings)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
p = &Provider{
|
|
domain: domain,
|
|
host: host,
|
|
ipVersion: ipVersion,
|
|
ipv6Suffix: ipv6Suffix,
|
|
key: extraSettings.Key,
|
|
useProviderIP: extraSettings.UseProviderIP,
|
|
}
|
|
err = p.isValid()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return p, nil
|
|
}
|
|
|
|
func (p *Provider) isValid() error {
|
|
if p.key == "" {
|
|
return fmt.Errorf("%w", errors.ErrKeyNotSet)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (p *Provider) String() string {
|
|
return utils.ToString(p.domain, p.host, constants.Njalla, 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://njal.la/\">Njalla</a>",
|
|
IPVersion: p.ipVersion.String(),
|
|
}
|
|
}
|
|
|
|
func (p *Provider) Update(ctx context.Context, client *http.Client, ip netip.Addr) (newIP netip.Addr, err error) {
|
|
u := url.URL{
|
|
Scheme: "https",
|
|
Host: "njal.la",
|
|
Path: "/update",
|
|
}
|
|
values := url.Values{}
|
|
values.Set("h", utils.BuildURLQueryHostname(p.host, p.domain))
|
|
values.Set("k", p.key)
|
|
updatingIP6 := ip.Is6()
|
|
useProviderIP := p.useProviderIP && (ip.Is4() || !p.ipv6Suffix.IsValid())
|
|
if useProviderIP {
|
|
values.Set("auto", "")
|
|
} else {
|
|
if updatingIP6 {
|
|
values.Set("aaaa", ip.String())
|
|
} else {
|
|
values.Set("a", 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)
|
|
headers.SetAccept(request, "application/json")
|
|
|
|
response, err := client.Do(request)
|
|
if err != nil {
|
|
return netip.Addr{}, fmt.Errorf("doing http request: %w", err)
|
|
}
|
|
defer response.Body.Close()
|
|
|
|
decoder := json.NewDecoder(response.Body)
|
|
var respBody struct {
|
|
Message string `json:"message"`
|
|
Value struct {
|
|
A string `json:"A"`
|
|
AAAA string `json:"AAAA"`
|
|
} `json:"value"`
|
|
}
|
|
err = decoder.Decode(&respBody)
|
|
if err != nil {
|
|
return netip.Addr{}, fmt.Errorf("json decoding response body: %w", err)
|
|
}
|
|
|
|
switch response.StatusCode {
|
|
case http.StatusOK:
|
|
if respBody.Message != "record updated" {
|
|
return netip.Addr{}, fmt.Errorf("%w: message received: %s", errors.ErrUnknownResponse, respBody.Message)
|
|
}
|
|
ipString := respBody.Value.A
|
|
if updatingIP6 {
|
|
ipString = respBody.Value.AAAA
|
|
}
|
|
newIP, err = netip.ParseAddr(ipString)
|
|
if err != nil {
|
|
return netip.Addr{}, fmt.Errorf("%w: %w", errors.ErrIPReceivedMalformed, err)
|
|
} else if !useProviderIP && ip.Compare(newIP) != 0 {
|
|
return netip.Addr{}, fmt.Errorf("%w: sent ip %s to update but received %s",
|
|
errors.ErrIPReceivedMismatch, ip, newIP)
|
|
}
|
|
return newIP, nil
|
|
case http.StatusUnauthorized:
|
|
return netip.Addr{}, fmt.Errorf("%w: %s", errors.ErrAuth, respBody.Message)
|
|
case http.StatusInternalServerError:
|
|
return netip.Addr{}, fmt.Errorf("%w: %s", errors.ErrBadRequest, respBody.Message)
|
|
}
|
|
|
|
return netip.Addr{}, fmt.Errorf("%w: %d: %s", errors.ErrHTTPStatusNotValid, response.StatusCode, respBody.Message)
|
|
}
|