2022-12-21 20:36:10 +00:00
|
|
|
from django.db import models
|
2022-12-21 20:48:39 +00:00
|
|
|
from django.template.defaultfilters import linebreaks_filter
|
2022-12-21 20:36:10 +00:00
|
|
|
|
2023-01-15 21:48:17 +00:00
|
|
|
from activities.models import FanOut
|
2023-01-15 23:15:57 +00:00
|
|
|
from core.files import resize_image
|
2022-12-21 20:48:39 +00:00
|
|
|
from core.html import strip_html
|
2023-01-15 21:48:17 +00:00
|
|
|
from users.models import (
|
|
|
|
Block,
|
|
|
|
BlockStates,
|
|
|
|
Domain,
|
|
|
|
Follow,
|
|
|
|
FollowStates,
|
|
|
|
Identity,
|
2023-01-16 18:53:40 +00:00
|
|
|
InboxMessage,
|
2023-01-15 21:48:17 +00:00
|
|
|
User,
|
|
|
|
)
|
2022-12-19 20:54:09 +00:00
|
|
|
|
|
|
|
|
|
|
|
class IdentityService:
|
|
|
|
"""
|
|
|
|
High-level helper methods for doing things to identities
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self, identity: Identity):
|
|
|
|
self.identity = identity
|
|
|
|
|
2023-01-15 21:48:17 +00:00
|
|
|
@classmethod
|
|
|
|
def create(
|
|
|
|
cls,
|
|
|
|
user: User,
|
|
|
|
username: str,
|
|
|
|
domain: Domain,
|
|
|
|
name: str,
|
|
|
|
discoverable: bool = True,
|
|
|
|
) -> Identity:
|
|
|
|
identity = Identity.objects.create(
|
|
|
|
actor_uri=f"https://{domain.uri_domain}/@{username}@{domain.domain}/",
|
|
|
|
username=username,
|
|
|
|
domain=domain,
|
|
|
|
name=name,
|
|
|
|
local=True,
|
|
|
|
discoverable=discoverable,
|
|
|
|
)
|
|
|
|
identity.users.add(user)
|
|
|
|
identity.generate_keypair()
|
|
|
|
# Send fanouts to all admin identities
|
|
|
|
for admin_identity in cls.admin_identities():
|
|
|
|
FanOut.objects.create(
|
|
|
|
type=FanOut.Types.identity_created,
|
|
|
|
identity=admin_identity,
|
|
|
|
subject_identity=identity,
|
|
|
|
)
|
|
|
|
return identity
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def admin_identities(cls) -> models.QuerySet[Identity]:
|
|
|
|
return Identity.objects.filter(users__admin=True).distinct()
|
|
|
|
|
2022-12-21 20:36:10 +00:00
|
|
|
def following(self) -> models.QuerySet[Identity]:
|
2022-12-21 20:56:52 +00:00
|
|
|
return (
|
|
|
|
Identity.objects.filter(inbound_follows__source=self.identity)
|
|
|
|
.not_deleted()
|
|
|
|
.order_by("username")
|
|
|
|
.select_related("domain")
|
|
|
|
)
|
2022-12-21 20:36:10 +00:00
|
|
|
|
|
|
|
def followers(self) -> models.QuerySet[Identity]:
|
2022-12-21 20:56:52 +00:00
|
|
|
return (
|
|
|
|
Identity.objects.filter(outbound_follows__target=self.identity)
|
|
|
|
.not_deleted()
|
|
|
|
.order_by("username")
|
|
|
|
.select_related("domain")
|
|
|
|
)
|
2022-12-21 20:36:10 +00:00
|
|
|
|
2022-12-30 22:03:11 +00:00
|
|
|
def follow_from(self, from_identity: Identity, boosts=True) -> Follow:
|
2022-12-19 20:54:09 +00:00
|
|
|
"""
|
|
|
|
Follows a user (or does nothing if already followed).
|
|
|
|
Returns the follow.
|
|
|
|
"""
|
2023-01-20 16:31:15 +00:00
|
|
|
if from_identity == self.identity:
|
|
|
|
raise ValueError("You cannot follow yourself")
|
2023-01-15 20:35:45 +00:00
|
|
|
return Follow.create_local(from_identity, self.identity, boosts=boosts)
|
2022-12-19 20:54:09 +00:00
|
|
|
|
|
|
|
def unfollow_from(self, from_identity: Identity):
|
|
|
|
"""
|
|
|
|
Unfollows a user (or does nothing if not followed).
|
|
|
|
"""
|
2023-01-20 16:31:15 +00:00
|
|
|
if from_identity == self.identity:
|
|
|
|
raise ValueError("You cannot unfollow yourself")
|
2022-12-19 20:54:09 +00:00
|
|
|
existing_follow = Follow.maybe_get(from_identity, self.identity)
|
|
|
|
if existing_follow:
|
|
|
|
existing_follow.transition_perform(FollowStates.undone)
|
2023-01-16 18:53:40 +00:00
|
|
|
InboxMessage.create_internal(
|
|
|
|
{
|
|
|
|
"type": "ClearTimeline",
|
|
|
|
"actor": from_identity.pk,
|
|
|
|
"object": self.identity.pk,
|
|
|
|
}
|
|
|
|
)
|
2022-12-19 20:54:09 +00:00
|
|
|
|
2023-01-15 20:35:45 +00:00
|
|
|
def block_from(self, from_identity: Identity) -> Block:
|
|
|
|
"""
|
|
|
|
Blocks a user.
|
|
|
|
"""
|
2023-01-20 16:31:15 +00:00
|
|
|
if from_identity == self.identity:
|
|
|
|
raise ValueError("You cannot block yourself")
|
2023-01-15 20:35:45 +00:00
|
|
|
self.unfollow_from(from_identity)
|
2023-01-16 18:53:40 +00:00
|
|
|
block = Block.create_local_block(from_identity, self.identity)
|
|
|
|
InboxMessage.create_internal(
|
|
|
|
{
|
|
|
|
"type": "ClearTimeline",
|
|
|
|
"actor": from_identity.pk,
|
|
|
|
"object": self.identity.pk,
|
|
|
|
"fullErase": True,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
return block
|
2023-01-15 20:35:45 +00:00
|
|
|
|
|
|
|
def unblock_from(self, from_identity: Identity):
|
|
|
|
"""
|
|
|
|
Unlocks a user
|
|
|
|
"""
|
2023-01-20 16:31:15 +00:00
|
|
|
if from_identity == self.identity:
|
|
|
|
raise ValueError("You cannot unblock yourself")
|
2023-01-15 20:35:45 +00:00
|
|
|
existing_block = Block.maybe_get(from_identity, self.identity, mute=False)
|
|
|
|
if existing_block and existing_block.active:
|
|
|
|
existing_block.transition_perform(BlockStates.undone)
|
|
|
|
|
|
|
|
def mute_from(
|
|
|
|
self,
|
|
|
|
from_identity: Identity,
|
|
|
|
duration: int = 0,
|
|
|
|
include_notifications: bool = False,
|
|
|
|
) -> Block:
|
|
|
|
"""
|
|
|
|
Mutes a user.
|
|
|
|
"""
|
2023-01-20 16:31:15 +00:00
|
|
|
if from_identity == self.identity:
|
|
|
|
raise ValueError("You cannot mute yourself")
|
2023-01-15 20:35:45 +00:00
|
|
|
return Block.create_local_mute(
|
|
|
|
from_identity,
|
|
|
|
self.identity,
|
|
|
|
duration=duration or None,
|
|
|
|
include_notifications=include_notifications,
|
|
|
|
)
|
|
|
|
|
|
|
|
def unmute_from(self, from_identity: Identity):
|
|
|
|
"""
|
|
|
|
Unmutes a user
|
|
|
|
"""
|
2023-01-20 16:31:15 +00:00
|
|
|
if from_identity == self.identity:
|
|
|
|
raise ValueError("You cannot unmute yourself")
|
2023-01-15 20:35:45 +00:00
|
|
|
existing_block = Block.maybe_get(from_identity, self.identity, mute=True)
|
|
|
|
if existing_block and existing_block.active:
|
|
|
|
existing_block.transition_perform(BlockStates.undone)
|
|
|
|
|
|
|
|
def relationships(self, from_identity: Identity):
|
|
|
|
"""
|
|
|
|
Returns a dict of any active relationships from the given identity.
|
|
|
|
"""
|
|
|
|
return {
|
|
|
|
"outbound_follow": Follow.maybe_get(
|
|
|
|
from_identity, self.identity, require_active=True
|
|
|
|
),
|
|
|
|
"inbound_follow": Follow.maybe_get(
|
|
|
|
self.identity, from_identity, require_active=True
|
|
|
|
),
|
|
|
|
"outbound_block": Block.maybe_get(
|
|
|
|
from_identity, self.identity, mute=False, require_active=True
|
|
|
|
),
|
|
|
|
"inbound_block": Block.maybe_get(
|
|
|
|
self.identity, from_identity, mute=False, require_active=True
|
|
|
|
),
|
|
|
|
"outbound_mute": Block.maybe_get(
|
|
|
|
from_identity, self.identity, mute=True, require_active=True
|
|
|
|
),
|
|
|
|
}
|
|
|
|
|
2022-12-19 20:54:09 +00:00
|
|
|
def mastodon_json_relationship(self, from_identity: Identity):
|
|
|
|
"""
|
|
|
|
Returns a Relationship object for the from_identity's relationship
|
|
|
|
with this identity.
|
|
|
|
"""
|
2023-01-15 20:35:45 +00:00
|
|
|
relationships = self.relationships(from_identity)
|
2022-12-19 20:54:09 +00:00
|
|
|
return {
|
|
|
|
"id": self.identity.pk,
|
2023-01-15 20:35:45 +00:00
|
|
|
"following": relationships["outbound_follow"] is not None,
|
|
|
|
"followed_by": relationships["inbound_follow"] is not None,
|
|
|
|
"showing_reblogs": (
|
|
|
|
relationships["outbound_follow"]
|
|
|
|
and relationships["outbound_follow"].boosts
|
|
|
|
or False
|
|
|
|
),
|
2022-12-19 20:54:09 +00:00
|
|
|
"notifying": False,
|
2023-01-15 20:35:45 +00:00
|
|
|
"blocking": relationships["outbound_block"] is not None,
|
|
|
|
"blocked_by": relationships["inbound_block"] is not None,
|
|
|
|
"muting": relationships["outbound_mute"] is not None,
|
2022-12-19 20:54:09 +00:00
|
|
|
"muting_notifications": False,
|
|
|
|
"requested": False,
|
|
|
|
"domain_blocking": False,
|
|
|
|
"endorsed": False,
|
2023-01-15 20:35:45 +00:00
|
|
|
"note": (
|
|
|
|
relationships["outbound_follow"]
|
|
|
|
and relationships["outbound_follow"].note
|
|
|
|
or ""
|
|
|
|
),
|
2022-12-19 20:54:09 +00:00
|
|
|
}
|
2022-12-21 20:48:39 +00:00
|
|
|
|
|
|
|
def set_summary(self, summary: str):
|
|
|
|
"""
|
|
|
|
Safely sets a summary and turns linebreaks into HTML
|
|
|
|
"""
|
2022-12-21 21:56:45 +00:00
|
|
|
if summary:
|
|
|
|
self.identity.summary = linebreaks_filter(strip_html(summary))
|
|
|
|
else:
|
|
|
|
self.identity.summary = None
|
2022-12-21 20:48:39 +00:00
|
|
|
self.identity.save()
|
2023-01-15 23:15:57 +00:00
|
|
|
|
|
|
|
def set_icon(self, file):
|
|
|
|
"""
|
|
|
|
Sets the user's avatar image
|
|
|
|
"""
|
|
|
|
self.identity.icon.save(
|
|
|
|
file.name,
|
|
|
|
resize_image(file, size=(400, 400)),
|
|
|
|
)
|
|
|
|
|
|
|
|
def set_image(self, file):
|
|
|
|
"""
|
|
|
|
Sets the user's header image
|
|
|
|
"""
|
|
|
|
self.identity.image.save(
|
|
|
|
file.name,
|
|
|
|
resize_image(file, size=(1500, 500)),
|
|
|
|
)
|