lib.itmens/common/utils.py

156 lines
5 KiB
Python
Raw Normal View History

2023-08-13 23:11:12 -04:00
import functools
2021-08-01 12:36:03 +02:00
import uuid
2023-07-20 21:59:49 -04:00
from typing import TYPE_CHECKING
2023-08-13 18:00:10 -04:00
from django.http import Http404, HttpRequest, HttpResponseRedirect
2021-08-01 12:36:03 +02:00
from django.utils import timezone
from django.utils.baseconv import base62
2021-08-01 12:36:03 +02:00
2023-07-20 21:59:49 -04:00
if TYPE_CHECKING:
from users.models import APIdentity, User
class AuthedHttpRequest(HttpRequest):
"""
A subclass of HttpRequest for type-checking only
"""
user: "User"
target_identity: "APIdentity"
2023-01-11 19:11:31 -05:00
2023-08-13 18:00:10 -04:00
class HTTPResponseHXRedirect(HttpResponseRedirect):
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
self["HX-Redirect"] = self["Location"]
status_code = 200
2023-12-09 16:25:41 -05:00
def user_identity_required(func): # TODO make this a middleware
@functools.wraps(func)
def wrapper(request, *args, **kwargs):
from users.models import APIdentity
identity = None
if request.user.is_authenticated:
try:
identity = APIdentity.objects.get(user=request.user)
except APIdentity.DoesNotExist:
return HttpResponseRedirect("/account/register")
request.identity = identity
return func(request, *args, **kwargs)
return wrapper
2023-08-13 23:11:12 -04:00
def target_identity_required(func):
@functools.wraps(func)
def wrapper(request, user_name, *args, **kwargs):
from users.models import APIdentity
from users.views import render_user_blocked, render_user_not_found
try:
target = APIdentity.get_by_handler(user_name)
except APIdentity.DoesNotExist:
return render_user_not_found(request)
target_user = target.user
2023-12-09 16:25:41 -05:00
viewer = None
if target_user and not target_user.is_active:
return render_user_not_found(request)
2023-12-09 16:25:41 -05:00
if request.user.is_authenticated:
try:
viewer = APIdentity.objects.get(user=request.user)
except APIdentity.DoesNotExist:
return HttpResponseRedirect("/account/register")
if request.user != target_user:
if target.is_blocking(viewer) or target.is_blocked_by(viewer):
return render_user_blocked(request)
else:
viewer = None
2023-08-13 23:11:12 -04:00
request.target_identity = target
2023-12-09 16:25:41 -05:00
request.identity = viewer
2023-08-13 23:11:12 -04:00
return func(request, user_name, *args, **kwargs)
return wrapper
2020-07-03 15:36:23 +08:00
class PageLinksGenerator:
# TODO inherit django paginator
"""
Calculate the pages for multiple links pagination.
length -- the number of page links in pagination
"""
2023-01-11 19:11:31 -05:00
def __init__(self, length: int, current_page: int, total_pages: int):
2020-07-03 15:36:23 +08:00
current_page = int(current_page)
self.current_page = current_page
2022-01-22 14:04:21 -05:00
self.previous_page = current_page - 1 if current_page > 1 else None
self.next_page = current_page + 1 if current_page < total_pages else None
self.start_page = 1
self.end_page = 1
2020-07-03 15:36:23 +08:00
self.page_range = None
self.has_prev = None
self.has_next = None
start_page = current_page - length // 2
end_page = current_page + length // 2
# decision is based on the start page and the end page
# both sides overflow
2023-01-11 19:11:31 -05:00
if (start_page < 1 and end_page > total_pages) or length >= total_pages:
2020-07-03 15:36:23 +08:00
self.start_page = 1
self.end_page = total_pages
self.has_prev = False
self.has_next = False
2023-01-11 19:11:31 -05:00
2020-07-03 15:36:23 +08:00
elif start_page < 1 and not end_page > total_pages:
self.start_page = 1
# this won't overflow because the total pages are more than the length
self.end_page = end_page - (start_page - 1)
self.has_prev = False
if end_page == total_pages:
self.has_next = False
else:
self.has_next = True
2023-01-11 19:11:31 -05:00
2020-07-03 15:36:23 +08:00
elif not start_page < 1 and end_page > total_pages:
self.end_page = total_pages
self.start_page = start_page - (end_page - total_pages)
self.has_next = False
if start_page == 1:
self.has_prev = False
else:
self.has_prev = True
# both sides do not overflow
elif not start_page < 1 and not end_page > total_pages:
self.start_page = start_page
self.end_page = end_page
self.has_prev = True
self.has_next = True
self.first_page = 1
self.last_page = total_pages
self.page_range = range(self.start_page, self.end_page + 1)
2020-10-03 23:27:41 +02:00
# assert self.has_prev is not None and self.has_next is not None
2021-08-01 12:36:03 +02:00
def GenerateDateUUIDMediaFilePath(instance, filename, path_root):
2023-01-11 19:11:31 -05:00
ext = filename.split(".")[-1]
2021-08-01 12:36:03 +02:00
filename = "%s.%s" % (uuid.uuid4(), ext)
2023-01-11 19:11:31 -05:00
root = ""
if path_root.endswith("/"):
2021-08-01 12:36:03 +02:00
root = path_root
else:
2023-01-11 19:11:31 -05:00
root = path_root + "/"
return root + timezone.now().strftime("%Y/%m/%d") + f"{filename}"
def get_uuid_or_404(uuid_b62):
try:
i = base62.decode(uuid_b62)
return uuid.UUID(int=i)
except ValueError:
raise Http404("Malformed Base62 UUID")