""" Track randomize ordered dict """ from threading import RLock from time import time from binascii import hexlify try: import helper_random except ImportError: from . import helper_random class RandomTrackingDict(object): """ Dict with randomised order and tracking. Keeps a track of how many items have been requested from the dict, and timeouts. Resets after all objects have been retrieved and timed out. The main purpose of this isn't as much putting related code together as performance optimisation and anonymisation of downloading of objects from other peers. If done using a standard dict or array, it takes too much CPU (and looks convoluted). Randomisation helps with anonymity. """ # pylint: disable=too-many-instance-attributes maxPending = 10 pendingTimeout = 60 def __init__(self): self.dictionary = {} self.indexDict = [] self.len = 0 self.pendingLen = 0 self.lastPoll = 0 self.lastObject = 0 self.lock = RLock() def __len__(self): return self.len def __contains__(self, key): hex_key = hexlify(key).decode('ascii') return hex_key in self.dictionary def __getitem__(self, key): hex_key = hexlify(key).decode('ascii') return self.dictionary[hex_key][1] def _swap(self, i1, i2): with self.lock: key1 = self.indexDict[i1] key2 = self.indexDict[i2] self.indexDict[i1] = key2 self.indexDict[i2] = key1 hex_key1 = hexlify(key1).decode('ascii') hex_key2 = hexlify(key2).decode('ascii') self.dictionary[hex_key1][0] = i2 self.dictionary[hex_key2][0] = i1 # for quick reassignment return i2 def __setitem__(self, key, value): with self.lock: hex_key = hexlify(key).decode('ascii') if hex_key in self.dictionary: self.dictionary[hex_key][1] = value else: self.indexDict.append(key) self.dictionary[hex_key] = [self.len, value] self._swap(self.len, self.len - self.pendingLen) self.len += 1 def __delitem__(self, key): hex_key = hexlify(key).decode('ascii') if hex_key not in self.dictionary: raise KeyError with self.lock: index = self.dictionary[hex_key][0] # not pending if index < self.len - self.pendingLen: # left of pending part index = self._swap(index, self.len - self.pendingLen - 1) # pending else: self.pendingLen -= 1 # end self._swap(index, self.len - 1) # if the following del is batched, performance of this single # operation can improve 4x, but it's already very fast so we'll # ignore it for the time being del self.indexDict[-1] del self.dictionary[hex_key] self.len -= 1 def setMaxPending(self, maxPending): """ Sets maximum number of objects that can be retrieved from the class simultaneously as long as there is no timeout """ self.maxPending = maxPending def setPendingTimeout(self, pendingTimeout): """Sets how long to wait for a timeout if max pending is reached (or all objects have been retrieved)""" self.pendingTimeout = pendingTimeout def setLastObject(self): """Update timestamp for tracking of received objects""" self.lastObject = time() def randomKeys(self, count=1): """Retrieve count random keys from the dict that haven't already been retrieved""" if self.len == 0 or ( (self.pendingLen >= self.maxPending or self.pendingLen == self.len) and self.lastPoll + self.pendingTimeout > time()): raise KeyError # pylint: disable=redefined-outer-name with self.lock: # reset if we've requested all # and if last object received too long time ago if self.pendingLen == self.len and self.lastObject + \ self.pendingTimeout < time(): self.pendingLen = 0 self.setLastObject() available = self.len - self.pendingLen if count > available: count = available randomIndex = helper_random.randomsample( range(self.len - self.pendingLen), count) retval = [self.indexDict[i] for i in randomIndex] for i in sorted(randomIndex, reverse=True): # swap with one below lowest pending self._swap(i, self.len - self.pendingLen - 1) self.pendingLen += 1 self.lastPoll = time() return retval