mirror of
https://github.com/qdm12/gluetun.git
synced 2026-05-07 04:20:12 +02:00
117 lines
3.5 KiB
Go
117 lines
3.5 KiB
Go
package updater
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"regexp"
|
|
"sort"
|
|
"strings"
|
|
"time"
|
|
|
|
"golang.org/x/mod/semver"
|
|
)
|
|
|
|
// getMostRecentStableWebAccountTag finds the most recent proton-account stable tag version,
|
|
// in order to use it in the x-pm-appversion http request header. Because if we do
|
|
// fall behind on versioning, Proton doesn't like it because they like to create
|
|
// complications where there is no need for it. Hence this function.
|
|
func getMostRecentStableWebAccountTag(ctx context.Context, client *http.Client) (version string, err error) {
|
|
page := 1
|
|
regexVersion := regexp.MustCompile(`^proton-account@(\d+\.\d+\.\d+\.\d+)$`)
|
|
for ctx.Err() == nil {
|
|
// Define a timeout since the default client has a large timeout and we don't
|
|
// want to wait too long.
|
|
const timeout = 5 * time.Second
|
|
ctx, cancel := context.WithTimeout(ctx, timeout)
|
|
defer cancel()
|
|
|
|
url := "https://api.github.com/repos/ProtonMail/WebClients/tags?per_page=30&page=" + fmt.Sprint(page)
|
|
|
|
request, err := http.NewRequestWithContext(ctx, http.MethodGet, url, nil)
|
|
if err != nil {
|
|
return "", fmt.Errorf("creating request: %w", err)
|
|
}
|
|
request.Header.Set("Accept", "application/vnd.github.v3+json")
|
|
|
|
response, err := client.Do(request)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
defer response.Body.Close()
|
|
|
|
data, err := io.ReadAll(response.Body)
|
|
if err != nil {
|
|
return "", fmt.Errorf("reading response body: %w", err)
|
|
}
|
|
|
|
if response.StatusCode != http.StatusOK {
|
|
return "", fmt.Errorf("%w: %s: %s", ErrHTTPStatusCodeNotOK, response.Status, data)
|
|
}
|
|
|
|
var tags []struct {
|
|
Name string `json:"name"`
|
|
}
|
|
err = json.Unmarshal(data, &tags)
|
|
if err != nil {
|
|
return "", fmt.Errorf("decoding JSON response: %w", err)
|
|
}
|
|
|
|
for _, tag := range tags {
|
|
if !regexVersion.MatchString(tag.Name) {
|
|
continue
|
|
}
|
|
version := "web-account@" + strings.TrimPrefix(tag.Name, "proton-account@")
|
|
return version, nil
|
|
}
|
|
|
|
page++
|
|
}
|
|
|
|
return "", fmt.Errorf("%w (queried %d pages)", context.Canceled, page)
|
|
}
|
|
|
|
// getMostRecentStableVPNGtkAppTag finds the latest proton-vpn-gtk-app semver tag,
|
|
// in order to use it in the x-pm-appversion http request header ONLY to fetch servers
|
|
// data. Because if we do fall behind on versioning, Proton doesn't like it because they like
|
|
// to create complications where there is no need for it. Hence this function.
|
|
func getMostRecentStableVPNGtkAppTag(ctx context.Context, client *http.Client) (version string, err error) {
|
|
const url = "https://api.github.com/repos/ProtonVPN/proton-vpn-gtk-app/tags?per_page=30"
|
|
|
|
request, err := http.NewRequestWithContext(ctx, http.MethodGet, url, nil)
|
|
if err != nil {
|
|
return "", fmt.Errorf("creating request: %w", err)
|
|
}
|
|
request.Header.Set("Accept", "application/vnd.github.v3+json")
|
|
|
|
response, err := client.Do(request)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
defer response.Body.Close()
|
|
|
|
if response.StatusCode != http.StatusOK {
|
|
return "", fmt.Errorf("%w: %s", ErrHTTPStatusCodeNotOK, response.Status)
|
|
}
|
|
|
|
decoder := json.NewDecoder(response.Body)
|
|
var data []struct {
|
|
Name string `json:"name"`
|
|
}
|
|
err = decoder.Decode(&data)
|
|
if err != nil {
|
|
return "", fmt.Errorf("decoding JSON response: %w", err)
|
|
}
|
|
|
|
// Sort tags by semver. Invalid tags are placed at the end and we ignore them.
|
|
// Yes, proton does push invalid semver tag names sometimes. Good job yet again.
|
|
sort.Slice(data, func(i, j int) bool {
|
|
return semver.Compare(data[i].Name, data[j].Name) > 0
|
|
})
|
|
|
|
version = "linux-vpn@" + data[0].Name[1:] // remove leading v
|
|
return version, nil
|
|
}
|