commit c207f38171e122a869c564701950e223f34d4e0a Author: aagbsn aagbsn@extc.org Date: Thu Dec 15 03:21:59 2011 -0800
4297 - Add ipv6 keyword to EmailBasedDistributor
Updates the EmailBasedDistributor to support 'ipv6' keyword in the subject or mail body --- lib/bridgedb/Dist.py | 133 ++++++++++++++++++++++++++++++++----------------- 1 files changed, 87 insertions(+), 46 deletions(-)
diff --git a/lib/bridgedb/Dist.py b/lib/bridgedb/Dist.py index e2d1559..285e088 100644 --- a/lib/bridgedb/Dist.py +++ b/lib/bridgedb/Dist.py @@ -38,7 +38,7 @@ class IPBasedDistributor(bridgedb.Bridges.BridgeHolder): def __init__(self, areaMapper, nClusters, key, ipCategories=(), answerParameters=None): self.areaMapper = areaMapper - self.answerParameters = answerParameters + self.answerParameters = answerParameters
self.rings = [] self.categoryRings = [] #DOCDDOC @@ -50,13 +50,15 @@ class IPBasedDistributor(bridgedb.Bridges.BridgeHolder): key4 = bridgedb.Bridges.get_hmac(key, "Assign-Areas-To-Rings") self.areaClusterHmac = bridgedb.Bridges.get_hmac_fn(key4, hex=True)
- # add splitter and cache the default rings - # plus leave room for dynamic filters + # add splitter and cache the default rings + # plus leave room for dynamic filters ring_cache_size = nClusters + len(ipCategories) + 5 self.splitter = bridgedb.Bridges.FilteredBridgeSplitter(key2, - max_cached_rings=ring_cache_size) + max_cached_rings=ring_cache_size)
- # assign bridges using a filter function + logging.debug("added splitter %s" % self.splitter) + + # assign bridges using a filter function def filterAssignBridgesToRing(hmac, numRings, assignedRing): def f(bridge): digest = hmac(bridge.getID()) @@ -69,27 +71,24 @@ class IPBasedDistributor(bridgedb.Bridges.BridgeHolder): for n in xrange(nClusters): key1 = bridgedb.Bridges.get_hmac(key, "Order-Bridges-In-Ring-%d"%n) ring = bridgedb.Bridges.BridgeRing(key1, answerParameters) - ring.setName("IP ring %s"%n) #XXX: should be n+1 for consistency? + ring.setName("IP ring %s"%n) #XXX: should be n+1 for consistency?
- g = filterAssignBridgesToRing(self.splitter.hmac, - nClusters + len(ipCategories), n) - self.splitter.addRing(ring, ring.name, g) - self.rings.append(ring) + g = filterAssignBridgesToRing(self.splitter.hmac, + nClusters + len(ipCategories), n) + self.splitter.addRing(ring, ring.name, g) + self.rings.append(ring)
n = nClusters for c in ipCategories: logging.info("Building ring: Order-Bridges-In-Ring-%d"%n) key1 = bridgedb.Bridges.get_hmac(key, "Order-Bridges-In-Ring-%d"%n) ring = bridgedb.Bridges.BridgeRing(key1, answerParameters) - ring.setName("IP category ring %s"%n) #XXX: should be n+1 for consistency? + ring.setName("IP category ring %s"%n) #XXX: should be n+1 for consistency? self.categoryRings.append( ring ) self.categories.append( (c, ring) ) - - g = filterAssignBridgesToRing(self.splitter.hmac, - nClusters + len(ipCategories), n) - - self.splitter.addRing(ring, ring.name, g) - + g = filterAssignBridgesToRing(self.splitter.hmac, + nClusters + len(ipCategories), n) + self.splitter.addRing(ring, ring.name, g) n += 1
def clear(self): @@ -100,14 +99,16 @@ class IPBasedDistributor(bridgedb.Bridges.BridgeHolder): self.splitter.insert(bridge)
def getBridgesForIP(self, ip, epoch, N=1, countryCode=None, - bridgeFilterRules=None): + bridgeFilterRules=None): """Return a list of bridges to give to a user. ip -- the user's IP address, as a dotted quad. epoch -- the time period when we got this request. This can be any string, so long as it changes with every period. N -- the number of bridges to try to give back. """ + logging.debug("getBridgesForIP(%s, %s, %s, %s" % (ip, epoch, N, bridgeFilterRules)) if not len(self.splitter): + logging.debug("bailing without splitter") return []
area = self.areaMapper(ip) @@ -121,29 +122,30 @@ class IPBasedDistributor(bridgedb.Bridges.BridgeHolder): pos = self.areaOrderHmac("category<%s>%s"%(epoch,area)) return ring.getBridges(pos, N, countryCode)
- # dynamic filter construction - if bridgeFilterRules: - ruleset = frozenset(bridgeFilterRules) - if ruleset in self.splitter.filterRings.keys(): - # cache hit - _,ring = self.splitter.filterRings[ruleset] - else: - # cache miss, add new ring - def filterBridgesByRules(rules): - def g(x): - r = [f(x) for f in rules] - if False in r: return False - return True - return g - # add new ring - #XXX what key do we use here? does it matter? + # dynamic filter construction + if bridgeFilterRules: + logging.debug("bridgeFilterRules: %s" % bridgeFilterRules) + ruleset = frozenset(bridgeFilterRules) + if ruleset in self.splitter.filterRings.keys(): + logging.debug("Cache hit %s" % ruleset) + _,ring = self.splitter.filterRings[ruleset] + else: + logging.debug("Cache miss %s" % ruleset) + def filterBridgesByRules(rules): + def g(x): + r = [f(x) for f in rules] + if False in r: return False + return True + return g + # add new ring + #XXX what key do we use here? does it matter? key1 = bridgedb.Bridges.get_hmac(self.splitter.key, str(bridgeFilterRules)) - ring = bridgedb.Bridges.BridgeRing(key1, self.answerParameters) - # debug log: cache miss - self.splitter.addRing(ring, ruleset, filterBridgesByRules(bridgeFilterRules), - populate_from=self.splitter.bridges) + ring = bridgedb.Bridges.BridgeRing(key1, self.answerParameters) + # debug log: cache miss + self.splitter.addRing(ring, ruleset, filterBridgesByRules(bridgeFilterRules), + populate_from=self.splitter.bridges)
- else: + else: # Which bridge cluster should we look at? h = int( self.areaClusterHmac(area)[:8], 16) clusterNum = h % len(self.rings) @@ -158,7 +160,7 @@ class IPBasedDistributor(bridgedb.Bridges.BridgeHolder): return ring.getBridges(pos, N)
def __len__(self): - #XXX does not include ip categories or filtered rings + #XXX does not include ip categories or filtered rings return sum(len(r) for r in self.rings)
def dumpAssignments(self, f, description=""): @@ -286,14 +288,26 @@ class EmailBasedDistributor(bridgedb.Bridges.BridgeHolder): self.domainmap = domainmap self.domainrules = domainrules
+ #XXX cache options not implemented + self.splitter = bridgedb.Bridges.FilteredBridgeSplitter(key2, + max_cached_rings=5) + + #XXX default ring. + def g(x): + return True + + self.splitter.addRing(self.ring, self.ring.name, g) + def clear(self): - self.ring.clear() + self.splitter.clear() + self.ring.clear() # should be take care of by above
def insert(self, bridge): """Assign a bridge to this distributor.""" - self.ring.insert(bridge) + self.splitter.insert(bridge)
- def getBridgesForEmail(self, emailaddress, epoch, N=1, parameters=None, countryCode=None): + def getBridgesForEmail(self, emailaddress, epoch, N=1, parameters=None, countryCode=None, + bridgeFilterRules=None): """Return a list of bridges to give to a user. emailaddress -- the user's email address, as given in a from line. epoch -- the time period when we got this request. This can @@ -332,7 +346,35 @@ class EmailBasedDistributor(bridgedb.Bridges.BridgeHolder): db.setWarnedEmail(emailaddress, False)
pos = self.emailHmac("<%s>%s" % (epoch, emailaddress)) - result = self.ring.getBridges(pos, N, countryCode) + + ring = None + if bridgeFilterRules: + ruleset = frozenset(bridgeFilterRules) + if ruleset in self.splitter.filterRings.keys(): + logging.debug("Cache hit %s" % ruleset) + _,ring = self.splitter.filterRings[ruleset] + else: + # cache miss, add new ring + logging.debug("Cache miss %s" % ruleset) + def filterBridgesByRules(rules): + def g(x): + r = [f(x) for f in rules] + if False in r: return False + return True + return g + # add new ring + #XXX what key do we use here? does it matter? + key1 = bridgedb.Bridges.get_hmac(self.splitter.key, + str(bridgeFilterRules)) + ring = bridgedb.Bridges.BridgeRing(key1, self.answerParameters) + # debug log: cache miss + self.splitter.addRing(ring, ruleset, + filterBridgesByRules(bridgeFilterRules), + populate_from=self.splitter.bridges) + else: + ring = self.ring + + result = ring.getBridges(pos, N, countryCode)
db.setEmailTime(emailaddress, now) db.commit() @@ -353,5 +395,4 @@ class EmailBasedDistributor(bridgedb.Bridges.BridgeHolder): db.commit()
def dumpAssignments(self, f, description=""): - self.ring.dumpAssignments(f, description) - + self.splitter.dumpAssignments(f, description)
tor-commits@lists.torproject.org