-
Notifications
You must be signed in to change notification settings - Fork 48
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
RedisBackend should probably not raise errors #189
Comments
what does the memcached backend do? it returns None for an object that's in the cache that's in an invalid pickle format? I can see the wisdom for both behaviors and this should be a configurable option only, or perhaps there can be a built-in proxybackend that squashes all errors. I am not too crazy about "all errors suppressed by default", like if memcached can't even get to the server, it returns None, is that right? seems familiar. I'd hate that. I think that the two backends have different default behaviors is an unfortunate side-effect of the fact that dogpile is interacting with different services of very different philosophies so I favor an option to make things behave differently but not by default, that would be more disruptive at this point. |
Yes, the memcached backend will swallow any exception, and it can configured at most to write a log to sys.stderr.
I don't like the fact that the memcache Client does that either. I think that any client (as in memcache.Client, redis.StrictRedis, etc.) should have strict behaviour (that is, raise all the errors), and it should be up to the layer using that client to decide what to do with the errors.
I am not so sure it would be disruptive at all, since the use case for dogpile.cache is to apply caching to functions. Caching is an optimization, but IMO decorated functions should not fail if the caching backend is not available/has errors. If we emit warnings when error occur, users can still configure a filter that makes them errors, and keep the old behaviour. But I can't think of a reason why one would expect to see errors bubbling up in case the cache backend is not available. |
caching is an optimization sure but dogpile.cache has been around for years and certainly different developers have different expectations. I'm +1 on an option to change the behavior, no change in default behavior. if i were to change the default behavior, it would be that everything raises by default and that is certainly what most developers would expect. memcached is not making that possible so we do the best we can and consider memcached to be the outlier. |
I understand that people may use the backends directly, so yes in that case it would break compatibility. |
"strict" mode means....raises? or warns ? or silently passes? is this flag at the backend level or at the Region level? Region level is fine but we'd have to build a generic exception catching mechanism for all of that, right? |
I would say that with strict=True (or passthrough_errors=True) should raise (if the underlying library somehow informs us of an error). I would have the flag at the backend level, not sure why it would be on the Region level (you can always configure a region and forward the "strict" flag to the backend, right?). |
Probably "strict" is not the best wording, as the default should be off (in my opinion), while "strict" gives more the idea that it should be turned on. |
but that means "strict=True" silently fails on memcached then, right? it seems like we really can't provide consistent behavior in both directions. which is why it seems like we can add a flag, or maybe just a proxy backend, that says, "exceptions become None" or something like that. I really dont know how to do that correctly, though. this really seems like a custom proxy backend thing to me. |
I would say that it should not be allowed to instantiate a backend passing strict=True in case it's not supported (memcached). Maybe we can just allow this boolean to be configured for RedisBackend, DBMBackend, MemoryBackend (although the only way it can fail is if the user is tampering with the internals). |
then what is the default value of "strict=True" ? different on the two backends? that's not a good idea. i think it should likely be simpler such as "suppress_exceptions" and it's only on backends that are not the memcached backend. but this really doesn't give you the "every backend works the same" experience you're looking for and honestly I dont think it's a realistic expectation.
|
"suppress_exceptions" sounds way better.
If we want to keep backwards compatibility, "suppress_exceptions" it has to default to False. If we want consistency, it should default to True.
Why not? I would opt-in for suppress_exception in the backends that support it. |
yes you'd need to opt in on this. the option would need some way to configure logging and all that. people will want different behaviors here. also I have no resources to work on this myself so you'd have to work on a PR with tests. |
I would be happy to work on it, I just wanted to know first if that would be accepted. |
My main concern is the burden of implementing this on every backend. new backends should not have to implement this flag also, idealy. it seems there should be either something in region itself, or a generic proxy backend, that implements this. |
True, but what if we have this option until the next major release, and after that exceptions are just going to be suppressed (still logged of course)? |
The question is more about what is the best behaviour of the backends. Should they raise on exception or not in case the cache backend is not available/has errors? |
the backends should absolutely raise by default. I think a pluggable solution, in the style of many other dogpile.cache elements, where an "exception handler" callable can be plugged into the region would be best. that way no changes need to occur to any backends and when users don't like the particular exceptions we are logging, or not, or how they are being logged, they can change it since the mechanism by which it works is transparent and pluggable. |
other examples of this in dogpile: RegionInvalidationStrategy key_mangler function_key_generator async_creation_runner etc. I can also see people wanting an exception handler that does a SQLAlchemy-like thing where it tries to relate the exception classes from different backends to each other. |
Right, I understand. Should I start working on a PR with this new argument for the Region? |
sure! pluggable impl makes me non-worried. :) |
FYI, the Redis backend also raises a lot of annoying exceptions on lock timeouts and race conditions with expensive processes. I need to find time to make a new testcase for this one day, but I think the situation was because a first process has a lock that times out during generation, a second process generates a new lock, and then the first process tries to release the lock it doesn't own (Exception!) and then the second process tries to release a lock that no longer exists (Exception!). The problem was due to somewhat incompatible implementation details of dogpile and Redis. The underlying Redis library love to raise Exceptions, and the points where they bubble up within dogpile require one of the two projects to be altered - it can't be handled properly by anyone simply implementing dogpile cache. I wish I could remember which mailing lists I was talking about this on. There were a handful of other people who kept battling the same issues as me. |
This was the ticket I once opened with redis. I should have noted it here earlier. I never got around to making the test case, but I do recall the overall issue is because of how/where redis is leveraged in the dogpile code. |
I find it a little bit inconsistent the fact that the RedisBackend can raise errors while trying to get a value from the cache.
In comparison, the MemcachedBackend doesn't raise any error.
Dogpile.cache is about caching, I think it should do best-effort to try to get the value from the cache, but it should not raise exceptions in case of errors while getting the value.
This example shows the issue.
Just run it in 2 different python versions that have a different pickle.HIGHEST_PROTOCOL defined, like python2.7 and python3.6 for example:
Dogpile cache version: 0.9.0 (but the same happens with the latest version, 1.0.2)
I can make a PR if you think the behaviour should be more best-effort.
The text was updated successfully, but these errors were encountered: