whatismyip/internal/setting/app.go

168 lines
4.7 KiB
Go
Raw Permalink Normal View History

2021-11-10 19:06:12 +00:00
package setting
import (
"bytes"
"errors"
2021-11-10 19:06:12 +00:00
"flag"
"fmt"
"os"
"time"
"github.com/dcarrillo/whatismyip/internal/core"
2024-04-12 17:26:48 +00:00
"gopkg.in/yaml.v3"
2021-11-10 19:06:12 +00:00
)
2025-01-02 19:13:41 +00:00
type geodbConf struct {
City string
ASN string
Token *string
2021-11-10 19:06:12 +00:00
}
type serverSettings struct {
ReadTimeout time.Duration
WriteTimeout time.Duration
}
2024-04-12 17:26:48 +00:00
type resolver struct {
Domain string `yaml:"domain"`
ResourceRecords []string `yaml:"resource_records"`
RedirectPort string `yaml:"redirect_port,omitempty"`
Ipv4 []string `yaml:"ipv4,omitempty"`
Ipv6 []string `yaml:"ipv6,omitempty"`
}
2021-11-10 19:06:12 +00:00
type settings struct {
2025-01-02 19:13:41 +00:00
GeodbPath geodbConf
TemplatePath string
BindAddress string
TLSAddress string
TLSCrtPath string
TLSKeyPath string
TrustedHeader string
TrustedPortHeader string
EnableSecureHeaders bool
EnableHTTP3 bool
Server serverSettings
2024-04-12 17:26:48 +00:00
Resolver resolver
version bool
2021-11-10 19:06:12 +00:00
}
const defaultAddress = ":8080"
var ErrVersion = errors.New("setting: version requested")
var App = settings{
// hard-coded for the time being
Server: serverSettings{
ReadTimeout: 10 * time.Second,
WriteTimeout: 10 * time.Second,
},
}
func Setup(args []string) (output string, err error) {
flags := flag.NewFlagSet("whatismyip", flag.ContinueOnError)
var buf bytes.Buffer
2024-04-12 17:26:48 +00:00
var resolverConf string
flags.SetOutput(&buf)
2021-11-10 19:06:12 +00:00
2025-01-02 19:13:41 +00:00
flags.StringVar(&App.GeodbPath.City, "geoip2-city", "", "Path to GeoIP2 city database. Enables geo information (--geoip2-asn becomes mandatory)")
flags.StringVar(&App.GeodbPath.ASN, "geoip2-asn", "", "Path to GeoIP2 ASN database. Enables ASN information. (--geoip2-city becomes mandatory)")
2024-04-12 17:26:48 +00:00
flags.StringVar(&App.TemplatePath, "template", "", "Path to the template file")
flags.StringVar(
&resolverConf,
"resolver",
"",
2024-05-09 18:01:40 +00:00
"Path to the resolver configuration. It actually enables the resolver for DNS client discovery.")
flags.StringVar(
&App.BindAddress,
"bind",
defaultAddress,
"Listening address (see https://pkg.go.dev/net?#Listen)",
)
flags.StringVar(
&App.TLSAddress,
"tls-bind",
"",
"Listening address for TLS (see https://pkg.go.dev/net?#Listen)",
)
flags.StringVar(&App.TLSCrtPath, "tls-crt", "", "When using TLS, path to certificate file")
flags.StringVar(&App.TLSKeyPath, "tls-key", "", "When using TLS, path to private key file")
flags.StringVar(
&App.TrustedHeader,
"trusted-header",
"",
"Trusted request header for remote IP (e.g. X-Real-IP). When using this feature if -trusted-port-header is not set the client port is shown as 'unknown'",
)
flags.StringVar(
&App.TrustedPortHeader,
"trusted-port-header",
"",
"Trusted request header for remote client port (e.g. X-Real-Port). When this parameter is set -trusted-header becomes mandatory",
)
flags.BoolVar(&App.version, "version", false, "Output version information and exit")
flags.BoolVar(
&App.EnableSecureHeaders,
"enable-secure-headers",
false,
"Add sane security-related headers to every response",
)
flags.BoolVar(
&App.EnableHTTP3,
"enable-http3",
false,
"Enable HTTP/3 protocol. HTTP/3 requires --tls-bind set, as HTTP/3 starts as a TLS connection that then gets upgraded to UDP. The UDP port is the same as the one used for the TLS server.",
)
2021-11-10 19:06:12 +00:00
err = flags.Parse(args)
if err != nil {
return buf.String(), err
2021-11-10 19:06:12 +00:00
}
if App.version {
return fmt.Sprintf("whatismyip version %s", core.Version), ErrVersion
2021-11-10 19:06:12 +00:00
}
2025-01-02 19:13:41 +00:00
if (App.GeodbPath.City != "" && App.GeodbPath.ASN == "") || (App.GeodbPath.City == "" && App.GeodbPath.ASN != "") {
return "", fmt.Errorf("both --geoip2-city and --geoip2-asn are mandatory to enable geo information")
}
2025-01-02 19:13:41 +00:00
if App.TrustedPortHeader != "" && App.TrustedHeader == "" {
return "", fmt.Errorf("truster-header is mandatory when truster-port-header is set")
2021-11-10 19:06:12 +00:00
}
if (App.TLSAddress != "") && (App.TLSCrtPath == "" || App.TLSKeyPath == "") {
2023-03-16 18:59:04 +00:00
return "", fmt.Errorf("in order to use TLS, the -tls-crt and -tls-key flags are mandatory")
2021-11-10 19:06:12 +00:00
}
if App.EnableHTTP3 && App.TLSAddress == "" {
return "", fmt.Errorf("in order to use HTTP3, the -tls-bind is mandatory")
}
if App.TemplatePath != "" {
info, err := os.Stat(App.TemplatePath)
if os.IsNotExist(err) {
2023-03-16 18:59:04 +00:00
return "", fmt.Errorf("%s no such file or directory", App.TemplatePath)
}
if info.IsDir() {
2023-03-16 18:59:04 +00:00
return "", fmt.Errorf("%s must be a file", App.TemplatePath)
}
2021-11-10 19:06:12 +00:00
}
2024-04-12 17:26:48 +00:00
if resolverConf != "" {
var err error
App.Resolver, err = readYAML(resolverConf)
if err != nil {
2025-01-02 19:13:41 +00:00
return "", fmt.Errorf("error reading resolver configuration %w", err)
2024-04-12 17:26:48 +00:00
}
}
return buf.String(), nil
2021-11-10 19:06:12 +00:00
}
2024-04-12 17:26:48 +00:00
func readYAML(path string) (resolver resolver, err error) {
yamlFile, err := os.ReadFile(path)
if err != nil {
return resolver, err
}
return resolver, yaml.Unmarshal(yamlFile, &resolver)
}