mirror of
https://github.com/tahoe-lafs/tahoe-lafs.git
synced 2025-01-19 03:06:33 +00:00
add misc/simulate_load.py in an attempt to understand why permuting the peerlist per each storage index matters
This commit is contained in:
parent
c46f5bc634
commit
b483c4710d
111
misc/simulate_load.py
Normal file
111
misc/simulate_load.py
Normal file
@ -0,0 +1,111 @@
|
||||
#!/usr/bin/env python
|
||||
|
||||
|
||||
import random
|
||||
|
||||
class Server:
|
||||
def __init__(self):
|
||||
self.si = random.randrange(0, 2**31)
|
||||
self.used = 0
|
||||
self.max = 2**40
|
||||
self.full_at_tick = None
|
||||
|
||||
def __repr__(self):
|
||||
if self.full_at_tick is not None:
|
||||
return "<%s %s full at %d>" % (self.__class__.__name__, self.si, self.full_at_tick)
|
||||
else:
|
||||
return "<%s %s>" % (self.__class__.__name__, self.si)
|
||||
|
||||
SERVERS = 40
|
||||
K = 3
|
||||
N = 10
|
||||
|
||||
MAXSHARESIZE = 2**34 / K
|
||||
|
||||
def go(permutedpeerlist):
|
||||
servers = [ Server() for x in range(SERVERS) ]
|
||||
servers.sort(cmp=lambda x,y: cmp(x.si, y.si))
|
||||
|
||||
tick = 0
|
||||
fullservers = 0
|
||||
while True:
|
||||
nextsharesize = random.randrange(40, MAXSHARESIZE)
|
||||
if permutedpeerlist:
|
||||
random.shuffle(servers)
|
||||
else:
|
||||
# rotate a random number
|
||||
rot = random.randrange(0, len(servers))
|
||||
servers = servers[rot:] + servers[:rot]
|
||||
|
||||
i = 0
|
||||
sharestoput = K
|
||||
while sharestoput:
|
||||
server = servers[i]
|
||||
if server.used + nextsharesize < server.max:
|
||||
server.used += nextsharesize
|
||||
sharestoput -= 1
|
||||
else:
|
||||
if server.full_at_tick is None:
|
||||
server.full_at_tick = tick
|
||||
fullservers += 1
|
||||
if fullservers == len(servers):
|
||||
# print "Couldn't place share -- all servers full. Stopping."
|
||||
return servers
|
||||
|
||||
i = (i + 1) % len(servers)
|
||||
|
||||
tick += 1
|
||||
|
||||
def test(permutedpeerlist, iters):
|
||||
# the i'th element of the filledat list is how many servers got full on the tick numbered 4500 + i * 9
|
||||
filledat = [0] * 77
|
||||
for test in range(iters):
|
||||
servers = go(permutedpeerlist)
|
||||
for server in servers:
|
||||
fidx = (server.full_at_tick - 4500) / 9
|
||||
if fidx >= len(filledat):
|
||||
filledat.extend([0]*(fidx-len(filledat)+1))
|
||||
filledat[fidx] += 1
|
||||
|
||||
# the i'th element of the fullat list is how many servers were full by the tick numbered 4500 + i * 9 (on average)
|
||||
fullat = [0] * 77
|
||||
for idx, num in enumerate(filledat):
|
||||
for fidx in range(idx, len(fullat)):
|
||||
fullat[fidx] += num
|
||||
|
||||
for idx in range(len(fullat)):
|
||||
fullat[idx] = fullat[idx] / float(iters)
|
||||
|
||||
# Now print it out as an ascii art graph.
|
||||
import sys
|
||||
for serversfull in range(40, 0, -1):
|
||||
sys.stdout.write("%2d" % serversfull)
|
||||
for numfull in fullat:
|
||||
if int(numfull) == serversfull:
|
||||
sys.stdout.write("*")
|
||||
else:
|
||||
sys.stdout.write(" ")
|
||||
sys.stdout.write("\n")
|
||||
|
||||
sys.stdout.write(" ^-- servers full\n")
|
||||
idx = 0
|
||||
while idx < len(fullat):
|
||||
sys.stdout.write("%d--^ " % (4500 + idx * 9))
|
||||
idx += 8
|
||||
|
||||
sys.stdout.write("\nfiles uploaded --> \n")
|
||||
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
import sys
|
||||
iters = 16
|
||||
for arg in sys.argv:
|
||||
if arg.startswith("--iters="):
|
||||
iters = int(arg[8:])
|
||||
if "--permute" in sys.argv:
|
||||
print "doing permuted peerlist, iterations: %d" % iters
|
||||
test(True, iters)
|
||||
else:
|
||||
print "doing simple ring, iterations: %d" % iters
|
||||
test(False, iters)
|
Loading…
Reference in New Issue
Block a user