Skip to content
This repository has been archived by the owner on Apr 26, 2024. It is now read-only.

Harden ResponseCache (and fix #9507) #9522

Closed
wants to merge 5 commits into from
Closed
Show file tree
Hide file tree
Changes from 4 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions changelog.d/9522.bugfix
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Fix #9507 by hardening ResponseCache against errors in execution of to-be-cached functions.
ShadowJonathan marked this conversation as resolved.
Show resolved Hide resolved
65 changes: 48 additions & 17 deletions synapse/util/caches/response_cache.py
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@
from typing import TYPE_CHECKING, Any, Callable, Dict, Generic, Optional, Set, TypeVar

from twisted.internet import defer
from twisted.python.failure import Failure

from synapse.logging.context import make_deferred_yieldable, run_in_background
from synapse.util.async_helpers import ObservableDeferred
Expand Down Expand Up @@ -80,6 +81,42 @@ def get(self, key: T) -> Optional[defer.Deferred]:
self._metrics.inc_misses()
return None

@staticmethod
def _safe_call(func: Callable[[Any], bool], ret_val: Any, key: T) -> bool:
"""
Wraps a conditional checking function with a try-catch, returns False
if the conditional function threw an exception during evaluation.
"""
try:
return func(ret_val)
except Exception as e:
logger.warning(
"conditional %r threw exception on %s, ignoring and invalidating cache...",
func,
key,
exc_info=e,
)
return False

def _remove_cache(self, key: T):
self.pending_result_cache.pop(key, None)

def _errback_remove_cache(self, f: Failure, key: T):
self._remove_cache(key)
return f

def _test_cache(self, r: Any, key: T):
should_cache = all(
self._safe_call(func, r, key=key)
for func in self.pending_conditionals.pop(key, [])
)
Comment on lines +109 to +112
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

for links: I've argued in #synapse-dev and https://github.com/matrix-org/synapse/pull/9358/files#r585481702 that we should only support one conditional per key, which simplifies this code.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'll add that to the "small changes to ResponseCache PR", this one is focused on fixing #9507, i dont wanna drag everything else into here.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Created #9525 for this


if self.timeout_sec and should_cache:
self.clock.call_later(self.timeout_sec, self._remove_cache, key)
else:
self._remove_cache(key)
return r

def set(self, key: T, deferred: defer.Deferred) -> defer.Deferred:
"""Set the entry for the given key to the given deferred.

Expand All @@ -101,20 +138,8 @@ def set(self, key: T, deferred: defer.Deferred) -> defer.Deferred:
result = ObservableDeferred(deferred, consumeErrors=True)
self.pending_result_cache[key] = result

def remove(r):
should_cache = all(
func(r) for func in self.pending_conditionals.pop(key, [])
)

if self.timeout_sec and should_cache:
self.clock.call_later(
self.timeout_sec, self.pending_result_cache.pop, key, None
)
else:
self.pending_result_cache.pop(key, None)
return r

result.addBoth(remove)
result.addCallback(self._test_cache, key)
result.addErrback(self._errback_remove_cache, key)
return result.observe()

def add_conditional(self, key: T, conditional: Callable[[Any], bool]):
Expand All @@ -130,8 +155,12 @@ def wrap_conditional(
) -> defer.Deferred:
"""The same as wrap(), but adds a conditional to the final execution.

When the final execution completes, *all* conditionals need to return True for it to properly cache,
else it'll not be cached in a timed fashion.
When the final execution completes, *all* conditionals need to return
True for it to properly cache (apart from it also not having thrown an
uncaught exception), else the cache will instantly expire.

(Note that also no conditional must throw an uncaught exception for it to
cache)
"""

# See if there's already a result on this key that hasn't yet completed. Due to the single-threaded nature of
Expand All @@ -145,7 +174,9 @@ def wrap_conditional(
def wrap(
self, key: T, callback: "Callable[..., Any]", *args: Any, **kwargs: Any
) -> defer.Deferred:
"""Wrap together a *get* and *set* call, taking care of logcontexts
"""Wrap together a *get* and *set* call, taking care of logcontexts.

Does not cache if the wrapped function throws an uncaught exception.

First looks up the key in the cache, and if it is present makes it
follow the synapse logcontext rules and returns it.
Expand Down