2024-02-05 19:39:08 +13:00
from time import sleep
from flask import current_app , json
from app import celery , db
2024-05-04 21:16:55 +01:00
from app . activitypub . signature import post_request , default_context
from app . activitypub . util import actor_json_to_model
2024-02-05 19:39:08 +13:00
from app . community . util import send_to_remote_instance
2024-03-13 16:40:20 +13:00
from app . models import User , CommunityMember , Community , Instance , Site , utcnow , ActivityPubLog , BannedInstances
from app . utils import gibberish , ap_datetime , instance_banned , get_request
2024-02-05 19:39:08 +13:00
def purge_user_then_delete ( user_id ) :
if current_app . debug :
purge_user_then_delete_task ( user_id )
else :
purge_user_then_delete_task . delay ( user_id )
@celery.task
def purge_user_then_delete_task ( user_id ) :
user = User . query . get ( user_id )
if user :
# posts
for post in user . posts :
if not post . community . local_only :
delete_json = {
' id ' : f " https:// { current_app . config [ ' SERVER_NAME ' ] } /activities/delete/ { gibberish ( 15 ) } " ,
' type ' : ' Delete ' ,
' actor ' : user . profile_id ( ) ,
' audience ' : post . community . profile_id ( ) ,
' to ' : [ post . community . profile_id ( ) , ' https://www.w3.org/ns/activitystreams#Public ' ] ,
' published ' : ap_datetime ( utcnow ( ) ) ,
' cc ' : [
user . followers_url ( )
] ,
' object ' : post . ap_id ,
}
if not post . community . is_local ( ) : # this is a remote community, send it to the instance that hosts it
success = post_request ( post . community . ap_inbox_url , delete_json , user . private_key ,
user . ap_profile_id + ' #main-key ' )
else : # local community - send it to followers on remote instances, using Announce
announce = {
" id " : f " https:// { current_app . config [ ' SERVER_NAME ' ] } /activities/announce/ { gibberish ( 15 ) } " ,
" type " : ' Announce ' ,
" to " : [
" https://www.w3.org/ns/activitystreams#Public "
] ,
" actor " : post . community . ap_profile_id ,
" cc " : [
post . community . ap_followers_url
] ,
' @context ' : default_context ( ) ,
' object ' : delete_json
}
for instance in post . community . following_instances ( ) :
if instance . inbox and not instance_banned ( instance . domain ) :
send_to_remote_instance ( instance . id , post . community . id , announce )
# unsubscribe
communities = CommunityMember . query . filter_by ( user_id = user_id ) . all ( )
for membership in communities :
community = Community . query . get ( membership . community_id )
unsubscribe_from_community ( community , user )
# federate deletion of account
if user . is_local ( ) :
instances = Instance . query . all ( )
payload = {
" @context " : default_context ( ) ,
" actor " : user . ap_profile_id ,
" id " : f " { user . ap_profile_id } #delete " ,
" object " : user . ap_profile_id ,
" to " : [
" https://www.w3.org/ns/activitystreams#Public "
] ,
" type " : " Delete "
}
for instance in instances :
if instance . inbox and instance . id != 1 :
2024-02-25 15:31:16 +13:00
post_request ( instance . inbox , payload , user . private_key , user . ap_profile_id + ' #main-key ' )
2024-02-05 19:39:08 +13:00
sleep ( 100 ) # wait a while for any related activitypub traffic to die down.
user . deleted = True
user . delete_dependencies ( )
user . purge_content ( )
db . session . commit ( )
def unsubscribe_from_community ( community , user ) :
undo_id = f " https:// { current_app . config [ ' SERVER_NAME ' ] } /activities/undo/ " + gibberish ( 15 )
follow = {
2024-06-04 09:44:10 +12:00
" actor " : user . profile_id ( ) ,
2024-02-05 19:39:08 +13:00
" to " : [ community . ap_profile_id ] ,
" object " : community . ap_profile_id ,
" type " : " Follow " ,
" id " : f " https:// { current_app . config [ ' SERVER_NAME ' ] } /activities/follow/ { gibberish ( 15 ) } "
}
undo = {
' actor ' : user . profile_id ( ) ,
' to ' : [ community . ap_profile_id ] ,
' type ' : ' Undo ' ,
' id ' : undo_id ,
' object ' : follow
}
activity = ActivityPubLog ( direction = ' out ' , activity_id = undo_id , activity_type = ' Undo ' ,
activity_json = json . dumps ( undo ) , result = ' processing ' )
db . session . add ( activity )
db . session . commit ( )
post_request ( community . ap_inbox_url , undo , user . private_key , user . profile_id ( ) + ' #main-key ' )
activity . result = ' success '
2024-03-13 16:40:20 +13:00
db . session . commit ( )
def search_for_user ( address : str ) :
if ' @ ' in address :
name , server = address . lower ( ) . split ( ' @ ' )
else :
name = address
server = ' '
if server :
banned = BannedInstances . query . filter_by ( domain = server ) . first ( )
if banned :
reason = f " Reason: { banned . reason } " if banned . reason is not None else ' '
raise Exception ( f " { server } is blocked. { reason } " )
already_exists = User . query . filter_by ( ap_id = address ) . first ( )
else :
already_exists = User . query . filter_by ( user_name = name ) . first ( )
if already_exists :
return already_exists
# Look up the profile address of the user using WebFinger
# todo: try, except block around every get_request
webfinger_data = get_request ( f " https:// { server } /.well-known/webfinger " ,
params = { ' resource ' : f " acct: { address [ 1 : ] } " } )
if webfinger_data . status_code == 200 :
webfinger_json = webfinger_data . json ( )
for links in webfinger_json [ ' links ' ] :
if ' rel ' in links and links [ ' rel ' ] == ' self ' : # this contains the URL of the activitypub profile
type = links [ ' type ' ] if ' type ' in links else ' application/activity+json '
# retrieve the activitypub profile
user_data = get_request ( links [ ' href ' ] , headers = { ' Accept ' : type } )
# to see the structure of the json contained in community_data, do a GET to https://lemmy.world/c/technology with header Accept: application/activity+json
if user_data . status_code == 200 :
user_json = user_data . json ( )
user_data . close ( )
2024-03-20 11:34:25 +00:00
if user_json [ ' type ' ] == ' Person ' or user_json [ ' type ' ] == ' Service ' :
2024-03-13 16:40:20 +13:00
user = actor_json_to_model ( user_json , name , server )
return user
return None