Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Do a major refactor of the library (#9)
* Do a major refactor of the library I think there was a significant complexity in the implementation of the library. I get that we tried to aim to reuse some code for sync and async implementation but it made the hard to follow, and I couldn't really tell what was public API and what was not, behind these long inheritance chains. I also believe that the way the public API is structured was wrong. Due to the way the code was structured, we had to duplicate some docstrings & code in various places. So, I changed the public API significantly and made it similar to the TypeScript implementation, where we only export the Ratelimit class, which you can configure with different rate limiters. That API sounds much more natural to me and makes it simple to declare & document what is public. I have also changed some parameter names, from things like max_number_of_tokens to max_tokens, to make it more idiomatic. Also, I have renamed metadata methods like remaining and reset to get_remaining and get_reset. Without the get prefix, I believe the names of these methods were confusing. I have also updated the README and examples with these changes. Also, I have improved the tests. There were no tests for the sliding window, and existing tests were failing from time to time. I tried to make them more robust. I have also enabled the mypy checks, after fixing the issues. I also found the return value of the reset method in case the identifier not rate limited confusing. The value of -1 can cause problems if the docs were not read carefully. Lastly, I have changed the rate limit algorithms we had slightly, as they had some bugs. For sliding window: - The previous window calculation was wrong. After quantizing the current time and determining the current window, the previous window is current window - 1, not current window - window duration. - The return value for the remaining tokens was wrong. It was not taking the number of requests * weight from the previous window into account. Because of that, it was returning a remaining token count larger than the actual value. For token bucket: - The bucket was not evicted at all. That was essentially a data leak. The implementation in the TypeScript implementation is also wrong, it is evicting the token in fixed intervals no matter what. What we need is not TTL but max idle with the time equal to the number of intervals required to fill the bucket to max tokens. Then, after this interval, the bucket can be safely evicted as after that point never evicting the bucket and recreating the bucket becomes the same thing: Bucket always includes token equal to the max tokens. Since redis does not have max idle feature for keys, we have to implement it with TTL by resetting the TTL each time we access the key. * do not use generic generator type for python 3.8 compatibility * address review comments * fix mypy errors * relax the test
- Loading branch information