2023-08-29 22:01:06 +12:00
from datetime import datetime
2023-10-10 22:25:37 +13:00
from typing import List
2023-08-29 22:01:06 +12:00
from app import db
2023-10-10 22:25:37 +13:00
from app . models import Community , File , BannedInstances , PostReply
2023-08-29 22:01:06 +12:00
from app . utils import get_request
2023-10-10 22:25:37 +13:00
from sqlalchemy import desc
2023-08-29 22:01:06 +12:00
def search_for_community ( address : str ) :
if address . startswith ( ' ! ' ) :
name , server = address [ 1 : ] . split ( ' @ ' )
banned = BannedInstances . query . filter_by ( domain = server ) . first ( )
if banned :
reason = f " Reason: { banned . reason } " if banned . reason is not None else ' '
2023-09-05 20:25:02 +12:00
raise Exception ( f " { server } is blocked. { reason } " ) # todo: create custom exception class hierarchy
2023-08-29 22:01:06 +12:00
already_exists = Community . query . filter_by ( ap_id = address [ 1 : ] ) . first ( )
if already_exists :
return already_exists
# Look up the profile address of the community 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 ' ] :
2023-09-05 20:25:02 +12:00
if ' rel ' in links and links [ ' rel ' ] == ' self ' : # this contains the URL of the activitypub profile
2023-08-29 22:01:06 +12:00
type = links [ ' type ' ] if ' type ' in links else ' application/activity+json '
# retrieve the activitypub profile
community_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 community_data . status_code == 200 :
community_json = community_data . json ( )
if community_json [ ' type ' ] == ' Group ' :
community = Community ( name = community_json [ ' preferredUsername ' ] ,
title = community_json [ ' name ' ] ,
description = community_json [ ' summary ' ] ,
nsfw = community_json [ ' sensitive ' ] ,
restricted_to_mods = community_json [ ' postingRestrictedToMods ' ] ,
created_at = community_json [ ' published ' ] ,
last_active = community_json [ ' updated ' ] ,
ap_id = f " { address [ 1 : ] } " ,
ap_public_url = community_json [ ' id ' ] ,
ap_profile_id = community_json [ ' id ' ] ,
ap_followers_url = community_json [ ' followers ' ] ,
ap_inbox_url = community_json [ ' endpoints ' ] [ ' sharedInbox ' ] ,
ap_fetched_at = datetime . utcnow ( ) ,
ap_domain = server ,
public_key = community_json [ ' publicKey ' ] [ ' publicKeyPem ' ] ,
# language=community_json['language'][0]['identifier'] # todo: language
)
if ' icon ' in community_json :
# todo: retrieve icon, save to disk, save more complete File record
icon = File ( source_url = community_json [ ' icon ' ] [ ' url ' ] )
community . icon = icon
db . session . add ( icon )
if ' image ' in community_json :
# todo: retrieve image, save to disk, save more complete File record
image = File ( source_url = community_json [ ' image ' ] [ ' url ' ] )
community . image = image
db . session . add ( image )
db . session . add ( community )
db . session . commit ( )
return community
return None
2023-09-05 20:25:02 +12:00
def community_url_exists ( url ) - > bool :
2023-09-17 21:19:51 +12:00
community = Community . query . filter_by ( ap_profile_id = url ) . first ( )
2023-09-05 20:25:02 +12:00
return community is not None
2023-09-17 21:19:51 +12:00
def actor_to_community ( actor ) - > Community :
actor = actor . strip ( )
if ' @ ' in actor :
community = Community . query . filter_by ( banned = False , ap_id = actor ) . first ( )
else :
community = Community . query . filter_by ( name = actor , banned = False , ap_id = None ) . first ( )
return community
2023-10-10 22:25:37 +13:00
# replies to a post, in a tree, sorted by a variety of methods
def post_replies ( post_id : int , sort_by : str , show_first : int = 0 ) - > List [ PostReply ] :
comments = PostReply . query . filter_by ( post_id = post_id )
if sort_by == ' hot ' :
comments = comments . order_by ( desc ( PostReply . ranking ) )
elif sort_by == ' top ' :
comments = comments . order_by ( desc ( PostReply . score ) )
elif sort_by == ' new ' :
comments = comments . order_by ( desc ( PostReply . posted_at ) )
comments_dict = { comment . id : { ' comment ' : comment , ' replies ' : [ ] } for comment in comments . all ( ) }
for comment in comments :
if comment . parent_id is not None :
parent_comment = comments_dict . get ( comment . parent_id )
if parent_comment :
parent_comment [ ' replies ' ] . append ( comments_dict [ comment . id ] )
return [ comment for comment in comments_dict . values ( ) if comment [ ' comment ' ] . parent_id is None ]
2023-10-16 21:38:36 +13:00
def get_comment_branch ( post_id : int , comment_id : int , sort_by : str ) - > List [ PostReply ] :
# Fetch the specified parent comment and its replies
parent_comment = PostReply . query . get ( comment_id )
if parent_comment is None :
return [ ]
comments = PostReply . query . filter ( PostReply . post_id == post_id )
if sort_by == ' hot ' :
comments = comments . order_by ( desc ( PostReply . ranking ) )
elif sort_by == ' top ' :
comments = comments . order_by ( desc ( PostReply . score ) )
elif sort_by == ' new ' :
comments = comments . order_by ( desc ( PostReply . posted_at ) )
comments_dict = { comment . id : { ' comment ' : comment , ' replies ' : [ ] } for comment in comments . all ( ) }
for comment in comments :
if comment . parent_id is not None :
parent_comment = comments_dict . get ( comment . parent_id )
if parent_comment :
parent_comment [ ' replies ' ] . append ( comments_dict [ comment . id ] )
return [ comment for comment in comments_dict . values ( ) if comment [ ' comment ' ] . id == comment_id ]