commit e96440eda20e1bbd89ed542654b4f080b8ecb064
Author: Damian Johnson <atagar(a)torproject.org>
Date: Sat May 11 18:36:44 2013 -0700
Revisions for cache thread safety
Handful of changes for the prior couple commits. Most are stylistic, but
there's a couple funcitonal ones:
* _set_cache() could accidently remove 'exit_policy'
* _confchanged_listener didn't use the new thread safe operations
---
stem/control.py | 83 ++++++++++++++++++++++++++----------------------------
1 files changed, 40 insertions(+), 43 deletions(-)
diff --git a/stem/control.py b/stem/control.py
index 71f790e..08ce196 100644
--- a/stem/control.py
+++ b/stem/control.py
@@ -679,14 +679,10 @@ class Controller(BaseController):
def _confchanged_listener(event):
if self.is_caching_enabled():
- for param, value in event.config.items():
- cache_key = "getconf.%s" % param.lower()
+ self._set_cache(dict((k, None) for k in event.config), "getconf")
- if cache_key in self._request_cache:
- del self._request_cache[cache_key]
-
- if param.lower() == "exitpolicy" and "exit_policy" in self._request_cache:
- del self._request_cache["exit_policy"]
+ if "exitpolicy" in event.config.keys():
+ self._set_cache({"exitpolicy": None})
self.add_event_listener(_confchanged_listener, EventType.CONF_CHANGED)
@@ -755,12 +751,12 @@ class Controller(BaseController):
params = set(params)
# check for cached results
+
from_cache = [param.lower() for param in params]
cached_results = self._get_cache_map(from_cache, "getinfo")
- reply = {}
- for key in cached_results.keys():
- user_expected_key = _case_insensitive_lookup(params, key, key)
+ for key in cached_results:
+ user_expected_key = _case_insensitive_lookup(params, key)
reply[user_expected_key] = cached_results[key]
params.remove(user_expected_key)
@@ -789,6 +785,7 @@ class Controller(BaseController):
if self.is_caching_enabled():
to_cache = {}
+
for key, value in response.entries.items():
key = key.lower() # make case insensitive
@@ -881,6 +878,7 @@ class Controller(BaseController):
with self._msg_lock:
try:
config_policy = self._get_cache("exit_policy")
+
if not config_policy:
policy = []
@@ -896,6 +894,7 @@ class Controller(BaseController):
policy += policy_line.split(",")
policy += self.get_info("exit-policy/default").split(",")
+
config_policy = stem.exit_policy.get_config_policy(policy)
self._set_cache({"exit_policy": config_policy})
@@ -1328,12 +1327,12 @@ class Controller(BaseController):
lookup_params = set([MAPPED_CONFIG_KEYS.get(entry, entry) for entry in params])
# check for cached results
+
from_cache = [param.lower() for param in lookup_params]
cached_results = self._get_cache_map(from_cache, "getconf")
- reply = {}
- for key in cached_results.keys():
- user_expected_key = _case_insensitive_lookup(lookup_params, key, key)
+ for key in cached_results:
+ user_expected_key = _case_insensitive_lookup(lookup_params, key)
reply[user_expected_key] = cached_results[key]
lookup_params.remove(user_expected_key)
@@ -1348,15 +1347,9 @@ class Controller(BaseController):
reply.update(response.entries)
if self.is_caching_enabled():
- to_cache = {}
- for param, value in response.entries.items():
- param = param.lower()
-
- if isinstance(value, (bytes, unicode)):
- value = [value]
- to_cache[param] = value
-
+ to_cache = dict((k.lower(), v) for k, v in response.entries.items())
self._set_cache(to_cache, "getconf")
+
# Maps the entries back to the parameters that the user requested so the
# capitalization matches (ie, if they request "exitpolicy" then that
# should be the key rather than "ExitPolicy"). When the same
@@ -1502,10 +1495,17 @@ class Controller(BaseController):
if self.is_caching_enabled():
to_cache = {}
+
for param, value in params:
+ param = param.lower()
+
if isinstance(value, (bytes, unicode)):
value = [value]
- to_cache[param.lower()] = value
+
+ to_cache[param] = value
+
+ if param == "exitpolicy":
+ self._set_cache({"exitpolicy": None})
self._set_cache(to_cache, "getconf")
else:
@@ -1596,25 +1596,24 @@ class Controller(BaseController):
if not response.is_ok():
raise stem.ProtocolError("SETEVENTS received unexpected response\n%s" % response)
- def _get_cache(self, param, func=None):
+ def _get_cache(self, param, namespace = None):
"""
- Queries cache for a configuration option.
+ Queries our request cache for the given key.
- :param str param: key to be queried in cache
- :param str func: function prefix to keys
+ :param str param: key to be queried
+ :param str namespace: namespace in which to check for the key
- :returns: cached value corresponding to key or None if key wasn't found
+ :returns: cached value corresponding to key or **None** if the key wasn't found
"""
- return self._get_cache_map([param], func).get(param, None)
+ return self._get_cache_map([param], namespace).get(param, None)
- def _get_cache_map(self, params, func=None):
+ def _get_cache_map(self, params, namespace = None):
"""
- Queries multiple configuration options in cache atomically, returning a
- mapping of those options to their values.
+ Queries our request cache for multiple entries.
- :param list params: keys to be queried in cache
- :param str func: function prefix to keys
+ :param list params: keys to be queried
+ :param str namespace: namespace in which to check for the keys
:returns: **dict** of 'param => cached value' pairs of keys present in cache
"""
@@ -1623,8 +1622,8 @@ class Controller(BaseController):
cached_values = {}
for param in params:
- if func:
- cache_key = "%s.%s" % (func, param)
+ if namespace:
+ cache_key = "%s.%s" % (namespace, param)
else:
cache_key = param
@@ -1633,18 +1632,19 @@ class Controller(BaseController):
return cached_values
- def _set_cache(self, params, func=None):
+ def _set_cache(self, params, namespace = None):
"""
- Changes the value of tor configuration options in cache atomically.
+ Sets the given request cache entries. If the new cache value is **None**
+ then it is removed from our cache.
:param dict params: **dict** of 'cache_key => value' pairs to be cached
- :param str func: function prefix to keys
+ :param str namespace: namespace for the keys
"""
with self._cache_lock:
for key, value in params.items():
- if func:
- cache_key = "%s.%s" % (func, key)
+ if namespace:
+ cache_key = "%s.%s" % (namespace, key)
else:
cache_key = key
@@ -1654,9 +1654,6 @@ class Controller(BaseController):
else:
self._request_cache[cache_key] = value
- if key.lower() == "exitpolicy" and "exit_policy" in self._request_cache:
- del self._request_cache["exit_policy"]
-
def is_caching_enabled(self):
"""
**True** if caching has been enabled, **False** otherwise.