aboutsummaryrefslogtreecommitdiff
path: root/pkg/packets/packets.go
blob: 33350d5b0c837ee6eda9d7adacde65ab1720e21b (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
package packets

import (
	"github.com/pektezol/bitreader"
	"github.com/pektezol/demoparser/pkg/classes"
	"github.com/pektezol/demoparser/pkg/messages"
	"github.com/pektezol/demoparser/pkg/writer"
)

type PacketMessageInfo struct {
	PacketType uint8
	TickNumber int32
	SlotNumber uint8
	Data       any
}

const MSSC int = 2

func ParsePackets(reader *bitreader.Reader) PacketMessageInfo {
	packetType := reader.TryReadUInt8()
	tickNumber := reader.TryReadSInt32()
	slotNumber := reader.TryReadUInt8()
	var packetData any
	switch packetType {
	case 1: // SignOn
		signOn := SignOn{}
		writer.AppendLine("[%d] %s (%d):", tickNumber, "SIGNON", packetType)
		for count := 0; count < MSSC; count++ {
			signOn.PacketInfo = append(signOn.PacketInfo, classes.ParseCmdInfo(reader))
		}
		signOn.InSequence = int32(reader.TryReadBits(32))
		signOn.OutSequence = int32(reader.TryReadBits(32))
		signOn.Size = int32(reader.TryReadSInt32())
		data := reader.TryReadBytesToSlice(uint64(signOn.Size))
		packetReader := bitreader.NewReaderFromBytes(data, true)
		for {
			messageType, err := packetReader.ReadBits(6)
			if err != nil {
				break
			}
			signOn.Data = append(signOn.Data, messages.ParseMessages(int(messageType), packetReader))
		}
		packetData = signOn
	case 2: // Packet
		packet := Packet{}
		writer.AppendLine("[%d] %s (%d):", tickNumber, "PACKET", packetType)
		for count := 0; count < MSSC; count++ {
			packet.PacketInfo = append(packet.PacketInfo, classes.ParseCmdInfo(reader))
		}
		packet.InSequence = int32(reader.TryReadBits(32))
		packet.OutSequence = int32(reader.TryReadBits(32))
		packet.Size = int32(reader.TryReadSInt32())
		data := reader.TryReadBytesToSlice(uint64(packet.Size))
		packetReader := bitreader.NewReaderFromBytes(data, true)
		for {
			messageType, err := packetReader.ReadBits(6)
			if err != nil {
				break
			}
			packet.Data = append(packet.Data, messages.ParseMessages(int(messageType), packetReader))
		}
		packetData = packet
	case 3: // SyncTick
		syncTick := SyncTick{}
		writer.AppendLine("[%d] %s (%d):", tickNumber, "SYNCTICK", packetType)
		packetData = syncTick
	case 4: // ConsoleCmd
		consoleCmd := ConsoleCmd{}
		writer.AppendLine("[%d] %s (%d):", tickNumber, "CONSOLECMD", packetType)
		consoleCmd.Size = reader.TryReadSInt32()
		consoleCmd.Data = reader.TryReadStringLength(uint64(consoleCmd.Size))
		writer.AppendLine("\t%s", consoleCmd.Data)
		packetData = consoleCmd
	case 5: // UserCmd TODO: usercmdinfo refactor
		userCmd := UserCmd{}
		writer.AppendLine("[%d] %s (%d):", tickNumber, "USERCMD", packetType)
		userCmd.Cmd = int32(reader.TryReadSInt32())
		userCmd.Size = int32(reader.TryReadSInt32())
		data := reader.TryReadBytesToSlice(uint64(userCmd.Size))
		userCmd.Data = classes.ParseUserCmdInfo(data)
		packetData = userCmd
	case 6: // DataTables TODO: prop stuff
		dataTables := DataTables{}
		writer.AppendLine("[%d] %s (%d):", tickNumber, "DATATABLES", packetType)
		dataTables.Size = int32(reader.TryReadSInt32())
		data := reader.TryReadBytesToSlice(uint64(dataTables.Size))
		dataTableReader := bitreader.NewReaderFromBytes(data, true)
		count := 0
		for dataTableReader.TryReadBool() {
			count++
			dataTables.SendTable = append(dataTables.SendTable, classes.ParseSendTable(dataTableReader))
		}
		writer.AppendLine("\t%d Send Tables:", count)
		writer.AppendOutputFromTemp()
		numOfClasses := dataTableReader.TryReadBits(16)
		for count = 0; count < int(numOfClasses); count++ {
			dataTables.ServerClassInfo = append(dataTables.ServerClassInfo, classes.ParseServerClassInfo(dataTableReader, count, int(numOfClasses)))
		}
		writer.AppendLine("\t%d Classes:", count)
		writer.AppendOutputFromTemp()
		packetData = dataTables
	case 7: // Stop
		stop := Stop{}
		writer.AppendLine("[%d] %s (%d):", tickNumber, "STOP", packetType)
		if reader.TryReadBool() {
			stop.RemainingData = reader.TryReadBitsToSlice(uint64(reader.TryReadRemainingBits()))
			writer.AppendLine("\tRemaining Data: %v", stop.RemainingData)
		}
		packetData = stop
	case 8: // CustomData TODO: not sar data
		customData := CustomData{}
		customData.Type = reader.TryReadSInt32()
		customData.Size = reader.TryReadSInt32()
		if customData.Type != 0 || customData.Size == 8 {
			// Not SAR data
			writer.AppendLine("[%d] %s (%d):", tickNumber, "CUSTOMDATA", packetType)
			customData.Data = string(reader.TryReadBytesToSlice(uint64(customData.Size)))
			writer.AppendLine("\t%s", customData.Data)
			packetData = customData
			break
		}
		// SAR data
		sarData := classes.SarData{}
		writer.AppendLine("[%d] %s (%d):", tickNumber, "SARDATA", packetType)
		data := reader.TryReadBytesToSlice(uint64(customData.Size))
		sarReader := bitreader.NewReaderFromBytes(data, true)
		sarData.ParseSarData(sarReader)
		packetData = sarData
	case 9: // StringTables TODO: parsing string table data
		stringTables := StringTables{}
		writer.AppendLine("[%d] %s (%d):", tickNumber, "STRINGTABLES", packetType)
		stringTables.Size = reader.TryReadSInt32()
		data := reader.TryReadBytesToSlice(uint64(stringTables.Size))
		stringTableReader := bitreader.NewReaderFromBytes(data, true)
		stringTables.Data = classes.ParseStringTables(stringTableReader)
		packetData = stringTables
	default: // invalid
		writer.AppendLine("[%d] %s (%d):", tickNumber, "INVALID", packetType)
		panic("invalid packet type")
	}
	return PacketMessageInfo{
		PacketType: packetType,
		TickNumber: tickNumber,
		SlotNumber: slotNumber,
		Data:       packetData,
	}
}