concurrent DNS bootstrap: implement CLI

This commit is contained in:
Vladislav Yarmak 2024-08-01 12:01:24 +03:00
parent a1d38df01c
commit 34dde845dd
2 changed files with 54 additions and 9 deletions

60
main.go
View file

@ -1,6 +1,7 @@
package main package main
import ( import (
"bytes"
"context" "context"
"crypto/tls" "crypto/tls"
"crypto/x509" "crypto/x509"
@ -8,6 +9,7 @@ import (
"errors" "errors"
"flag" "flag"
"fmt" "fmt"
"io"
"io/ioutil" "io/ioutil"
"log" "log"
"net" "net"
@ -43,6 +45,37 @@ func arg_fail(msg string) {
os.Exit(2) os.Exit(2)
} }
type CSVArg struct {
values []string
}
func (a *CSVArg) String() string {
if len(a.values) == 0 {
return ""
}
buf := new(bytes.Buffer)
wr := csv.NewWriter(buf)
wr.Write(a.values)
wr.Flush()
return strings.TrimRight(buf.String(), "\n")
}
func (a *CSVArg) Set(line string) error {
rd := csv.NewReader(strings.NewReader(line))
rd.FieldsPerRecord = -1
rd.TrimLeadingSpace = true
values, err := rd.Read()
if err == io.EOF {
a.values = nil
return nil
}
if err != nil {
return fmt.Errorf("unable to parse comma-separated argument: %w", err)
}
a.values = values
return nil
}
type CLIArgs struct { type CLIArgs struct {
country string country string
listCountries bool listCountries bool
@ -55,15 +88,26 @@ type CLIArgs struct {
apiLogin string apiLogin string
apiPassword string apiPassword string
apiAddress string apiAddress string
bootstrapDNS string bootstrapDNS *CSVArg
refresh time.Duration refresh time.Duration
refreshRetry time.Duration refreshRetry time.Duration
certChainWorkaround bool certChainWorkaround bool
caFile string caFile string
} }
func parse_args() CLIArgs { func parse_args() *CLIArgs {
var args CLIArgs args := &CLIArgs{
bootstrapDNS: &CSVArg{
values: []string{
"https://1.1.1.3/dns-query",
"https://security.cloudflare-dns.com/dns-query",
"https://wikimedia-dns.org/dns-query",
"https://dns.adguard-dns.com/dns-query",
"https://dns.quad9.net/dns-query",
"https://doh.cleanbrowsing.org/doh/adult-filter/",
},
},
}
flag.StringVar(&args.country, "country", "EU", "desired proxy location") flag.StringVar(&args.country, "country", "EU", "desired proxy location")
flag.BoolVar(&args.listCountries, "list-countries", false, "list available countries and exit") flag.BoolVar(&args.listCountries, "list-countries", false, "list available countries and exit")
flag.BoolVar(&args.listProxies, "list-proxies", false, "output proxy list and exit") flag.BoolVar(&args.listProxies, "list-proxies", false, "output proxy list and exit")
@ -78,10 +122,10 @@ func parse_args() CLIArgs {
flag.StringVar(&args.apiLogin, "api-login", "se0316", "SurfEasy API login") flag.StringVar(&args.apiLogin, "api-login", "se0316", "SurfEasy API login")
flag.StringVar(&args.apiPassword, "api-password", "SILrMEPBmJuhomxWkfm3JalqHX2Eheg1YhlEZiMh8II", "SurfEasy API password") flag.StringVar(&args.apiPassword, "api-password", "SILrMEPBmJuhomxWkfm3JalqHX2Eheg1YhlEZiMh8II", "SurfEasy API password")
flag.StringVar(&args.apiAddress, "api-address", "", fmt.Sprintf("override IP address of %s", API_DOMAIN)) flag.StringVar(&args.apiAddress, "api-address", "", fmt.Sprintf("override IP address of %s", API_DOMAIN))
flag.StringVar(&args.bootstrapDNS, "bootstrap-dns", "https://1.1.1.3/dns-query", flag.Var(args.bootstrapDNS, "bootstrap-dns",
"DNS/DoH/DoT/DoQ resolver for initial discovering of SurfEasy API address. "+ "comma-separated list of DNS/DoH/DoT/DoQ resolvers for initial discovery of SurfEasy API address. "+
"See https://github.com/ameshkov/dnslookup/ for upstream DNS URL format. "+ "See https://github.com/ameshkov/dnslookup/ for upstream DNS URL format. "+
"Examples: https://1.1.1.1/dns-query, quic://dns.adguard.com") "Examples: https://1.1.1.1/dns-query,quic://dns.adguard.com")
flag.DurationVar(&args.refresh, "refresh", 4*time.Hour, "login refresh interval") flag.DurationVar(&args.refresh, "refresh", 4*time.Hour, "login refresh interval")
flag.DurationVar(&args.refreshRetry, "refresh-retry", 5*time.Second, "login refresh retry interval") flag.DurationVar(&args.refreshRetry, "refresh-retry", 5*time.Second, "login refresh retry interval")
flag.BoolVar(&args.certChainWorkaround, "certchain-workaround", true, flag.BoolVar(&args.certChainWorkaround, "certchain-workaround", true,
@ -147,13 +191,13 @@ func run() int {
} }
seclientDialer := dialer seclientDialer := dialer
if args.apiAddress != "" || args.bootstrapDNS != "" { if args.apiAddress != "" || len(args.bootstrapDNS.values) > 0 {
var apiAddress string var apiAddress string
if args.apiAddress != "" { if args.apiAddress != "" {
apiAddress = args.apiAddress apiAddress = args.apiAddress
mainLogger.Info("Using fixed API host IP address = %s", apiAddress) mainLogger.Info("Using fixed API host IP address = %s", apiAddress)
} else { } else {
resolver, err := NewResolver(args.bootstrapDNS, args.timeout) resolver, err := NewResolver(args.bootstrapDNS.values[0], args.timeout)
if err != nil { if err != nil {
mainLogger.Critical("Unable to instantiate DNS resolver: %v", err) mainLogger.Critical("Unable to instantiate DNS resolver: %v", err)
return 4 return 4

View file

@ -1,9 +1,10 @@
package main package main
import ( import (
"time"
"github.com/AdguardTeam/dnsproxy/upstream" "github.com/AdguardTeam/dnsproxy/upstream"
"github.com/miekg/dns" "github.com/miekg/dns"
"time"
) )
type Resolver struct { type Resolver struct {