Files
goodlink/pro/local.go
2025-09-22 16:25:08 +08:00

207 lines
4.6 KiB
Go

package pro
import (
"errors"
"go2"
"goodlink/config"
"goodlink/netstack"
"goodlink/utils"
"goodlink2/tun"
"log"
"net"
"strings"
"time"
"github.com/quic-go/quic-go"
)
var (
m_local_state = 0 //0: 停止, 1: 启动, 2: 连接成功
)
func GetLocalQuicConn(conn *net.UDPConn, addr *tun.AddrType, count int) (*tun.TunActive, *tun.TunPassive, quic.Connection, quic.Stream, error) {
var tun_active *tun.TunActive
var tun_passive *tun.TunPassive
SessionID := string(go2.RandomBytes(24))
utils.Log().DebugF("会话ID: %s", SessionID)
redisJson := RedisJsonType{
LocalVersion: GetVersion(),
State: 0,
SessionID: SessionID,
ConnectCount: count,
}
conn_type := 0 // 被动连接
if addr.WanPort1 == addr.WanPort2 {
conn_type = 1 // 主动连接
}
switch conn_type {
case 0:
utils.Log().Debug("请求连接Remote端")
default:
redisJson.LocalAddr = *addr
utils.Log().DebugF("发送Local端地址: %v", redisJson.LocalAddr)
}
RedisSet(15*time.Second, &redisJson)
for m_local_state == 1 {
time.Sleep(1 * time.Second)
if RedisGet(&redisJson) != nil {
log.Println("会话超时")
return tun_active, tun_passive, nil, nil, nil
}
//log.Printf("状态消息: %v", redisJson)
utils.Log().SetDebugSate(redisJson.State)
if !strings.EqualFold(redisJson.SessionID, SessionID) {
utils.Log().Debug("会话被重置")
return tun_active, tun_passive, nil, nil, nil
}
switch redisJson.State {
case 1:
if redisJson.RemoteVersion != GetVersion() {
utils.Log().DebugF("两端版本不兼容: %v", redisJson)
RedisDel()
return tun_active, tun_passive, nil, nil, errors.New("两端版本不兼容")
}
utils.Log().DebugF("收到Remote端地址: %v", redisJson.RemoteAddr)
switch conn_type {
case 0:
if tun_passive != nil {
tun_passive.Release()
}
tun_active = nil
redisJson.LocalAddr = *addr
tun_passive = tun.CreateTunPassive([]byte(redisJson.SessionID), conn, &redisJson.LocalAddr, &redisJson.RemoteAddr, redisJson.SendPortCount, time.Duration(config.Arg_conn_passive_send_time)*time.Millisecond)
tun_passive.Start()
redisJson.State = 2
utils.Log().DebugF("发送Local端地址: %v", redisJson.LocalAddr)
RedisSet(redisJson.RedisTimeOut, &redisJson)
default:
if tun_active != nil {
tun_active.Release()
}
tun_passive = nil
tun_active = tun.CreateTunActive([]byte(redisJson.SessionID), conn, &redisJson.LocalAddr, &redisJson.RemoteAddr, time.Duration(config.Arg_conn_active_send_time)*time.Millisecond)
tun_active.Start()
redisJson.State = 2
RedisSet(redisJson.RedisTimeOut, &redisJson)
}
case 3:
if tun_passive != nil {
if tun_passive.TunQuicConn != nil {
utils.Log().DebugF("连接成功")
return tun_active, tun_passive, tun_passive.TunQuicConn, tun_passive.TunHealthStream, nil
}
}
if tun_active != nil {
if tun_active.TunQuicConn != nil {
utils.Log().DebugF("连接成功")
return tun_active, tun_passive, tun_active.TunQuicConn, tun_active.TunHealthStream, nil
}
}
utils.Log().Debug("连接失败")
return tun_active, tun_passive, nil, nil, nil
case 4:
utils.Log().Debug("连接超时")
return tun_active, tun_passive, nil, nil, nil
default:
utils.Log().DebugF("等待Remote端状态: Local: %v => Remote: %v", redisJson.LocalAddr, redisJson.RemoteAddr)
}
}
return tun_active, tun_passive, nil, nil, nil
}
func GetLocalStats() int {
return m_local_state
}
var (
m_tun_active *tun.TunActive
m_tun_passive *tun.TunPassive
)
func StopLocal() error {
m_local_state = 0
Release(m_tun_active, m_tun_passive)
return nil
}
func RunLocal(tun_key string) error {
m_local_state = 1
m_tun_key = tun_key
m_md5_tun_key = go2.Md5Encode(tun_key)
count := 0
var udp_conn *net.UDPConn
var addr tun.AddrType
if err := netstack.Start(); err != nil {
return err
}
for m_local_state == 1 {
if udp_conn != nil {
udp_conn.Close()
}
udp_conn, addr = GetUDPAddr()
log.Printf("Local端地址: %v", addr)
count++
tun_active, tun_passive, quic_conn, health, err := GetLocalQuicConn(udp_conn, &addr, count)
if err != nil {
Release(tun_active, tun_passive)
return err
}
if quic_conn == nil {
Release(tun_active, tun_passive)
continue
}
m_tun_active = tun_active
m_tun_passive = tun_passive
netstack.SetForWarder(quic_conn)
utils.Log().DebugF("Remote端IP: %s", netstack.GetRemoteIP())
m_local_state = 2
tun.ProcessHealth(health)
if m_local_state != 0 {
m_local_state = 1
}
utils.Log().DebugF("释放连接: %v", quic_conn.LocalAddr())
Release(tun_active, tun_passive)
netstack.SetForWarder(nil)
count = 0
}
return nil
}