2022-06-09 23:47:12 +00:00
|
|
|
package updater
|
2021-05-08 00:59:42 +00:00
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"context"
|
|
|
|
|
"errors"
|
|
|
|
|
"fmt"
|
2022-06-06 01:57:44 +00:00
|
|
|
"sort"
|
2021-05-08 00:59:42 +00:00
|
|
|
|
2022-04-23 10:59:29 +00:00
|
|
|
"github.com/qdm12/gluetun/internal/constants/vpn"
|
2021-05-08 00:59:42 +00:00
|
|
|
"github.com/qdm12/gluetun/internal/models"
|
2022-05-28 22:02:18 +00:00
|
|
|
"github.com/qdm12/gluetun/internal/provider/common"
|
2021-05-08 00:59:42 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
var (
|
2022-05-28 22:02:18 +00:00
|
|
|
ErrNotIPv4 = errors.New("IP address is not IPv4")
|
2021-05-08 00:59:42 +00:00
|
|
|
)
|
|
|
|
|
|
2022-06-07 16:47:35 +00:00
|
|
|
func (u *Updater) FetchServers(ctx context.Context, minServers int) (
|
2022-05-28 20:58:50 +00:00
|
|
|
servers []models.Server, err error) {
|
2023-06-08 07:39:07 +00:00
|
|
|
const recommended = true
|
|
|
|
|
const limit = 0
|
|
|
|
|
data, err := fetchAPI(ctx, u.client, recommended, limit)
|
2021-05-08 00:59:42 +00:00
|
|
|
if err != nil {
|
2022-05-28 20:58:50 +00:00
|
|
|
return nil, err
|
2021-05-08 00:59:42 +00:00
|
|
|
}
|
|
|
|
|
|
2022-04-16 22:25:36 +02:00
|
|
|
servers = make([]models.Server, 0, len(data))
|
2021-05-08 00:59:42 +00:00
|
|
|
|
|
|
|
|
for _, jsonServer := range data {
|
2023-06-08 07:39:07 +00:00
|
|
|
if jsonServer.Status != "online" {
|
|
|
|
|
u.warner.Warn(fmt.Sprintf("ignoring offline server %s", jsonServer.Name))
|
2021-05-08 00:59:42 +00:00
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
2023-06-08 07:39:07 +00:00
|
|
|
server := models.Server{
|
|
|
|
|
Country: jsonServer.country(),
|
|
|
|
|
Region: jsonServer.region(),
|
|
|
|
|
City: jsonServer.city(),
|
|
|
|
|
Hostname: jsonServer.Hostname,
|
|
|
|
|
IPs: jsonServer.ips(),
|
2021-05-08 00:59:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
number, err := parseServerName(jsonServer.Name)
|
2023-06-08 07:39:07 +00:00
|
|
|
switch {
|
|
|
|
|
case errors.Is(err, ErrNoIDInServerName):
|
|
|
|
|
u.warner.Warn(fmt.Sprintf("%s - leaving server number as 0", err))
|
|
|
|
|
case err != nil:
|
|
|
|
|
u.warner.Warn(fmt.Sprintf("failed parsing server name: %s", err))
|
|
|
|
|
continue
|
|
|
|
|
default: // no error
|
|
|
|
|
server.Number = number
|
2021-05-08 00:59:42 +00:00
|
|
|
}
|
|
|
|
|
|
2023-06-08 07:39:07 +00:00
|
|
|
var wireguardFound, openvpnFound bool
|
|
|
|
|
wireguardServer := server
|
|
|
|
|
wireguardServer.VPN = vpn.Wireguard
|
|
|
|
|
openVPNServer := server // accumulate UDP+TCP technologies
|
|
|
|
|
openVPNServer.VPN = vpn.OpenVPN
|
|
|
|
|
|
|
|
|
|
for _, technology := range jsonServer.Technologies {
|
|
|
|
|
switch technology.Identifier {
|
|
|
|
|
case "openvpn_udp":
|
|
|
|
|
openvpnFound = true
|
|
|
|
|
openVPNServer.UDP = true
|
|
|
|
|
case "openvpn_tcp":
|
|
|
|
|
openvpnFound = true
|
|
|
|
|
openVPNServer.TCP = true
|
|
|
|
|
case "wireguard_udp":
|
|
|
|
|
wireguardFound = true
|
|
|
|
|
wireguardServer.WgPubKey, err = jsonServer.wireguardPublicKey()
|
|
|
|
|
if err != nil {
|
|
|
|
|
u.warner.Warn(fmt.Sprintf("ignoring Wireguard server %s: %s",
|
|
|
|
|
jsonServer.Name, err))
|
|
|
|
|
wireguardFound = false
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
default: // Ignore other technologies
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if openvpnFound {
|
|
|
|
|
servers = append(servers, openVPNServer)
|
|
|
|
|
}
|
|
|
|
|
if wireguardFound {
|
|
|
|
|
servers = append(servers, wireguardServer)
|
2021-05-08 00:59:42 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if len(servers) < minServers {
|
2022-05-28 20:58:50 +00:00
|
|
|
return nil, fmt.Errorf("%w: %d and expected at least %d",
|
2022-05-28 22:02:18 +00:00
|
|
|
common.ErrNotEnoughServers, len(servers), minServers)
|
2021-05-08 00:59:42 +00:00
|
|
|
}
|
|
|
|
|
|
2022-06-06 01:57:44 +00:00
|
|
|
sort.Sort(models.SortableServers(servers))
|
2021-05-08 00:59:42 +00:00
|
|
|
|
2022-05-28 20:58:50 +00:00
|
|
|
return servers, nil
|
2021-05-08 00:59:42 +00:00
|
|
|
}
|