add admission option for listener

This commit is contained in:
ginuerzh 2022-03-13 00:08:16 +08:00
parent 4007e80811
commit bfc1f8472c
22 changed files with 348 additions and 5 deletions

View File

@ -58,6 +58,7 @@ func NewAdmissionPatterns(reversed bool, patterns []string, opts ...Option) Admi
func (p *admission) Admit(addr string) bool { func (p *admission) Admit(addr string) bool {
if addr == "" || p == nil || len(p.matchers) == 0 { if addr == "" || p == nil || len(p.matchers) == 0 {
p.options.logger.Debugf("admission: %v is denied", addr)
return false return false
} }
@ -81,5 +82,8 @@ func (p *admission) Admit(addr string) bool {
b := !p.reversed && matched || b := !p.reversed && matched ||
p.reversed && !matched p.reversed && !matched
if !b {
p.options.logger.Debugf("admission: %v is denied", addr)
}
return b return b
} }

View File

@ -0,0 +1,223 @@
package admission
import (
"errors"
"io"
"net"
"syscall"
"github.com/go-gost/gost/pkg/admission"
)
var (
errUnsupport = errors.New("unsupported operation")
)
type packetConn struct {
net.PacketConn
admission admission.Admission
}
func WrapPacketConn(admission admission.Admission, pc net.PacketConn) net.PacketConn {
if admission == nil {
return pc
}
return &packetConn{
PacketConn: pc,
admission: admission,
}
}
func (c *packetConn) ReadFrom(p []byte) (n int, addr net.Addr, err error) {
for {
n, addr, err = c.PacketConn.ReadFrom(p)
if err != nil {
return
}
if c.admission != nil &&
!c.admission.Admit(addr.String()) {
continue
}
return
}
}
type udpConn struct {
net.PacketConn
admission admission.Admission
}
func WrapUDPConn(admission admission.Admission, pc net.PacketConn) UDPConn {
return &udpConn{
PacketConn: pc,
admission: admission,
}
}
func (c *udpConn) RemoteAddr() net.Addr {
if nc, ok := c.PacketConn.(remoteAddr); ok {
return nc.RemoteAddr()
}
return nil
}
func (c *udpConn) SetReadBuffer(n int) error {
if nc, ok := c.PacketConn.(setBuffer); ok {
return nc.SetReadBuffer(n)
}
return errUnsupport
}
func (c *udpConn) SetWriteBuffer(n int) error {
if nc, ok := c.PacketConn.(setBuffer); ok {
return nc.SetWriteBuffer(n)
}
return errUnsupport
}
func (c *udpConn) Read(b []byte) (n int, err error) {
if nc, ok := c.PacketConn.(io.Reader); ok {
n, err = nc.Read(b)
return
}
err = errUnsupport
return
}
func (c *udpConn) ReadFrom(p []byte) (n int, addr net.Addr, err error) {
for {
n, addr, err = c.PacketConn.ReadFrom(p)
if err != nil {
return
}
if c.admission != nil &&
!c.admission.Admit(addr.String()) {
continue
}
return
}
}
func (c *udpConn) ReadFromUDP(b []byte) (n int, addr *net.UDPAddr, err error) {
if nc, ok := c.PacketConn.(readUDP); ok {
for {
n, addr, err = nc.ReadFromUDP(b)
if err != nil {
return
}
if c.admission != nil &&
!c.admission.Admit(addr.String()) {
continue
}
return
}
}
err = errUnsupport
return
}
func (c *udpConn) ReadMsgUDP(b, oob []byte) (n, oobn, flags int, addr *net.UDPAddr, err error) {
if nc, ok := c.PacketConn.(readUDP); ok {
for {
n, oobn, flags, addr, err = nc.ReadMsgUDP(b, oob)
if err != nil {
return
}
if c.admission != nil &&
!c.admission.Admit(addr.String()) {
continue
}
return
}
}
err = errUnsupport
return
}
func (c *udpConn) Write(b []byte) (n int, err error) {
if nc, ok := c.PacketConn.(io.Writer); ok {
n, err = nc.Write(b)
return
}
err = errUnsupport
return
}
func (c *udpConn) WriteTo(p []byte, addr net.Addr) (n int, err error) {
n, err = c.PacketConn.WriteTo(p, addr)
return
}
func (c *udpConn) WriteToUDP(b []byte, addr *net.UDPAddr) (n int, err error) {
if nc, ok := c.PacketConn.(writeUDP); ok {
n, err = nc.WriteToUDP(b, addr)
return
}
err = errUnsupport
return
}
func (c *udpConn) WriteMsgUDP(b, oob []byte, addr *net.UDPAddr) (n, oobn int, err error) {
if nc, ok := c.PacketConn.(writeUDP); ok {
n, oobn, err = nc.WriteMsgUDP(b, oob, addr)
return
}
err = errUnsupport
return
}
func (c *udpConn) SyscallConn() (rc syscall.RawConn, err error) {
if nc, ok := c.PacketConn.(syscallConn); ok {
return nc.SyscallConn()
}
err = errUnsupport
return
}
func (c *udpConn) SetDSCP(n int) error {
if nc, ok := c.PacketConn.(setDSCP); ok {
return nc.SetDSCP(n)
}
return nil
}
type UDPConn interface {
net.PacketConn
io.Reader
io.Writer
readUDP
writeUDP
setBuffer
syscallConn
remoteAddr
}
type setBuffer interface {
SetReadBuffer(bytes int) error
SetWriteBuffer(bytes int) error
}
type readUDP interface {
ReadFromUDP(b []byte) (n int, addr *net.UDPAddr, err error)
ReadMsgUDP(b, oob []byte) (n, oobn, flags int, addr *net.UDPAddr, err error)
}
type writeUDP interface {
WriteToUDP(b []byte, addr *net.UDPAddr) (int, error)
WriteMsgUDP(b, oob []byte, addr *net.UDPAddr) (n, oobn int, err error)
}
type syscallConn interface {
SyscallConn() (syscall.RawConn, error)
}
type remoteAddr interface {
RemoteAddr() net.Addr
}
// tcpraw.TCPConn
type setDSCP interface {
SetDSCP(int) error
}

View File

@ -0,0 +1,37 @@
package admission
import (
"net"
"github.com/go-gost/gost/pkg/admission"
)
type listener struct {
net.Listener
admission admission.Admission
}
func WrapListener(admission admission.Admission, ln net.Listener) net.Listener {
if admission == nil {
return ln
}
return &listener{
Listener: ln,
admission: admission,
}
}
func (ln *listener) Accept() (net.Conn, error) {
for {
c, err := ln.Listener.Accept()
if err != nil {
return nil, err
}
if ln.admission != nil &&
!ln.admission.Admit(c.RemoteAddr().String()) {
c.Close()
continue
}
return c, err
}
}

41
pkg/common/net/udp.go Normal file
View File

@ -0,0 +1,41 @@
package net
import (
"io"
"net"
"syscall"
)
type UDPConn interface {
net.PacketConn
io.Reader
io.Writer
readUDP
writeUDP
setBuffer
syscallConn
remoteAddr
}
type setBuffer interface {
SetReadBuffer(bytes int) error
SetWriteBuffer(bytes int) error
}
type readUDP interface {
ReadFromUDP(b []byte) (n int, addr *net.UDPAddr, err error)
ReadMsgUDP(b, oob []byte) (n, oobn, flags int, addr *net.UDPAddr, err error)
}
type writeUDP interface {
WriteToUDP(b []byte, addr *net.UDPAddr) (int, error)
WriteMsgUDP(b, oob []byte, addr *net.UDPAddr) (n, oobn int, err error)
}
type syscallConn interface {
SyscallConn() (syscall.RawConn, error)
}
type remoteAddr interface {
RemoteAddr() net.Addr
}

View File

@ -54,10 +54,11 @@ func ParseService(cfg *config.ServiceConfig) (service.Service, error) {
ln := registry.ListenerRegistry().Get(cfg.Listener.Type)( ln := registry.ListenerRegistry().Get(cfg.Listener.Type)(
listener.AddrOption(cfg.Addr), listener.AddrOption(cfg.Addr),
listener.ChainOption(registry.ChainRegistry().Get(cfg.Listener.Chain)),
listener.AutherOption(auther), listener.AutherOption(auther),
listener.AuthOption(parseAuth(cfg.Listener.Auth)), listener.AuthOption(parseAuth(cfg.Listener.Auth)),
listener.TLSConfigOption(tlsConfig), listener.TLSConfigOption(tlsConfig),
listener.AdmissionOption(registry.AdmissionRegistry().Get(cfg.Admission)),
listener.ChainOption(registry.ChainRegistry().Get(cfg.Listener.Chain)),
listener.LoggerOption(listenerLogger), listener.LoggerOption(listenerLogger),
listener.ServiceOption(cfg.Name), listener.ServiceOption(cfg.Name),
) )

View File

@ -3,6 +3,7 @@ package grpc
import ( import (
"net" "net"
"github.com/go-gost/gost/pkg/common/admission"
"github.com/go-gost/gost/pkg/common/metrics" "github.com/go-gost/gost/pkg/common/metrics"
pb "github.com/go-gost/gost/pkg/common/util/grpc/proto" pb "github.com/go-gost/gost/pkg/common/util/grpc/proto"
"github.com/go-gost/gost/pkg/listener" "github.com/go-gost/gost/pkg/listener"
@ -48,6 +49,7 @@ func (l *grpcListener) Init(md md.Metadata) (err error) {
return return
} }
ln = metrics.WrapListener(l.options.Service, ln) ln = metrics.WrapListener(l.options.Service, ln)
ln = admission.WrapListener(l.options.Admission, ln)
var opts []grpc.ServerOption var opts []grpc.ServerOption
if !l.md.insecure { if !l.md.insecure {

View File

@ -7,6 +7,7 @@ import (
"net/http" "net/http"
"net/http/httputil" "net/http/httputil"
"github.com/go-gost/gost/pkg/common/admission"
"github.com/go-gost/gost/pkg/common/metrics" "github.com/go-gost/gost/pkg/common/metrics"
"github.com/go-gost/gost/pkg/listener" "github.com/go-gost/gost/pkg/listener"
"github.com/go-gost/gost/pkg/logger" "github.com/go-gost/gost/pkg/logger"
@ -70,6 +71,7 @@ func (l *h2Listener) Init(md md.Metadata) (err error) {
} }
l.addr = ln.Addr() l.addr = ln.Addr()
ln = metrics.WrapListener(l.options.Service, ln) ln = metrics.WrapListener(l.options.Service, ln)
ln = admission.WrapListener(l.options.Admission, ln)
if l.h2c { if l.h2c {
l.server.Handler = h2c.NewHandler( l.server.Handler = h2c.NewHandler(

View File

@ -5,6 +5,7 @@ import (
"net" "net"
"net/http" "net/http"
"github.com/go-gost/gost/pkg/common/admission"
"github.com/go-gost/gost/pkg/common/metrics" "github.com/go-gost/gost/pkg/common/metrics"
http2_util "github.com/go-gost/gost/pkg/internal/util/http2" http2_util "github.com/go-gost/gost/pkg/internal/util/http2"
"github.com/go-gost/gost/pkg/listener" "github.com/go-gost/gost/pkg/listener"
@ -59,6 +60,7 @@ func (l *http2Listener) Init(md md.Metadata) (err error) {
} }
l.addr = ln.Addr() l.addr = ln.Addr()
ln = metrics.WrapListener(l.options.Service, ln) ln = metrics.WrapListener(l.options.Service, ln)
ln = admission.WrapListener(l.options.Admission, ln)
ln = tls.NewListener( ln = tls.NewListener(
ln, ln,

View File

@ -4,6 +4,7 @@ import (
"context" "context"
"net" "net"
"github.com/go-gost/gost/pkg/common/admission"
"github.com/go-gost/gost/pkg/common/metrics" "github.com/go-gost/gost/pkg/common/metrics"
icmp_pkg "github.com/go-gost/gost/pkg/internal/util/icmp" icmp_pkg "github.com/go-gost/gost/pkg/internal/util/icmp"
"github.com/go-gost/gost/pkg/listener" "github.com/go-gost/gost/pkg/listener"
@ -55,6 +56,7 @@ func (l *icmpListener) Init(md md.Metadata) (err error) {
} }
conn = icmp_pkg.ServerConn(conn) conn = icmp_pkg.ServerConn(conn)
conn = metrics.WrapPacketConn(l.options.Service, conn) conn = metrics.WrapPacketConn(l.options.Service, conn)
conn = admission.WrapPacketConn(l.options.Admission, conn)
config := &quic.Config{ config := &quic.Config{
KeepAlive: l.md.keepAlive, KeepAlive: l.md.keepAlive,

View File

@ -4,6 +4,7 @@ import (
"net" "net"
"time" "time"
"github.com/go-gost/gost/pkg/common/admission"
"github.com/go-gost/gost/pkg/common/metrics" "github.com/go-gost/gost/pkg/common/metrics"
kcp_util "github.com/go-gost/gost/pkg/common/util/kcp" kcp_util "github.com/go-gost/gost/pkg/common/util/kcp"
"github.com/go-gost/gost/pkg/listener" "github.com/go-gost/gost/pkg/listener"
@ -64,6 +65,7 @@ func (l *kcpListener) Init(md md.Metadata) (err error) {
} }
conn = metrics.WrapUDPConn(l.options.Service, conn) conn = metrics.WrapUDPConn(l.options.Service, conn)
conn = admission.WrapUDPConn(l.options.Admission, conn)
ln, err := kcp.ServeConn( ln, err := kcp.ServeConn(
kcp_util.BlockCrypt(config.Key, config.Crypt, kcp_util.DefaultSalt), kcp_util.BlockCrypt(config.Key, config.Crypt, kcp_util.DefaultSalt),

View File

@ -3,6 +3,7 @@ package http
import ( import (
"net" "net"
"github.com/go-gost/gost/pkg/common/admission"
"github.com/go-gost/gost/pkg/common/metrics" "github.com/go-gost/gost/pkg/common/metrics"
"github.com/go-gost/gost/pkg/listener" "github.com/go-gost/gost/pkg/listener"
"github.com/go-gost/gost/pkg/logger" "github.com/go-gost/gost/pkg/logger"
@ -42,6 +43,7 @@ func (l *obfsListener) Init(md md.Metadata) (err error) {
return return
} }
ln = metrics.WrapListener(l.options.Service, ln) ln = metrics.WrapListener(l.options.Service, ln)
ln = admission.WrapListener(l.options.Admission, ln)
l.Listener = ln l.Listener = ln
return return

View File

@ -3,6 +3,7 @@ package tls
import ( import (
"net" "net"
"github.com/go-gost/gost/pkg/common/admission"
"github.com/go-gost/gost/pkg/common/metrics" "github.com/go-gost/gost/pkg/common/metrics"
"github.com/go-gost/gost/pkg/listener" "github.com/go-gost/gost/pkg/listener"
"github.com/go-gost/gost/pkg/logger" "github.com/go-gost/gost/pkg/logger"
@ -42,6 +43,7 @@ func (l *obfsListener) Init(md md.Metadata) (err error) {
return return
} }
ln = metrics.WrapListener(l.options.Service, ln) ln = metrics.WrapListener(l.options.Service, ln)
ln = admission.WrapListener(l.options.Admission, ln)
l.Listener = ln l.Listener = ln
return return

View File

@ -4,6 +4,7 @@ import (
"crypto/tls" "crypto/tls"
"net/url" "net/url"
"github.com/go-gost/gost/pkg/admission"
"github.com/go-gost/gost/pkg/auth" "github.com/go-gost/gost/pkg/auth"
"github.com/go-gost/gost/pkg/chain" "github.com/go-gost/gost/pkg/chain"
"github.com/go-gost/gost/pkg/logger" "github.com/go-gost/gost/pkg/logger"
@ -14,6 +15,7 @@ type Options struct {
Auther auth.Authenticator Auther auth.Authenticator
Auth *url.Userinfo Auth *url.Userinfo
TLSConfig *tls.Config TLSConfig *tls.Config
Admission admission.Admission
Chain chain.Chainer Chain chain.Chainer
Logger logger.Logger Logger logger.Logger
Service string Service string
@ -45,6 +47,12 @@ func TLSConfigOption(tlsConfig *tls.Config) Option {
} }
} }
func AdmissionOption(admission admission.Admission) Option {
return func(opts *Options) {
opts.Admission = admission
}
}
func ChainOption(chain chain.Chainer) Option { func ChainOption(chain chain.Chainer) Option {
return func(opts *Options) { return func(opts *Options) {
opts.Chain = chain opts.Chain = chain

View File

@ -5,6 +5,7 @@ import (
"net" "net"
"time" "time"
"github.com/go-gost/gost/pkg/common/admission"
"github.com/go-gost/gost/pkg/common/metrics" "github.com/go-gost/gost/pkg/common/metrics"
ssh_util "github.com/go-gost/gost/pkg/internal/util/ssh" ssh_util "github.com/go-gost/gost/pkg/internal/util/ssh"
"github.com/go-gost/gost/pkg/listener" "github.com/go-gost/gost/pkg/listener"
@ -50,6 +51,7 @@ func (l *sshListener) Init(md md.Metadata) (err error) {
} }
ln = metrics.WrapListener(l.options.Service, ln) ln = metrics.WrapListener(l.options.Service, ln)
ln = admission.WrapListener(l.options.Admission, ln)
l.Listener = ln l.Listener = ln
config := &ssh.ServerConfig{ config := &ssh.ServerConfig{

View File

@ -7,6 +7,7 @@ import (
"strconv" "strconv"
"time" "time"
"github.com/go-gost/gost/pkg/common/admission"
"github.com/go-gost/gost/pkg/common/metrics" "github.com/go-gost/gost/pkg/common/metrics"
ssh_util "github.com/go-gost/gost/pkg/internal/util/ssh" ssh_util "github.com/go-gost/gost/pkg/internal/util/ssh"
sshd_util "github.com/go-gost/gost/pkg/internal/util/sshd" sshd_util "github.com/go-gost/gost/pkg/internal/util/sshd"
@ -59,6 +60,7 @@ func (l *sshdListener) Init(md md.Metadata) (err error) {
} }
ln = metrics.WrapListener(l.options.Service, ln) ln = metrics.WrapListener(l.options.Service, ln)
ln = admission.WrapListener(l.options.Admission, ln)
l.Listener = ln l.Listener = ln
config := &ssh.ServerConfig{ config := &ssh.ServerConfig{

View File

@ -41,7 +41,9 @@ func (l *tcpListener) Init(md md.Metadata) (err error) {
if err != nil { if err != nil {
return return
} }
l.Listener = metrics.WrapListener(l.options.Service, ln)
ln = metrics.WrapListener(l.options.Service, ln)
l.Listener = ln
return return
} }

View File

@ -4,6 +4,7 @@ import (
"crypto/tls" "crypto/tls"
"net" "net"
"github.com/go-gost/gost/pkg/common/admission"
"github.com/go-gost/gost/pkg/common/metrics" "github.com/go-gost/gost/pkg/common/metrics"
"github.com/go-gost/gost/pkg/listener" "github.com/go-gost/gost/pkg/listener"
"github.com/go-gost/gost/pkg/logger" "github.com/go-gost/gost/pkg/logger"
@ -43,6 +44,7 @@ func (l *tlsListener) Init(md md.Metadata) (err error) {
return return
} }
ln = metrics.WrapListener(l.options.Service, ln) ln = metrics.WrapListener(l.options.Service, ln)
ln = admission.WrapListener(l.options.Admission, ln)
l.Listener = tls.NewListener(ln, l.options.TLSConfig) l.Listener = tls.NewListener(ln, l.options.TLSConfig)

View File

@ -4,6 +4,7 @@ import (
"crypto/tls" "crypto/tls"
"net" "net"
"github.com/go-gost/gost/pkg/common/admission"
"github.com/go-gost/gost/pkg/common/metrics" "github.com/go-gost/gost/pkg/common/metrics"
"github.com/go-gost/gost/pkg/listener" "github.com/go-gost/gost/pkg/listener"
"github.com/go-gost/gost/pkg/logger" "github.com/go-gost/gost/pkg/logger"
@ -47,6 +48,7 @@ func (l *mtlsListener) Init(md md.Metadata) (err error) {
} }
ln = metrics.WrapListener(l.options.Service, ln) ln = metrics.WrapListener(l.options.Service, ln)
ln = admission.WrapListener(l.options.Admission, ln)
l.Listener = tls.NewListener(ln, l.options.TLSConfig) l.Listener = tls.NewListener(ln, l.options.TLSConfig)
l.cqueue = make(chan net.Conn, l.md.backlog) l.cqueue = make(chan net.Conn, l.md.backlog)

View File

@ -43,13 +43,15 @@ func (l *udpListener) Init(md md.Metadata) (err error) {
return return
} }
conn, err := net.ListenUDP("udp", laddr) var conn net.PacketConn
conn, err = net.ListenUDP("udp", laddr)
if err != nil { if err != nil {
return return
} }
conn = metrics.WrapPacketConn(l.options.Service, conn)
l.Listener = udp.NewListener( l.Listener = udp.NewListener(
metrics.WrapPacketConn(l.options.Service, conn), conn,
laddr, laddr,
l.md.backlog, l.md.backlog,
l.md.readQueueSize, l.md.readBufferSize, l.md.readQueueSize, l.md.readBufferSize,

View File

@ -6,6 +6,7 @@ import (
"net/http" "net/http"
"net/http/httputil" "net/http/httputil"
"github.com/go-gost/gost/pkg/common/admission"
"github.com/go-gost/gost/pkg/common/metrics" "github.com/go-gost/gost/pkg/common/metrics"
ws_util "github.com/go-gost/gost/pkg/internal/util/ws" ws_util "github.com/go-gost/gost/pkg/internal/util/ws"
"github.com/go-gost/gost/pkg/listener" "github.com/go-gost/gost/pkg/listener"
@ -84,6 +85,7 @@ func (l *wsListener) Init(md md.Metadata) (err error) {
return return
} }
ln = metrics.WrapListener(l.options.Service, ln) ln = metrics.WrapListener(l.options.Service, ln)
ln = admission.WrapListener(l.options.Admission, ln)
if l.tlsEnabled { if l.tlsEnabled {
ln = tls.NewListener(ln, l.options.TLSConfig) ln = tls.NewListener(ln, l.options.TLSConfig)

View File

@ -6,6 +6,7 @@ import (
"net/http" "net/http"
"net/http/httputil" "net/http/httputil"
"github.com/go-gost/gost/pkg/common/admission"
"github.com/go-gost/gost/pkg/common/metrics" "github.com/go-gost/gost/pkg/common/metrics"
ws_util "github.com/go-gost/gost/pkg/internal/util/ws" ws_util "github.com/go-gost/gost/pkg/internal/util/ws"
"github.com/go-gost/gost/pkg/listener" "github.com/go-gost/gost/pkg/listener"
@ -89,6 +90,7 @@ func (l *mwsListener) Init(md md.Metadata) (err error) {
return return
} }
ln = metrics.WrapListener(l.options.Service, ln) ln = metrics.WrapListener(l.options.Service, ln)
ln = admission.WrapListener(l.options.Admission, ln)
if l.tlsEnabled { if l.tlsEnabled {
ln = tls.NewListener(ln, l.options.TLSConfig) ln = tls.NewListener(ln, l.options.TLSConfig)

View File

@ -93,7 +93,6 @@ func (s *service) Serve() error {
if s.options.admission != nil && if s.options.admission != nil &&
!s.options.admission.Admit(conn.RemoteAddr().String()) { !s.options.admission.Admit(conn.RemoteAddr().String()) {
s.options.logger.Infof("admission: %s is denied", conn.RemoteAddr())
conn.Close() conn.Close()
continue continue
} }