Compare commits
11 Commits
v20220306.
...
v20220306.
Author | SHA1 | Date | |
---|---|---|---|
6c839d7196 | |||
d2de3225b0 | |||
130a048394 | |||
0be7f66582 | |||
0f014cf9f3 | |||
230e89e2b7 | |||
20e1092604 | |||
2998e1d39d | |||
5821d890b8 | |||
2dfdb1ad74 | |||
8d07304c03 |
3
.github/workflows/go.yml
vendored
3
.github/workflows/go.yml
vendored
@ -9,6 +9,8 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v2
|
- uses: actions/checkout@v2
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
|
||||||
- name: Set up Go
|
- name: Set up Go
|
||||||
uses: actions/setup-go@v2
|
uses: actions/setup-go@v2
|
||||||
@ -22,7 +24,6 @@ jobs:
|
|||||||
run: |
|
run: |
|
||||||
git tag "v$(git show -s --format=%cd --date=format:%Y%m%d.%H%M%S)"
|
git tag "v$(git show -s --format=%cd --date=format:%Y%m%d.%H%M%S)"
|
||||||
git push --tags
|
git push --tags
|
||||||
sudo mv snyk-linux* /usr/local/bin/
|
|
||||||
|
|
||||||
- name: Release binaries with GoReleaser
|
- name: Release binaries with GoReleaser
|
||||||
uses: goreleaser/goreleaser-action@v2
|
uses: goreleaser/goreleaser-action@v2
|
||||||
|
21
LICENSE
Normal file
21
LICENSE
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
MIT License
|
||||||
|
|
||||||
|
Copyright (c) 2022 Bastian Doetsch
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
22
README.md
Normal file
22
README.md
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
# mullvad-best-server
|
||||||
|

|
||||||
|
|
||||||
|
Determines the mullvat.net wireguard server with the lowest latency.
|
||||||
|
|
||||||
|
## Installation
|
||||||
|
|
||||||
|
Download binary from releases for your platform and unpack.
|
||||||
|
|
||||||
|
## Usage
|
||||||
|
### Default usage
|
||||||
|
Execute `mullvad-best-server`. It outputs the code, e.g. `de05`. You can then connect to it with e.g. wireguard using the normal shell scripts.
|
||||||
|
|
||||||
|
### Extended Output
|
||||||
|
If you want the full server information, execute `mullvad-best-server -o json`. It returns the full json output of the server information.
|
||||||
|
|
||||||
|
### Specify Country
|
||||||
|
The -c flag allows to give a country code. Else de/at/ch will be used.
|
||||||
|
|
||||||
|
## Background
|
||||||
|
The program uses `https://api.mullvad.net/www/relays/wireguard/` to get the current server list, pings the ones with the right country
|
||||||
|
and outputs the server with the lowest ping.
|
40
main.go
40
main.go
@ -2,36 +2,58 @@ package main
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"flag"
|
||||||
|
"fmt"
|
||||||
"github.com/go-ping/ping"
|
"github.com/go-ping/ping"
|
||||||
|
"github.com/rs/zerolog"
|
||||||
"github.com/rs/zerolog/log"
|
"github.com/rs/zerolog/log"
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
var pings = make(map[string]time.Duration)
|
var pings = make(map[string]time.Duration)
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
|
zerolog.SetGlobalLevel(zerolog.InfoLevel)
|
||||||
|
var outputFlag = flag.String("o", "short", "Output format. 'json' outputs server json")
|
||||||
|
var countryFlag = flag.String("c", "ch", "Server country code, e.g. ch for Switzerland")
|
||||||
|
flag.Parse()
|
||||||
|
|
||||||
servers := getServers()
|
servers := getServers()
|
||||||
bestIndex := selectBestServerIndex(servers)
|
bestIndex := selectBestServerIndex(servers, countryFlag)
|
||||||
log.Info().Interface("best", servers[bestIndex]).Msg("Best Latency Server found.")
|
log.Debug().Interface("server", servers[bestIndex]).Msg("Best latency server found.")
|
||||||
|
hostname := strings.Split(servers[bestIndex].Hostname, "-")[0]
|
||||||
|
if *outputFlag != "json" {
|
||||||
|
fmt.Println(hostname)
|
||||||
|
} else {
|
||||||
|
serverJson, err := json.Marshal(servers[bestIndex])
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal().Err(err)
|
||||||
|
}
|
||||||
|
fmt.Println(string(serverJson))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func selectBestServerIndex(servers []server) int {
|
func selectBestServerIndex(servers []server, country *string) int {
|
||||||
best := servers[0].Hostname
|
best := servers[0].Hostname
|
||||||
bestIndex := 0
|
bestIndex := -1
|
||||||
allowedCountries := map[string]string{}
|
allowedCountries := map[string]string{}
|
||||||
allowedCountries["de"] = "1"
|
if *country == "" {
|
||||||
allowedCountries["ch"] = "1"
|
allowedCountries["de"] = "1"
|
||||||
allowedCountries["at"] = "1"
|
allowedCountries["ch"] = "1"
|
||||||
|
allowedCountries["at"] = "1"
|
||||||
|
} else {
|
||||||
|
allowedCountries[*country] = "1"
|
||||||
|
}
|
||||||
for i, server := range servers {
|
for i, server := range servers {
|
||||||
if server.Active && allowedCountries[server.CountryCode] != "" {
|
if server.Active && allowedCountries[server.CountryCode] != "" {
|
||||||
duration, err := serverLatency(server)
|
duration, err := serverLatency(server)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
pings[server.Hostname] = duration
|
pings[server.Hostname] = duration
|
||||||
if best == "" || pings[best] > pings[server.Hostname] {
|
if bestIndex == -1 || pings[best] > pings[server.Hostname] {
|
||||||
best = server.Hostname
|
best = server.Hostname
|
||||||
bestIndex = i
|
bestIndex = i
|
||||||
}
|
}
|
||||||
@ -69,7 +91,7 @@ func serverLatency(s server) (time.Duration, error) {
|
|||||||
}
|
}
|
||||||
var duration time.Duration
|
var duration time.Duration
|
||||||
pinger.OnRecv = func(pkt *ping.Packet) {
|
pinger.OnRecv = func(pkt *ping.Packet) {
|
||||||
log.Info().Str("Server", s.Hostname).IPAddr("IP", pkt.IPAddr.IP).Dur("RTT", pkt.Rtt).Msg("Added server latency.")
|
log.Debug().Str("Server", s.Hostname).IPAddr("IP", pkt.IPAddr.IP).Dur("RTT", pkt.Rtt).Msg("Added server latency.")
|
||||||
duration = pkt.Rtt
|
duration = pkt.Rtt
|
||||||
}
|
}
|
||||||
err = pinger.Run()
|
err = pinger.Run()
|
||||||
|
Reference in New Issue
Block a user