1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
|
from __future__ import annotations
from collections import defaultdict
from contextlib import contextmanager
from dataclasses import dataclass, field
from typing import Any, Callable, DefaultDict, Generator, List, Tuple
AuditHookType = Callable[[str, Tuple[Any, ...]], Any]
@dataclass
class AuditHookDispatcher:
handlers: DefaultDict[str, List[AuditHookType]] = field(
default_factory=lambda: defaultdict(list)
)
def audit(self, name: str, args: Tuple[Any, ...]) -> Any:
handlers = self.handlers[name]
for handler in handlers:
handler(name, args)
@contextmanager
def ctx_hook(self, name: str, hook: AuditHookType) -> Generator[None, None, None]:
self.handlers[name].append(hook)
try:
yield None
finally:
self.handlers[name].remove(hook)
|