Commit 2177ef77 authored by Jeremy Hylton's avatar Jeremy Hylton

Apparently functional version that reads from the end.

Starts near the end of the file and reads forward, looking for a
complete snapshot.  If it doesn't find one, move farther from the end
and repeat.

XXX This approach does N**2 reads, where N is the number of times we
need to move further back and retry.  When we move backwards, we
double the distance from the end of the file.
parent 898f8392
#! /usr/bin/env python #! /usr/bin/env python
"""Report on the number of currently waiting clients in the ZEO queue.""" """Report on the number of currently waiting clients in the ZEO queue."""
import fileinput import getopt
import re import re
import sys import sys
import time import time
# pick arbitrary buffer size that isn't too big
BUFSIZE = 8 * 1024 * 1024
rx_time = re.compile('(\d\d\d\d-\d\d-\d\d)T(\d\d:\d\d:\d\d)') rx_time = re.compile('(\d\d\d\d-\d\d-\d\d)T(\d\d:\d\d:\d\d)')
def parse_time(line): def parse_time(line):
...@@ -81,6 +84,15 @@ class Status: ...@@ -81,6 +84,15 @@ class Status:
self.t_restart = None self.t_restart = None
self.txns = {} self.txns = {}
def iscomplete(self):
# The status report will always be complete if we encounter an
# explicit restart.
if self.t_restart is not None:
return 1
# If we haven't seen a restart, assume that seeing a finished
# transaction is good enough.
return self.commit is not None
def report(self): def report(self):
print "Blocked transactions:", self.n_blocked print "Blocked transactions:", self.n_blocked
if not VERBOSE: if not VERBOSE:
...@@ -98,12 +110,13 @@ class Status: ...@@ -98,12 +110,13 @@ class Status:
L = [(txn.begin, txn) for txn in self.txns.values()] L = [(txn.begin, txn) for txn in self.txns.values()]
L.sort() L.sort()
first_txn = L[0][1] for x, txn in L:
if first_txn.isactive(): if txn.isactive():
began = first_txn.begin began = txn.begin
print "Blocked transaction began at:", time.ctime(began) print "Blocked transaction began at:", time.ctime(began)
print "Hint:", first_txn.hint print "Hint:", txn.hint
print "Idle time: %d sec" % int(time.time() - began) print "Idle time: %d sec" % int(time.time() - began)
break
def process(self, line): def process(self, line):
if line.find("calling") != -1: if line.find("calling") != -1:
...@@ -213,21 +226,36 @@ class Status: ...@@ -213,21 +226,36 @@ class Status:
pass pass
self.commit = self.commit_or_abort = txn self.commit = self.commit_or_abort = txn
def process_from(f, pos):
s = Status()
f.seek(-pos, 2)
f.readline()
for line in f.readlines(BUFSIZE):
s.process(line)
return s
def main(): def main():
global VERBOSE global VERBOSE
# decide whether -v was passed on the command line
try:
i = sys.argv.index("-v")
except ValueError:
VERBOSE = 0 VERBOSE = 0
else: opts, args = getopt.getopt(sys.argv[1:], 'v')
VERBOSE = 1 for k, v in opts:
# fileinput assumes all of sys.argv[1:] is files it should read if k == '-v':
del sys.argv[i] VERBOSE += 1
s = Status() path = args[0]
for line in fileinput.input(): f = open(path, "rb")
s.process(line)
# Start at pos bytes from the end of the file and read forwards.
# If we read enough log data to have a complete snapshot of the
# server state, stop and print a report. If not, move twice as
# far from the end of the file and repeat.
pos = 16 * 1024
while 1:
s = process_from(f, pos)
if s.iscomplete():
break
pos *= 2
s.report() s.report()
if __name__ == "__main__": if __name__ == "__main__":
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment