mirror of
https://github.com/qdm12/ddns-updater.git
synced 2026-03-31 06:24:00 -04:00
59 lines
1.5 KiB
Go
59 lines
1.5 KiB
Go
package update
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"net/netip"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/qdm12/ddns-updater/pkg/publicip/ipversion"
|
|
)
|
|
|
|
type getIPFunc func(ctx context.Context) (ip netip.Addr, err error)
|
|
|
|
var ErrIPv6NotSupported = errors.New("IPv6 is not supported on this system")
|
|
|
|
func tryAndRepeatGettingIP(ctx context.Context, getIPFunc getIPFunc,
|
|
logger Logger, version ipversion.IPVersion,
|
|
) (ip netip.Addr, err error) {
|
|
const tries = 3
|
|
logMessagePrefix := "obtaining " + version.String() + " address"
|
|
errs := make([]error, 0, tries)
|
|
for try := range tries {
|
|
ip, err = getIPFunc(ctx)
|
|
if err != nil {
|
|
errs = append(errs, err)
|
|
logger.Debug(logMessagePrefix + ": try " + strconv.Itoa(try+1) + " of " +
|
|
strconv.Itoa(tries) + " failed: " + err.Error())
|
|
continue
|
|
} else if try == 0 {
|
|
return ip, nil
|
|
}
|
|
|
|
tryWord := "try"
|
|
if try > 1 {
|
|
tryWord = "tries"
|
|
}
|
|
logger.Info(logMessagePrefix + " succeeded after " +
|
|
strconv.Itoa(try) + " failed " + tryWord)
|
|
return ip, nil
|
|
}
|
|
|
|
allErrorsAreIPv6NotSupported := true
|
|
for _, err := range errs {
|
|
const ipv6NotSupportedMessage = "connect: cannot assign requested address"
|
|
if !strings.Contains(err.Error(), ipv6NotSupportedMessage) {
|
|
allErrorsAreIPv6NotSupported = false
|
|
break
|
|
}
|
|
}
|
|
|
|
err = &joinedErrors{errs: errs}
|
|
if allErrorsAreIPv6NotSupported {
|
|
return ip, fmt.Errorf("%w: after %d tries, errors were: %w", ErrIPv6NotSupported, tries, err)
|
|
}
|
|
return ip, fmt.Errorf("%s: after %d tries, errors were: %w", logMessagePrefix, tries, err)
|
|
}
|