utg-base 1.21.2__py3-none-any.whl → 1.22.0__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
|
@@ -1,8 +1,6 @@
|
|
|
1
1
|
from functools import wraps
|
|
2
2
|
from typing import Literal, Dict, Union, List
|
|
3
3
|
|
|
4
|
-
from rest_framework.exceptions import PermissionDenied
|
|
5
|
-
|
|
6
4
|
from .utils import has_perms as has_permissions
|
|
7
5
|
|
|
8
6
|
|
|
@@ -13,15 +11,20 @@ def has_perm(*permission: str, operator: Literal["OR", "AND"] = "OR"):
|
|
|
13
11
|
has_perm('PERM1', 'PERM2')
|
|
14
12
|
|
|
15
13
|
:param permission: One or many permissions.
|
|
14
|
+
:param operator: Any of (OR, AND)
|
|
16
15
|
"""
|
|
16
|
+
|
|
17
17
|
def decorator(view_func):
|
|
18
18
|
view_func._perms = permission
|
|
19
|
+
|
|
19
20
|
@wraps(view_func)
|
|
20
21
|
def _wrapped_view(self, request, *args, **kwargs):
|
|
21
22
|
if not has_permissions(user_id=request.user.id, perms=list(permission), operator=operator, request=request):
|
|
22
23
|
self.permission_denied(request)
|
|
23
24
|
return view_func(self, request, *args, **kwargs)
|
|
25
|
+
|
|
24
26
|
return _wrapped_view
|
|
27
|
+
|
|
25
28
|
return decorator
|
|
26
29
|
|
|
27
30
|
|
|
@@ -34,6 +37,7 @@ def has_class_perm(permissions_map: Dict[str, Union[str, List[str]]], operator:
|
|
|
34
37
|
'update': ['perm4']
|
|
35
38
|
})
|
|
36
39
|
"""
|
|
40
|
+
|
|
37
41
|
def decorator(cls):
|
|
38
42
|
for action, perms in permissions_map.items():
|
|
39
43
|
if isinstance(perms, str):
|
utg_base/utils/data.py
CHANGED
|
@@ -1,8 +1,12 @@
|
|
|
1
|
+
import datetime
|
|
2
|
+
from collections import defaultdict
|
|
1
3
|
from collections.abc import Iterable
|
|
2
4
|
from copy import deepcopy
|
|
3
5
|
|
|
4
6
|
from django.db.models import QuerySet
|
|
5
7
|
|
|
8
|
+
from utg_base.utils.date import UDate, UDateTime
|
|
9
|
+
|
|
6
10
|
|
|
7
11
|
def deep_map(data: dict | list, func_cond, func_map, in_place=True):
|
|
8
12
|
if not in_place:
|
|
@@ -153,3 +157,35 @@ def compute_change_percent(current, previous, ndigits=2):
|
|
|
153
157
|
if not current or previous in (None, 0):
|
|
154
158
|
return None
|
|
155
159
|
return round(((current - previous) / previous) * 100, ndigits)
|
|
160
|
+
|
|
161
|
+
|
|
162
|
+
def list_to_dict(rows: Iterable[dict], key: str, key_func=None) -> dict:
|
|
163
|
+
if not key_func:
|
|
164
|
+
key_func = lambda x: x
|
|
165
|
+
return {row[key_func(key)]: row for row in rows}
|
|
166
|
+
|
|
167
|
+
|
|
168
|
+
def generate_series(start: UDate | UDateTime, end: UDate | UDateTime, step: datetime.timedelta):
|
|
169
|
+
current = start
|
|
170
|
+
if step.total_seconds() == 0:
|
|
171
|
+
raise ValueError("step cannot be zero")
|
|
172
|
+
|
|
173
|
+
series = []
|
|
174
|
+
while current <= end:
|
|
175
|
+
series.append(current)
|
|
176
|
+
current += step
|
|
177
|
+
|
|
178
|
+
return series
|
|
179
|
+
|
|
180
|
+
|
|
181
|
+
def fill_series(series: Iterable, data: dict, keys: str | list[str]):
|
|
182
|
+
result = defaultdict(list)
|
|
183
|
+
|
|
184
|
+
for term in series:
|
|
185
|
+
for key in keys:
|
|
186
|
+
result[key].append(data.get(term, {}).get(key, None))
|
|
187
|
+
|
|
188
|
+
data = []
|
|
189
|
+
for key in keys:
|
|
190
|
+
data.append(result[key])
|
|
191
|
+
return data
|
|
@@ -40,7 +40,7 @@ utg_base/models/__init__.py,sha256=1zXygGICiR3iUCKdkNal9d3i3kNp654gFgBf_VlR2gI,6
|
|
|
40
40
|
utg_base/models/jwt_user.py,sha256=6TQ5wB_OZBtGhRL-2MonBGZm0n0Y86s4BRTxiRlUJOk,375
|
|
41
41
|
utg_base/models/timestamp.py,sha256=AkCliNXnvs8Z17b1mcS7gOK7v6h3Jul6WCyGyVAkb-w,217
|
|
42
42
|
utg_base/permissions/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
|
43
|
-
utg_base/permissions/decorators.py,sha256=
|
|
43
|
+
utg_base/permissions/decorators.py,sha256=db26UrbOk66AuQBRFyxcNoQDSC3EWkencptRHk0ikUM,2105
|
|
44
44
|
utg_base/permissions/folder.py,sha256=uJv40FVb7R379qss66a5oUcLK7KCUIL6DPbzEcGOw38,694
|
|
45
45
|
utg_base/permissions/utils.py,sha256=Rp8_ZuU5SdXri4amfFlMmrBoJUjwwQ4GYxDujRpt8YU,3264
|
|
46
46
|
utg_base/references_api/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
|
@@ -60,7 +60,7 @@ utg_base/u_services/migrations/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5
|
|
|
60
60
|
utg_base/u_services/models.py,sha256=seaxBj9vNWrPu1Y8r7NzHEuXemtSvqLkaw4S1Xgr58I,1310
|
|
61
61
|
utg_base/u_services/u_requests.py,sha256=fGNWWQ3RJOOARgNHkScsqlGXwP6sI-HnS3Y-HnSoVS4,1709
|
|
62
62
|
utg_base/utils/__init__.py,sha256=zRTdSLGpukInlb7oRD6vcI1jKQkkBR3jTCiOiwOmRyA,159
|
|
63
|
-
utg_base/utils/data.py,sha256=
|
|
63
|
+
utg_base/utils/data.py,sha256=8w0Lr8gj4Klu8903XWZlZeMzz9HxNX9_W08KVeIQMuw,5039
|
|
64
64
|
utg_base/utils/date.py,sha256=fS4o5Nd4HyNyHBJTSsasTlqSMNjnNHwocSYdinkKyeY,4371
|
|
65
65
|
utg_base/utils/dict_util.py,sha256=ipdCZO8aTukGQ319OWHb2Ij5MNtV-FioJQ4qCX3Th48,758
|
|
66
66
|
utg_base/utils/response_processors.py,sha256=WdZQL49wOJqCIY2MucAI6sez_llCqih0v_ltQa-mv7k,687
|
|
@@ -69,6 +69,6 @@ utg_base/utils/sql.py,sha256=rqIWcSjdjIMszdRnsnhV5TTYB8W17RPOujIQA9rKC_Y,762
|
|
|
69
69
|
utg_base/utils/string.py,sha256=ATwIo9uLa00p85h_NjRYLcjRs8o3KSGF7s2yhTg5GiA,1073
|
|
70
70
|
utg_base/utils/thread.py,sha256=4RqRnwtyHymY-dNcuPrMSTamE2V7wCMVfzzyIb0P4TI,2191
|
|
71
71
|
utg_base/utils/translation.py,sha256=GxJHUt0iar_0E7RWBPbeLFQ4DhgXBjffHCmxfKyjFtk,463
|
|
72
|
-
utg_base-1.
|
|
73
|
-
utg_base-1.
|
|
74
|
-
utg_base-1.
|
|
72
|
+
utg_base-1.22.0.dist-info/METADATA,sha256=3mO-4yLX73EtPVU5-gCl8NMCF8wmdw8Z5xzYYs9gZsY,960
|
|
73
|
+
utg_base-1.22.0.dist-info/WHEEL,sha256=Nq82e9rUAnEjt98J6MlVmMCZb-t9cYE2Ir1kpBmnWfs,88
|
|
74
|
+
utg_base-1.22.0.dist-info/RECORD,,
|
|
File without changes
|