Commit eab7252b authored by Tim Peters's avatar Tim Peters

_PackWhileWriting(): Save a lot more info to display in case of failure.

Forcing the output even when a test passes is interesting on my box.
When ZEO is involved, a single ClientThread tends to run all the way to
completion before another ClientThread manages to sneak in -- apart from
their first transactions, it's as if we ran one thread, then started the
next after the first finished (etc).  When ZEO isn't involved, threads
take turns.  When I see failures, they involve ZEO.  It's possible that
30 seconds just isn't long enough to wait on this box.
parent 28361e79
...@@ -175,7 +175,10 @@ class PackableStorage(PackableStorageBase): ...@@ -175,7 +175,10 @@ class PackableStorage(PackableStorageBase):
root[i].value = MinPO(i) root[i].value = MinPO(i)
get_transaction().commit() get_transaction().commit()
threads = [ClientThread(db, choices) for i in range(4)] NUM_LOOP_TRIP = 50
timer = ElapsedTimer(time.time())
threads = [ClientThread(db, choices, NUM_LOOP_TRIP, timer, i)
for i in range(4)]
for t in threads: for t in threads:
t.start() t.start()
...@@ -190,24 +193,44 @@ class PackableStorage(PackableStorageBase): ...@@ -190,24 +193,44 @@ class PackableStorage(PackableStorageBase):
if True in liveness: if True in liveness:
# They should have finished by now. # They should have finished by now.
print 'Liveness:', liveness print 'Liveness:', liveness
# Display how far each thread got, one column per thread. # Combine the outcomes, and sort by start time.
i = 0 outcomes = []
fmt = "(%2s %10s)" for t in threads:
while True: outcomes.extend(t.outcomes)
found_one = False # each outcome list has as many of these as a loop trip got thru:
columns = [] # thread_id
for t in threads: # elapsed millis at loop top
if i < len(t.outcomes): # elapsed millis at attempt to assign to self.root[index]
found_one = True # index into self.root getting replaced
result = tuple(t.outcomes[i]) # elapsed millis when outcome known
else: # 'OK' or 'Conflict'
result = ('--', '--') # True if we got beyond this line, False if it raised an
columns.append(fmt % result) # exception (one possible Conflict cause):
if found_one: # self.root[index].value = MinPO(j)
print ' '.join(columns) def cmp_by_time(a, b):
i += 1 return cmp((a[1], a[0]), (b[1], b[0]))
else: outcomes.sort(cmp_by_time)
break counts = [0] * 4
for outcome in outcomes:
n = len(outcome)
assert n >= 2
tid = outcome[0]
print 'tid:%d top:%5d' % (tid, outcome[1]),
if n > 2:
print 'commit:%5d' % outcome[2],
if n > 3:
print 'index:%2d' % outcome[3],
if n > 4:
print 'known:%5d' % outcome[4],
if n > 5:
print '%8s' % outcome[5],
if n > 6:
print 'assigned:%5s' % outcome[6],
counts[tid] += 1
if counts[tid] == NUM_LOOP_TRIP:
print 'thread %d done' % tid,
print
self.fail('a thread is still alive') self.fail('a thread is still alive')
# Iterate over the storage to make sure it's sane, but not every # Iterate over the storage to make sure it's sane, but not every
...@@ -575,22 +598,49 @@ class PackableUndoStorage(PackableStorageBase): ...@@ -575,22 +598,49 @@ class PackableUndoStorage(PackableStorageBase):
# random write activity while the main thread is packing it. # random write activity while the main thread is packing it.
class ClientThread(TestThread): class ClientThread(TestThread):
def __init__(self, db, choices): def __init__(self, db, choices, loop_trip, timer, thread_id):
TestThread.__init__(self) TestThread.__init__(self)
self.root = db.open().root() self.root = db.open().root()
self.choices = choices self.choices = choices
# list of pairs, (root index written to, 'OK' or 'Conflict') self.loop_trip = loop_trip
self.millis = timer.elapsed_millis
self.thread_id = thread_id
# list of lists; each list has as many of these as a loop trip
# got thru:
# thread_id
# elapsed millis at loop top
# elapsed millis at attempt
# index into self.root getting replaced
# elapsed millis when outcome known
# 'OK' or 'Conflict'
# True if we got beyond this line, False if it raised an exception:
# self.root[index].value = MinPO(j)
self.outcomes = [] self.outcomes = []
def runtest(self): def runtest(self):
from random import choice from random import choice
for j in range(50): for j in range(self.loop_trip):
assign_worked = False
alist = [self.thread_id, self.millis()]
self.outcomes.append(alist)
try: try:
index = choice(self.choices) index = choice(self.choices)
alist.extend([self.millis(), index])
self.root[index].value = MinPO(j) self.root[index].value = MinPO(j)
assign_worked = True
get_transaction().commit() get_transaction().commit()
self.outcomes.append((index, 'OK')) alist.append(self.millis())
alist.append('OK')
except ConflictError: except ConflictError:
alist.append(self.millis())
alist.append('Conflict')
get_transaction().abort() get_transaction().abort()
self.outcomes.append((index, 'Conflict')) alist.append(assign_worked)
class ElapsedTimer:
def __init__(self, start_time):
self.start_time = start_time
def elapsed_millis(self):
return int((time.time() - self.start_time) * 1000)
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