This repository has been archived on 2023-09-24. You can view files and clone it, but cannot push or open issues or pull requests.
takahe/activities/views/search.py

147 lines
4.9 KiB
Python
Raw Normal View History

2022-11-29 05:41:36 +01:00
from typing import Set
2022-12-05 05:13:33 +01:00
import httpx
from asgiref.sync import async_to_sync
2022-11-18 02:52:00 +01:00
from django import forms
from django.views.generic import FormView
2022-12-05 05:13:33 +01:00
from activities.models import Hashtag, Post
from core.ld import canonicalise
from users.models import Domain, Identity, IdentityStates
2022-12-05 05:13:33 +01:00
from users.models.system_actor import SystemActor
2022-11-18 02:52:00 +01:00
class Search(FormView):
template_name = "activities/search.html"
class form_class(forms.Form):
2022-11-29 05:41:36 +01:00
query = forms.CharField(
2022-12-05 05:13:33 +01:00
help_text="Search for:\nA user by @username@domain or their profile URL\nA hashtag by #tagname\nA post by its URL",
2022-12-04 03:47:09 +01:00
widget=forms.TextInput(attrs={"type": "search", "autofocus": "autofocus"}),
2022-11-29 05:41:36 +01:00
)
2022-12-05 05:13:33 +01:00
def search_identities_handle(self, query: str):
"""
Searches for identities by their handles
"""
# Short circuit if it's obviously not for us
if "://" in query:
return set()
# Try to fetch the user by handle
2022-11-29 05:41:36 +01:00
query = query.lstrip("@")
results: Set[Identity] = set()
2022-11-18 02:52:00 +01:00
if "@" in query:
username, domain = query.split("@", 1)
# Resolve the domain to the display domain
domain_instance = Domain.get_domain(domain)
try:
if domain_instance is None:
raise Identity.DoesNotExist()
identity = Identity.objects.get(
domain=domain_instance, username=username
)
except Identity.DoesNotExist:
if self.request.identity is not None:
# Allow authenticated users to fetch remote
identity = Identity.by_username_and_domain(
username, domain, fetch=True
)
if identity and identity.state == IdentityStates.outdated:
async_to_sync(identity.fetch_actor)()
else:
identity = None
if identity:
2022-11-29 05:41:36 +01:00
results.add(identity)
2022-11-18 02:52:00 +01:00
else:
for identity in Identity.objects.filter(username=query)[:20]:
2022-11-29 05:41:36 +01:00
results.add(identity)
2022-11-18 02:52:00 +01:00
for identity in Identity.objects.filter(username__startswith=query)[:20]:
2022-11-29 05:41:36 +01:00
results.add(identity)
return results
2022-12-05 05:13:33 +01:00
def search_url(self, query: str) -> Post | Identity | None:
"""
Searches for an identity or post by URL.
"""
# Short circuit if it's obviously not for us
if "://" not in query:
return None
# Clean up query
query = query.strip()
# Fetch the provided URL as the system actor to retrieve the AP JSON
try:
response = async_to_sync(SystemActor().signed_request)(
method="get", uri=query
)
except (httpx.RequestError, httpx.ConnectError):
return None
if response.status_code >= 400:
return None
document = canonicalise(response.json(), include_security=True)
type = document.get("type", "unknown").lower()
# Is it an identity?
if type == "person":
# Try and retrieve the profile by actor URI
identity = Identity.by_actor_uri(document["id"], create=True)
if identity and identity.state == IdentityStates.outdated:
async_to_sync(identity.fetch_actor)()
return identity
# Is it a post?
elif type == "note":
# Try and retrieve the post by URI
# (we do not trust the JSON we just got - fetch from source!)
try:
2022-12-05 05:22:20 +01:00
return Post.by_object_uri(document["id"], fetch=True)
2022-12-05 05:13:33 +01:00
except Post.DoesNotExist:
return None
# Dunno what it is
else:
return None
2022-11-29 05:41:36 +01:00
def search_hashtags(self, query: str):
2022-12-05 05:13:33 +01:00
"""
Searches for hashtags by their name
"""
2022-11-29 05:41:36 +01:00
2022-12-05 05:13:33 +01:00
# Short circuit out if it's obviously not a hashtag
if "@" in query or "://" in query:
return set()
2022-11-29 05:41:36 +01:00
2022-12-05 05:13:33 +01:00
results: Set[Hashtag] = set()
2022-11-29 05:41:36 +01:00
query = query.lstrip("#")
for hashtag in Hashtag.objects.public().hashtag_or_alias(query)[:10]:
results.add(hashtag)
for hashtag in Hashtag.objects.public().filter(hashtag__startswith=query)[:10]:
results.add(hashtag)
return results
def form_valid(self, form):
query = form.cleaned_data["query"].lower()
results = {
2022-12-05 05:13:33 +01:00
"identities": self.search_identities_handle(query),
2022-11-29 05:41:36 +01:00
"hashtags": self.search_hashtags(query),
2022-12-05 05:13:33 +01:00
"posts": set(),
2022-11-29 05:41:36 +01:00
}
2022-12-05 05:13:33 +01:00
url_result = self.search_url(query)
if isinstance(url_result, Identity):
results["identities"].add(url_result)
if isinstance(url_result, Post):
results["posts"].add(url_result)
2022-11-18 02:52:00 +01:00
# Render results
context = self.get_context_data(form=form)
context["results"] = results
return self.render_to_response(context)