2017-01-18 17:02:44 +00:00
|
|
|
package server
|
|
|
|
|
|
|
|
import (
|
2017-03-26 19:51:37 +00:00
|
|
|
"crypto/tls"
|
2017-01-18 17:02:44 +00:00
|
|
|
"errors"
|
|
|
|
"net"
|
|
|
|
"net/http"
|
2017-03-26 19:51:37 +00:00
|
|
|
"net/http/httputil"
|
2017-01-18 17:02:44 +00:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"git.xeserv.us/xena/route/database"
|
2017-03-26 19:51:37 +00:00
|
|
|
"git.xeserv.us/xena/route/lib/tun2"
|
|
|
|
"github.com/mtneug/pkg/ulid"
|
|
|
|
"golang.org/x/crypto/acme/autocert"
|
2017-01-18 17:02:44 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// RPC constants
|
|
|
|
const (
|
|
|
|
RPCPort uint16 = 39453
|
|
|
|
)
|
|
|
|
|
|
|
|
// Server is the main server type
|
|
|
|
type Server struct {
|
2017-01-20 00:31:22 +00:00
|
|
|
cfg *Config
|
2017-04-28 23:27:34 +00:00
|
|
|
db database.Storage
|
|
|
|
ts *tun2.Server
|
2017-01-18 17:02:44 +00:00
|
|
|
|
2017-04-28 23:27:34 +00:00
|
|
|
autocert.Manager
|
2017-01-18 17:02:44 +00:00
|
|
|
}
|
|
|
|
|
2017-01-20 00:31:22 +00:00
|
|
|
// Config configures Server
|
|
|
|
type Config struct {
|
2017-04-28 23:27:34 +00:00
|
|
|
BoltDBPath string `env:"BOLTDB_PATH,required"`
|
2017-03-27 04:39:19 +00:00
|
|
|
|
2017-04-28 23:27:34 +00:00
|
|
|
WebAddr string `env:"WEB_ADDR,required"`
|
|
|
|
SSLAddr string `env:"SSL_ADDR,required"`
|
|
|
|
BackendTCPAddr string `env:"BACKEND_TCP_ADDR,required"`
|
|
|
|
BackendKCPAddr string `env:"BACKEND_KCP_ADDR,required"`
|
|
|
|
GRPCAddr string `env:"GRPC_ADDR,required"`
|
2017-03-27 04:39:19 +00:00
|
|
|
|
2017-04-28 23:27:34 +00:00
|
|
|
DomainSuffix string `env:"DOMAIN_SUFFIX,required"`
|
|
|
|
ACMEEmail string `env:"ACME_EMAIL,required"`
|
2017-03-27 04:39:19 +00:00
|
|
|
CertKey *[32]byte
|
2017-01-18 17:02:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// New creates a new Server
|
2017-01-20 00:31:22 +00:00
|
|
|
func New(cfg Config) (*Server, error) {
|
2017-04-28 23:27:34 +00:00
|
|
|
if cfg.CertKey == nil {
|
|
|
|
return nil, errors.New("no cert decryption key, can't do anything")
|
2017-01-18 17:02:44 +00:00
|
|
|
}
|
|
|
|
|
2017-04-28 23:27:34 +00:00
|
|
|
db, err := database.NewBoltStorage(cfg.BoltDBPath, cfg.CertKey)
|
2017-01-18 17:02:44 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2017-03-26 19:51:37 +00:00
|
|
|
m := autocert.Manager{
|
|
|
|
Prompt: autocert.AcceptTOS,
|
2017-04-28 23:27:34 +00:00
|
|
|
Cache: database.Cache(db),
|
2017-03-26 19:51:37 +00:00
|
|
|
HostPolicy: nil,
|
2017-03-27 04:39:19 +00:00
|
|
|
Email: cfg.ACMEEmail,
|
2017-03-26 19:51:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
tcfg := &tun2.ServerConfig{
|
2017-04-28 23:27:34 +00:00
|
|
|
TCPAddr: cfg.BackendTCPAddr,
|
|
|
|
KCPAddr: cfg.BackendKCPAddr,
|
2017-03-26 19:51:37 +00:00
|
|
|
TLSConfig: &tls.Config{
|
|
|
|
GetCertificate: m.GetCertificate,
|
|
|
|
},
|
2017-04-28 23:27:34 +00:00
|
|
|
Storage: &storageWrapper{
|
|
|
|
Storage: db,
|
|
|
|
},
|
2017-03-26 19:51:37 +00:00
|
|
|
}
|
2017-01-18 17:02:44 +00:00
|
|
|
|
2017-03-26 19:51:37 +00:00
|
|
|
ts, err := tun2.NewServer(tcfg)
|
2017-01-18 17:02:44 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2017-04-28 23:27:34 +00:00
|
|
|
s := &Server{
|
|
|
|
cfg: &cfg,
|
|
|
|
db: db,
|
|
|
|
ts: ts,
|
2017-03-26 19:51:37 +00:00
|
|
|
|
2017-04-28 23:27:34 +00:00
|
|
|
Manager: m,
|
2017-01-28 00:14:05 +00:00
|
|
|
}
|
|
|
|
|
2017-04-28 23:27:34 +00:00
|
|
|
s.ts = ts
|
|
|
|
go ts.ListenAndServe()
|
2017-01-18 17:02:44 +00:00
|
|
|
|
2017-04-28 23:27:34 +00:00
|
|
|
return s, nil
|
2017-01-18 17:02:44 +00:00
|
|
|
}
|
|
|
|
|
2017-04-28 23:27:34 +00:00
|
|
|
// Director removes headers that are typically stripped off at request ingress.
|
|
|
|
func (s *Server) Director(r *http.Request) {
|
2017-01-22 17:49:14 +00:00
|
|
|
r.Header.Del("X-Forwarded-For")
|
2017-03-26 20:47:42 +00:00
|
|
|
r.Header.Del("X-Client-Ip")
|
2017-04-28 23:27:34 +00:00
|
|
|
}
|
2017-01-22 17:49:14 +00:00
|
|
|
|
2017-04-28 23:27:34 +00:00
|
|
|
func (s *Server) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
2017-01-22 17:49:14 +00:00
|
|
|
if r.Header.Get("X-Tor2web") != "" {
|
|
|
|
http.Error(w, "tor2web proxy use is not allowed", 400)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
host, _, err := net.SplitHostPort(r.RemoteAddr)
|
|
|
|
if err != nil {
|
|
|
|
http.Error(w, err.Error(), 500)
|
|
|
|
return
|
|
|
|
}
|
2017-03-26 20:47:42 +00:00
|
|
|
r.Header.Set("X-Remote-Ip", host)
|
|
|
|
r.Header.Set("X-Request-Ingress", time.Now().Format(time.RFC3339))
|
2017-01-20 01:17:18 +00:00
|
|
|
|
2017-03-26 19:51:37 +00:00
|
|
|
rid := ulid.New().String()
|
2017-01-23 16:03:14 +00:00
|
|
|
r.Header.Set("X-Request-Id", rid)
|
|
|
|
w.Header().Set("X-Request-Id", rid)
|
|
|
|
|
2017-02-08 02:25:14 +00:00
|
|
|
// http://www.gnuterrypratchett.com/
|
2017-03-26 19:51:37 +00:00
|
|
|
w.Header().Set("X-Clacks-Overhead", "GNU Ashlynn")
|
2017-02-08 02:25:14 +00:00
|
|
|
|
2017-03-26 19:51:37 +00:00
|
|
|
rp := &httputil.ReverseProxy{
|
2017-03-26 20:02:22 +00:00
|
|
|
Director: s.Director,
|
2017-03-26 19:51:37 +00:00
|
|
|
Transport: s.ts,
|
|
|
|
FlushInterval: 1 * time.Second,
|
|
|
|
}
|
|
|
|
|
|
|
|
rp.ServeHTTP(w, r)
|
2017-01-18 17:02:44 +00:00
|
|
|
}
|