mirror of
https://github.com/yusing/godoxy.git
synced 2026-04-17 05:59:42 +02:00
Introduce reusable `inbound_mtls_profiles` in root config and support `entrypoint.inbound_mtls_profile` to require client certificates for all HTTPS traffic on an entrypoint. Profiles can trust the system CA store, custom PEM CA files, or both, and are compiled into TLS client-auth pools during entrypoint initialization. Also add route-scoped `inbound_mtls_profile` support for HTTP-based routes when no global entrypoint profile is configured. Route-level mTLS selection is driven by TLS SNI, preserves existing behavior for open and unmatched hosts, and returns the intended 421 response when secure requests omit SNI or when Host and SNI resolve to different routes. Add validation for missing profile references and unsupported non-HTTP route usage, update config and route documentation/examples, expand inbound mTLS handshake and routing regression coverage, and bump `goutils` for HTTPS listener test support.
151 lines
3.5 KiB
Go
151 lines
3.5 KiB
Go
package entrypoint
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"net"
|
|
"strconv"
|
|
|
|
"github.com/yusing/godoxy/internal/common"
|
|
"github.com/yusing/godoxy/internal/types"
|
|
)
|
|
|
|
func (ep *Entrypoint) IterRoutes(yield func(r types.Route) bool) {
|
|
for _, r := range ep.HTTPRoutes().Iter {
|
|
if !yield(r) {
|
|
return
|
|
}
|
|
}
|
|
for _, r := range ep.streamRoutes.Iter {
|
|
if !yield(r) {
|
|
return
|
|
}
|
|
}
|
|
for _, r := range ep.excludedRoutes.Iter {
|
|
if !yield(r) {
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
func (ep *Entrypoint) NumRoutes() int {
|
|
return ep.HTTPRoutes().Size() + ep.streamRoutes.Size() + ep.excludedRoutes.Size()
|
|
}
|
|
|
|
func (ep *Entrypoint) GetRoute(alias string) (types.Route, bool) {
|
|
if r, ok := ep.HTTPRoutes().Get(alias); ok {
|
|
return r, true
|
|
}
|
|
if r, ok := ep.streamRoutes.Get(alias); ok {
|
|
return r, true
|
|
}
|
|
if r, ok := ep.excludedRoutes.Get(alias); ok {
|
|
return r, true
|
|
}
|
|
return nil, false
|
|
}
|
|
|
|
func (ep *Entrypoint) StartAddRoute(r types.Route) error {
|
|
if r.ShouldExclude() {
|
|
ep.excludedRoutes.Add(r)
|
|
r.Task().OnCancel("remove_route", func() {
|
|
ep.excludedRoutes.Del(r)
|
|
})
|
|
return nil
|
|
}
|
|
switch r := r.(type) {
|
|
case types.HTTPRoute:
|
|
if err := ep.AddHTTPRoute(r); err != nil {
|
|
return err
|
|
}
|
|
ep.shortLinkMatcher.AddRoute(r.Key())
|
|
r.Task().OnCancel("remove_route", func() {
|
|
ep.delHTTPRoute(r)
|
|
ep.shortLinkMatcher.DelRoute(r.Key())
|
|
})
|
|
case types.StreamRoute:
|
|
err := r.ListenAndServe(r.Task().Context(), nil, nil)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
ep.streamRoutes.Add(r)
|
|
|
|
r.Task().OnCancel("remove_route", func() {
|
|
r.Stream().Close()
|
|
ep.streamRoutes.Del(r)
|
|
})
|
|
default:
|
|
return fmt.Errorf("unknown route type: %T", r)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func getAddr(route types.HTTPRoute) (httpAddr, httpsAddr string) {
|
|
if port := route.ListenURL().Port(); port == "" || port == "0" {
|
|
host := route.ListenURL().Hostname()
|
|
if host == "" {
|
|
httpAddr = common.ProxyHTTPAddr
|
|
httpsAddr = common.ProxyHTTPSAddr
|
|
} else {
|
|
httpAddr = net.JoinHostPort(host, strconv.Itoa(common.ProxyHTTPPort))
|
|
httpsAddr = net.JoinHostPort(host, strconv.Itoa(common.ProxyHTTPSPort))
|
|
}
|
|
return httpAddr, httpsAddr
|
|
}
|
|
|
|
httpsAddr = route.ListenURL().Host
|
|
return
|
|
}
|
|
|
|
// AddHTTPRoute adds a HTTP route to the entrypoint's server.
|
|
//
|
|
// If the server does not exist, it will be created, started and return any error.
|
|
func (ep *Entrypoint) AddHTTPRoute(route types.HTTPRoute) error {
|
|
httpAddr, httpsAddr := getAddr(route)
|
|
var httpErr, httpsErr error
|
|
if httpAddr != "" {
|
|
httpErr = ep.addHTTPRoute(route, httpAddr, HTTPProtoHTTP)
|
|
}
|
|
if httpsAddr != "" {
|
|
httpsErr = ep.addHTTPRoute(route, httpsAddr, HTTPProtoHTTPS)
|
|
}
|
|
return errors.Join(httpErr, httpsErr)
|
|
}
|
|
|
|
func (ep *Entrypoint) addHTTPRoute(route types.HTTPRoute, addr string, proto HTTPProto) error {
|
|
return ep.addHTTPRouteWithListener(route, addr, proto, nil)
|
|
}
|
|
|
|
func (ep *Entrypoint) addHTTPRouteWithListener(route types.HTTPRoute, addr string, proto HTTPProto, listener net.Listener) error {
|
|
var err error
|
|
srv, _ := ep.servers.LoadOrCompute(addr, func() (newSrv *httpServer, cancel bool) {
|
|
newSrv = newHTTPServer(ep)
|
|
err = newSrv.listen(addr, proto, listener)
|
|
cancel = err != nil
|
|
return
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
srv.AddRoute(route)
|
|
return nil
|
|
}
|
|
|
|
func (ep *Entrypoint) delHTTPRoute(route types.HTTPRoute) {
|
|
httpAddr, httpsAddr := getAddr(route)
|
|
if httpAddr != "" {
|
|
srv, _ := ep.servers.Load(httpAddr)
|
|
if srv != nil {
|
|
srv.DelRoute(route)
|
|
}
|
|
}
|
|
if httpsAddr != "" {
|
|
srv, _ := ep.servers.Load(httpsAddr)
|
|
if srv != nil {
|
|
srv.DelRoute(route)
|
|
}
|
|
}
|
|
// TODO: close server if no routes are left
|
|
}
|