wttr.in/srv.go

204 lines
4.6 KiB
Go
Raw Normal View History

package main
import (
"crypto/tls"
2022-12-03 16:53:28 +00:00
"errors"
"fmt"
2022-11-20 16:53:51 +00:00
"io"
"log"
"net/http"
"time"
2022-11-29 20:38:39 +00:00
2022-12-03 16:53:28 +00:00
"github.com/alecthomas/kong"
2022-11-29 20:38:39 +00:00
"github.com/chubin/wttr.in/internal/config"
2022-12-04 20:16:23 +00:00
geoip "github.com/chubin/wttr.in/internal/geo/ip"
geoloc "github.com/chubin/wttr.in/internal/geo/location"
2022-12-03 15:08:43 +00:00
"github.com/chubin/wttr.in/internal/logging"
"github.com/chubin/wttr.in/internal/processor"
)
2022-12-03 16:53:28 +00:00
var cli struct {
ConfigCheck bool `name:"config-check" help:"Check configuration"`
ConfigDump bool `name:"config-dump" help:"Dump configuration"`
GeoResolve string `name:"geo-resolve" help:"Resolve location"`
ConfigFile string `name:"config-file" arg:"" optional:"" help:"Name of configuration file"`
2022-12-04 20:16:23 +00:00
ConvertGeoIPCache bool `name:"convert-geo-ip-cache" help:"Convert Geo IP data cache to SQlite"`
2022-12-03 16:53:28 +00:00
}
2022-11-20 16:53:51 +00:00
const logLineStart = "LOG_LINE_START "
2020-05-30 16:14:17 +00:00
func copyHeader(dst, src http.Header) {
for k, vv := range src {
for _, v := range vv {
dst.Add(k, v)
}
}
}
2022-11-20 16:53:51 +00:00
func serveHTTP(mux *http.ServeMux, port int, logFile io.Writer, errs chan<- error) {
srv := &http.Server{
Addr: fmt.Sprintf(":%d", port),
2022-11-20 16:53:51 +00:00
ErrorLog: log.New(logFile, logLineStart, log.LstdFlags),
ReadTimeout: 5 * time.Second,
WriteTimeout: 10 * time.Second,
IdleTimeout: 1 * time.Second,
Handler: mux,
}
errs <- srv.ListenAndServe()
2022-11-20 09:03:31 +00:00
}
2022-12-03 16:53:50 +00:00
func serveHTTPS(mux *http.ServeMux, port int, certFile, keyFile string, logFile io.Writer, errs chan<- error) {
tlsConfig := &tls.Config{
2022-11-29 20:38:39 +00:00
// CipherSuites: []uint16{
// tls.TLS_CHACHA20_POLY1305_SHA256,
// tls.TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256,
// tls.TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256,
// },
// MinVersion: tls.VersionTLS13,
}
srv := &http.Server{
Addr: fmt.Sprintf(":%d", port),
2022-11-20 16:53:51 +00:00
ErrorLog: log.New(logFile, logLineStart, log.LstdFlags),
ReadTimeout: 5 * time.Second,
WriteTimeout: 20 * time.Second,
IdleTimeout: 1 * time.Second,
TLSConfig: tlsConfig,
Handler: mux,
}
2022-12-03 16:53:50 +00:00
errs <- srv.ListenAndServeTLS(certFile, keyFile)
2022-11-20 09:03:31 +00:00
}
2022-12-03 16:53:50 +00:00
func serve(conf *config.Config) error {
2022-11-20 09:03:31 +00:00
var (
// mux is main HTTP/HTTP requests multiplexer.
mux *http.ServeMux = http.NewServeMux()
// logger is optimized requests logger.
2022-12-03 16:53:50 +00:00
logger *logging.RequestLogger
2022-11-19 18:20:35 +00:00
2022-12-03 15:08:43 +00:00
rp *processor.RequestProcessor
2022-11-27 14:51:41 +00:00
2022-11-20 09:03:31 +00:00
// errs is the servers errors channel.
errs chan error = make(chan error, 1)
// numberOfServers started. If 0, exit.
numberOfServers int
2022-11-20 16:53:51 +00:00
2022-12-03 16:53:50 +00:00
errorsLog *logging.LogSuppressor
2022-11-27 14:51:41 +00:00
err error
2022-11-20 09:03:31 +00:00
)
2022-12-03 16:53:50 +00:00
// logger is optimized requests logger.
logger = logging.NewRequestLogger(
conf.Logging.AccessLog,
time.Duration(conf.Logging.Interval)*time.Second)
errorsLog = logging.NewLogSuppressor(
conf.Logging.ErrorsLog,
[]string{
"error reading preface from client",
"TLS handshake error from",
},
logLineStart,
)
rp, err = processor.NewRequestProcessor(conf)
2022-11-27 14:51:41 +00:00
if err != nil {
log.Fatalln("log processor initialization:", err)
}
err = errorsLog.Open()
2022-11-20 16:53:51 +00:00
if err != nil {
log.Fatalln("errors log:", err)
}
2022-11-27 14:51:41 +00:00
rp.Start()
2022-11-20 09:03:31 +00:00
mux.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
2022-11-19 18:20:35 +00:00
if err := logger.Log(r); err != nil {
log.Println(err)
}
2020-05-30 16:14:17 +00:00
// printStat()
2022-11-27 14:51:41 +00:00
response, err := rp.ProcessRequest(r)
if err != nil {
log.Println(err)
return
}
if response.StatusCode == 0 {
log.Println("status code 0", response)
return
}
copyHeader(w.Header(), response.Header)
w.Header().Set("Access-Control-Allow-Origin", "*")
w.WriteHeader(response.StatusCode)
w.Write(response.Body)
})
2022-12-03 16:53:50 +00:00
if conf.Server.PortHTTP != 0 {
go serveHTTP(mux, conf.Server.PortHTTP, errorsLog, errs)
2022-11-20 09:03:31 +00:00
numberOfServers++
}
2022-12-03 16:53:50 +00:00
if conf.Server.PortHTTPS != 0 {
go serveHTTPS(mux, conf.Server.PortHTTPS, conf.Server.TLSCertFile, conf.Server.TLSKeyFile, errorsLog, errs)
2022-11-20 09:03:31 +00:00
numberOfServers++
}
if numberOfServers == 0 {
2022-12-03 16:53:50 +00:00
return errors.New("no servers configured")
}
return <-errs // block until one of the servers writes an error
}
func main() {
var (
conf *config.Config
err error
)
ctx := kong.Parse(&cli)
if cli.ConfigFile != "" {
conf, err = config.Load(cli.ConfigFile)
if err != nil {
log.Fatalf("reading config from %s: %s\n", cli.ConfigFile, err)
}
} else {
conf = config.Default()
}
if cli.ConfigDump {
fmt.Print(string(conf.Dump()))
}
if cli.ConfigCheck || cli.ConfigDump {
2022-11-20 09:03:31 +00:00
return
}
2022-12-03 16:53:50 +00:00
2022-12-04 20:16:23 +00:00
if cli.ConvertGeoIPCache {
2022-12-06 17:52:26 +00:00
geoIPCache, err := geoip.NewCache(conf)
if err != nil {
ctx.FatalIfErrorf(err)
}
2022-12-04 20:16:23 +00:00
ctx.FatalIfErrorf(geoIPCache.ConvertCache())
return
}
if cli.GeoResolve != "" {
sr := geoloc.NewSearcher(conf)
loc, err := sr.Search(cli.GeoResolve)
ctx.FatalIfErrorf(err)
if loc != nil {
fmt.Println(*loc)
}
}
2022-12-03 16:53:50 +00:00
err = serve(conf)
ctx.FatalIfErrorf(err)
}