Files
proxy/gateway/fwd/data.go

178 lines
3.5 KiB
Go
Raw Normal View History

2025-03-07 15:50:57 +08:00
package fwd
import (
"bufio"
"context"
"encoding/hex"
"errors"
"fmt"
2025-03-07 15:50:57 +08:00
"io"
"log/slog"
"net"
"proxy-server/gateway/app"
"proxy-server/gateway/core"
"proxy-server/gateway/debug"
"proxy-server/gateway/env"
"proxy-server/gateway/fwd/metrics"
"proxy-server/utils"
2025-03-07 15:50:57 +08:00
"strconv"
2025-03-08 10:59:31 +08:00
"time"
2025-03-07 15:50:57 +08:00
)
func ListenData(ctx context.Context) error {
2025-03-07 15:50:57 +08:00
dataPort := env.AppDataPort
// 监听端口
ls, err := net.Listen("tcp", ":"+strconv.Itoa(int(dataPort)))
if err != nil {
return fmt.Errorf("监听数据通道失败: %w", err)
2025-03-07 15:50:57 +08:00
}
defer utils.Close(ls)
2025-03-07 15:50:57 +08:00
// 异步等待连接
var connCh = make(chan net.Conn)
2025-03-07 15:50:57 +08:00
go func() {
for {
conn, err := ls.Accept()
if errors.Is(err, net.ErrClosed) {
slog.Debug("数据通道监听关闭")
return
}
if err != nil {
slog.Error("接受数据通道连接失败", "err", err)
return
}
select {
case connCh <- conn:
case <-ctx.Done():
utils.Close(conn)
return
}
}
2025-03-07 15:50:57 +08:00
}()
// 处理连接
2025-03-07 15:50:57 +08:00
for {
select {
case <-ctx.Done():
2025-03-07 15:50:57 +08:00
return nil
case conn := <-connCh:
app.DataConnWg.Add(1)
2025-03-07 15:50:57 +08:00
go func() {
defer app.DataConnWg.Done()
defer utils.Close(conn)
err := processDataConn(ctx, conn)
2025-03-07 15:50:57 +08:00
if err != nil {
slog.Error("处理数据通道连接失败", "err", err)
2025-03-07 15:50:57 +08:00
}
}()
}
}
}
func processDataConn(ctx context.Context, edge net.Conn) error {
var reader = bufio.NewReader(edge)
2025-03-07 15:50:57 +08:00
// 接收连接结果
var buf = make([]byte, 17)
_, err := io.ReadFull(reader, buf)
2025-03-07 15:50:57 +08:00
if err != nil {
return fmt.Errorf("从节点获取连接结果失败: %w", err)
2025-03-07 15:50:57 +08:00
}
tag := hex.EncodeToString(buf[0:16])
status := buf[16]
2025-03-07 15:50:57 +08:00
// 加载用户连接
user, ok := app.UserConnMap.LoadAndDelete(tag)
2025-03-07 15:50:57 +08:00
if !ok {
if status == 1 {
return fmt.Errorf("用户连接已关闭tag%s", tag)
} else {
return nil
}
2025-03-07 15:50:57 +08:00
}
defer utils.Close(user)
2025-03-07 15:50:57 +08:00
// 检查状态
if status != 1 {
return errors.New("目标地址建立连接失败")
}
data := time.Now()
// 复制用户流量进行访问目标分析
userCopyFrom, userCopyTo := io.Pipe()
defer utils.Close(userCopyTo)
teeUser := io.TeeReader(user, userCopyTo)
2025-03-07 15:50:57 +08:00
go func() {
err := analysisAndLog(user, userCopyFrom)
2025-03-07 15:50:57 +08:00
if err != nil {
slog.Error("数据解析失败", "err", err)
}
}()
// 复制节点数据到用户
var waitEdge = make(chan error)
2025-03-07 15:50:57 +08:00
go func() {
_, err := io.Copy(user, reader)
if ok, err := utils.WarpConnErr(err); !ok {
slog.Error("读取节点数据失败", "err", err)
2025-03-07 15:50:57 +08:00
}
waitEdge <- err
2025-03-07 15:50:57 +08:00
}()
// 复制用户数据到节点
var waitUser = make(chan error)
2025-03-07 15:50:57 +08:00
go func() {
_, err := io.Copy(edge, teeUser)
if ok, err := utils.WarpConnErr(err); !ok {
slog.Error("读取用户数据失败", "err", err)
2025-03-07 15:50:57 +08:00
}
waitUser <- err
2025-03-07 15:50:57 +08:00
}()
// 等待数据转发完成,关闭数据通道的时机:
2025-03-07 15:50:57 +08:00
select {
case <-ctx.Done():
slog.Debug("服务关闭")
case <-waitEdge:
case <-waitUser:
storeConnMatrics(user, data)
}
return nil
}
2025-03-07 15:50:57 +08:00
func storeConnMatrics(user *core.Conn, data time.Time) {
proxy := time.Now()
2025-03-08 10:59:31 +08:00
start, startOk := metrics.TimerStart.Load(user.Conn)
auth, authOk := metrics.TimerAuth.Load(user.Conn)
2025-03-08 10:59:31 +08:00
var authDuration time.Duration
if startOk && authOk {
authDuration = auth.(time.Time).Sub(start.(time.Time))
}
2025-03-08 10:59:31 +08:00
var dataDuration time.Duration
if authOk {
dataDuration = data.Sub(auth.(time.Time))
}
2025-03-08 10:59:31 +08:00
proxyDuration := proxy.Sub(data)
2025-03-08 10:59:31 +08:00
var totalDuration time.Duration
if startOk {
totalDuration = proxy.Sub(start.(time.Time))
}
2025-03-08 10:59:31 +08:00
debug.ConsumingCh <- debug.Consuming{
Auth: authDuration,
Data: dataDuration,
Proxy: proxyDuration,
Total: totalDuration,
2025-03-08 10:59:31 +08:00
}
2025-03-07 15:50:57 +08:00
}