Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/alisaifee/limits
Rate limiting using various strategies and storage backends such as redis & memcached
https://github.com/alisaifee/limits
async etcd memcached mongodb python rate-limiting redis
Last synced: 5 days ago
JSON representation
Rate limiting using various strategies and storage backends such as redis & memcached
- Host: GitHub
- URL: https://github.com/alisaifee/limits
- Owner: alisaifee
- License: mit
- Created: 2015-01-08T04:34:40.000Z (about 10 years ago)
- Default Branch: master
- Last Pushed: 2024-05-01T10:00:18.000Z (9 months ago)
- Last Synced: 2024-05-02T02:12:00.243Z (9 months ago)
- Topics: async, etcd, memcached, mongodb, python, rate-limiting, redis
- Language: Python
- Homepage: https://limits.readthedocs.org
- Size: 1.14 MB
- Stars: 346
- Watchers: 8
- Forks: 56
- Open Issues: 5
-
Metadata Files:
- Readme: README.rst
- Changelog: HISTORY.rst
- Funding: .github/FUNDING.yml
- License: LICENSE.txt
- Code of conduct: CODE_OF_CONDUCT.md
Awesome Lists containing this project
README
.. |ci| image:: https://github.com/alisaifee/limits/workflows/CI/badge.svg?branch=master
:target: https://github.com/alisaifee/limits/actions?query=branch%3Amaster+workflow%3ACI
.. |codecov| image:: https://codecov.io/gh/alisaifee/limits/branch/master/graph/badge.svg
:target: https://codecov.io/gh/alisaifee/limits
.. |pypi| image:: https://img.shields.io/pypi/v/limits.svg?style=flat-square
:target: https://pypi.python.org/pypi/limits
.. |pypi-versions| image:: https://img.shields.io/pypi/pyversions/limits?style=flat-square
:target: https://pypi.python.org/pypi/limits
.. |license| image:: https://img.shields.io/pypi/l/limits.svg?style=flat-square
:target: https://pypi.python.org/pypi/limits
.. |docs| image:: https://readthedocs.org/projects/limits/badge/?version=latest
:target: https://limits.readthedocs.orglimits
------
|docs| |ci| |codecov| |pypi| |pypi-versions| |license|**limits** is a python library to perform rate limiting with commonly used storage backends (Redis, Memcached, MongoDB & Etcd).
Supported Strategies
====================
`Fixed Window `_
This strategy resets at a fixed interval (start of minute, hour, day etc).
For example, given a rate limit of ``10/minute`` the strategy will:- Allow 10 requests between ``00:01:00`` and ``00:02:00``
- Allow 10 requests at ``00:00:59`` and 10 more requests at ``00:01:00```Fixed Window (Elastic) `_
Identical to Fixed window, except every breach of rate limit results in an extension
to the time out. For example a rate limit of `1/minute` hit twice within a minute will
result in a lock-out for two minutes.`Moving Window `_
Sliding window strategy enforces a rate limit of N/(m time units)
on the **last m** time units at the second granularity.For example, with a rate limit of ``10/minute``:
- Allow 9 requests that arrive at ``00:00:59``
- Allow another request that arrives at ``00:01:00``
- Reject the request that arrives at ``00:01:01``Storage backends
================- `Redis `_
- `Memcached `_
- `MongoDB `_
- `Etcd `_
- `In-Memory `_Dive right in
=============Initialize the storage backend
.. code-block:: python
from limits import storage
memory_storage = storage.MemoryStorage()
# or memcached
memcached_storage = storage.MemcachedStorage("memcached://localhost:11211")
# or redis
redis_storage = storage.RedisStorage("redis://localhost:6379")
# or use the factory
storage_uri = "memcached://localhost:11211"
some_storage = storage.storage_from_string(storage_uri)Initialize a rate limiter with the Moving Window Strategy
.. code-block:: python
from limits import strategies
moving_window = strategies.MovingWindowRateLimiter(memory_storage)Initialize a rate limit
.. code-block:: python
from limits import parse
one_per_minute = parse("1/minute")Initialize a rate limit explicitly
.. code-block:: python
from limits import RateLimitItemPerSecond
one_per_second = RateLimitItemPerSecond(1, 1)Test the limits
.. code-block:: python
assert True == moving_window.hit(one_per_minute, "test_namespace", "foo")
assert False == moving_window.hit(one_per_minute, "test_namespace", "foo")
assert True == moving_window.hit(one_per_minute, "test_namespace", "bar")assert True == moving_window.hit(one_per_second, "test_namespace", "foo")
assert False == moving_window.hit(one_per_second, "test_namespace", "foo")
time.sleep(1)
assert True == moving_window.hit(one_per_second, "test_namespace", "foo")Check specific limits without hitting them
.. code-block:: python
assert True == moving_window.hit(one_per_second, "test_namespace", "foo")
while not moving_window.test(one_per_second, "test_namespace", "foo"):
time.sleep(0.01)
assert True == moving_window.hit(one_per_second, "test_namespace", "foo")Query available capacity and reset time for a limit
.. code-block:: python
assert True == moving_window.hit(one_per_minute, "test_namespace", "foo")
window = moving_window.get_window_stats(one_per_minute, "test_namespace", "foo")
assert window.remaining == 0
assert False == moving_window.hit(one_per_minute, "test_namespace", "foo")
time.sleep(window.reset_time - time.time())
assert True == moving_window.hit(one_per_minute, "test_namespace", "foo")Links
=====* `Documentation `_
* `Changelog `_