mirror of
https://github.com/DeBrosOfficial/orama.git
synced 2026-06-16 23:14:13 +00:00
- add `orama-sni-router` binary to build process - introduce `cmd/sni-router` for TLS-level SNI routing - add documentation for stealth turn deployment architecture
243 lines
6.5 KiB
Go
243 lines
6.5 KiB
Go
// Command sni-router is a TLS-level Server Name Indication router.
|
|
//
|
|
// It listens on a public TCP port (typically :443), peeks at the TLS
|
|
// ClientHello SNI on each connection, and forwards the raw stream to
|
|
// a configured backend. It does NOT terminate TLS — encrypted bytes
|
|
// pass through verbatim. This lets one port serve multiple TLS-speaking
|
|
// backends (HTTPS for the gateway, TURN-over-TLS for stealth WebRTC).
|
|
//
|
|
// See pkg/sniproxy for the underlying library.
|
|
//
|
|
// Configuration: YAML file at --config (defaults to ~/.orama/sni-router.yaml).
|
|
//
|
|
// Example sni-router.yaml:
|
|
//
|
|
// listen: ":443"
|
|
// client_hello_timeout: 5s
|
|
// backend_dial_timeout: 5s
|
|
// max_concurrent_conns: 10000
|
|
// fallback:
|
|
// name: caddy
|
|
// addr: "127.0.0.1:8443"
|
|
// routes:
|
|
// - match: "cdn.example.com"
|
|
// backend:
|
|
// name: turn-tls
|
|
// addr: "127.0.0.1:5349"
|
|
// - match: "turn.example.com"
|
|
// backend:
|
|
// name: turn-tls
|
|
// addr: "127.0.0.1:5349"
|
|
// - match: "*.ns-myapp.example.com"
|
|
// backend:
|
|
// name: gateway
|
|
// addr: "127.0.0.1:8443"
|
|
package main
|
|
|
|
import (
|
|
"flag"
|
|
"fmt"
|
|
"net"
|
|
"os"
|
|
"os/signal"
|
|
"path/filepath"
|
|
"strings"
|
|
"syscall"
|
|
"time"
|
|
|
|
"github.com/DeBrosOfficial/network/pkg/config"
|
|
"github.com/DeBrosOfficial/network/pkg/logging"
|
|
"github.com/DeBrosOfficial/network/pkg/sniproxy"
|
|
"go.uber.org/zap"
|
|
)
|
|
|
|
var (
|
|
version = "dev"
|
|
commit = "unknown"
|
|
)
|
|
|
|
// yamlBackend mirrors sniproxy.Backend for YAML decoding.
|
|
type yamlBackend struct {
|
|
Name string `yaml:"name"`
|
|
Network string `yaml:"network"`
|
|
Addr string `yaml:"addr"`
|
|
}
|
|
|
|
// yamlRoute mirrors sniproxy.Route for YAML decoding.
|
|
type yamlRoute struct {
|
|
Match string `yaml:"match"`
|
|
Backend yamlBackend `yaml:"backend"`
|
|
}
|
|
|
|
// yamlConfig is the on-disk configuration shape.
|
|
type yamlConfig struct {
|
|
Listen string `yaml:"listen"`
|
|
ClientHelloTimeout time.Duration `yaml:"client_hello_timeout"`
|
|
BackendDialTimeout time.Duration `yaml:"backend_dial_timeout"`
|
|
MaxConcurrentConns int `yaml:"max_concurrent_conns"`
|
|
Fallback yamlBackend `yaml:"fallback"`
|
|
Routes []yamlRoute `yaml:"routes"`
|
|
}
|
|
|
|
func main() {
|
|
logger, err := logging.NewColoredLogger(logging.ComponentSNI, true)
|
|
if err != nil {
|
|
fmt.Fprintf(os.Stderr, "failed to init logger: %v\n", err)
|
|
os.Exit(1)
|
|
}
|
|
|
|
logger.ComponentInfo(logging.ComponentSNI, "Starting SNI router",
|
|
zap.String("version", version),
|
|
zap.String("commit", commit))
|
|
|
|
cfg := parseConfig(logger)
|
|
|
|
router := sniproxy.NewRouter(toBackend(cfg.Fallback))
|
|
router.Replace(toRoutes(cfg.Routes), toBackend(cfg.Fallback))
|
|
|
|
srv := sniproxy.NewServer(router, sniproxy.Config{
|
|
ClientHelloTimeout: cfg.ClientHelloTimeout,
|
|
BackendDialTimeout: cfg.BackendDialTimeout,
|
|
MaxConcurrentConns: cfg.MaxConcurrentConns,
|
|
}, logger.Logger)
|
|
|
|
ln, err := net.Listen("tcp", cfg.Listen)
|
|
if err != nil {
|
|
logger.ComponentError(logging.ComponentSNI, "Failed to listen",
|
|
zap.String("addr", cfg.Listen), zap.Error(err))
|
|
os.Exit(1)
|
|
}
|
|
|
|
logger.ComponentInfo(logging.ComponentSNI, "SNI router listening",
|
|
zap.String("addr", cfg.Listen),
|
|
zap.Int("routes", len(cfg.Routes)),
|
|
zap.String("fallback", cfg.Fallback.Addr),
|
|
)
|
|
|
|
// Run Serve in a goroutine so the main goroutine can wait on signals.
|
|
serveErrCh := make(chan error, 1)
|
|
go func() {
|
|
serveErrCh <- srv.Serve(ln)
|
|
}()
|
|
|
|
// Wait for termination signal or unrecoverable Serve error.
|
|
quit := make(chan os.Signal, 1)
|
|
signal.Notify(quit, os.Interrupt, syscall.SIGTERM)
|
|
|
|
select {
|
|
case sig := <-quit:
|
|
logger.ComponentInfo(logging.ComponentSNI, "Shutdown signal received",
|
|
zap.String("signal", sig.String()))
|
|
case err := <-serveErrCh:
|
|
logger.ComponentError(logging.ComponentSNI, "Serve returned",
|
|
zap.Error(err))
|
|
}
|
|
|
|
// Stop accepting new connections, then drain in-flight ones.
|
|
_ = ln.Close()
|
|
srv.Close()
|
|
|
|
logger.ComponentInfo(logging.ComponentSNI, "SNI router shutdown complete")
|
|
}
|
|
|
|
func parseConfig(logger *logging.ColoredLogger) yamlConfig {
|
|
configFlag := flag.String("config", "", "Config file path (absolute or filename in ~/.orama)")
|
|
flag.Parse()
|
|
|
|
var configPath string
|
|
var err error
|
|
if *configFlag != "" {
|
|
if filepath.IsAbs(*configFlag) {
|
|
configPath = *configFlag
|
|
} else {
|
|
configPath, err = config.DefaultPath(*configFlag)
|
|
if err != nil {
|
|
logger.ComponentError(logging.ComponentSNI, "Failed to determine config path",
|
|
zap.Error(err))
|
|
os.Exit(1)
|
|
}
|
|
}
|
|
} else {
|
|
configPath, err = config.DefaultPath("sni-router.yaml")
|
|
if err != nil {
|
|
logger.ComponentError(logging.ComponentSNI, "Failed to determine config path",
|
|
zap.Error(err))
|
|
os.Exit(1)
|
|
}
|
|
}
|
|
|
|
data, err := os.ReadFile(configPath)
|
|
if err != nil {
|
|
logger.ComponentError(logging.ComponentSNI, "Config file not found",
|
|
zap.String("path", configPath), zap.Error(err))
|
|
fmt.Fprintf(os.Stderr, "\nConfig file not found at %s\n", configPath)
|
|
os.Exit(1)
|
|
}
|
|
|
|
var y yamlConfig
|
|
if err := config.DecodeStrict(strings.NewReader(string(data)), &y); err != nil {
|
|
logger.ComponentError(logging.ComponentSNI, "Failed to parse SNI router config",
|
|
zap.Error(err))
|
|
fmt.Fprintf(os.Stderr, "Configuration parse error: %v\n", err)
|
|
os.Exit(1)
|
|
}
|
|
|
|
if errs := validateConfig(&y); len(errs) > 0 {
|
|
fmt.Fprintf(os.Stderr, "\nSNI router configuration errors (%d):\n", len(errs))
|
|
for _, e := range errs {
|
|
fmt.Fprintf(os.Stderr, " - %s\n", e)
|
|
}
|
|
fmt.Fprintf(os.Stderr, "\nPlease fix the configuration and try again.\n")
|
|
os.Exit(1)
|
|
}
|
|
|
|
logger.ComponentInfo(logging.ComponentSNI, "Loaded SNI router configuration",
|
|
zap.String("path", configPath),
|
|
)
|
|
|
|
return y
|
|
}
|
|
|
|
// validateConfig returns a non-empty slice of human-readable errors on misconfig.
|
|
func validateConfig(y *yamlConfig) []string {
|
|
var errs []string
|
|
if y.Listen == "" {
|
|
errs = append(errs, "listen: required (e.g. \":443\")")
|
|
}
|
|
if y.Fallback.Addr == "" {
|
|
errs = append(errs, "fallback.addr: required (where to send unmatched SNIs, typically Caddy)")
|
|
}
|
|
for i, r := range y.Routes {
|
|
if r.Match == "" {
|
|
errs = append(errs, fmt.Sprintf("routes[%d].match: required", i))
|
|
}
|
|
if r.Backend.Addr == "" {
|
|
errs = append(errs, fmt.Sprintf("routes[%d].backend.addr: required", i))
|
|
}
|
|
}
|
|
return errs
|
|
}
|
|
|
|
func toBackend(b yamlBackend) sniproxy.Backend {
|
|
network := b.Network
|
|
if network == "" {
|
|
network = "tcp"
|
|
}
|
|
return sniproxy.Backend{
|
|
Name: b.Name,
|
|
Network: network,
|
|
Addr: b.Addr,
|
|
}
|
|
}
|
|
|
|
func toRoutes(in []yamlRoute) []sniproxy.Route {
|
|
out := make([]sniproxy.Route, len(in))
|
|
for i, r := range in {
|
|
out[i] = sniproxy.Route{
|
|
Match: r.Match,
|
|
Backend: toBackend(r.Backend),
|
|
}
|
|
}
|
|
return out
|
|
}
|