-
Notifications
You must be signed in to change notification settings - Fork 196
/
index.js
116 lines (111 loc) · 3.99 KB
/
index.js
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
const backend = require('./backend')
const PacketByPacket = require('p-by-p')
const PacketParser = require('./backend/packetparser')
const utils = require('./backend/utils')
const gamestate = require('./backend/gamestate')
const bunchStash = require('./backend/bunchstash')
const CONSTS = require('./backend/constants')
const Cap = require('cap').Cap
function printUsage () {
console.log(
`Usage:
* node index.js sniff <interface> <game_pc_ip>
* node index.js playback <pcapfile>
Example: node index.js playback '/SOME/COOL/DIR/pubg-game1.pcap' | pino
node index.js sniff en5 192.168.0.100 | pino`
)
process.exit(1)
}
function startWebServer () {
const apiServerPort = 20086
backend.listen(apiServerPort, () => {
console.log('科学吃鸡 listening on http://localhost:' + apiServerPort)
})
}
function printStateOnConsole() {
setInterval(() => {
console.log(`[${utils.toLocalISOString(new Date())}] - playbackState: ${gamestate.playbackState}, speed: ${gamestate.playbackSpeed}, playbackIndex: ${gamestate.playbackIndex}, processedEvents: ${gamestate.totalProcessedEvents}/${gamestate.playbackEvents.length}`)
}, 5000)
}
function main () {
const args = process.argv.slice(2)
if (args.length < 2 || !['sniff', 'playback'].includes(args[0])) {
printUsage()
}
if (args[0] === 'sniff') {
if (args.length !== 3) {
printUsage()
}
const c = new Cap()
const device = args[1]
const filter = `(src host ${args[2]} and udp dst portrange 7000-7999) or (dst host ${args[2]} and udp src portrange 7000-7999)`
const bufSize = 10 * 1024 * 1024
const capBuffer = Buffer.alloc(65535)
const linkType = c.open(device, filter, bufSize, capBuffer)
c.setMinBytes && c.setMinBytes(0)
const parser = PacketParser()
let pIndex = 0
c.on('packet', function (nbytes, trunc) {
// raw packet data === buffer.slice(0, nbytes)
pIndex++
const rawPacketData = capBuffer.slice(0, nbytes)
// one packet can generate 1 or 0 event
const result = parser.parse(rawPacketData, new Date().getTime(), pIndex)
if (result) {
// result can only be UEBunch event
const l2Evts = bunchStash.feedEvent(result)
if (l2Evts) {
for (const l2Evt of l2Evts) {
gamestate.processPUBGEvent(l2Evt)
}
}
}
})
startWebServer()
} else if (args[0] === 'playback') {
const parser = PacketParser()
// read the file and get all the events out first
const pbyp = PacketByPacket(args[1])
const allEvents = []
pbyp.on('packet', packet => {
// packet contains { header, data }
const result = parser.parse(packet.data, packet.header.timestamp)
if (result != null) {
allEvents.push(result)
}
})
pbyp.on('error', err => {
console.log('ERROR happened during parsing the pcap file', err)
})
pbyp.on('end', result => {
console.log('pcap file processing finished.', result)
console.log(`Got ${allEvents.length} layer-1 events.`)
if (allEvents.length > 0) {
console.log(
`Time range: ${utils.toLocalISOString(allEvents[0].time)} -> ${utils.toLocalISOString(allEvents[allEvents.length - 1].time)}`
)
// we have SelfLoc and UEBunches now, need to process the 2nd pass
const finalEvents = []
for (const l1Evt of allEvents) {
const l2Evts = bunchStash.feedEvent(l1Evt)
if (l2Evts) {
for (const l2Evt of l2Evts) {
if (l2Evt.type === CONSTS.EventTypes.GAMESTART) {
// reset the state, so that stash can process further games
gamestate.resetGameState()
}
finalEvents.push(l2Evt)
}
}
}
console.log(`Got ${finalEvents.length} layer-2 events.`)
// now set the mode to playback and start the api server
gamestate.setPlaybackMode(finalEvents)
printStateOnConsole()
startWebServer()
}
})
pbyp.resume()
}
}
main()