From 875c78b3aa8278da174d4c44e107821c63fda342 Mon Sep 17 00:00:00 2001 From: Bruno Souza <12611098+brnosouza@users.noreply.github.com> Date: Thu, 1 Aug 2024 16:32:16 -0300 Subject: [PATCH] Making sure the cache key accepts any data (#110) Co-authored-by: Bruno Souza --- pyproject.toml | 2 +- security/auth_throttling/__init__.py | 5 +++-- tests/tests.py | 29 +++++++++++++++++++++------- 3 files changed, 26 insertions(+), 10 deletions(-) diff --git a/pyproject.toml b/pyproject.toml index 5c6c9c0..b9f90f9 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -1,6 +1,6 @@ [tool.poetry] name = "django-security" -version = "1.0.2" +version = "1.0.3" homepage = "https://github.com/sdelements/django-security" description = "Models, views, middlewares and forms to facilitate security hardening of Django applications." authors = ["Security Compass "] diff --git a/security/auth_throttling/__init__.py b/security/auth_throttling/__init__.py index f82675e..2ab498b 100644 --- a/security/auth_throttling/__init__.py +++ b/security/auth_throttling/__init__.py @@ -1,6 +1,7 @@ import hashlib import logging import time +import typing import urllib.parse from math import ceil @@ -44,8 +45,8 @@ def delay_message(remainder): return _("%d seconds") % ceil(remainder) -def _to_ascii_compatible(value: str): - if not value.isascii(): +def _to_ascii_compatible(value: typing.Any): + if isinstance(value, str) and not value.isascii(): value = urllib.parse.quote(value) return value diff --git a/tests/tests.py b/tests/tests.py index d645b90..82f2db2 100644 --- a/tests/tests.py +++ b/tests/tests.py @@ -1078,13 +1078,28 @@ class UnicodeDataTests(TestCase): IP_ADDRESS = "127.0.0.1" def test_unicode_data_in_cache_key(self): - self._execute_with_unicode_data(self.USERNAME, self.IP_ADDRESS) + self._execute_with_data(self.USERNAME, self.IP_ADDRESS) - def _execute_with_unicode_data(self, username, ip): + def test_types_in_cache_key(self): + """ + We can send any kind of data for the downstream functions, + usually strings (maybe the username or email) and ints (maybe the userId) + """ + + self._execute_with_data(1, self.IP_ADDRESS) + self._execute_with_data(2.67, self.IP_ADDRESS) + self._execute_with_data(bool, self.IP_ADDRESS) + self._execute_with_data({"key": "value"}, self.IP_ADDRESS) + self._execute_with_data([1], self.IP_ADDRESS) + self._execute_with_data({1, 2}, self.IP_ADDRESS) + self._execute_with_data((1, 2), self.IP_ADDRESS) + self._execute_with_data("some_string", self.IP_ADDRESS) + + def _execute_with_data(self, data, ip): try: - increment_counters(username=username, ip=ip) - reset_counters(username=username, ip=ip) - throttling_delay(username=username, ip=ip) - attempt_count(attempt_type="auth", id=username) + increment_counters(key=data, ip=ip) + reset_counters(key=data, ip=ip) + throttling_delay(username=data, ip=ip) + attempt_count(attempt_type="auth", id=data) except Exception: - self.fail("Unicode data not allowed") + self.fail("Unicode or incompatible data not allowed")