2025-03-07 15:50:57 +08:00
|
|
|
|
package fwd
|
|
|
|
|
|
|
|
|
|
|
|
import (
|
2025-05-16 15:13:16 +08:00
|
|
|
|
"bufio"
|
2025-05-19 09:41:41 +08:00
|
|
|
|
"context"
|
2025-05-23 18:59:53 +08:00
|
|
|
|
"encoding/hex"
|
2025-05-16 16:59:33 +08:00
|
|
|
|
"errors"
|
2025-05-15 09:53:23 +08:00
|
|
|
|
"fmt"
|
2025-03-07 15:50:57 +08:00
|
|
|
|
"io"
|
|
|
|
|
|
"log/slog"
|
|
|
|
|
|
"net"
|
2025-05-19 09:41:41 +08:00
|
|
|
|
"proxy-server/gateway/app"
|
2025-05-23 18:59:53 +08:00
|
|
|
|
"proxy-server/gateway/core"
|
2025-05-16 17:04:03 +08:00
|
|
|
|
"proxy-server/gateway/debug"
|
|
|
|
|
|
"proxy-server/gateway/env"
|
|
|
|
|
|
"proxy-server/gateway/fwd/metrics"
|
2025-05-19 09:41:41 +08:00
|
|
|
|
"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
|
|
|
|
)
|
|
|
|
|
|
|
2025-05-19 09:41:41 +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 {
|
2025-05-15 09:53:23 +08:00
|
|
|
|
return fmt.Errorf("监听数据通道失败: %w", err)
|
2025-03-07 15:50:57 +08:00
|
|
|
|
}
|
2025-05-19 09:41:41 +08:00
|
|
|
|
defer utils.Close(ls)
|
2025-03-07 15:50:57 +08:00
|
|
|
|
|
2025-05-16 16:59:33 +08:00
|
|
|
|
// 异步等待连接
|
|
|
|
|
|
var connCh = make(chan net.Conn)
|
2025-03-07 15:50:57 +08:00
|
|
|
|
go func() {
|
2025-05-16 16:59:33 +08:00
|
|
|
|
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:
|
2025-05-19 09:41:41 +08:00
|
|
|
|
case <-ctx.Done():
|
|
|
|
|
|
utils.Close(conn)
|
2025-05-16 16:59:33 +08:00
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-03-07 15:50:57 +08:00
|
|
|
|
}()
|
|
|
|
|
|
|
2025-05-16 16:59:33 +08:00
|
|
|
|
// 处理连接
|
2025-03-07 15:50:57 +08:00
|
|
|
|
for {
|
|
|
|
|
|
select {
|
2025-05-19 09:41:41 +08:00
|
|
|
|
case <-ctx.Done():
|
2025-03-07 15:50:57 +08:00
|
|
|
|
return nil
|
2025-05-16 16:59:33 +08:00
|
|
|
|
case conn := <-connCh:
|
2025-05-19 09:41:41 +08:00
|
|
|
|
app.DataConnWg.Add(1)
|
2025-03-07 15:50:57 +08:00
|
|
|
|
go func() {
|
2025-05-19 09:41:41 +08:00
|
|
|
|
defer app.DataConnWg.Done()
|
2025-05-23 18:59:53 +08:00
|
|
|
|
defer func() {
|
|
|
|
|
|
utils.Close(conn)
|
|
|
|
|
|
slog.Debug("关闭数据通道连接")
|
|
|
|
|
|
}()
|
2025-05-19 09:41:41 +08:00
|
|
|
|
err := processDataConn(ctx, conn)
|
2025-03-07 15:50:57 +08:00
|
|
|
|
if err != nil {
|
2025-05-16 16:59:33 +08:00
|
|
|
|
slog.Error("处理数据通道连接失败", "err", err)
|
2025-03-07 15:50:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
}()
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-26 16:37:54 +08:00
|
|
|
|
func processDataConn(ctx context.Context, edge net.Conn) error {
|
|
|
|
|
|
var reader = bufio.NewReader(edge)
|
2025-03-07 15:50:57 +08:00
|
|
|
|
|
2025-05-16 15:13:16 +08:00
|
|
|
|
// 接收连接结果
|
|
|
|
|
|
var buf = make([]byte, 17)
|
|
|
|
|
|
_, err := io.ReadFull(reader, buf)
|
2025-03-07 15:50:57 +08:00
|
|
|
|
if err != nil {
|
2025-05-17 10:00:28 +08:00
|
|
|
|
return fmt.Errorf("从节点获取连接结果失败: %w", err)
|
2025-03-07 15:50:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-23 18:59:53 +08:00
|
|
|
|
tag := hex.EncodeToString(buf[0:16])
|
2025-05-16 15:13:16 +08:00
|
|
|
|
status := buf[16]
|
2025-03-07 15:50:57 +08:00
|
|
|
|
|
2025-05-16 15:13:16 +08:00
|
|
|
|
// 加载用户连接
|
2025-05-23 18:59:53 +08:00
|
|
|
|
user, ok := app.UserConnMap.LoadAndDelete(tag)
|
2025-03-07 15:50:57 +08:00
|
|
|
|
if !ok {
|
2025-05-23 18:59:53 +08:00
|
|
|
|
return fmt.Errorf("用户连接已关闭,tag:%s", tag)
|
2025-03-07 15:50:57 +08:00
|
|
|
|
}
|
2025-05-23 18:59:53 +08:00
|
|
|
|
defer func() {
|
|
|
|
|
|
utils.Close(user)
|
|
|
|
|
|
slog.Debug("关闭用户连接")
|
|
|
|
|
|
}()
|
2025-03-07 15:50:57 +08:00
|
|
|
|
|
|
|
|
|
|
// 检查状态
|
|
|
|
|
|
if status != 1 {
|
|
|
|
|
|
return errors.New("目标地址建立连接失败")
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-16 15:13:16 +08:00
|
|
|
|
data := time.Now()
|
|
|
|
|
|
|
2025-05-23 18:59:53 +08:00
|
|
|
|
// 复制用户流量进行访问目标分析
|
|
|
|
|
|
userCopyFrom, userCopyTo := io.Pipe()
|
|
|
|
|
|
defer utils.Close(userCopyTo)
|
2025-05-16 16:59:33 +08:00
|
|
|
|
|
2025-05-23 18:59:53 +08:00
|
|
|
|
teeUser := io.TeeReader(user, userCopyTo)
|
2025-03-07 15:50:57 +08:00
|
|
|
|
go func() {
|
2025-05-23 18:59:53 +08:00
|
|
|
|
err := analysisAndLog(user, userCopyFrom)
|
2025-03-07 15:50:57 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
|
slog.Error("数据解析失败", "err", err)
|
|
|
|
|
|
}
|
|
|
|
|
|
}()
|
|
|
|
|
|
|
2025-05-23 18:59:53 +08:00
|
|
|
|
// 复制节点数据到用户
|
|
|
|
|
|
var waitEdge = make(chan error)
|
2025-03-07 15:50:57 +08:00
|
|
|
|
go func() {
|
2025-05-23 18:59:53 +08:00
|
|
|
|
_, err := io.Copy(user, reader)
|
|
|
|
|
|
switch {
|
|
|
|
|
|
case errors.Is(err, net.ErrClosed):
|
2025-05-29 14:44:06 +08:00
|
|
|
|
slog.Warn("节点连接意外关闭")
|
2025-05-23 18:59:53 +08:00
|
|
|
|
case err != nil:
|
|
|
|
|
|
slog.Error("读取节点数据失败", "err", err)
|
|
|
|
|
|
default:
|
|
|
|
|
|
slog.Debug("节点数据读取完成")
|
2025-03-07 15:50:57 +08:00
|
|
|
|
}
|
2025-05-23 18:59:53 +08:00
|
|
|
|
waitEdge <- err
|
2025-03-07 15:50:57 +08:00
|
|
|
|
}()
|
2025-05-23 18:59:53 +08:00
|
|
|
|
|
|
|
|
|
|
// 复制用户数据到节点
|
|
|
|
|
|
var waitUser = make(chan error)
|
2025-03-07 15:50:57 +08:00
|
|
|
|
go func() {
|
2025-05-26 16:37:54 +08:00
|
|
|
|
_, err := io.Copy(edge, teeUser)
|
2025-05-23 18:59:53 +08:00
|
|
|
|
switch {
|
|
|
|
|
|
case errors.Is(err, net.ErrClosed):
|
2025-05-29 14:44:06 +08:00
|
|
|
|
slog.Warn("用户连接意外关闭")
|
2025-05-23 18:59:53 +08:00
|
|
|
|
case err != nil:
|
|
|
|
|
|
slog.Error("读取用户数据失败", "err", err)
|
|
|
|
|
|
default:
|
|
|
|
|
|
slog.Debug("用户数据读取完成")
|
2025-03-07 15:50:57 +08:00
|
|
|
|
}
|
2025-05-23 18:59:53 +08:00
|
|
|
|
waitUser <- err
|
2025-03-07 15:50:57 +08:00
|
|
|
|
}()
|
|
|
|
|
|
|
2025-05-23 18:59:53 +08:00
|
|
|
|
// 等待数据转发完成,关闭数据通道的时机:
|
2025-03-07 15:50:57 +08:00
|
|
|
|
select {
|
2025-05-19 09:41:41 +08:00
|
|
|
|
case <-ctx.Done():
|
2025-05-23 18:59:53 +08:00
|
|
|
|
slog.Debug("服务关闭")
|
|
|
|
|
|
case <-waitEdge:
|
|
|
|
|
|
case <-waitUser:
|
|
|
|
|
|
storeConnMatrics(user, data)
|
|
|
|
|
|
}
|
2025-05-16 16:59:33 +08:00
|
|
|
|
|
2025-05-23 18:59:53 +08:00
|
|
|
|
return nil
|
|
|
|
|
|
}
|
2025-03-07 15:50:57 +08:00
|
|
|
|
|
2025-05-23 18:59:53 +08:00
|
|
|
|
func storeConnMatrics(user *core.Conn, data time.Time) {
|
|
|
|
|
|
proxy := time.Now()
|
2025-03-08 10:59:31 +08:00
|
|
|
|
|
2025-05-23 18:59:53 +08:00
|
|
|
|
start, startOk := metrics.TimerStart.Load(user.Conn)
|
|
|
|
|
|
auth, authOk := metrics.TimerAuth.Load(user.Conn)
|
2025-03-08 10:59:31 +08:00
|
|
|
|
|
2025-05-23 18:59:53 +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
|
|
|
|
|
2025-05-23 18:59:53 +08:00
|
|
|
|
var dataDuration time.Duration
|
|
|
|
|
|
if authOk {
|
|
|
|
|
|
dataDuration = data.Sub(auth.(time.Time))
|
|
|
|
|
|
}
|
2025-03-08 10:59:31 +08:00
|
|
|
|
|
2025-05-23 18:59:53 +08:00
|
|
|
|
proxyDuration := proxy.Sub(data)
|
2025-03-08 10:59:31 +08:00
|
|
|
|
|
2025-05-23 18:59:53 +08:00
|
|
|
|
var totalDuration time.Duration
|
|
|
|
|
|
if startOk {
|
|
|
|
|
|
totalDuration = proxy.Sub(start.(time.Time))
|
|
|
|
|
|
}
|
2025-03-08 10:59:31 +08:00
|
|
|
|
|
2025-05-23 18:59:53 +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
|
|
|
|
}
|