2013-03-20 15:31:27 +01:00
|
|
|
import hashlib
|
2019-12-16 06:27:34 +01:00
|
|
|
import re
|
2020-09-05 04:02:13 +02:00
|
|
|
import secrets
|
2021-08-14 03:32:16 +02:00
|
|
|
from typing import Any, Callable, List, Optional, TypeVar
|
2013-08-08 16:50:58 +02:00
|
|
|
|
2013-04-16 22:57:50 +02:00
|
|
|
from django.conf import settings
|
|
|
|
|
2021-02-12 08:20:45 +01:00
|
|
|
T = TypeVar("T")
|
2016-06-03 18:39:57 +02:00
|
|
|
|
2021-02-12 08:19:30 +01:00
|
|
|
|
|
|
|
def statsd_key(val: str, clean_periods: bool = False) -> str:
|
2021-02-12 08:20:45 +01:00
|
|
|
if ":" in val:
|
|
|
|
val = val.split(":")[0]
|
|
|
|
val = val.replace("-", "_")
|
2013-04-30 23:58:59 +02:00
|
|
|
if clean_periods:
|
2021-02-12 08:20:45 +01:00
|
|
|
val = val.replace(".", "_")
|
2013-04-16 22:57:50 +02:00
|
|
|
|
|
|
|
return val
|
|
|
|
|
2021-02-12 08:19:30 +01:00
|
|
|
|
2017-11-05 11:37:41 +01:00
|
|
|
class StatsDWrapper:
|
2013-04-16 22:57:50 +02:00
|
|
|
"""Transparently either submit metrics to statsd
|
|
|
|
or do nothing without erroring out"""
|
|
|
|
|
|
|
|
# Backported support for gauge deltas
|
|
|
|
# as our statsd server supports them but supporting
|
|
|
|
# pystatsd is not released yet
|
2021-02-12 08:19:30 +01:00
|
|
|
def _our_gauge(self, stat: str, value: float, rate: float = 1, delta: bool = False) -> None:
|
2019-01-31 14:32:37 +01:00
|
|
|
"""Set a gauge value."""
|
|
|
|
from django_statsd.clients import statsd
|
2021-02-12 08:19:30 +01:00
|
|
|
|
2019-01-31 14:32:37 +01:00
|
|
|
if delta:
|
2021-02-12 08:20:45 +01:00
|
|
|
value_str = f"{value:+g}|g"
|
2019-01-31 14:32:37 +01:00
|
|
|
else:
|
2021-02-12 08:20:45 +01:00
|
|
|
value_str = f"{value:g}|g"
|
2019-01-31 14:32:37 +01:00
|
|
|
statsd._send(stat, value_str, rate)
|
2013-04-16 22:57:50 +02:00
|
|
|
|
2017-11-05 11:15:10 +01:00
|
|
|
def __getattr__(self, name: str) -> Any:
|
2013-04-16 22:57:50 +02:00
|
|
|
# Hand off to statsd if we have it enabled
|
|
|
|
# otherwise do nothing
|
2021-02-12 08:20:45 +01:00
|
|
|
if name in ["timer", "timing", "incr", "decr", "gauge"]:
|
|
|
|
if settings.STATSD_HOST != "":
|
2013-04-16 22:57:50 +02:00
|
|
|
from django_statsd.clients import statsd
|
2021-02-12 08:19:30 +01:00
|
|
|
|
2021-02-12 08:20:45 +01:00
|
|
|
if name == "gauge":
|
2013-04-16 22:57:50 +02:00
|
|
|
return self._our_gauge
|
|
|
|
else:
|
|
|
|
return getattr(statsd, name)
|
|
|
|
else:
|
|
|
|
return lambda *args, **kwargs: None
|
|
|
|
|
|
|
|
raise AttributeError
|
|
|
|
|
2021-02-12 08:19:30 +01:00
|
|
|
|
2013-04-16 22:57:50 +02:00
|
|
|
statsd = StatsDWrapper()
|
2013-03-12 17:51:35 +01:00
|
|
|
|
2021-02-12 08:19:30 +01:00
|
|
|
|
|
|
|
def make_safe_digest(string: str, hash_func: Callable[[bytes], Any] = hashlib.sha1) -> str:
|
2013-03-20 15:31:27 +01:00
|
|
|
"""
|
|
|
|
return a hex digest of `string`.
|
|
|
|
"""
|
|
|
|
# hashlib.sha1, md5, etc. expect bytes, so non-ASCII strings must
|
|
|
|
# be encoded.
|
2021-08-02 23:20:39 +02:00
|
|
|
return hash_func(string.encode()).hexdigest()
|
2013-04-16 22:57:50 +02:00
|
|
|
|
|
|
|
|
2017-11-05 11:15:10 +01:00
|
|
|
def log_statsd_event(name: str) -> None:
|
2013-04-16 22:57:50 +02:00
|
|
|
"""
|
|
|
|
Sends a single event to statsd with the desired name and the current timestamp
|
|
|
|
|
|
|
|
This can be used to provide vertical lines in generated graphs,
|
|
|
|
for example when doing a prod deploy, bankruptcy request, or
|
|
|
|
other one-off events
|
|
|
|
|
|
|
|
Note that to draw this event as a vertical line in graphite
|
|
|
|
you can use the drawAsInfinite() command
|
|
|
|
"""
|
2020-06-10 06:41:04 +02:00
|
|
|
event_name = f"events.{name}"
|
2013-06-07 23:53:20 +02:00
|
|
|
statsd.incr(event_name)
|
2013-08-08 16:50:58 +02:00
|
|
|
|
2021-02-12 08:19:30 +01:00
|
|
|
|
2018-08-01 11:18:37 +02:00
|
|
|
def generate_api_key() -> str:
|
2020-09-05 04:02:13 +02:00
|
|
|
api_key = ""
|
|
|
|
while len(api_key) < 32:
|
|
|
|
# One iteration suffices 99.4992% of the time.
|
|
|
|
api_key += secrets.token_urlsafe(3 * 9).replace("_", "").replace("-", "")
|
|
|
|
return api_key[:32]
|
2018-08-01 11:18:37 +02:00
|
|
|
|
2021-02-12 08:19:30 +01:00
|
|
|
|
2019-12-16 06:27:34 +01:00
|
|
|
def has_api_key_format(key: str) -> bool:
|
|
|
|
return bool(re.fullmatch(r"([A-Za-z0-9]){32}", key))
|
|
|
|
|
2021-02-12 08:19:30 +01:00
|
|
|
|
2021-07-25 11:20:48 +02:00
|
|
|
def assert_is_not_none(value: Optional[T]) -> T:
|
|
|
|
assert value is not None
|
|
|
|
return value
|
|
|
|
|
|
|
|
|
2021-02-12 08:19:30 +01:00
|
|
|
def process_list_in_batches(
|
2023-04-26 03:18:14 +02:00
|
|
|
lst: List[T], chunk_size: int, process_batch: Callable[[List[T]], None]
|
2021-02-12 08:19:30 +01:00
|
|
|
) -> None:
|
2018-10-15 14:24:13 +02:00
|
|
|
offset = 0
|
|
|
|
|
|
|
|
while True:
|
2021-02-12 08:19:30 +01:00
|
|
|
items = lst[offset : offset + chunk_size]
|
2018-10-15 14:24:13 +02:00
|
|
|
if not items:
|
|
|
|
break
|
|
|
|
process_batch(items)
|
|
|
|
offset += chunk_size
|