first commit

This commit is contained in:
2024-07-23 10:23:43 +08:00
commit 7b4c2521a3
126 changed files with 15931 additions and 0 deletions

142
pkg/proxy/io.go Normal file
View File

@ -0,0 +1,142 @@
package proxy
import (
"context"
"golang.org/x/time/rate"
"io"
"net"
"runtime/debug"
"sync"
"time"
)
const burstLimit = 1000 * 1000 * 1000
type Reader struct {
r io.Reader
limiter *rate.Limiter
ctx context.Context
}
func NewReader(r io.Reader) *Reader {
return &Reader{
r: r,
ctx: context.Background(),
}
}
func (s *Reader) SetRateLimit(bytesPerSec float64) {
s.limiter = rate.NewLimiter(rate.Limit(bytesPerSec), burstLimit)
s.limiter.AllowN(time.Now(), burstLimit) // spend initial burst
}
func (s *Reader) Read(p []byte) (int, error) {
if s.limiter == nil {
return s.r.Read(p)
}
n, err := s.r.Read(p)
if err != nil {
return n, err
}
if err := s.limiter.WaitN(s.ctx, n); err != nil {
return n, err
}
return n, nil
}
func ConnectHost(hostAndPort string, timeout int) (conn net.Conn, err error) {
conn, err = net.DialTimeout("tcp", hostAndPort, time.Duration(timeout)*time.Millisecond)
return
}
func CloseConn(conn net.Conn) {
if conn != nil {
_ = conn.SetDeadline(time.Now().Add(time.Millisecond))
_ = conn.Close()
}
}
func IoBind(dst io.ReadWriter, src io.ReadWriter, fn func(isSrcErr bool, err error), cfn func(count int, isPositive bool), bytesPreSec float64) {
var one = &sync.Once{}
go func() {
defer func() {
if e := recover(); e != nil {
logger.Sugar().Errorf("IoBind crashed , err : %s , \ntrace:%s", e, string(debug.Stack()))
}
}()
var err error
var isSrcErr bool
if bytesPreSec > 0 {
newReader := NewReader(src)
newReader.SetRateLimit(bytesPreSec)
_, isSrcErr, err = IoCopy(dst, newReader, func(c int) {
cfn(c, false)
})
} else {
_, isSrcErr, err = IoCopy(dst, src, func(c int) {
cfn(c, false)
})
}
if err != nil {
one.Do(func() {
fn(isSrcErr, err)
})
}
}()
go func() {
defer func() {
if e := recover(); e != nil {
logger.Sugar().Errorf("IoBind crashed , err : %s , \ntrace:%s", e, string(debug.Stack()))
}
}()
var err error
var isSrcErr bool
if bytesPreSec > 0 {
newReader := NewReader(dst)
newReader.SetRateLimit(bytesPreSec)
_, isSrcErr, err = IoCopy(src, newReader, func(c int) {
cfn(c, true)
})
} else {
_, isSrcErr, err = IoCopy(src, dst, func(c int) {
cfn(c, true)
})
}
if err != nil {
one.Do(func() {
fn(isSrcErr, err)
})
}
}()
}
func IoCopy(dst io.Writer, src io.Reader, fn ...func(count int)) (written int64, isSrcErr bool, err error) {
buf := make([]byte, 32*1024)
for {
nr, er := src.Read(buf)
if nr > 0 {
nw, ew := dst.Write(buf[0:nr])
if nw > 0 {
written += int64(nw)
if len(fn) == 1 {
fn[0](nw)
}
}
if ew != nil {
err = ew
break
}
if nr != nw {
err = io.ErrShortWrite
break
}
}
if er != nil {
err = er
isSrcErr = true
break
}
}
return written, isSrcErr, err
}

105
pkg/proxy/service.go Normal file
View File

@ -0,0 +1,105 @@
package proxy
import (
"fmt"
"go.uber.org/zap"
"net"
"runtime/debug"
"sync"
)
var (
servicesMap = new(sync.Map)
logger *zap.Logger
)
type Service struct {
TCPConn TCP
Name string
}
func (s *Service) Stop() {
servicesMap.Delete(s.Name)
s.TCPConn.Close()
}
func Run(name string, args TCPArgs, zapLogger *zap.Logger) *Service {
logger = zapLogger
service := &Service{
TCPConn: &tcp{cfg: args},
Name: name,
}
store, loaded := servicesMap.LoadOrStore(name, service)
if loaded {
service = store.(*Service)
}
go func() {
defer func() {
recoverErr := recover()
if recoverErr != nil {
logger.Sugar().Errorf("%s servcie crashed, ERR: %s\ntrace:%s", name, recoverErr, string(debug.Stack()))
}
}()
startErr := service.TCPConn.Start()
if startErr != nil {
logger.Sugar().Errorf("%s servcie fail, ERR: %s", name, startErr)
}
}()
return service
}
///////////////////////////////////////////////////////////////////////////
type Listener struct {
ip string
port int
Listener net.Listener
errAcceptHandler func(err error)
}
func NewListener(ip string, port int) Listener {
return Listener{
ip: ip,
port: port,
errAcceptHandler: func(err error) {
logger.Sugar().Errorf("accept error , ERR:%s", err)
},
}
}
func (sc *Listener) ListenTCP(fn func(conn net.Conn)) (err error) {
sc.Listener, err = net.Listen("tcp", fmt.Sprintf("%s:%d", sc.ip, sc.port))
if err == nil {
go func() {
defer func() {
if e := recover(); e != nil {
logger.Sugar().Infof("ListenTCP crashed , err : %s , \ntrace:%s", e, string(debug.Stack()))
}
}()
for {
var conn net.Conn
conn, err = sc.Listener.Accept()
if err == nil {
go func() {
defer func() {
if e := recover(); e != nil {
logger.Sugar().Infof("connection handler crashed , err : %s , \ntrace:%s", e, string(debug.Stack()))
}
}()
fn(conn)
}()
} else {
sc.errAcceptHandler(err)
break
}
}
}()
}
return
}
func (sc *Listener) CloseListen() error {
return sc.Listener.Close()
}

94
pkg/proxy/tcp.go Normal file
View File

@ -0,0 +1,94 @@
package proxy
import (
"net"
"strconv"
)
var _ TCP = (*tcp)(nil)
type TCPArgs struct {
Local string //监听地址
Parent string //被代理地址
Timeout int //拨号超时(毫秒)
OutCallback func() bool //回调 //是否允许代理
}
type TCP interface {
Start() (err error)
Close()
callback(inConn net.Conn)
outToTCP(inConn net.Conn) (err error)
}
type tcp struct {
inConn net.Conn
outConn net.Conn
listen Listener
cfg TCPArgs
}
func (s *tcp) Start() (err error) {
host, port, _ := net.SplitHostPort(s.cfg.Local)
p, _ := strconv.Atoi(port)
s.listen = NewListener(host, p)
err = s.listen.ListenTCP(s.callback)
if err != nil {
return
}
return
}
func (s *tcp) Close() {
if s.inConn != nil {
CloseConn(s.inConn)
}
if s.outConn != nil {
CloseConn(s.outConn)
}
_ = s.listen.CloseListen()
}
func (s *tcp) callback(inConn net.Conn) {
defer func() {
if err := recover(); err != nil {
logger.Sugar().Infof("conn handler crashed with err : %s", err)
}
}()
if s.cfg.OutCallback != nil {
if !s.cfg.OutCallback() {
CloseConn(inConn)
return
}
}
err := s.outToTCP(inConn)
if err != nil {
CloseConn(inConn)
}
s.inConn = inConn
}
func (s *tcp) outToTCP(inConn net.Conn) error {
outConn, err := ConnectHost(s.cfg.Parent, s.cfg.Timeout)
if err != nil {
return err
}
inAddr := inConn.RemoteAddr().String()
inLocalAddr := inConn.LocalAddr().String()
outAddr := outConn.RemoteAddr().String()
outLocalAddr := outConn.LocalAddr().String()
IoBind(inConn, outConn, func(isSrcErr bool, err error) {
CloseConn(inConn)
CloseConn(outConn)
logger.Sugar().Infof("conn %s - %s - %s -%s released", inAddr, inLocalAddr, outLocalAddr, outAddr)
}, func(n int, d bool) {}, 0)
logger.Sugar().Infof("conn %s - %s - %s -%s connected", inAddr, inLocalAddr, outLocalAddr, outAddr)
return nil
}