2022-12-29 18:27:43 -08:00
|
|
|
from django.db.models import Q
|
|
|
|
from django.http import HttpRequest, HttpResponse, JsonResponse
|
2022-12-11 10:22:06 -08:00
|
|
|
from django.shortcuts import get_object_or_404
|
2022-12-21 13:54:49 -08:00
|
|
|
from ninja import Field
|
2022-12-11 10:22:06 -08:00
|
|
|
|
2023-01-03 00:01:20 -08:00
|
|
|
from activities.models import Post
|
2022-12-21 13:54:49 -08:00
|
|
|
from activities.services import SearchService
|
2022-12-11 10:22:06 -08:00
|
|
|
from api import schemas
|
2022-12-11 11:37:28 -08:00
|
|
|
from api.decorators import identity_required
|
2022-12-11 23:38:02 -08:00
|
|
|
from api.pagination import MastodonPaginator
|
2022-12-11 10:22:06 -08:00
|
|
|
from api.views.base import api_router
|
|
|
|
from users.models import Identity
|
2022-12-19 12:54:09 -08:00
|
|
|
from users.services import IdentityService
|
2022-12-11 10:22:06 -08:00
|
|
|
|
2022-12-10 23:25:48 -08:00
|
|
|
|
2022-12-11 10:22:06 -08:00
|
|
|
@api_router.get("/v1/accounts/verify_credentials", response=schemas.Account)
|
2022-12-10 23:25:48 -08:00
|
|
|
@identity_required
|
|
|
|
def verify_credentials(request):
|
|
|
|
return request.identity.to_mastodon_json()
|
2022-12-11 10:22:06 -08:00
|
|
|
|
|
|
|
|
|
|
|
@api_router.get("/v1/accounts/relationships", response=list[schemas.Relationship])
|
|
|
|
@identity_required
|
|
|
|
def account_relationships(request):
|
|
|
|
ids = request.GET.getlist("id[]")
|
|
|
|
result = []
|
|
|
|
for id in ids:
|
|
|
|
identity = get_object_or_404(Identity, pk=id)
|
|
|
|
result.append(
|
2022-12-19 12:54:09 -08:00
|
|
|
IdentityService(identity).mastodon_json_relationship(request.identity)
|
2022-12-11 10:22:06 -08:00
|
|
|
)
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
2022-12-21 08:22:17 -08:00
|
|
|
@api_router.get(
|
|
|
|
"/v1/accounts/familiar_followers", response=list[schemas.FamiliarFollowers]
|
|
|
|
)
|
|
|
|
@identity_required
|
|
|
|
def familiar_followers(request):
|
|
|
|
"""
|
|
|
|
Returns people you follow that also follow given account IDs
|
|
|
|
"""
|
|
|
|
ids = request.GET.getlist("id[]")
|
|
|
|
result = []
|
|
|
|
for id in ids:
|
|
|
|
target_identity = get_object_or_404(Identity, pk=id)
|
|
|
|
result.append(
|
|
|
|
{
|
|
|
|
"id": id,
|
|
|
|
"accounts": [
|
|
|
|
identity.to_mastodon_json()
|
|
|
|
for identity in Identity.objects.filter(
|
|
|
|
inbound_follows__source=request.identity,
|
|
|
|
outbound_follows__target=target_identity,
|
|
|
|
)[:20]
|
|
|
|
],
|
|
|
|
}
|
|
|
|
)
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
2022-12-21 13:54:49 -08:00
|
|
|
@api_router.get("/v1/accounts/search", response=list[schemas.Account])
|
|
|
|
@identity_required
|
|
|
|
def search(
|
|
|
|
request,
|
|
|
|
q: str,
|
|
|
|
fetch_identities: bool = Field(False, alias="resolve"),
|
|
|
|
following: bool = False,
|
|
|
|
limit: int = 20,
|
|
|
|
offset: int = 0,
|
|
|
|
):
|
|
|
|
"""
|
|
|
|
Handles searching for accounts by username or handle
|
|
|
|
"""
|
|
|
|
if limit > 40:
|
|
|
|
limit = 40
|
|
|
|
if offset:
|
|
|
|
return []
|
|
|
|
searcher = SearchService(q, request.identity)
|
|
|
|
search_result = searcher.search_identities_handle()
|
|
|
|
return [i.to_mastodon_json() for i in search_result]
|
|
|
|
|
|
|
|
|
2022-12-29 18:27:43 -08:00
|
|
|
@api_router.get("/v1/accounts/lookup", response=schemas.Account)
|
|
|
|
def lookup(request: HttpRequest, acct: str):
|
|
|
|
"""
|
|
|
|
Quickly lookup a username to see if it is available, skipping WebFinger
|
|
|
|
resolution.
|
|
|
|
"""
|
|
|
|
acct = acct.lstrip("@")
|
|
|
|
host = request.get_host()
|
|
|
|
|
|
|
|
identity = Identity.objects.filter(
|
|
|
|
Q(domain__service_domain__iexact=host) | Q(domain__domain__iexact=host),
|
|
|
|
local=True,
|
|
|
|
username__iexact=acct,
|
|
|
|
).first()
|
|
|
|
|
|
|
|
if not identity:
|
|
|
|
return JsonResponse({"error": "Record not found"}, status=404)
|
|
|
|
|
|
|
|
return identity.to_mastodon_json()
|
|
|
|
|
|
|
|
|
2022-12-11 10:22:06 -08:00
|
|
|
@api_router.get("/v1/accounts/{id}", response=schemas.Account)
|
|
|
|
@identity_required
|
|
|
|
def account(request, id: str):
|
2022-12-16 18:42:48 -08:00
|
|
|
identity = get_object_or_404(
|
|
|
|
Identity.objects.exclude(restriction=Identity.Restriction.blocked), pk=id
|
|
|
|
)
|
2022-12-11 10:22:06 -08:00
|
|
|
return identity.to_mastodon_json()
|
|
|
|
|
|
|
|
|
|
|
|
@api_router.get("/v1/accounts/{id}/statuses", response=list[schemas.Status])
|
|
|
|
@identity_required
|
|
|
|
def account_statuses(
|
2022-12-29 09:31:32 -08:00
|
|
|
request: HttpRequest,
|
|
|
|
response: HttpResponse,
|
2022-12-11 10:22:06 -08:00
|
|
|
id: str,
|
|
|
|
exclude_reblogs: bool = False,
|
|
|
|
exclude_replies: bool = False,
|
|
|
|
only_media: bool = False,
|
|
|
|
pinned: bool = False,
|
|
|
|
tagged: str | None = None,
|
|
|
|
max_id: str | None = None,
|
|
|
|
since_id: str | None = None,
|
|
|
|
min_id: str | None = None,
|
|
|
|
limit: int = 20,
|
|
|
|
):
|
2022-12-16 18:42:48 -08:00
|
|
|
identity = get_object_or_404(
|
|
|
|
Identity.objects.exclude(restriction=Identity.Restriction.blocked), pk=id
|
|
|
|
)
|
2022-12-11 23:38:02 -08:00
|
|
|
queryset = (
|
2022-12-11 11:37:28 -08:00
|
|
|
identity.posts.not_hidden()
|
|
|
|
.unlisted(include_replies=not exclude_replies)
|
2022-12-11 10:22:06 -08:00
|
|
|
.select_related("author")
|
|
|
|
.prefetch_related("attachments")
|
|
|
|
.order_by("-created")
|
|
|
|
)
|
|
|
|
if pinned:
|
|
|
|
return []
|
|
|
|
if only_media:
|
2022-12-11 23:38:02 -08:00
|
|
|
queryset = queryset.filter(attachments__pk__isnull=False)
|
2022-12-11 10:22:06 -08:00
|
|
|
if tagged:
|
2022-12-11 23:38:02 -08:00
|
|
|
queryset = queryset.tagged_with(tagged)
|
2023-01-03 00:01:20 -08:00
|
|
|
# Get user posts with pagination
|
2022-12-29 13:00:37 -08:00
|
|
|
paginator = MastodonPaginator(Post, sort_attribute="published")
|
2022-12-29 09:31:32 -08:00
|
|
|
pager = paginator.paginate(
|
2022-12-11 23:38:02 -08:00
|
|
|
queryset,
|
|
|
|
min_id=min_id,
|
|
|
|
max_id=max_id,
|
|
|
|
since_id=since_id,
|
|
|
|
limit=limit,
|
|
|
|
)
|
2023-01-03 00:01:20 -08:00
|
|
|
# Convert those to the JSON form
|
|
|
|
pager.jsonify_posts(identity=request.identity)
|
|
|
|
# Add a link header if we need to
|
2022-12-29 09:31:32 -08:00
|
|
|
if pager.results:
|
2022-12-29 10:46:25 -08:00
|
|
|
response.headers["Link"] = pager.link_header(
|
|
|
|
request,
|
|
|
|
[
|
|
|
|
"limit",
|
|
|
|
"id",
|
|
|
|
"exclude_reblogs",
|
|
|
|
"exclude_replies",
|
|
|
|
"only_media",
|
|
|
|
"pinned",
|
|
|
|
"tagged",
|
|
|
|
],
|
2022-12-29 09:31:32 -08:00
|
|
|
)
|
2023-01-03 00:01:20 -08:00
|
|
|
return pager.json_results
|
2022-12-19 12:54:09 -08:00
|
|
|
|
|
|
|
|
|
|
|
@api_router.post("/v1/accounts/{id}/follow", response=schemas.Relationship)
|
|
|
|
@identity_required
|
2022-12-30 14:03:11 -08:00
|
|
|
def account_follow(request, id: str, reblogs: bool = True):
|
2022-12-19 12:54:09 -08:00
|
|
|
identity = get_object_or_404(
|
|
|
|
Identity.objects.exclude(restriction=Identity.Restriction.blocked), pk=id
|
|
|
|
)
|
|
|
|
service = IdentityService(identity)
|
2022-12-30 14:03:11 -08:00
|
|
|
service.follow_from(request.identity, boosts=reblogs)
|
2022-12-19 12:54:09 -08:00
|
|
|
return service.mastodon_json_relationship(request.identity)
|
|
|
|
|
|
|
|
|
|
|
|
@api_router.post("/v1/accounts/{id}/unfollow", response=schemas.Relationship)
|
|
|
|
@identity_required
|
|
|
|
def account_unfollow(request, id: str):
|
|
|
|
identity = get_object_or_404(
|
|
|
|
Identity.objects.exclude(restriction=Identity.Restriction.blocked), pk=id
|
|
|
|
)
|
|
|
|
service = IdentityService(identity)
|
|
|
|
service.unfollow_from(request.identity)
|
|
|
|
return service.mastodon_json_relationship(request.identity)
|
2022-12-29 18:27:43 -08:00
|
|
|
|
|
|
|
|
|
|
|
@api_router.get("/v1/accounts/{id}/following", response=list[schemas.Account])
|
|
|
|
def account_following(
|
|
|
|
request: HttpRequest,
|
|
|
|
response: HttpResponse,
|
|
|
|
id: str,
|
|
|
|
max_id: str | None = None,
|
|
|
|
since_id: str | None = None,
|
|
|
|
min_id: str | None = None,
|
|
|
|
limit: int = 40,
|
|
|
|
):
|
|
|
|
identity = get_object_or_404(
|
|
|
|
Identity.objects.exclude(restriction=Identity.Restriction.blocked), pk=id
|
|
|
|
)
|
|
|
|
|
|
|
|
if not identity.config_identity.visible_follows and request.identity != identity:
|
|
|
|
return []
|
|
|
|
|
|
|
|
service = IdentityService(identity)
|
|
|
|
|
|
|
|
paginator = MastodonPaginator(Identity, max_limit=80, sort_attribute="username")
|
|
|
|
pager = paginator.paginate(
|
|
|
|
service.following(),
|
|
|
|
min_id=min_id,
|
|
|
|
max_id=max_id,
|
|
|
|
since_id=since_id,
|
|
|
|
limit=limit,
|
|
|
|
)
|
2023-01-03 00:01:20 -08:00
|
|
|
pager.jsonify_identities()
|
2022-12-29 18:27:43 -08:00
|
|
|
|
|
|
|
if pager.results:
|
|
|
|
response.headers["Link"] = pager.link_header(
|
|
|
|
request,
|
|
|
|
["limit"],
|
|
|
|
)
|
|
|
|
|
2023-01-03 00:01:20 -08:00
|
|
|
return pager.json_results
|