-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathping.lua
executable file
·362 lines (273 loc) · 10.7 KB
/
ping.lua
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
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
#!/usr/bin/lua
print("start:")
-- if maxTime (ms) is specified and ping TIME statistic is available then
-- the array of ping records will be truncated respectively.
-- Eg: if maxTime=10000 and ping TIME=15000 then the last third of the
-- records in the array will be cut off
--
-- afterwards, if minSeq is specified then the array of ping records will
-- be extended with NA values (reflecting a lost ping reply) so that
-- its size has at least minSeq entries
local in_log_files={
{["file"]="ping6-olsr1.log", ["exp"]=1, ["minSeq"]=200},
{["file"]="ping6-bmx.log", ["exp"]=1, ["minSeq"]=200},
{["file"]="ping6-batadv.log", ["exp"]=1, ["minSeq"]=200},
-- {["file"]="ping6-babel.log", ["exp"]=1, ["minSeq"]=200},
{["file"]="ping6-olsr2.log", ["exp"]=1, ["minSeq"]=200},
}
local out_data_file="./ping.data"
local out_stat_file="./ping.stat"
local protocols={ ["olsr1"] = "fdba:11:",
["bmx6"] = "fdba:12:",
["batadv"] = "fdbb::",
["babel"] = "fdba:14:",
["olsr2"] = "fdba:15:",
}
local function tableLength(t)
assert(type(t)=="table")
local n,c=0,0
for n in pairs(t) do c=c+1 end
return c
end
local in_begin_keyword1="PING .* data bytes"
local in_end_keyword1="Terminated"
local in_failure_keyword="ping: sendmsg: Network is unreachable"
local min_pings = 1
local min_protocols = 1 -- tableLength(protocols) -- per group
local MAX_TTL=64
local out_data_fd=io.open(out_data_file,"w+")
local data_hdr_format="%6s %8s %4s %4s %8s %6s %4s %4s\n"
local data_lin_format="%6s %8s %4s %4s %8s %6s %4s %4s\n"
out_data_fd:write(string.format(data_hdr_format,
"SEQ", "PROTO", "EXP", "NODE", "GRP", "RTT", "TTL", "HOPS"))
local out_stat_fd=io.open(out_stat_file,"w+")
local stat_hdr_format="%-7s %3s %-6s %-3s %-4s %4s %4s %4s %5s %7s %6s %6s %6s %4s %3s %4s %6s %6s %6s %6s %6s %s\n"
local stat_lin_format="%-7s %3s %-6s %-3s %-4s %4s %4s %4s %5s %7s %6s %6d %6s %4s %3s %4s %6s %6.2f %6s %6s %6.1f %s\n"
out_stat_fd:write(string.format(stat_hdr_format,
"PROTO", "EXP", "NODE", "GRP", "SIZE", --"ADDR",
"SEND", "RCVD", "LOSS", "TIME",
"maxTime", "minSeq", "maxSeq",
"SEQMAX", "UNIQ", "DUP", "REOR",
"HOPMAX", "HOPAVG", "RTTMIN", "RTTMAX", "RTTAVG", "LOGFILE"))
local function init_ping()
return {
beg_key_found=false,
file=nil,
exp=nil,
minSeq=nil,
maxTime=nil,
maxSeq=nil,
prot=nil,
node_id=nil,
bytes=nil,
addr=nil,
max_seq=0,
last_seq=0,
min_ttl=nil,
max_ttl=nil,
tot_ttl=0,
min_hop=nil,
max_hop=nil,
tot_hop=0,
min_time=nil,
max_time=nil,
tot_time=0,
double_succeeds=0,
unique_succeeds=0,
reorders_last=0,
reorders_max=0,
unmatched=0,
failure_after_succeeds=0,
lost=0,
pstats = nil,
data_table = {}
}
end
local global_data = { prev_nodeid=nil, prev_exp=nil, curr_group=0, curr_pings=nil }
local function complete_group()
print("complete_group(): "..tostring(global_data.curr_pings).." "..tableLength(global_data.curr_pings or {}))
if not global_data.curr_pings or tableLength(global_data.curr_pings) < min_protocols then
return
end
global_data.curr_group = global_data.curr_group + 1
for p,a in pairs(global_data.curr_pings) do
assert( type(global_data.curr_pings[p])=="table" )
local ping = global_data.curr_pings[p]
ping.maxSeq =
(ping.maxTime and ping.pstats and ping.pstats.time_ms and tonumber(ping.pstats.time_ms) > ping.maxTime) and
((ping.max_seq * ping.maxTime) / ping.pstats.time_ms) or (ping.max_seq)
for i = 1,((ping.minSeq and ping.minSeq > ping.maxSeq) and ping.minSeq or ping.maxSeq) do
if i <= ping.maxSeq and ping.data_table[i] then
out_data_fd:write(string.format(data_lin_format,
i,
ping.prot,
ping.node_id,
ping.exp,
global_data.curr_group,
ping.data_table[i].time,
ping.data_table[i].ttl,
ping.data_table[i].hop
))
else
out_data_fd:write(string.format(data_lin_format,
i,
ping.prot,
ping.node_id,
ping.exp,
global_data.curr_group,
"NA", "NA", "NA"
))
end
end
out_stat_fd:write(string.format(stat_lin_format,
ping.prot,
ping.exp,
ping.node_id,
global_data.curr_group,
(ping.bytes or "NA"),
(ping.pstats and ping.pstats.transmitted or "NA"),
(ping.pstats and ping.pstats.received or "NA"),
(ping.pstats and ping.pstats.loss_percent or "NA"),
(ping.pstats and ping.pstats.time_ms or "NA"),
(ping.maxTime or "NA"),
(ping.minSeq or "NA"),
(ping.maxSeq or "NA"),
ping.max_seq,
ping.unique_succeeds,
ping.double_succeeds,
ping.reorders_last,
(ping.max_hop or "NA"),
(ping.tot_hop/ping.unique_succeeds),
(ping.min_time or "NA"),
(ping.max_time or "NA"),
(ping.tot_time/ping.unique_succeeds),
ping.file
) )
end
end
local function eval_pings( ping )
print("eval_pings() max_seq="..tostring(ping and ping.max_seq))
if ping and ping.file then
print("Summary: log=".. ping.file .." bytes="..(ping.bytes or "NA") .. " addr="..(ping.addr or "NA") ..
" max_seq="..ping.max_seq.."/"..(ping.unique_succeeds+ping.lost) ..
" unique="..ping.unique_succeeds .." double="..ping.double_succeeds ..
" failure="..ping.unmatched.."/"..ping.failure_after_succeeds.. " lost="..ping.lost..
" reorders="..ping.reorders_last.."/"..ping.reorders_max..
" ttl="..(ping.min_ttl or "NA") .. "/"..(ping.max_ttl or "NA").."/"..(ping.tot_ttl/ping.unique_succeeds)..
" rtt="..(ping.min_time or "NA").. "/"..(ping.max_time or "NA").."/"..(ping.tot_time/ping.unique_succeeds) )
end
if type(ping)=="table" and ping.max_seq >= min_pings then
print("prev_nodeid="..tostring(global_data.prev_nodeid).." node_id="..tostring(ping and ping.node_id).." prev_exp="..tostring(global_data.prev_exp).." exp="..tostring(ping and ping.exp))
if global_data.prev_nodeid ~= ping.node_id or global_data.prev_exp ~= ping.exp then
complete_group()
global_data.prev_nodeid = ping.node_id
global_data.prev_exp = ping.exp
global_data.curr_pings = {}
end
global_data.curr_pings[ping.prot] = ping
end
return init_ping()
end
local function iterate_log_file( log, ping )
local line
local curr_line=0
for line in io.lines(log.file) do
curr_line = curr_line + 1
if string.match( line, in_begin_keyword1 ) then
ping = eval_pings(ping)
ping.beg_key_found=true
print("Found begin: "..log.file)
end
if ping.beg_key_found and ping.addr then
if line:match( " "..ping.addr.." ping statistics ") then
ping.pstats = { line=curr_line }
elseif type(ping.pstats)=="table" then
if ping.pstats.line==(curr_line - 1) and
line:match( "transmitted") and line:match("received") and line:match("loss") then
ping.pstats.transmitted = ((line:match("[%d]+ packets transmitted")) or "NA"):match("[%d]+")
ping.pstats.received = ((line:match("[%d]+ received")) or "NA"):match("[%d]+")
ping.pstats.loss_percent = ((line:match("[%d]+%% packet loss")) or "NA"):match("[%d]+")
ping.pstats.time_ms = ((line:match("time [%d]+ms")) or "NA"):match("[%d]+")
end
end
end
if ping.beg_key_found then
local probe_bytes = (line:match( "[%d]+ bytes") or "NA" ):match("[%d]+")
local probe_addr = (line:match( "from .*: ") or "NA" ):gsub("from ",""):gsub(": ","")
local probe_seq = (line:match( "icmp_seq=[%d]+") or "NA" ):match("[%d]+")
local probe_ttl = (line:match( "ttl=[%d]+") or "NA" ):match("[%d]+")
local probe_time = (line:match( "time=[%d]+.[%d]+ ms") or "NA" ):match("[%d]+.[%d]+")
local probe_hop
if (probe_bytes and probe_addr and probe_seq and probe_ttl and probe_time) then
assert((not ping.file) or (ping.file==log.file))
ping.file = log.file
ping.exp = log.exp
ping.maxTime = log.maxTime
ping.minSeq = log.minSeq
for p,a in pairs(protocols) do
assert( type(p)=="string" and type(a)=="string" and type(probe_addr)=="string" )
-- print( "probe_addr="..probe_addr.." a="..a)
if probe_addr:match( "^"..a ) then
local node_id = probe_addr:gsub(a,""):gsub("::1","")
assert( (not ping.node_id) or (ping.node_id==node_id))
assert( (not ping.prot) or (ping.prot==p))
if (not ping.prot) then
ping.node_id = node_id
ping.prot = p
end
end
end
assert(ping.prot and ping.node_id)
assert((not ping.bytes) or probe_bytes:match(ping.bytes), line)
ping.bytes = ping.bytes and ping.bytes or probe_bytes
assert((not ping.addr) or (ping.addr==probe_addr), log.file.." "..line)
ping.addr = probe_addr
probe_bytes = tonumber(probe_bytes)
probe_seq = tonumber(probe_seq)
probe_ttl = tonumber(probe_ttl)
probe_hop = 1 + (MAX_TTL - probe_ttl)
ping.max_seq = math.max((ping.max_seq or probe_seq),probe_seq)
ping.min_ttl = math.min((ping.min_ttl or probe_ttl),probe_ttl)
ping.max_ttl = math.max((ping.max_ttl or probe_ttl),probe_ttl)
ping.tot_ttl = ping.tot_ttl + probe_ttl
ping.min_hop = math.min((ping.min_hop or probe_hop),probe_hop)
ping.max_hop = math.max((ping.max_hop or probe_hop),probe_hop)
ping.tot_hop = ping.tot_hop + probe_hop
ping.min_time = math.min((ping.min_time or probe_time),probe_time)
ping.max_time = math.max((ping.max_time or probe_time),probe_time)
ping.tot_time = ping.tot_time + probe_time
if ping.data_table[probe_seq] then
ping.double_succeeds = ping.double_succeeds + 1
print("file="..ping.file.." line="..curr_line .. " DOUBLE: " .. line)
else
--print( curr_line .. " UNIQUE:" ..
-- " bytes="..probe_bytes .. " addr="..probe_addr ..
-- " seq="..probe_seq ..
-- " ttl="..probe_ttl ..
-- " time="..probe_time ..
-- "")
ping.unique_succeeds = ping.unique_succeeds + 1
ping.failure_after_succeeds = 0
ping.reorders_last = (ping.last_seq > probe_seq) and (ping.reorders_last + 1) or ping.reorders_last
ping.last_seq = probe_seq
ping.reorders_max = (ping.max_seq > probe_seq) and (ping.reorders_max + 1) or ping.reorders_max
ping.data_table[probe_seq] = {seq=probe_seq, ttl=probe_ttl, hop=probe_hop, time=probe_time}
end
else
--print(curr_line .. " UNMATCHED: " .. line)
ping.unmatched = ping.unmatched + 1
ping.failure_after_succeeds = ping.failure_after_succeeds + 1
end
end
end
return ping
end
local ping = init_ping()
for i,p in ipairs(in_log_files) do
ping = iterate_log_file(p, ping)
end
ping = eval_pings(ping)
complete_group()
out_data_fd:close()
out_stat_fd:close()
for line in io.lines(out_stat_file) do print(line) end