2024-09-17 10:10:06 +00:00
|
|
|
from typing import Annotated
|
|
|
|
|
|
|
|
import annotated_types
|
2024-07-18 18:23:30 +00:00
|
|
|
from django.conf import settings
|
2024-09-25 10:08:11 +00:00
|
|
|
from django.db.models import F
|
2024-07-18 18:23:30 +00:00
|
|
|
from django.http import HttpResponse
|
2024-08-01 17:02:29 +00:00
|
|
|
from ninja import Query
|
|
|
|
from ninja_extra import ControllerBase, api_controller, paginate, route
|
2024-07-18 18:23:30 +00:00
|
|
|
from ninja_extra.exceptions import PermissionDenied
|
2024-08-01 17:02:29 +00:00
|
|
|
from ninja_extra.pagination import PageNumberPaginationExtra
|
|
|
|
from ninja_extra.schemas import PaginatedResponseSchema
|
2024-07-18 18:23:30 +00:00
|
|
|
|
|
|
|
from club.models import Mailing
|
2025-01-10 20:37:12 +00:00
|
|
|
from core.auth.api_permissions import CanAccessLookup, CanView
|
2024-10-20 18:55:05 +00:00
|
|
|
from core.models import Group, SithFile, User
|
2024-09-17 10:10:06 +00:00
|
|
|
from core.schemas import (
|
|
|
|
FamilyGodfatherSchema,
|
2024-10-20 18:55:05 +00:00
|
|
|
GroupSchema,
|
2024-09-17 10:10:06 +00:00
|
|
|
MarkdownSchema,
|
2024-10-19 22:18:53 +00:00
|
|
|
SithFileSchema,
|
2024-09-17 10:10:06 +00:00
|
|
|
UserFamilySchema,
|
2024-08-01 17:02:29 +00:00
|
|
|
UserFilterSchema,
|
|
|
|
UserProfileSchema,
|
2024-09-17 10:10:06 +00:00
|
|
|
)
|
2024-07-18 18:23:30 +00:00
|
|
|
from core.templatetags.renderer import markdown
|
|
|
|
|
|
|
|
|
|
|
|
@api_controller("/markdown")
|
|
|
|
class MarkdownController(ControllerBase):
|
|
|
|
@route.post("", url_name="markdown")
|
|
|
|
def render_markdown(self, body: MarkdownSchema):
|
|
|
|
"""Convert the markdown text into html."""
|
|
|
|
return HttpResponse(markdown(body.text), content_type="text/html")
|
|
|
|
|
|
|
|
|
|
|
|
@api_controller("/mailings")
|
|
|
|
class MailingListController(ControllerBase):
|
|
|
|
@route.get("", response=str)
|
|
|
|
def fetch_mailing_lists(self, key: str):
|
|
|
|
if key != settings.SITH_MAILING_FETCH_KEY:
|
|
|
|
raise PermissionDenied
|
|
|
|
mailings = Mailing.objects.filter(
|
|
|
|
is_moderated=True, club__is_active=True
|
|
|
|
).prefetch_related("subscriptions")
|
|
|
|
data = "\n".join(m.fetch_format() for m in mailings)
|
|
|
|
return data
|
2024-09-17 10:10:06 +00:00
|
|
|
|
|
|
|
|
2024-10-19 22:18:53 +00:00
|
|
|
@api_controller("/user", permissions=[CanAccessLookup])
|
2024-08-01 17:02:29 +00:00
|
|
|
class UserController(ControllerBase):
|
|
|
|
@route.get("", response=list[UserProfileSchema])
|
|
|
|
def fetch_profiles(self, pks: Query[set[int]]):
|
|
|
|
return User.objects.filter(pk__in=pks)
|
|
|
|
|
2024-09-23 19:26:34 +00:00
|
|
|
@route.get(
|
|
|
|
"/search",
|
|
|
|
response=PaginatedResponseSchema[UserProfileSchema],
|
|
|
|
url_name="search_users",
|
|
|
|
)
|
2024-08-01 17:02:29 +00:00
|
|
|
@paginate(PageNumberPaginationExtra, page_size=20)
|
|
|
|
def search_users(self, filters: Query[UserFilterSchema]):
|
2024-09-25 10:08:11 +00:00
|
|
|
return filters.filter(
|
|
|
|
User.objects.order_by(F("last_login").desc(nulls_last=True))
|
|
|
|
)
|
2024-08-01 17:02:29 +00:00
|
|
|
|
|
|
|
|
2024-10-19 22:18:53 +00:00
|
|
|
@api_controller("/file")
|
|
|
|
class SithFileController(ControllerBase):
|
|
|
|
@route.get(
|
|
|
|
"/search",
|
|
|
|
response=PaginatedResponseSchema[SithFileSchema],
|
|
|
|
permissions=[CanAccessLookup],
|
|
|
|
)
|
|
|
|
@paginate(PageNumberPaginationExtra, page_size=50)
|
2024-10-20 21:25:56 +00:00
|
|
|
def search_files(self, search: Annotated[str, annotated_types.MinLen(1)]):
|
|
|
|
return SithFile.objects.filter(is_in_sas=False).filter(name__icontains=search)
|
2024-10-19 22:18:53 +00:00
|
|
|
|
|
|
|
|
2024-10-20 18:55:05 +00:00
|
|
|
@api_controller("/group")
|
|
|
|
class GroupController(ControllerBase):
|
|
|
|
@route.get(
|
|
|
|
"/search",
|
|
|
|
response=PaginatedResponseSchema[GroupSchema],
|
|
|
|
permissions=[CanAccessLookup],
|
|
|
|
)
|
|
|
|
@paginate(PageNumberPaginationExtra, page_size=50)
|
|
|
|
def search_group(self, search: Annotated[str, annotated_types.MinLen(1)]):
|
|
|
|
return Group.objects.filter(name__icontains=search).values()
|
|
|
|
|
|
|
|
|
2024-09-17 10:10:06 +00:00
|
|
|
DepthValue = Annotated[int, annotated_types.Ge(0), annotated_types.Le(10)]
|
|
|
|
DEFAULT_DEPTH = 4
|
|
|
|
|
|
|
|
|
|
|
|
@api_controller("/family")
|
|
|
|
class FamilyController(ControllerBase):
|
|
|
|
@route.get(
|
|
|
|
"/{user_id}",
|
|
|
|
permissions=[CanView],
|
|
|
|
response=UserFamilySchema,
|
|
|
|
url_name="family_graph",
|
|
|
|
)
|
|
|
|
def get_family_graph(
|
|
|
|
self,
|
|
|
|
user_id: int,
|
|
|
|
godfathers_depth: DepthValue = DEFAULT_DEPTH,
|
|
|
|
godchildren_depth: DepthValue = DEFAULT_DEPTH,
|
|
|
|
):
|
|
|
|
user: User = self.get_object_or_exception(User, pk=user_id)
|
|
|
|
|
|
|
|
relations = user.get_family(godfathers_depth, godchildren_depth)
|
|
|
|
if not relations:
|
|
|
|
# If the user has no relations, return only the user
|
|
|
|
# He is alone in its family, but the family exists nonetheless
|
|
|
|
return {"users": [user], "relationships": []}
|
|
|
|
|
|
|
|
user_ids = {r.from_user_id for r in relations} | {
|
|
|
|
r.to_user_id for r in relations
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
"users": User.objects.filter(id__in=user_ids).distinct(),
|
|
|
|
"relationships": (
|
|
|
|
[
|
|
|
|
FamilyGodfatherSchema(
|
|
|
|
godchild=r.from_user_id, godfather=r.to_user_id
|
|
|
|
)
|
|
|
|
for r in relations
|
|
|
|
]
|
|
|
|
),
|
|
|
|
}
|