2023-01-15 12:45:59 -08:00
|
|
|
from django.http import HttpRequest, HttpResponse
|
2022-12-11 10:22:06 -08:00
|
|
|
from django.shortcuts import get_object_or_404
|
2023-01-15 12:35:45 -08:00
|
|
|
from ninja import Field, Schema
|
2022-12-11 10:22:06 -08:00
|
|
|
|
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
|
2023-01-15 12:45:59 -08:00
|
|
|
from users.shortcuts import by_handle_or_404
|
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.
|
|
|
|
"""
|
2023-01-15 12:45:59 -08:00
|
|
|
identity = by_handle_or_404(request, handle=acct, local=False)
|
2022-12-29 18:27:43 -08:00
|
|
|
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)
|
2023-01-05 21:51:02 -08:00
|
|
|
.select_related("author", "author__domain")
|
|
|
|
.prefetch_related(
|
|
|
|
"attachments",
|
|
|
|
"mentions__domain",
|
|
|
|
"emojis",
|
|
|
|
"author__inbound_follows",
|
|
|
|
"author__outbound_follows",
|
|
|
|
"author__posts",
|
|
|
|
)
|
2022-12-11 10:22:06 -08:00
|
|
|
.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
|
2023-01-08 22:06:09 -08:00
|
|
|
paginator = MastodonPaginator()
|
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)
|
2023-01-15 12:35:45 -08:00
|
|
|
|
|
|
|
|
|
|
|
@api_router.post("/v1/accounts/{id}/block", response=schemas.Relationship)
|
|
|
|
@identity_required
|
|
|
|
def account_block(request, id: str):
|
|
|
|
identity = get_object_or_404(Identity, pk=id)
|
|
|
|
service = IdentityService(identity)
|
|
|
|
service.block_from(request.identity)
|
|
|
|
return service.mastodon_json_relationship(request.identity)
|
|
|
|
|
|
|
|
|
|
|
|
@api_router.post("/v1/accounts/{id}/unblock", response=schemas.Relationship)
|
|
|
|
@identity_required
|
|
|
|
def account_unblock(request, id: str):
|
|
|
|
identity = get_object_or_404(Identity, pk=id)
|
|
|
|
service = IdentityService(identity)
|
|
|
|
service.unblock_from(request.identity)
|
|
|
|
return service.mastodon_json_relationship(request.identity)
|
|
|
|
|
|
|
|
|
|
|
|
class MuteDetailsSchema(Schema):
|
|
|
|
notifications: bool = True
|
|
|
|
duration: int = 0
|
|
|
|
|
|
|
|
|
|
|
|
@api_router.post("/v1/accounts/{id}/mute", response=schemas.Relationship)
|
|
|
|
@identity_required
|
|
|
|
def account_mute(request, id: str, details: MuteDetailsSchema):
|
|
|
|
identity = get_object_or_404(Identity, pk=id)
|
|
|
|
service = IdentityService(identity)
|
|
|
|
service.mute_from(
|
|
|
|
request.identity,
|
|
|
|
duration=details.duration,
|
|
|
|
include_notifications=details.notifications,
|
|
|
|
)
|
|
|
|
return service.mastodon_json_relationship(request.identity)
|
|
|
|
|
|
|
|
|
|
|
|
@api_router.post("/v1/accounts/{id}/unmute", response=schemas.Relationship)
|
|
|
|
@identity_required
|
|
|
|
def account_unmute(request, id: str):
|
|
|
|
identity = get_object_or_404(Identity, pk=id)
|
|
|
|
service = IdentityService(identity)
|
|
|
|
service.unmute_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)
|
|
|
|
|
2023-01-08 22:06:09 -08:00
|
|
|
paginator = MastodonPaginator(max_limit=80)
|
2022-12-29 18:27:43 -08:00
|
|
|
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
|