Files
go-ts/internal/client/client.go

106 lines
2.2 KiB
Go
Raw Normal View History

2026-01-15 16:49:16 +01:00
package client
import (
"log"
"time"
"go-ts/pkg/protocol"
"go-ts/pkg/transport"
)
type Channel struct {
ID uint64
ParentID uint64
Name string
Order uint64
}
type Client struct {
Conn *transport.TS3Conn
Handshake *HandshakeState
Nickname string
ClientID uint16
// Counters
PacketIDCounterC2S uint16
// State
Connected bool
2026-01-15 17:09:32 +01:00
// Fragment reassembly
FragmentBuffer []byte
FragmentStartPktID uint16
FragmentCompressed bool
Fragmenting bool
2026-01-15 16:49:16 +01:00
// Server Data
Channels map[uint64]*Channel
}
func NewClient(nickname string) *Client {
return &Client{
Nickname: nickname,
PacketIDCounterC2S: 1,
Channels: make(map[uint64]*Channel),
}
}
func (c *Client) Connect(address string) error {
conn, err := transport.NewTS3Conn(address)
if err != nil {
return err
}
c.Conn = conn
log.Printf("Connected to UDP. Starting Handshake...")
// Initialize Handshake State
2026-01-15 16:49:16 +01:00
hs, err := NewHandshakeState(c.Conn)
if err != nil {
return err
}
c.Handshake = hs
// Improve Identity Security Level to 8 (Standard Requirement)
c.Handshake.ImproveSecurityLevel(8)
// Send Init1
2026-01-15 16:49:16 +01:00
if err := c.Handshake.SendPacket0(); err != nil {
return err
}
// Listen Loop
pktChan := c.Conn.PacketChan()
2026-01-15 16:49:16 +01:00
ticker := time.NewTicker(3 * time.Second)
defer ticker.Stop()
for {
select {
case pkt := <-pktChan:
2026-01-15 16:49:16 +01:00
if err := c.handlePacket(pkt); err != nil {
log.Printf("Error handling packet: %v", err)
}
case <-ticker.C:
ping := protocol.NewPacket(protocol.PacketTypePing, nil)
c.PacketIDCounterC2S++
2026-01-15 16:49:16 +01:00
ping.Header.PacketID = c.PacketIDCounterC2S
ping.Header.ClientID = c.ClientID
// Must NOT have NewProtocol (0x20) flag for Pings/Pongs
ping.Header.Type = uint8(protocol.PacketTypePing) | protocol.PacketFlagUnencrypted
2026-01-15 16:49:16 +01:00
// Use SharedMac if available, otherwise zeros (as per ts3j InitPacketTransformation)
if c.Handshake != nil && len(c.Handshake.SharedMac) > 0 {
copy(ping.Header.MAC[:], c.Handshake.SharedMac)
2026-01-15 16:49:16 +01:00
} else {
// Initialize Header.MAC with zeros
for i := 0; i < 8; i++ {
ping.Header.MAC[i] = 0
2026-01-15 17:09:32 +01:00
}
}
2026-01-15 16:49:16 +01:00
log.Printf("Sending KeepAlive Ping (PID=%d)", ping.Header.PacketID)
c.Conn.SendPacket(ping)
}
2026-01-15 16:49:16 +01:00
}
}