140 lines
4.7 KiB
Python
140 lines
4.7 KiB
Python
|
from typing import Literal
|
||
|
|
||
|
from django.forms import ValidationError
|
||
|
from django.shortcuts import get_object_or_404
|
||
|
from ninja import Schema
|
||
|
|
||
|
from activities.models import (
|
||
|
Post,
|
||
|
PostAttachment,
|
||
|
PostInteraction,
|
||
|
PostStates,
|
||
|
TimelineEvent,
|
||
|
)
|
||
|
from api import schemas
|
||
|
from api.views.base import api_router
|
||
|
from core.models import Config
|
||
|
|
||
|
from ..decorators import identity_required
|
||
|
|
||
|
|
||
|
class PostStatusSchema(Schema):
|
||
|
status: str
|
||
|
in_reply_to_id: str | None = None
|
||
|
sensitive: bool = False
|
||
|
spoiler_text: str | None = None
|
||
|
visibility: Literal["public", "unlisted", "private", "direct"] = "public"
|
||
|
language: str | None = None
|
||
|
scheduled_at: str | None = None
|
||
|
media_ids: list[str] = []
|
||
|
|
||
|
|
||
|
@api_router.post("/v1/statuses", response=schemas.Status)
|
||
|
@identity_required
|
||
|
def post_status(request, details: PostStatusSchema):
|
||
|
# Check text length
|
||
|
if len(details.status) > Config.system.post_length:
|
||
|
raise ValidationError("Status is too long")
|
||
|
if len(details.status) == 0 and not details.media_ids:
|
||
|
raise ValidationError("Status is empty")
|
||
|
# Grab attachments
|
||
|
attachments = [get_object_or_404(PostAttachment, pk=id) for id in details.media_ids]
|
||
|
# Create the Post
|
||
|
visibility_map = {
|
||
|
"public": Post.Visibilities.public,
|
||
|
"unlisted": Post.Visibilities.unlisted,
|
||
|
"private": Post.Visibilities.followers,
|
||
|
"direct": Post.Visibilities.mentioned,
|
||
|
}
|
||
|
reply_post = None
|
||
|
if details.in_reply_to_id:
|
||
|
try:
|
||
|
reply_post = Post.objects.get(pk=details.in_reply_to_id)
|
||
|
except Post.DoesNotExist:
|
||
|
pass
|
||
|
post = Post.create_local(
|
||
|
author=request.identity,
|
||
|
content=details.status,
|
||
|
summary=details.spoiler_text,
|
||
|
sensitive=details.sensitive,
|
||
|
visibility=visibility_map[details.visibility],
|
||
|
reply_to=reply_post,
|
||
|
attachments=attachments,
|
||
|
)
|
||
|
# Add their own timeline event for immediate visibility
|
||
|
TimelineEvent.add_post(request.identity, post)
|
||
|
return post.to_mastodon_json()
|
||
|
|
||
|
|
||
|
@api_router.get("/v1/statuses/{id}", response=schemas.Status)
|
||
|
@identity_required
|
||
|
def status(request, id: str):
|
||
|
post = get_object_or_404(Post, pk=id)
|
||
|
interactions = PostInteraction.get_post_interactions([post], request.identity)
|
||
|
return post.to_mastodon_json(interactions=interactions)
|
||
|
|
||
|
|
||
|
@api_router.delete("/v1/statuses/{id}", response=schemas.Status)
|
||
|
@identity_required
|
||
|
def delete_status(request, id: str):
|
||
|
post = get_object_or_404(Post, pk=id)
|
||
|
post.transition_perform(PostStates.deleted)
|
||
|
TimelineEvent.objects.filter(subject_post=post, identity=request.identity).delete()
|
||
|
return post.to_mastodon_json()
|
||
|
|
||
|
|
||
|
@api_router.get("/v1/statuses/{id}/context", response=schemas.Context)
|
||
|
@identity_required
|
||
|
def status_context(request, id: str):
|
||
|
post = get_object_or_404(Post, pk=id)
|
||
|
parent = post.in_reply_to_post()
|
||
|
ancestors = []
|
||
|
if parent:
|
||
|
ancestors.append(parent)
|
||
|
descendants = list(Post.objects.filter(in_reply_to=post.object_uri)[:40])
|
||
|
interactions = PostInteraction.get_post_interactions(
|
||
|
[post] + ancestors + descendants, request.identity
|
||
|
)
|
||
|
return {
|
||
|
"ancestors": [p.to_mastodon_json(interactions=interactions) for p in ancestors],
|
||
|
"descendants": [
|
||
|
p.to_mastodon_json(interactions=interactions) for p in descendants
|
||
|
],
|
||
|
}
|
||
|
|
||
|
|
||
|
@api_router.post("/v1/statuses/{id}/favourite", response=schemas.Status)
|
||
|
@identity_required
|
||
|
def favourite_status(request, id: str):
|
||
|
post = get_object_or_404(Post, pk=id)
|
||
|
post.like_as(request.identity)
|
||
|
interactions = PostInteraction.get_post_interactions([post], request.identity)
|
||
|
return post.to_mastodon_json(interactions=interactions)
|
||
|
|
||
|
|
||
|
@api_router.post("/v1/statuses/{id}/unfavourite", response=schemas.Status)
|
||
|
@identity_required
|
||
|
def unfavourite_status(request, id: str):
|
||
|
post = get_object_or_404(Post, pk=id)
|
||
|
post.unlike_as(request.identity)
|
||
|
interactions = PostInteraction.get_post_interactions([post], request.identity)
|
||
|
return post.to_mastodon_json(interactions=interactions)
|
||
|
|
||
|
|
||
|
@api_router.post("/v1/statuses/{id}/reblog", response=schemas.Status)
|
||
|
@identity_required
|
||
|
def reblog_status(request, id: str):
|
||
|
post = get_object_or_404(Post, pk=id)
|
||
|
post.boost_as(request.identity)
|
||
|
interactions = PostInteraction.get_post_interactions([post], request.identity)
|
||
|
return post.to_mastodon_json(interactions=interactions)
|
||
|
|
||
|
|
||
|
@api_router.post("/v1/statuses/{id}/unreblog", response=schemas.Status)
|
||
|
@identity_required
|
||
|
def unreblog_status(request, id: str):
|
||
|
post = get_object_or_404(Post, pk=id)
|
||
|
post.unboost_as(request.identity)
|
||
|
interactions = PostInteraction.get_post_interactions([post], request.identity)
|
||
|
return post.to_mastodon_json(interactions=interactions)
|