2023-12-11 11:53:35 -08:00
from datetime import datetime , timedelta , date , timezone
2023-08-05 02:26:24 -07:00
from time import time
2024-03-22 19:12:51 -07:00
from typing import List , Union
2023-09-17 02:19:51 -07:00
2024-03-22 19:12:51 -07:00
import requests
2024-05-21 03:20:08 -07:00
from bs4 import BeautifulSoup
2024-02-18 18:01:53 -08:00
from flask import current_app , escape , url_for , render_template_string
2023-11-29 09:36:08 -08:00
from flask_login import UserMixin , current_user
2024-05-21 03:20:08 -07:00
from sqlalchemy import or_ , text , desc
2023-08-05 02:26:24 -07:00
from werkzeug . security import generate_password_hash , check_password_hash
from flask_babel import _ , lazy_gettext as _l
from sqlalchemy . orm import backref
from sqlalchemy_utils . types import TSVectorType # https://sqlalchemy-searchable.readthedocs.io/en/latest/installation.html
2024-03-30 18:15:10 -07:00
from sqlalchemy . dialects . postgresql import ARRAY
from sqlalchemy . ext . mutable import MutableList
2023-10-03 02:29:13 -07:00
from flask_sqlalchemy import BaseQuery
from sqlalchemy_searchable import SearchQueryMixin
2024-03-22 19:12:51 -07:00
from app import db , login , cache , celery
2023-08-05 02:26:24 -07:00
import jwt
2023-11-29 23:57:51 -08:00
import os
2024-05-18 00:41:20 -07:00
import math
2023-08-05 02:26:24 -07:00
2023-09-08 01:04:01 -07:00
from app . constants import SUBSCRIPTION_NONMEMBER , SUBSCRIPTION_MEMBER , SUBSCRIPTION_MODERATOR , SUBSCRIPTION_OWNER , \
2024-04-29 02:43:37 -07:00
SUBSCRIPTION_BANNED , SUBSCRIPTION_PENDING , NOTIF_USER , NOTIF_COMMUNITY , NOTIF_TOPIC , NOTIF_POST , NOTIF_REPLY
2023-08-29 03:01:06 -07:00
2023-08-05 02:26:24 -07:00
2023-12-16 23:33:27 -08:00
# datetime.utcnow() is depreciated in Python 3.12 so it will need to be swapped out eventually
2023-12-11 11:53:35 -08:00
def utcnow ( ) :
2023-12-16 23:33:27 -08:00
return datetime . utcnow ( )
2023-12-11 11:53:35 -08:00
2023-10-03 02:29:13 -07:00
class FullTextSearchQuery ( BaseQuery , SearchQueryMixin ) :
pass
2024-01-02 19:29:58 -08:00
class BannedInstances ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
domain = db . Column ( db . String ( 256 ) , index = True )
reason = db . Column ( db . String ( 256 ) )
initiator = db . Column ( db . String ( 256 ) )
created_at = db . Column ( db . DateTime , default = utcnow )
class AllowedInstances ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
domain = db . Column ( db . String ( 256 ) , index = True )
created_at = db . Column ( db . DateTime , default = utcnow )
class Instance ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
domain = db . Column ( db . String ( 256 ) , index = True )
inbox = db . Column ( db . String ( 256 ) )
shared_inbox = db . Column ( db . String ( 256 ) )
outbox = db . Column ( db . String ( 256 ) )
vote_weight = db . Column ( db . Float , default = 1.0 )
software = db . Column ( db . String ( 50 ) )
version = db . Column ( db . String ( 50 ) )
created_at = db . Column ( db . DateTime , default = utcnow )
updated_at = db . Column ( db . DateTime , default = utcnow )
last_seen = db . Column ( db . DateTime , default = utcnow ) # When an Activity was received from them
last_successful_send = db . Column ( db . DateTime ) # When we successfully sent them an Activity
failures = db . Column ( db . Integer , default = 0 ) # How many times we failed to send (reset to 0 after every successful send)
most_recent_attempt = db . Column ( db . DateTime ) # When the most recent failure was
2024-01-03 19:09:22 -08:00
dormant = db . Column ( db . Boolean , default = False ) # True once this instance is considered offline and not worth sending to any more
2024-01-02 19:29:58 -08:00
start_trying_again = db . Column ( db . DateTime ) # When to start trying again. Should grow exponentially with each failure.
2024-01-03 19:09:22 -08:00
gone_forever = db . Column ( db . Boolean , default = False ) # True once this instance is considered offline forever - never start trying again
2024-01-02 19:29:58 -08:00
ip_address = db . Column ( db . String ( 50 ) )
2024-02-22 19:52:17 -08:00
trusted = db . Column ( db . Boolean , default = False )
2024-04-18 01:51:08 -07:00
posting_warning = db . Column ( db . String ( 512 ) )
2024-05-23 06:19:23 -07:00
nodeinfo_href = db . Column ( db . String ( 100 ) )
2024-01-02 19:29:58 -08:00
posts = db . relationship ( ' Post ' , backref = ' instance ' , lazy = ' dynamic ' )
post_replies = db . relationship ( ' PostReply ' , backref = ' instance ' , lazy = ' dynamic ' )
communities = db . relationship ( ' Community ' , backref = ' instance ' , lazy = ' dynamic ' )
def online ( self ) :
return not self . dormant and not self . gone_forever
2024-02-13 15:31:44 -08:00
def user_is_admin ( self , user_id ) :
role = InstanceRole . query . filter_by ( instance_id = self . id , user_id = user_id ) . first ( )
return role and role . role == ' admin '
class InstanceRole ( db . Model ) :
instance_id = db . Column ( db . Integer , db . ForeignKey ( ' instance.id ' ) , primary_key = True )
user_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , primary_key = True )
role = db . Column ( db . String ( 50 ) , default = ' admin ' )
user = db . relationship ( ' User ' , lazy = ' joined ' )
2024-01-02 19:29:58 -08:00
class InstanceBlock ( db . Model ) :
user_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , primary_key = True )
instance_id = db . Column ( db . Integer , db . ForeignKey ( ' instance.id ' ) , primary_key = True )
created_at = db . Column ( db . DateTime , default = utcnow )
2024-02-18 18:01:53 -08:00
class Conversation ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
user_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , index = True )
reported = db . Column ( db . Boolean , default = False )
read = db . Column ( db . Boolean , default = False )
created_at = db . Column ( db . DateTime , default = utcnow )
updated_at = db . Column ( db . DateTime , default = utcnow )
initiator = db . relationship ( ' User ' , backref = db . backref ( ' conversations_initiated ' , lazy = ' dynamic ' ) ,
foreign_keys = [ user_id ] )
messages = db . relationship ( ' ChatMessage ' , backref = db . backref ( ' conversation ' ) , cascade = ' all,delete ' ,
lazy = ' dynamic ' )
def member_names ( self , user_id ) :
retval = [ ]
for member in self . members :
if member . id != user_id :
retval . append ( member . display_name ( ) )
return ' , ' . join ( retval )
def is_member ( self , user ) :
for member in self . members :
if member . id == user . id :
return True
return False
def instances ( self ) :
retval = [ ]
for member in self . members :
if member . instance . id != 1 and member . instance not in retval :
retval . append ( member . instance )
return retval
2024-02-18 18:56:56 -08:00
@staticmethod
def find_existing_conversation ( recipient , sender ) :
sql = """ SELECT
c . id AS conversation_id ,
c . created_at AS conversation_created_at ,
c . updated_at AS conversation_updated_at ,
cm1 . user_id AS user1_id ,
cm2 . user_id AS user2_id
FROM
public . conversation AS c
JOIN
public . conversation_member AS cm1 ON c . id = cm1 . conversation_id
JOIN
public . conversation_member AS cm2 ON c . id = cm2 . conversation_id
WHERE
cm1 . user_id = : user_id_1 AND
cm2 . user_id = : user_id_2 AND
cm1 . user_id < > cm2 . user_id ; """
ec = db . session . execute ( text ( sql ) , { ' user_id_1 ' : recipient . id , ' user_id_2 ' : sender . id } ) . fetchone ( )
return Conversation . query . get ( ec [ 0 ] ) if ec else None
2024-02-18 18:01:53 -08:00
conversation_member = db . Table ( ' conversation_member ' ,
db . Column ( ' user_id ' , db . Integer , db . ForeignKey ( ' user.id ' ) ) ,
db . Column ( ' conversation_id ' , db . Integer , db . ForeignKey ( ' conversation.id ' ) ) ,
db . PrimaryKeyConstraint ( ' user_id ' , ' conversation_id ' )
)
class ChatMessage ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
sender_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , index = True )
recipient_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , index = True )
conversation_id = db . Column ( db . Integer , db . ForeignKey ( ' conversation.id ' ) , index = True )
body = db . Column ( db . Text )
body_html = db . Column ( db . Text )
reported = db . Column ( db . Boolean , default = False )
read = db . Column ( db . Boolean , default = False )
encrypted = db . Column ( db . String ( 15 ) )
created_at = db . Column ( db . DateTime , default = utcnow )
sender = db . relationship ( ' User ' , foreign_keys = [ sender_id ] )
2024-04-16 02:23:19 -07:00
class Tag ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
2024-05-10 18:45:04 -07:00
name = db . Column ( db . String ( 256 ) , index = True ) # lowercase version of tag, e.g. solarstorm
display_as = db . Column ( db . String ( 256 ) ) # Version of tag with uppercase letters, e.g. SolarStorm
2024-05-11 18:02:45 -07:00
post_count = db . Column ( db . Integer , default = 0 )
banned = db . Column ( db . Boolean , default = False , index = True )
2024-04-16 02:23:19 -07:00
class Language ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
code = db . Column ( db . String ( 5 ) , index = True )
name = db . Column ( db . String ( 50 ) )
community_language = db . Table ( ' community_language ' , db . Column ( ' community_id ' , db . Integer , db . ForeignKey ( ' community.id ' ) ) ,
db . Column ( ' language_id ' , db . Integer , db . ForeignKey ( ' language.id ' ) ) ,
db . PrimaryKeyConstraint ( ' community_id ' , ' language_id ' )
)
post_tag = db . Table ( ' post_tag ' , db . Column ( ' post_id ' , db . Integer , db . ForeignKey ( ' post.id ' ) ) ,
db . Column ( ' tag_id ' , db . Integer , db . ForeignKey ( ' tag.id ' ) ) ,
db . PrimaryKeyConstraint ( ' post_id ' , ' tag_id ' )
)
2023-08-05 02:26:24 -07:00
class File ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
file_path = db . Column ( db . String ( 255 ) )
file_name = db . Column ( db . String ( 255 ) )
width = db . Column ( db . Integer )
height = db . Column ( db . Integer )
2024-05-25 03:44:43 -07:00
alt_text = db . Column ( db . String ( 1500 ) )
2024-02-12 09:50:25 -08:00
source_url = db . Column ( db . String ( 1024 ) )
2023-11-27 01:05:35 -08:00
thumbnail_path = db . Column ( db . String ( 255 ) )
thumbnail_width = db . Column ( db . Integer )
thumbnail_height = db . Column ( db . Integer )
2024-05-05 20:42:12 -07:00
def view_url ( self , resize = False ) :
2023-11-27 01:05:35 -08:00
if self . source_url :
2024-05-05 20:42:12 -07:00
if resize and ' /pictrs/ ' in self . source_url and ' ? ' not in self . source_url :
return f ' { self . source_url } ?thumbnail=1024 '
else :
return self . source_url
2023-11-27 01:05:35 -08:00
elif self . file_path :
file_path = self . file_path [ 4 : ] if self . file_path . startswith ( ' app/ ' ) else self . file_path
return f " https:// { current_app . config [ ' SERVER_NAME ' ] } / { file_path } "
else :
return ' '
2024-01-25 00:18:44 -08:00
def medium_url ( self ) :
if self . file_path is None :
return self . thumbnail_url ( )
file_path = self . file_path [ 4 : ] if self . file_path . startswith ( ' app/ ' ) else self . file_path
return f " https:// { current_app . config [ ' SERVER_NAME ' ] } / { file_path } "
2023-11-27 01:05:35 -08:00
def thumbnail_url ( self ) :
2023-12-21 01:14:43 -08:00
if self . thumbnail_path is None :
if self . source_url :
return self . source_url
else :
return ' '
2023-11-27 01:05:35 -08:00
thumbnail_path = self . thumbnail_path [ 4 : ] if self . thumbnail_path . startswith ( ' app/ ' ) else self . thumbnail_path
return f " https:// { current_app . config [ ' SERVER_NAME ' ] } / { thumbnail_path } "
2023-08-05 02:26:24 -07:00
2023-11-29 23:57:51 -08:00
def delete_from_disk ( self ) :
2024-03-22 19:12:51 -07:00
purge_from_cache = [ ]
2023-11-29 23:57:51 -08:00
if self . file_path and os . path . isfile ( self . file_path ) :
2024-04-06 14:39:50 -07:00
try :
os . unlink ( self . file_path )
except FileNotFoundError as e :
. . .
2024-03-22 19:12:51 -07:00
purge_from_cache . append ( self . file_path . replace ( ' app/ ' , f " https:// { current_app . config [ ' SERVER_NAME ' ] } / " ) )
2023-11-29 23:57:51 -08:00
if self . thumbnail_path and os . path . isfile ( self . thumbnail_path ) :
2024-04-06 14:39:50 -07:00
try :
os . unlink ( self . thumbnail_path )
except FileNotFoundError as e :
. . .
2024-03-22 19:12:51 -07:00
purge_from_cache . append ( self . thumbnail_path . replace ( ' app/ ' , f " https:// { current_app . config [ ' SERVER_NAME ' ] } / " ) )
2024-04-03 00:13:05 -07:00
if self . source_url and self . source_url . startswith ( ' http ' ) and current_app . config [ ' SERVER_NAME ' ] in self . source_url :
# self.source_url is always a url rather than a file path, which makes deleting the file a bit fiddly
2024-04-06 14:39:50 -07:00
try :
os . unlink ( self . source_url . replace ( f " https:// { current_app . config [ ' SERVER_NAME ' ] } / " , ' app/ ' ) )
except FileNotFoundError as e :
. . .
2024-04-03 00:13:05 -07:00
purge_from_cache . append ( self . source_url ) # otoh it makes purging the cdn cache super easy.
2024-03-22 19:12:51 -07:00
if purge_from_cache :
flush_cdn_cache ( purge_from_cache )
2023-11-29 23:57:51 -08:00
2024-02-09 14:42:18 -08:00
def filesize ( self ) :
size = 0
if self . file_path and os . path . exists ( self . file_path ) :
size + = os . path . getsize ( self . file_path )
if self . thumbnail_path and os . path . exists ( self . thumbnail_path ) :
size + = os . path . getsize ( self . thumbnail_path )
return size
2023-08-05 02:26:24 -07:00
2024-03-22 19:12:51 -07:00
def flush_cdn_cache ( url : Union [ str , List [ str ] ] ) :
zone_id = current_app . config [ ' CLOUDFLARE_ZONE_ID ' ]
token = current_app . config [ ' CLOUDFLARE_API_TOKEN ' ]
if zone_id and token :
if current_app . debug :
flush_cdn_cache_task ( url )
else :
flush_cdn_cache_task . delay ( url )
@celery.task
def flush_cdn_cache_task ( to_purge : Union [ str , List [ str ] ] ) :
zone_id = current_app . config [ ' CLOUDFLARE_ZONE_ID ' ]
token = current_app . config [ ' CLOUDFLARE_API_TOKEN ' ]
headers = {
' Authorization ' : f " Bearer { token } " ,
' Content-Type ' : ' application/json '
}
# url can be a string or a list of strings
body = ' '
if isinstance ( to_purge , str ) and to_purge == ' all ' :
body = {
' purge_everything ' : True
}
else :
if isinstance ( to_purge , str ) :
body = {
' files ' : [ to_purge ]
}
elif isinstance ( to_purge , list ) :
body = {
' files ' : to_purge
}
if body :
response = requests . request (
' POST ' ,
f ' https://api.cloudflare.com/client/v4/zones/ { zone_id } /purge_cache ' ,
headers = headers ,
json = body ,
timeout = 5 ,
)
2024-01-03 19:00:19 -08:00
class Topic ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
2024-01-27 21:11:32 -08:00
machine_name = db . Column ( db . String ( 50 ) , index = True )
2024-01-03 19:00:19 -08:00
name = db . Column ( db . String ( 50 ) )
num_communities = db . Column ( db . Integer , default = 0 )
2024-02-29 23:32:29 -08:00
parent_id = db . Column ( db . Integer )
2024-01-03 19:00:19 -08:00
communities = db . relationship ( ' Community ' , lazy = ' dynamic ' , backref = ' topic ' , cascade = " all, delete-orphan " )
2024-04-08 01:01:08 -07:00
def path ( self ) :
return_value = [ self . machine_name ]
parent_id = self . parent_id
while parent_id is not None :
parent_topic = Topic . query . get ( parent_id )
if parent_topic is None :
break
return_value . append ( parent_topic . machine_name )
parent_id = parent_topic . parent_id
return_value = list ( reversed ( return_value ) )
return ' / ' . join ( return_value )
2024-04-28 21:03:00 -07:00
def notify_new_posts ( self , user_id : int ) - > bool :
existing_notification = NotificationSubscription . query . filter ( NotificationSubscription . entity_id == self . id ,
NotificationSubscription . user_id == user_id ,
NotificationSubscription . type == NOTIF_TOPIC ) . first ( )
return existing_notification is not None
2024-01-03 19:00:19 -08:00
2023-08-05 02:26:24 -07:00
class Community ( db . Model ) :
2023-10-03 02:29:13 -07:00
query_class = FullTextSearchQuery
2023-08-05 02:26:24 -07:00
id = db . Column ( db . Integer , primary_key = True )
icon_id = db . Column ( db . Integer , db . ForeignKey ( ' file.id ' ) )
2023-08-29 03:01:06 -07:00
image_id = db . Column ( db . Integer , db . ForeignKey ( ' file.id ' ) )
2023-10-20 19:49:01 -07:00
user_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) )
2023-08-05 02:26:24 -07:00
name = db . Column ( db . String ( 256 ) , index = True )
title = db . Column ( db . String ( 256 ) )
2023-12-21 01:14:43 -08:00
description = db . Column ( db . Text ) # markdown
description_html = db . Column ( db . Text ) # html equivalent of above markdown
2023-08-05 02:26:24 -07:00
rules = db . Column ( db . Text )
2023-12-21 01:14:43 -08:00
rules_html = db . Column ( db . Text )
2023-12-13 00:04:11 -08:00
content_warning = db . Column ( db . Text ) # "Are you sure you want to view this community?"
2023-08-05 02:26:24 -07:00
subscriptions_count = db . Column ( db . Integer , default = 0 )
post_count = db . Column ( db . Integer , default = 0 )
post_reply_count = db . Column ( db . Integer , default = 0 )
nsfw = db . Column ( db . Boolean , default = False )
nsfl = db . Column ( db . Boolean , default = False )
2023-12-13 00:04:11 -08:00
instance_id = db . Column ( db . Integer , db . ForeignKey ( ' instance.id ' ) , index = True )
low_quality = db . Column ( db . Boolean , default = False ) # upvotes earned in low quality communities don't improve reputation
2023-12-11 11:53:35 -08:00
created_at = db . Column ( db . DateTime , default = utcnow )
last_active = db . Column ( db . DateTime , default = utcnow )
2023-08-05 02:26:24 -07:00
public_key = db . Column ( db . Text )
private_key = db . Column ( db . Text )
2023-12-30 15:09:20 -08:00
content_retention = db . Column ( db . Integer , default = - 1 )
2024-01-03 19:00:19 -08:00
topic_id = db . Column ( db . Integer , db . ForeignKey ( ' topic.id ' ) , index = True )
2024-01-20 18:44:13 -08:00
default_layout = db . Column ( db . String ( 15 ) )
2024-04-18 01:51:08 -07:00
posting_warning = db . Column ( db . String ( 512 ) )
2023-08-05 02:26:24 -07:00
ap_id = db . Column ( db . String ( 255 ) , index = True )
2023-08-22 02:24:11 -07:00
ap_profile_id = db . Column ( db . String ( 255 ) , index = True )
2023-08-05 02:26:24 -07:00
ap_followers_url = db . Column ( db . String ( 255 ) )
ap_preferred_username = db . Column ( db . String ( 255 ) )
ap_discoverable = db . Column ( db . Boolean , default = False )
ap_public_url = db . Column ( db . String ( 255 ) )
ap_fetched_at = db . Column ( db . DateTime )
ap_deleted_at = db . Column ( db . DateTime )
ap_inbox_url = db . Column ( db . String ( 255 ) )
2024-02-22 19:52:17 -08:00
ap_outbox_url = db . Column ( db . String ( 255 ) )
2024-03-19 00:34:19 -07:00
ap_featured_url = db . Column ( db . String ( 255 ) )
2023-12-03 01:41:15 -08:00
ap_moderators_url = db . Column ( db . String ( 255 ) )
2023-08-05 02:26:24 -07:00
ap_domain = db . Column ( db . String ( 255 ) )
banned = db . Column ( db . Boolean , default = False )
2023-08-10 02:13:37 -07:00
restricted_to_mods = db . Column ( db . Boolean , default = False )
2024-01-01 22:41:00 -08:00
local_only = db . Column ( db . Boolean , default = False ) # only users on this instance can post
2023-12-13 00:04:11 -08:00
new_mods_wanted = db . Column ( db . Boolean , default = False )
2023-08-05 02:26:24 -07:00
searchable = db . Column ( db . Boolean , default = True )
2023-09-05 01:25:02 -07:00
private_mods = db . Column ( db . Boolean , default = False )
2023-08-05 02:26:24 -07:00
2023-12-30 15:09:20 -08:00
# Which feeds posts from this community show up in
show_home = db . Column ( db . Boolean , default = False ) # For anonymous users. When logged in, the home feed shows posts from subscribed communities
show_popular = db . Column ( db . Boolean , default = True )
show_all = db . Column ( db . Boolean , default = True )
2024-05-08 02:07:22 -07:00
ignore_remote_language = db . Column ( db . Boolean , default = False )
2023-08-29 03:01:06 -07:00
search_vector = db . Column ( TSVectorType ( ' name ' , ' title ' , ' description ' , ' rules ' ) )
2023-08-05 02:26:24 -07:00
2024-01-01 22:41:00 -08:00
posts = db . relationship ( ' Post ' , lazy = ' dynamic ' , cascade = " all, delete-orphan " )
replies = db . relationship ( ' PostReply ' , lazy = ' dynamic ' , cascade = " all, delete-orphan " )
2023-08-29 03:01:06 -07:00
icon = db . relationship ( ' File ' , foreign_keys = [ icon_id ] , single_parent = True , backref = ' community ' , cascade = " all, delete-orphan " )
image = db . relationship ( ' File ' , foreign_keys = [ image_id ] , single_parent = True , cascade = " all, delete-orphan " )
2024-04-16 02:23:19 -07:00
languages = db . relationship ( ' Language ' , lazy = ' dynamic ' , secondary = community_language , backref = db . backref ( ' communities ' , lazy = ' dynamic ' ) )
2023-08-29 03:01:06 -07:00
2024-05-08 02:07:22 -07:00
def language_ids ( self ) :
return [ language . id for language in self . languages . all ( ) ]
2023-11-29 10:12:17 -08:00
@cache.memoize ( timeout = 500 )
2023-12-07 20:13:38 -08:00
def icon_image ( self , size = ' default ' ) - > str :
2023-08-29 03:01:06 -07:00
if self . icon_id is not None :
2023-12-07 20:13:38 -08:00
if size == ' default ' :
if self . icon . file_path is not None :
if self . icon . file_path . startswith ( ' app/ ' ) :
return self . icon . file_path . replace ( ' app/ ' , ' / ' )
else :
return self . icon . file_path
if self . icon . source_url is not None :
if self . icon . source_url . startswith ( ' app/ ' ) :
return self . icon . source_url . replace ( ' app/ ' , ' / ' )
else :
return self . icon . source_url
elif size == ' tiny ' :
if self . icon . thumbnail_path is not None :
if self . icon . thumbnail_path . startswith ( ' app/ ' ) :
return self . icon . thumbnail_path . replace ( ' app/ ' , ' / ' )
else :
return self . icon . thumbnail_path
if self . icon . source_url is not None :
if self . icon . source_url . startswith ( ' app/ ' ) :
return self . icon . source_url . replace ( ' app/ ' , ' / ' )
else :
return self . icon . source_url
2024-01-04 19:41:50 -08:00
return ' /static/images/1px.gif '
2023-08-29 03:01:06 -07:00
2023-11-29 10:12:17 -08:00
@cache.memoize ( timeout = 500 )
2023-08-29 03:01:06 -07:00
def header_image ( self ) - > str :
if self . image_id is not None :
if self . image . file_path is not None :
2023-12-07 20:13:38 -08:00
if self . image . file_path . startswith ( ' app/ ' ) :
return self . image . file_path . replace ( ' app/ ' , ' / ' )
else :
return self . image . file_path
2023-08-29 03:01:06 -07:00
if self . image . source_url is not None :
2023-12-07 20:13:38 -08:00
if self . image . source_url . startswith ( ' app/ ' ) :
return self . image . source_url . replace ( ' app/ ' , ' / ' )
else :
return self . image . source_url
2023-08-29 03:01:06 -07:00
return ' '
def display_name ( self ) - > str :
if self . ap_id is None :
return self . title
else :
return f " { self . title } @ { self . ap_domain } "
def link ( self ) - > str :
if self . ap_id is None :
return self . name
else :
2024-03-04 00:39:56 -08:00
return self . ap_id . lower ( )
2023-08-10 02:13:37 -07:00
2024-03-12 20:40:20 -07:00
@cache.memoize ( timeout = 3 )
2023-09-17 02:19:51 -07:00
def moderators ( self ) :
return CommunityMember . query . filter ( ( CommunityMember . community_id == self . id ) &
( or_ (
CommunityMember . is_owner ,
CommunityMember . is_moderator
2024-04-05 20:29:47 -07:00
) )
) . filter ( CommunityMember . is_banned == False ) . all ( )
2023-09-17 02:19:51 -07:00
2023-12-25 15:36:02 -08:00
def is_moderator ( self , user = None ) :
if user is None :
return any ( moderator . user_id == current_user . id for moderator in self . moderators ( ) )
else :
return any ( moderator . user_id == user . id for moderator in self . moderators ( ) )
2023-11-29 09:36:08 -08:00
2023-12-25 15:36:02 -08:00
def is_owner ( self , user = None ) :
if user is None :
return any ( moderator . user_id == current_user . id and moderator . is_owner for moderator in self . moderators ( ) )
else :
return any ( moderator . user_id == user . id and moderator . is_owner for moderator in self . moderators ( ) )
2023-12-21 01:14:43 -08:00
2024-02-13 15:31:44 -08:00
def is_instance_admin ( self , user ) :
if self . instance_id :
instance_role = InstanceRole . query . filter ( InstanceRole . instance_id == self . instance_id ,
InstanceRole . user_id == user . id ,
InstanceRole . role == ' admin ' ) . first ( )
return instance_role is not None
else :
return False
2024-01-06 15:47:06 -08:00
def user_is_banned ( self , user ) :
2024-04-28 21:03:00 -07:00
# use communities_banned_from() instead of this method, where possible. Redis caches the result of communities_banned_from()
2024-04-28 21:08:35 -07:00
# we cannot use communities_banned_from() in models.py because it causes a circular import
2024-04-28 21:03:00 -07:00
community_bans = CommunityBan . query . filter ( CommunityBan . user_id == user . id ) . all ( )
return self . id in [ cb . community_id for cb in community_bans ]
2024-01-06 15:47:06 -08:00
2023-12-07 20:13:38 -08:00
def profile_id ( self ) :
2024-03-04 00:39:56 -08:00
retval = self . ap_profile_id if self . ap_profile_id else f " https:// { current_app . config [ ' SERVER_NAME ' ] } /c/ { self . name } "
return retval . lower ( )
2023-12-07 20:13:38 -08:00
2024-03-23 19:12:34 -07:00
def public_url ( self ) :
result = self . ap_public_url if self . ap_public_url else f " https:// { current_app . config [ ' SERVER_NAME ' ] } /c/ { self . name } "
return result
2023-12-09 01:14:16 -08:00
def is_local ( self ) :
return self . ap_id is None or self . profile_id ( ) . startswith ( ' https:// ' + current_app . config [ ' SERVER_NAME ' ] )
2023-12-13 00:04:11 -08:00
def local_url ( self ) :
if self . is_local ( ) :
return self . ap_profile_id
else :
return f " https:// { current_app . config [ ' SERVER_NAME ' ] } /c/ { self . ap_id } "
2024-01-06 15:47:06 -08:00
def notify_new_posts ( self , user_id : int ) - > bool :
2024-04-22 01:53:03 -07:00
existing_notification = NotificationSubscription . query . filter ( NotificationSubscription . entity_id == self . id ,
NotificationSubscription . user_id == user_id ,
NotificationSubscription . type == NOTIF_COMMUNITY ) . first ( )
return existing_notification is not None
# ids of all the users who want to be notified when there is a post in this community
def notification_subscribers ( self ) :
return list ( db . session . execute ( text ( ' SELECT user_id FROM " notification_subscription " WHERE entity_id = :community_id AND type = :type ' ) ,
{ ' community_id ' : self . id , ' type ' : NOTIF_COMMUNITY } ) . scalars ( ) )
2024-01-06 15:47:06 -08:00
2024-01-02 19:29:58 -08:00
# instances that have users which are members of this community. (excluding the current instance)
def following_instances ( self , include_dormant = False ) - > List [ Instance ] :
instances = Instance . query . join ( User , User . instance_id == Instance . id ) . join ( CommunityMember , CommunityMember . user_id == User . id )
instances = instances . filter ( CommunityMember . community_id == self . id , CommunityMember . is_banned == False )
if not include_dormant :
instances = instances . filter ( Instance . dormant == False )
instances = instances . filter ( Instance . id != 1 , Instance . gone_forever == False )
return instances . all ( )
2023-12-26 00:39:52 -08:00
2024-03-24 09:38:20 -07:00
def has_followers_from_domain ( self , domain : str ) - > bool :
instances = Instance . query . join ( User , User . instance_id == Instance . id ) . join ( CommunityMember , CommunityMember . user_id == User . id )
instances = instances . filter ( CommunityMember . community_id == self . id , CommunityMember . is_banned == False )
for instance in instances :
if instance . domain == domain :
return True
return False
2024-05-12 02:31:04 -07:00
def loop_videos ( self ) - > bool :
return ' gifs ' in self . name
2023-12-21 01:14:43 -08:00
def delete_dependencies ( self ) :
for post in self . posts :
post . delete_dependencies ( )
db . session . delete ( post )
db . session . query ( CommunityBan ) . filter ( CommunityBan . community_id == self . id ) . delete ( )
db . session . query ( CommunityBlock ) . filter ( CommunityBlock . community_id == self . id ) . delete ( )
db . session . query ( CommunityJoinRequest ) . filter ( CommunityJoinRequest . community_id == self . id ) . delete ( )
db . session . query ( CommunityMember ) . filter ( CommunityMember . community_id == self . id ) . delete ( )
db . session . query ( Report ) . filter ( Report . suspect_community_id == self . id ) . delete ( )
2024-04-17 07:10:04 -07:00
2023-08-05 02:26:24 -07:00
2023-10-18 02:23:59 -07:00
user_role = db . Table ( ' user_role ' ,
db . Column ( ' user_id ' , db . Integer , db . ForeignKey ( ' user.id ' ) ) ,
db . Column ( ' role_id ' , db . Integer , db . ForeignKey ( ' role.id ' ) ) ,
db . PrimaryKeyConstraint ( ' user_id ' , ' role_id ' )
)
2023-08-05 02:26:24 -07:00
class User ( UserMixin , db . Model ) :
2023-10-03 02:29:13 -07:00
query_class = FullTextSearchQuery
2023-08-05 02:26:24 -07:00
id = db . Column ( db . Integer , primary_key = True )
2023-11-21 23:48:27 -08:00
user_name = db . Column ( db . String ( 255 ) , index = True )
2023-12-31 17:49:15 -08:00
title = db . Column ( db . String ( 256 ) )
2023-08-05 02:26:24 -07:00
email = db . Column ( db . String ( 255 ) , index = True )
password_hash = db . Column ( db . String ( 128 ) )
verified = db . Column ( db . Boolean , default = False )
2023-08-25 20:41:11 -07:00
verification_token = db . Column ( db . String ( 16 ) , index = True )
2023-08-05 02:26:24 -07:00
banned = db . Column ( db . Boolean , default = False )
deleted = db . Column ( db . Boolean , default = False )
2023-12-21 01:14:43 -08:00
about = db . Column ( db . Text ) # markdown
about_html = db . Column ( db . Text ) # html
2023-08-05 02:26:24 -07:00
keywords = db . Column ( db . String ( 256 ) )
2023-12-28 00:00:26 -08:00
matrix_user_id = db . Column ( db . String ( 256 ) )
2023-08-05 02:26:24 -07:00
show_nsfw = db . Column ( db . Boolean , default = False )
show_nsfl = db . Column ( db . Boolean , default = False )
2023-12-11 11:53:35 -08:00
created = db . Column ( db . DateTime , default = utcnow )
last_seen = db . Column ( db . DateTime , default = utcnow , index = True )
2024-02-12 20:22:03 -08:00
avatar_id = db . Column ( db . Integer , db . ForeignKey ( ' file.id ' ) , index = True )
cover_id = db . Column ( db . Integer , db . ForeignKey ( ' file.id ' ) , index = True )
2023-08-05 02:26:24 -07:00
public_key = db . Column ( db . Text )
private_key = db . Column ( db . Text )
newsletter = db . Column ( db . Boolean , default = True )
2024-02-22 19:52:17 -08:00
email_unread = db . Column ( db . Boolean , default = True ) # True if they want to receive 'unread notifications' emails
email_unread_sent = db . Column ( db . Boolean ) # True after a 'unread notifications' email has been sent. None for remote users
receive_message_mode = db . Column ( db . String ( 20 ) , default = ' Closed ' ) # possible values: Open, TrustedOnly, Closed
2023-08-05 02:26:24 -07:00
bounces = db . Column ( db . SmallInteger , default = 0 )
timezone = db . Column ( db . String ( 20 ) )
2023-09-10 01:20:53 -07:00
reputation = db . Column ( db . Float , default = 0.0 )
2023-12-13 00:04:11 -08:00
attitude = db . Column ( db . Float , default = 1.0 ) # (upvotes cast - downvotes cast) / (upvotes + downvotes). A number between 1 and -1 is the ratio between up and down votes they cast
2023-08-05 02:26:24 -07:00
stripe_customer_id = db . Column ( db . String ( 50 ) )
stripe_subscription_id = db . Column ( db . String ( 50 ) )
searchable = db . Column ( db . Boolean , default = True )
2023-08-25 18:10:01 -07:00
indexable = db . Column ( db . Boolean , default = False )
2023-10-07 01:32:19 -07:00
bot = db . Column ( db . Boolean , default = False )
ignore_bots = db . Column ( db . Boolean , default = False )
2023-11-30 02:21:37 -08:00
unread_notifications = db . Column ( db . Integer , default = 0 )
2023-12-29 22:03:44 -08:00
ip_address = db . Column ( db . String ( 50 ) )
2023-12-21 01:14:43 -08:00
instance_id = db . Column ( db . Integer , db . ForeignKey ( ' instance.id ' ) , index = True )
2023-12-31 19:26:57 -08:00
reports = db . Column ( db . Integer , default = 0 ) # how many times this user has been reported.
2024-01-14 21:26:22 -08:00
default_sort = db . Column ( db . String ( 25 ) , default = ' hot ' )
2024-02-06 20:31:12 -08:00
theme = db . Column ( db . String ( 20 ) , default = ' ' )
2024-02-22 19:52:17 -08:00
referrer = db . Column ( db . String ( 256 ) )
2024-02-26 00:26:19 -08:00
markdown_editor = db . Column ( db . Boolean , default = False )
2024-05-08 18:59:52 -07:00
interface_language = db . Column ( db . String ( 10 ) ) # a locale that the translation system understands e.g. 'en' or 'en-us'. If empty, use browser default
language_id = db . Column ( db . Integer , db . ForeignKey ( ' language.id ' ) ) # the default choice in the language dropdown when composing posts & comments
2023-08-05 02:26:24 -07:00
2023-12-10 23:46:38 -08:00
avatar = db . relationship ( ' File ' , lazy = ' joined ' , foreign_keys = [ avatar_id ] , single_parent = True , cascade = " all, delete-orphan " )
cover = db . relationship ( ' File ' , lazy = ' joined ' , foreign_keys = [ cover_id ] , single_parent = True , cascade = " all, delete-orphan " )
2023-12-29 14:36:24 -08:00
instance = db . relationship ( ' Instance ' , lazy = ' joined ' , foreign_keys = [ instance_id ] )
2024-02-18 18:01:53 -08:00
conversations = db . relationship ( ' Conversation ' , lazy = ' dynamic ' , secondary = conversation_member , backref = db . backref ( ' members ' , lazy = ' joined ' ) )
2023-09-08 01:04:01 -07:00
ap_id = db . Column ( db . String ( 255 ) , index = True ) # e.g. username@server
ap_profile_id = db . Column ( db . String ( 255 ) , index = True ) # e.g. https://server/u/username
ap_public_url = db . Column ( db . String ( 255 ) ) # e.g. https://server/u/username
2023-08-05 02:26:24 -07:00
ap_fetched_at = db . Column ( db . DateTime )
ap_followers_url = db . Column ( db . String ( 255 ) )
ap_preferred_username = db . Column ( db . String ( 255 ) )
2024-03-21 16:26:03 -07:00
ap_manually_approves_followers = db . Column ( db . Boolean , default = False )
2023-08-05 02:26:24 -07:00
ap_deleted_at = db . Column ( db . DateTime )
ap_inbox_url = db . Column ( db . String ( 255 ) )
ap_domain = db . Column ( db . String ( 255 ) )
2024-02-29 23:32:29 -08:00
search_vector = db . Column ( TSVectorType ( ' user_name ' , ' about ' , ' keywords ' ) )
2023-08-05 02:26:24 -07:00
activity = db . relationship ( ' ActivityLog ' , backref = ' account ' , lazy = ' dynamic ' , cascade = " all, delete-orphan " )
2023-11-29 10:12:17 -08:00
posts = db . relationship ( ' Post ' , lazy = ' dynamic ' , cascade = " all, delete-orphan " )
2023-12-26 18:47:17 -08:00
post_replies = db . relationship ( ' PostReply ' , lazy = ' dynamic ' , cascade = " all, delete-orphan " )
2023-08-05 02:26:24 -07:00
2023-10-18 02:23:59 -07:00
roles = db . relationship ( ' Role ' , secondary = user_role , lazy = ' dynamic ' , cascade = " all, delete " )
2023-08-05 02:26:24 -07:00
def __repr__ ( self ) :
2024-01-17 17:56:23 -08:00
return ' <User {} _ {} > ' . format ( self . user_name , self . id )
2023-08-05 02:26:24 -07:00
def set_password ( self , password ) :
self . password_hash = generate_password_hash ( password )
def check_password ( self , password ) :
try :
result = check_password_hash ( self . password_hash , password )
return result
except Exception :
return False
2024-01-11 15:34:08 -08:00
def get_id ( self ) :
if self . is_authenticated :
return self . id
else :
2024-01-11 16:49:40 -08:00
return 0
2024-01-11 15:34:08 -08:00
2023-10-20 19:49:01 -07:00
def display_name ( self ) :
if self . deleted is False :
2023-12-31 17:49:15 -08:00
if self . title :
return self . title
else :
return self . user_name
2023-10-20 19:49:01 -07:00
else :
return ' [deleted] '
2024-05-25 23:24:13 -07:00
@cache.memoize ( timeout = 500 )
2023-12-23 19:20:18 -08:00
def avatar_thumbnail ( self ) - > str :
if self . avatar_id is not None :
if self . avatar . thumbnail_path is not None :
if self . avatar . thumbnail_path . startswith ( ' app/ ' ) :
return self . avatar . thumbnail_path . replace ( ' app/ ' , ' / ' )
else :
return self . avatar . thumbnail_path
else :
return self . avatar_image ( )
return ' '
2024-05-25 23:24:13 -07:00
@cache.memoize ( timeout = 500 )
2023-10-07 01:32:19 -07:00
def avatar_image ( self ) - > str :
if self . avatar_id is not None :
if self . avatar . file_path is not None :
2023-12-07 20:13:38 -08:00
if self . avatar . file_path . startswith ( ' app/ ' ) :
return self . avatar . file_path . replace ( ' app/ ' , ' / ' )
else :
return self . avatar . file_path
2023-10-07 01:32:19 -07:00
if self . avatar . source_url is not None :
2023-12-07 20:13:38 -08:00
if self . avatar . source_url . startswith ( ' app/ ' ) :
return self . avatar . source_url . replace ( ' app/ ' , ' / ' )
else :
return self . avatar . source_url
2023-10-07 01:32:19 -07:00
return ' '
2024-05-25 23:24:13 -07:00
@cache.memoize ( timeout = 500 )
2023-10-07 01:32:19 -07:00
def cover_image ( self ) - > str :
if self . cover_id is not None :
2024-02-09 09:41:24 -08:00
if self . cover . thumbnail_path is not None :
if self . cover . thumbnail_path . startswith ( ' app/ ' ) :
return self . cover . thumbnail_path . replace ( ' app/ ' , ' / ' )
2023-12-07 20:13:38 -08:00
else :
2024-02-09 09:41:24 -08:00
return self . cover . thumbnail_path
2023-10-07 01:32:19 -07:00
if self . cover . source_url is not None :
2023-12-07 20:13:38 -08:00
if self . cover . source_url . startswith ( ' app/ ' ) :
return self . cover . source_url . replace ( ' app/ ' , ' / ' )
else :
return self . cover . source_url
2023-10-07 01:32:19 -07:00
return ' '
2024-02-09 14:42:18 -08:00
def filesize ( self ) :
size = 0
if self . avatar_id :
size + = self . avatar . filesize ( )
if self . cover_id :
size + = self . cover . filesize ( )
return size
def num_content ( self ) :
content = 0
2024-03-20 15:07:11 -07:00
content + = db . session . execute ( text ( ' SELECT COUNT(id) as c FROM " post " WHERE user_id = :user_id ' ) , { ' user_id ' : self . id } ) . scalar ( )
content + = db . session . execute ( text ( ' SELECT COUNT(id) as c FROM " post_reply " WHERE user_id = :user_id ' ) , { ' user_id ' : self . id } ) . scalar ( )
2024-02-09 14:42:18 -08:00
return content
2023-12-09 01:14:16 -08:00
def is_local ( self ) :
return self . ap_id is None or self . ap_profile_id . startswith ( ' https:// ' + current_app . config [ ' SERVER_NAME ' ] )
2024-02-01 18:30:03 -08:00
def waiting_for_approval ( self ) :
application = UserRegistration . query . filter_by ( user_id = self . id , status = 0 ) . first ( )
return application is not None
2024-01-02 19:29:58 -08:00
@cache.memoize ( timeout = 30 )
2023-12-21 01:14:43 -08:00
def is_admin ( self ) :
for role in self . roles :
if role . name == ' Admin ' :
return True
return False
2024-03-11 00:14:12 -07:00
def trustworthy ( self ) :
if self . is_admin ( ) :
return True
if self . created_recently ( ) or self . reputation < 100 :
return False
return True
2023-10-10 02:25:37 -07:00
def link ( self ) - > str :
2023-12-09 01:14:16 -08:00
if self . is_local ( ) :
2023-10-10 02:25:37 -07:00
return self . user_name
else :
2024-03-04 00:46:23 -08:00
return self . ap_id
2023-10-10 02:25:37 -07:00
2023-12-09 01:14:16 -08:00
def followers_url ( self ) :
if self . ap_followers_url :
return self . ap_followers_url
else :
return self . profile_id ( ) + ' /followers '
2023-08-05 02:26:24 -07:00
def get_reset_password_token ( self , expires_in = 600 ) :
return jwt . encode (
{ ' reset_password ' : self . id , ' exp ' : time ( ) + expires_in } ,
current_app . config [ ' SECRET_KEY ' ] ,
2023-09-02 21:30:20 -07:00
algorithm = ' HS256 ' )
2023-08-05 02:26:24 -07:00
def another_account_using_email ( self , email ) :
another_account = User . query . filter ( User . email == email , User . id != self . id ) . first ( )
return another_account is not None
def expires_soon ( self ) :
if self . expires is None :
return False
2023-12-11 11:53:35 -08:00
return self . expires < utcnow ( ) + timedelta ( weeks = 1 )
2023-08-05 02:26:24 -07:00
def is_expired ( self ) :
if self . expires is None :
return True
2023-12-11 11:53:35 -08:00
return self . expires < utcnow ( )
2023-08-05 02:26:24 -07:00
def expired_ages_ago ( self ) :
if self . expires is None :
return True
return self . expires < datetime ( 2019 , 9 , 1 )
2023-12-26 22:51:07 -08:00
def recalculate_attitude ( self ) :
upvotes = db . session . execute ( text ( ' SELECT COUNT(id) as c FROM " post_vote " WHERE user_id = :user_id AND effect > 0 ' ) ,
{ ' user_id ' : self . id } ) . scalar ( )
downvotes = db . session . execute ( text ( ' SELECT COUNT(id) as c FROM " post_vote " WHERE user_id = :user_id AND effect < 0 ' ) ,
{ ' user_id ' : self . id } ) . scalar ( )
if upvotes is None :
upvotes = 0
if downvotes is None :
downvotes = 0
comment_upvotes = db . session . execute ( text ( ' SELECT COUNT(id) as c FROM " post_reply_vote " WHERE user_id = :user_id AND effect > 0 ' ) ,
{ ' user_id ' : self . id } ) . scalar ( )
comment_downvotes = db . session . execute ( text ( ' SELECT COUNT(id) as c FROM " post_reply_vote " WHERE user_id = :user_id AND effect < 0 ' ) ,
{ ' user_id ' : self . id } ) . scalar ( )
if comment_upvotes is None :
comment_upvotes = 0
if comment_downvotes is None :
comment_downvotes = 0
total_upvotes = upvotes + comment_upvotes
total_downvotes = downvotes + comment_downvotes
if total_downvotes == 0 : # guard against division by zero
self . attitude = 1.0
else :
self . attitude = ( total_upvotes - total_downvotes ) / ( total_upvotes + total_downvotes )
2023-12-03 01:41:15 -08:00
def subscribed ( self , community_id : int ) - > int :
if community_id is None :
2023-08-29 03:01:06 -07:00
return False
2023-12-03 01:41:15 -08:00
subscription : CommunityMember = CommunityMember . query . filter_by ( user_id = self . id , community_id = community_id ) . first ( )
2023-08-29 03:01:06 -07:00
if subscription :
2023-09-08 01:04:01 -07:00
if subscription . is_banned :
return SUBSCRIPTION_BANNED
elif subscription . is_owner :
2023-08-29 03:01:06 -07:00
return SUBSCRIPTION_OWNER
elif subscription . is_moderator :
return SUBSCRIPTION_MODERATOR
else :
return SUBSCRIPTION_MEMBER
else :
2023-12-03 01:41:15 -08:00
join_request = CommunityJoinRequest . query . filter_by ( user_id = self . id , community_id = community_id ) . first ( )
if join_request :
return SUBSCRIPTION_PENDING
else :
return SUBSCRIPTION_NONMEMBER
2023-08-29 03:01:06 -07:00
2023-09-17 02:19:51 -07:00
def communities ( self ) - > List [ Community ] :
return Community . query . filter ( Community . banned == False ) . \
2024-01-27 21:11:32 -08:00
join ( CommunityMember ) . filter ( CommunityMember . is_banned == False , CommunityMember . user_id == self . id ) . all ( )
2023-09-17 02:19:51 -07:00
2023-11-17 01:02:44 -08:00
def profile_id ( self ) :
2024-06-03 14:44:10 -07:00
result = self . ap_profile_id if self . ap_profile_id else f " https:// { current_app . config [ ' SERVER_NAME ' ] } /u/ { self . user_name . lower ( ) } "
2024-03-04 00:46:23 -08:00
return result
2023-11-17 01:02:44 -08:00
2024-03-23 19:12:34 -07:00
def public_url ( self ) :
result = self . ap_public_url if self . ap_public_url else f " https:// { current_app . config [ ' SERVER_NAME ' ] } /u/ { self . user_name } "
return result
2023-11-29 08:14:22 -08:00
def created_recently ( self ) :
2023-12-11 11:53:35 -08:00
return self . created and self . created > utcnow ( ) - timedelta ( days = 7 )
2023-11-29 08:14:22 -08:00
2023-12-31 19:26:57 -08:00
def has_blocked_instance ( self , instance_id : int ) :
2023-12-26 00:39:52 -08:00
instance_block = InstanceBlock . query . filter_by ( user_id = self . id , instance_id = instance_id ) . first ( )
return instance_block is not None
2023-12-31 19:26:57 -08:00
def has_blocked_user ( self , user_id : int ) :
existing_block = UserBlock . query . filter_by ( blocker_id = self . id , blocked_id = user_id ) . first ( )
return existing_block is not None
2023-08-05 02:26:24 -07:00
@staticmethod
def verify_reset_password_token ( token ) :
try :
id = jwt . decode ( token , current_app . config [ ' SECRET_KEY ' ] ,
algorithms = [ ' HS256 ' ] ) [ ' reset_password ' ]
except :
return
return User . query . get ( id )
2023-12-31 17:49:15 -08:00
def delete_dependencies ( self ) :
if self . cover_id :
file = File . query . get ( self . cover_id )
file . delete_from_disk ( )
self . cover_id = None
db . session . delete ( file )
if self . avatar_id :
file = File . query . get ( self . avatar_id )
file . delete_from_disk ( )
self . avatar_id = None
db . session . delete ( file )
2024-02-01 18:30:03 -08:00
if self . waiting_for_approval ( ) :
db . session . query ( UserRegistration ) . filter ( UserRegistration . user_id == self . id ) . delete ( )
2024-04-22 01:53:03 -07:00
db . session . query ( NotificationSubscription ) . filter ( NotificationSubscription . user_id == self . id ) . delete ( )
db . session . query ( Notification ) . filter ( Notification . user_id == self . id ) . delete ( )
2024-05-18 00:41:20 -07:00
db . session . query ( PollChoiceVote ) . filter ( PollChoiceVote . user_id == self . id ) . delete ( )
2023-12-31 17:49:15 -08:00
2024-06-01 21:45:21 -07:00
def purge_content ( self , soft = True ) :
2023-11-30 02:21:37 -08:00
files = File . query . join ( Post ) . filter ( Post . user_id == self . id ) . all ( )
for file in files :
file . delete_from_disk ( )
2023-12-31 17:49:15 -08:00
self . delete_dependencies ( )
2024-01-08 23:44:08 -08:00
posts = Post . query . filter_by ( user_id = self . id ) . all ( )
for post in posts :
post . delete_dependencies ( )
2024-06-01 21:45:21 -07:00
if soft :
post . deleted = True
else :
db . session . delete ( post )
2024-04-14 02:49:42 -07:00
db . session . commit ( )
2024-02-08 15:52:16 -08:00
post_replies = PostReply . query . filter_by ( user_id = self . id ) . all ( )
for reply in post_replies :
2024-04-14 02:49:42 -07:00
reply . delete_dependencies ( )
2024-06-01 21:45:21 -07:00
if soft :
reply . deleted = True
else :
db . session . delete ( reply )
2024-01-08 23:44:08 -08:00
db . session . commit ( )
2023-10-20 19:49:01 -07:00
2024-03-23 19:19:49 -07:00
def mention_tag ( self ) :
if self . ap_domain is None :
return ' @ ' + self . user_name + ' @ ' + current_app . config [ ' SERVER_NAME ' ]
else :
return ' @ ' + self . user_name + ' @ ' + self . ap_domain
2024-04-19 01:06:08 -07:00
# True if user_id wants to be notified about posts by self
def notify_new_posts ( self , user_id ) :
existing_notification = NotificationSubscription . query . filter ( NotificationSubscription . entity_id == self . id ,
NotificationSubscription . user_id == user_id ,
NotificationSubscription . type == NOTIF_USER ) . first ( )
return existing_notification is not None
# ids of all the users who want to be notified when self makes a post
def notification_subscribers ( self ) :
2024-04-19 01:13:02 -07:00
return list ( db . session . execute ( text ( ' SELECT user_id FROM " notification_subscription " WHERE entity_id = :user_id AND type = :type ' ) ,
{ ' user_id ' : self . id , ' type ' : NOTIF_USER } ) . scalars ( ) )
2024-04-19 01:06:08 -07:00
2023-10-20 19:49:01 -07:00
2023-08-05 02:26:24 -07:00
class ActivityLog ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
user_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , index = True )
activity_type = db . Column ( db . String ( 64 ) )
activity = db . Column ( db . String ( 255 ) )
2023-12-11 11:53:35 -08:00
timestamp = db . Column ( db . DateTime , index = True , default = utcnow )
2023-08-05 02:26:24 -07:00
class Post ( db . Model ) :
2023-10-03 02:29:13 -07:00
query_class = FullTextSearchQuery
2023-08-05 02:26:24 -07:00
id = db . Column ( db . Integer , primary_key = True )
user_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , index = True )
community_id = db . Column ( db . Integer , db . ForeignKey ( ' community.id ' ) , index = True )
image_id = db . Column ( db . Integer , db . ForeignKey ( ' file.id ' ) , index = True )
domain_id = db . Column ( db . Integer , db . ForeignKey ( ' domain.id ' ) , index = True )
2023-12-13 00:04:11 -08:00
instance_id = db . Column ( db . Integer , db . ForeignKey ( ' instance.id ' ) , index = True )
2023-08-05 02:26:24 -07:00
slug = db . Column ( db . String ( 255 ) )
title = db . Column ( db . String ( 255 ) )
url = db . Column ( db . String ( 2048 ) )
body = db . Column ( db . Text )
2023-08-10 02:13:37 -07:00
body_html = db . Column ( db . Text )
2023-08-05 02:26:24 -07:00
type = db . Column ( db . Integer )
2023-09-16 00:09:04 -07:00
comments_enabled = db . Column ( db . Boolean , default = True )
2024-06-01 21:45:21 -07:00
deleted = db . Column ( db . Boolean , default = False , index = True )
2023-12-14 00:22:46 -08:00
mea_culpa = db . Column ( db . Boolean , default = False )
2023-08-05 02:26:24 -07:00
has_embed = db . Column ( db . Boolean , default = False )
reply_count = db . Column ( db . Integer , default = 0 )
2023-12-14 20:35:11 -08:00
score = db . Column ( db . Integer , default = 0 , index = True ) # used for 'top' ranking
2024-02-12 20:22:03 -08:00
nsfw = db . Column ( db . Boolean , default = False , index = True )
nsfl = db . Column ( db . Boolean , default = False , index = True )
2023-08-05 02:26:24 -07:00
sticky = db . Column ( db . Boolean , default = False )
2023-11-29 23:57:51 -08:00
notify_author = db . Column ( db . Boolean , default = True )
2024-03-12 00:58:47 -07:00
indexable = db . Column ( db . Boolean , default = True )
2024-02-12 20:22:03 -08:00
from_bot = db . Column ( db . Boolean , default = False , index = True )
2023-12-11 11:53:35 -08:00
created_at = db . Column ( db . DateTime , index = True , default = utcnow ) # this is when the content arrived here
posted_at = db . Column ( db . DateTime , index = True , default = utcnow ) # this is when the original server created it
last_active = db . Column ( db . DateTime , index = True , default = utcnow )
2023-08-05 02:26:24 -07:00
ip = db . Column ( db . String ( 50 ) )
up_votes = db . Column ( db . Integer , default = 0 )
down_votes = db . Column ( db . Integer , default = 0 )
2024-02-12 20:22:03 -08:00
ranking = db . Column ( db . Integer , default = 0 , index = True ) # used for 'hot' ranking
2023-08-05 02:26:24 -07:00
edited_at = db . Column ( db . DateTime )
2023-12-16 03:12:49 -08:00
reports = db . Column ( db . Integer , default = 0 ) # how many times this post has been reported. Set to -1 to ignore reports
2024-05-08 22:54:30 -07:00
language_id = db . Column ( db . Integer , db . ForeignKey ( ' language.id ' ) , index = True )
2024-03-30 18:15:10 -07:00
cross_posts = db . Column ( MutableList . as_mutable ( ARRAY ( db . Integer ) ) )
2024-04-16 02:23:19 -07:00
tags = db . relationship ( ' Tag ' , lazy = ' dynamic ' , secondary = post_tag , backref = db . backref ( ' posts ' , lazy = ' dynamic ' ) )
2023-08-05 02:26:24 -07:00
ap_id = db . Column ( db . String ( 255 ) , index = True )
2023-08-10 02:13:37 -07:00
ap_create_id = db . Column ( db . String ( 100 ) )
ap_announce_id = db . Column ( db . String ( 100 ) )
2023-08-05 02:26:24 -07:00
search_vector = db . Column ( TSVectorType ( ' title ' , ' body ' ) )
2023-11-29 10:12:17 -08:00
image = db . relationship ( File , lazy = ' joined ' , foreign_keys = [ image_id ] , cascade = " all, delete " )
domain = db . relationship ( ' Domain ' , lazy = ' joined ' , foreign_keys = [ domain_id ] )
2023-12-03 01:41:15 -08:00
author = db . relationship ( ' User ' , lazy = ' joined ' , overlaps = ' posts ' , foreign_keys = [ user_id ] )
2024-01-01 22:41:00 -08:00
community = db . relationship ( ' Community ' , lazy = ' joined ' , overlaps = ' posts ' , foreign_keys = [ community_id ] )
2023-12-09 18:10:09 -08:00
replies = db . relationship ( ' PostReply ' , lazy = ' dynamic ' , backref = ' post ' )
2024-05-08 22:54:30 -07:00
language = db . relationship ( ' Language ' , foreign_keys = [ language_id ] )
2023-08-10 02:13:37 -07:00
2023-12-09 01:14:16 -08:00
def is_local ( self ) :
return self . ap_id is None or self . ap_id . startswith ( ' https:// ' + current_app . config [ ' SERVER_NAME ' ] )
2023-09-16 00:09:04 -07:00
@classmethod
def get_by_ap_id ( cls , ap_id ) :
return cls . query . filter_by ( ap_id = ap_id ) . first ( )
2023-11-21 02:05:07 -08:00
def delete_dependencies ( self ) :
2024-05-18 00:41:20 -07:00
db . session . query ( PollChoiceVote ) . filter ( PollChoiceVote . post_id == self . id ) . delete ( )
db . session . query ( PollChoice ) . filter ( PollChoice . post_id == self . id ) . delete ( )
db . session . query ( Poll ) . filter ( Poll . post_id == self . id ) . delete ( )
2023-12-16 03:12:49 -08:00
db . session . query ( Report ) . filter ( Report . suspect_post_id == self . id ) . delete ( )
2023-11-21 02:05:07 -08:00
db . session . execute ( text ( ' DELETE FROM post_reply_vote WHERE post_reply_id IN (SELECT id FROM post_reply WHERE post_id = :post_id) ' ) ,
{ ' post_id ' : self . id } )
db . session . execute ( text ( ' DELETE FROM post_reply WHERE post_id = :post_id ' ) , { ' post_id ' : self . id } )
db . session . execute ( text ( ' DELETE FROM post_vote WHERE post_id = :post_id ' ) , { ' post_id ' : self . id } )
2023-11-29 23:57:51 -08:00
if self . image_id :
file = File . query . get ( self . image_id )
file . delete_from_disk ( )
2023-11-21 02:05:07 -08:00
2023-11-28 23:32:07 -08:00
def youtube_embed ( self ) :
if self . url :
vpos = self . url . find ( ' v= ' )
if vpos != - 1 :
return self . url [ vpos + 2 : vpos + 13 ]
2024-05-25 18:19:57 -07:00
def peertube_embed ( self ) :
if self . url :
return self . url . replace ( ' watch ' , ' embed ' )
2023-12-07 20:13:38 -08:00
def profile_id ( self ) :
2023-12-10 23:46:38 -08:00
if self . ap_id :
return self . ap_id
else :
return f " https:// { current_app . config [ ' SERVER_NAME ' ] } /post/ { self . id } "
2023-12-07 20:13:38 -08:00
2024-01-10 23:39:22 -08:00
def blocked_by_content_filter ( self , content_filters ) :
lowercase_title = self . title . lower ( )
for name , keywords in content_filters . items ( ) if content_filters else { } :
for keyword in keywords :
if keyword in lowercase_title :
return name
return False
2024-04-29 02:43:37 -07:00
def notify_new_replies ( self , user_id : int ) - > bool :
existing_notification = NotificationSubscription . query . filter ( NotificationSubscription . entity_id == self . id ,
NotificationSubscription . user_id == user_id ,
NotificationSubscription . type == NOTIF_POST ) . first ( )
return existing_notification is not None
2024-05-08 22:54:30 -07:00
def language_code ( self ) :
if self . language_id :
return self . language . code
else :
return ' en '
def language_name ( self ) :
if self . language_id :
return self . language . name
else :
return ' English '
2024-05-11 18:02:45 -07:00
def tags_for_activitypub ( self ) :
return_value = [ ]
for tag in self . tags :
return_value . append ( { ' type ' : ' Hashtag ' ,
' href ' : f ' https:// { current_app . config [ " SERVER_NAME " ] } /tag/ { tag . name } ' ,
' name ' : f ' # { tag . name } ' } )
return return_value
2023-08-05 02:26:24 -07:00
class PostReply ( db . Model ) :
2023-10-03 02:29:13 -07:00
query_class = FullTextSearchQuery
2023-08-05 02:26:24 -07:00
id = db . Column ( db . Integer , primary_key = True )
user_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , index = True )
post_id = db . Column ( db . Integer , db . ForeignKey ( ' post.id ' ) , index = True )
community_id = db . Column ( db . Integer , db . ForeignKey ( ' community.id ' ) , index = True )
2023-10-10 02:25:37 -07:00
domain_id = db . Column ( db . Integer , db . ForeignKey ( ' domain.id ' ) , index = True )
2023-08-05 02:26:24 -07:00
image_id = db . Column ( db . Integer , db . ForeignKey ( ' file.id ' ) , index = True )
2024-02-08 18:14:39 -08:00
parent_id = db . Column ( db . Integer , index = True )
2023-08-05 02:26:24 -07:00
root_id = db . Column ( db . Integer )
2023-10-10 02:25:37 -07:00
depth = db . Column ( db . Integer , default = 0 )
2023-12-27 23:00:07 -08:00
instance_id = db . Column ( db . Integer , db . ForeignKey ( ' instance.id ' ) , index = True )
2023-08-05 02:26:24 -07:00
body = db . Column ( db . Text )
2023-08-10 02:13:37 -07:00
body_html = db . Column ( db . Text )
2023-09-16 00:09:04 -07:00
body_html_safe = db . Column ( db . Boolean , default = False )
2023-12-14 20:35:11 -08:00
score = db . Column ( db . Integer , default = 0 , index = True ) # used for 'top' sorting
2023-08-05 02:26:24 -07:00
nsfw = db . Column ( db . Boolean , default = False )
nsfl = db . Column ( db . Boolean , default = False )
2023-11-29 23:57:51 -08:00
notify_author = db . Column ( db . Boolean , default = True )
2023-12-11 11:53:35 -08:00
created_at = db . Column ( db . DateTime , index = True , default = utcnow )
posted_at = db . Column ( db . DateTime , index = True , default = utcnow )
2024-06-01 21:45:21 -07:00
deleted = db . Column ( db . Boolean , default = False , index = True )
2023-08-05 02:26:24 -07:00
ip = db . Column ( db . String ( 50 ) )
2023-10-07 01:32:19 -07:00
from_bot = db . Column ( db . Boolean , default = False )
2023-08-05 02:26:24 -07:00
up_votes = db . Column ( db . Integer , default = 0 )
down_votes = db . Column ( db . Integer , default = 0 )
2024-01-07 00:36:04 -08:00
ranking = db . Column ( db . Float , default = 0.0 , index = True ) # used for 'hot' sorting
2024-05-08 22:54:30 -07:00
language_id = db . Column ( db . Integer , db . ForeignKey ( ' language.id ' ) , index = True )
2023-08-05 02:26:24 -07:00
edited_at = db . Column ( db . DateTime )
2023-12-16 03:12:49 -08:00
reports = db . Column ( db . Integer , default = 0 ) # how many times this post has been reported. Set to -1 to ignore reports
2023-08-05 02:26:24 -07:00
ap_id = db . Column ( db . String ( 255 ) , index = True )
2023-09-16 00:09:04 -07:00
ap_create_id = db . Column ( db . String ( 100 ) )
ap_announce_id = db . Column ( db . String ( 100 ) )
2023-08-05 02:26:24 -07:00
search_vector = db . Column ( TSVectorType ( ' body ' ) )
2023-12-26 19:58:30 -08:00
author = db . relationship ( ' User ' , lazy = ' joined ' , foreign_keys = [ user_id ] , single_parent = True , overlaps = " post_replies " )
2024-01-01 22:41:00 -08:00
community = db . relationship ( ' Community ' , lazy = ' joined ' , overlaps = ' replies ' , foreign_keys = [ community_id ] )
2024-05-08 22:54:30 -07:00
language = db . relationship ( ' Language ' , foreign_keys = [ language_id ] )
def language_code ( self ) :
if self . language_id :
return self . language . code
else :
return ' en '
def language_name ( self ) :
if self . language_id :
return self . language . name
else :
return ' English '
2023-12-26 18:47:17 -08:00
2023-12-09 01:14:16 -08:00
def is_local ( self ) :
return self . ap_id is None or self . ap_id . startswith ( ' https:// ' + current_app . config [ ' SERVER_NAME ' ] )
2023-09-16 00:09:04 -07:00
@classmethod
def get_by_ap_id ( cls , ap_id ) :
return cls . query . filter_by ( ap_id = ap_id ) . first ( )
2023-12-07 20:13:38 -08:00
def profile_id ( self ) :
2023-12-09 18:10:09 -08:00
if self . ap_id :
return self . ap_id
else :
return f " https:// { current_app . config [ ' SERVER_NAME ' ] } /comment/ { self . id } "
2023-12-07 20:13:38 -08:00
2023-12-09 01:14:16 -08:00
# the ap_id of the parent object, whether it's another PostReply or a Post
def in_reply_to ( self ) :
if self . parent_id is None :
return self . post . ap_id
else :
parent = PostReply . query . get ( self . parent_id )
return parent . ap_id
# the AP profile of the person who wrote the parent object, which could be another PostReply or a Post
def to ( self ) :
if self . parent_id is None :
return self . post . author . profile_id ( )
else :
parent = PostReply . query . get ( self . parent_id )
return parent . author . profile_id ( )
2023-12-25 15:36:02 -08:00
def delete_dependencies ( self ) :
2024-04-14 02:49:42 -07:00
for child_reply in self . child_replies ( ) :
child_reply . delete_dependencies ( )
db . session . delete ( child_reply )
2023-12-25 15:36:02 -08:00
db . session . query ( Report ) . filter ( Report . suspect_post_reply_id == self . id ) . delete ( )
db . session . execute ( text ( ' DELETE FROM post_reply_vote WHERE post_reply_id = :post_reply_id ' ) ,
{ ' post_reply_id ' : self . id } )
if self . image_id :
file = File . query . get ( self . image_id )
file . delete_from_disk ( )
2024-04-14 02:49:42 -07:00
def child_replies ( self ) :
return PostReply . query . filter_by ( parent_id = self . id ) . all ( )
2023-12-25 15:36:02 -08:00
def has_replies ( self ) :
2024-06-01 21:45:21 -07:00
reply = PostReply . query . filter_by ( parent_id = self . id ) . filter ( PostReply . deleted == False ) . first ( )
2023-12-25 15:36:02 -08:00
return reply is not None
2024-01-10 23:39:22 -08:00
def blocked_by_content_filter ( self , content_filters ) :
lowercase_body = self . body . lower ( )
for name , keywords in content_filters . items ( ) if content_filters else { } :
for keyword in keywords :
if keyword in lowercase_body :
return name
return False
2024-04-29 02:43:37 -07:00
def notify_new_replies ( self , user_id : int ) - > bool :
existing_notification = NotificationSubscription . query . filter ( NotificationSubscription . entity_id == self . id ,
NotificationSubscription . user_id == user_id ,
NotificationSubscription . type == NOTIF_REPLY ) . first ( )
return existing_notification is not None
2023-08-05 02:26:24 -07:00
class Domain ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
name = db . Column ( db . String ( 255 ) , index = True )
post_count = db . Column ( db . Integer , default = 0 )
2023-09-16 00:09:04 -07:00
banned = db . Column ( db . Boolean , default = False , index = True ) # Domains can be banned site-wide (by admin) or DomainBlock'ed by users
2023-12-29 14:36:24 -08:00
notify_mods = db . Column ( db . Boolean , default = False , index = True )
notify_admins = db . Column ( db . Boolean , default = False , index = True )
2023-08-05 02:26:24 -07:00
2024-02-01 19:52:23 -08:00
def blocked_by ( self , user ) :
block = DomainBlock . query . filter_by ( domain_id = self . id , user_id = user . id ) . first ( )
return block is not None
def purge_content ( self ) :
files = File . query . join ( Post ) . filter ( Post . domain_id == self . id ) . all ( )
for file in files :
file . delete_from_disk ( )
posts = Post . query . filter_by ( domain_id = self . id ) . all ( )
for post in posts :
post . delete_dependencies ( )
db . session . delete ( post )
db . session . commit ( )
2023-08-05 02:26:24 -07:00
class DomainBlock ( db . Model ) :
user_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , primary_key = True )
domain_id = db . Column ( db . Integer , db . ForeignKey ( ' domain.id ' ) , primary_key = True )
2023-12-11 11:53:35 -08:00
created_at = db . Column ( db . DateTime , default = utcnow )
2023-08-05 02:26:24 -07:00
class CommunityBlock ( db . Model ) :
user_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , primary_key = True )
community_id = db . Column ( db . Integer , db . ForeignKey ( ' community.id ' ) , primary_key = True )
2023-12-11 11:53:35 -08:00
created_at = db . Column ( db . DateTime , default = utcnow )
2023-08-05 02:26:24 -07:00
class CommunityMember ( db . Model ) :
user_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , primary_key = True )
community_id = db . Column ( db . Integer , db . ForeignKey ( ' community.id ' ) , primary_key = True )
is_moderator = db . Column ( db . Boolean , default = False )
is_owner = db . Column ( db . Boolean , default = False )
2024-02-12 20:22:03 -08:00
is_banned = db . Column ( db . Boolean , default = False , index = True )
2024-01-06 15:47:06 -08:00
notify_new_posts = db . Column ( db . Boolean , default = False )
2023-12-11 11:53:35 -08:00
created_at = db . Column ( db . DateTime , default = utcnow )
2023-08-05 02:26:24 -07:00
2024-04-29 08:13:29 -07:00
class UserFollower ( db . Model ) :
local_user_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , primary_key = True )
remote_user_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , primary_key = True )
is_accepted = db . Column ( db . Boolean , default = True ) # flip to ban remote user / reject follow
is_inward = db . Column ( db . Boolean , default = True ) # true = remote user is following a local one
created_at = db . Column ( db . DateTime , default = utcnow )
2023-09-09 01:46:40 -07:00
# people banned from communities
2023-09-08 01:04:01 -07:00
class CommunityBan ( db . Model ) :
2024-03-14 18:24:45 -07:00
user_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , primary_key = True ) # person who is banned, not the banner
2023-09-08 01:04:01 -07:00
community_id = db . Column ( db . Integer , db . ForeignKey ( ' community.id ' ) , primary_key = True )
banned_by = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) )
reason = db . Column ( db . String ( 50 ) )
2023-12-11 11:53:35 -08:00
created_at = db . Column ( db . DateTime , default = utcnow )
2023-09-08 01:04:01 -07:00
ban_until = db . Column ( db . DateTime )
2023-08-05 02:26:24 -07:00
class UserNote ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
2024-01-24 00:17:36 -08:00
user_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , index = True )
target_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , index = True )
2023-08-05 02:26:24 -07:00
body = db . Column ( db . Text )
2023-12-11 11:53:35 -08:00
created_at = db . Column ( db . DateTime , default = utcnow )
2023-08-05 02:26:24 -07:00
class UserBlock ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
2024-01-24 00:17:36 -08:00
blocker_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , index = True )
blocked_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , index = True )
2023-12-11 11:53:35 -08:00
created_at = db . Column ( db . DateTime , default = utcnow )
2023-08-05 02:26:24 -07:00
class Settings ( db . Model ) :
name = db . Column ( db . String ( 50 ) , primary_key = True )
value = db . Column ( db . String ( 1024 ) )
2023-09-05 01:25:02 -07:00
class Interest ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
name = db . Column ( db . String ( 50 ) )
communities = db . Column ( db . Text )
2023-09-08 01:04:01 -07:00
class CommunityJoinRequest ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
user_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) )
2024-01-24 00:17:36 -08:00
community_id = db . Column ( db . Integer , db . ForeignKey ( ' community.id ' ) , index = True )
2023-09-08 01:04:01 -07:00
class UserFollowRequest ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
user_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) )
follow_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) )
2024-02-01 18:30:03 -08:00
class UserRegistration ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
user_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , index = True )
answer = db . Column ( db . String ( 512 ) )
status = db . Column ( db . Integer , default = 0 , index = True ) # 0 = unapproved, 1 = approved
created_at = db . Column ( db . DateTime , default = utcnow )
approved_at = db . Column ( db . DateTime )
approved_by = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) )
user = db . relationship ( ' User ' , foreign_keys = [ user_id ] , lazy = ' joined ' )
2023-09-10 01:20:53 -07:00
class PostVote ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
2024-01-24 00:17:36 -08:00
user_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , index = True )
2023-09-10 01:20:53 -07:00
author_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) )
2024-01-24 00:17:36 -08:00
post_id = db . Column ( db . Integer , db . ForeignKey ( ' post.id ' ) , index = True )
2024-01-09 13:18:11 -08:00
effect = db . Column ( db . Float , index = True )
2023-12-11 11:53:35 -08:00
created_at = db . Column ( db . DateTime , default = utcnow )
2023-11-29 23:57:51 -08:00
post = db . relationship ( ' Post ' , foreign_keys = [ post_id ] )
2023-09-10 01:20:53 -07:00
class PostReplyVote ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
2024-01-24 00:17:36 -08:00
user_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , index = True ) # who voted
2023-11-24 01:52:42 -08:00
author_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) ) # the author of the reply voted on - who's reputation is affected
2024-01-24 00:17:36 -08:00
post_reply_id = db . Column ( db . Integer , db . ForeignKey ( ' post_reply.id ' ) , index = True )
2023-09-10 01:20:53 -07:00
effect = db . Column ( db . Float )
2023-12-11 11:53:35 -08:00
created_at = db . Column ( db . DateTime , default = utcnow )
2023-09-10 01:20:53 -07:00
2023-09-09 01:46:40 -07:00
# save every activity to a log, to aid debugging
class ActivityPubLog ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
direction = db . Column ( db . String ( 3 ) ) # 'in' or 'out'
2024-01-18 10:45:48 -08:00
activity_id = db . Column ( db . String ( 256 ) , index = True )
2023-09-09 01:46:40 -07:00
activity_type = db . Column ( db . String ( 50 ) ) # e.g. 'Follow', 'Accept', 'Like', etc
activity_json = db . Column ( db . Text ) # the full json of the activity
result = db . Column ( db . String ( 10 ) ) # 'success' or 'failure'
exception_message = db . Column ( db . Text )
2023-12-11 11:53:35 -08:00
created_at = db . Column ( db . DateTime , default = utcnow )
2023-09-09 01:46:40 -07:00
2023-10-02 02:16:44 -07:00
class Filter ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
title = db . Column ( db . String ( 50 ) )
2024-01-10 23:39:22 -08:00
filter_home = db . Column ( db . Boolean , default = True )
2023-10-02 02:16:44 -07:00
filter_posts = db . Column ( db . Boolean , default = True )
filter_replies = db . Column ( db . Boolean , default = False )
hide_type = db . Column ( db . Integer , default = 0 ) # 0 = hide with warning, 1 = hide completely
2024-01-24 00:17:36 -08:00
user_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , index = True )
2024-01-10 23:39:22 -08:00
expire_after = db . Column ( db . Date )
keywords = db . Column ( db . String ( 500 ) )
2023-10-02 02:16:44 -07:00
2024-01-10 23:39:22 -08:00
def keywords_string ( self ) :
if self . keywords is None or self . keywords == ' ' :
return ' '
2024-03-11 00:14:12 -07:00
split_keywords = [ kw . strip ( ) for kw in self . keywords . split ( ' \n ' ) ]
return ' , ' . join ( split_keywords )
2023-10-02 02:16:44 -07:00
2023-10-18 02:23:59 -07:00
class Role ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
name = db . Column ( db . String ( 50 ) )
weight = db . Column ( db . Integer , default = 0 )
permissions = db . relationship ( ' RolePermission ' )
class RolePermission ( db . Model ) :
role_id = db . Column ( db . Integer , db . ForeignKey ( ' role.id ' ) , primary_key = True )
permission = db . Column ( db . String , primary_key = True , index = True )
2023-10-02 02:16:44 -07:00
2023-11-29 23:57:51 -08:00
class Notification ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
title = db . Column ( db . String ( 50 ) )
url = db . Column ( db . String ( 512 ) )
read = db . Column ( db . Boolean , default = False )
2024-01-24 00:17:36 -08:00
user_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , index = True ) # who the notification should go to
2023-12-29 14:36:24 -08:00
author_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) ) # the person who caused the notification to happen
2023-12-11 11:53:35 -08:00
created_at = db . Column ( db . DateTime , default = utcnow )
2023-11-29 23:57:51 -08:00
2023-12-13 00:04:11 -08:00
class Report ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
reasons = db . Column ( db . String ( 256 ) )
description = db . Column ( db . String ( 256 ) )
2024-03-26 02:18:05 -07:00
status = db . Column ( db . Integer , default = 0 ) # 0 = new, 1 = escalated to admin, 2 = being appealed, 3 = resolved, 4 = discarded
2024-02-18 18:01:53 -08:00
type = db . Column ( db . Integer , default = 0 ) # 0 = user, 1 = post, 2 = reply, 3 = community, 4 = conversation
2023-12-13 00:04:11 -08:00
reporter_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) )
2024-03-18 01:05:13 -07:00
suspect_community_id = db . Column ( db . Integer , db . ForeignKey ( ' community.id ' ) )
2023-12-13 00:04:11 -08:00
suspect_user_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) )
suspect_post_id = db . Column ( db . Integer , db . ForeignKey ( ' post.id ' ) )
2023-12-25 15:36:02 -08:00
suspect_post_reply_id = db . Column ( db . Integer , db . ForeignKey ( ' post_reply.id ' ) )
2024-02-18 18:01:53 -08:00
suspect_conversation_id = db . Column ( db . Integer , db . ForeignKey ( ' conversation.id ' ) )
2024-03-18 01:05:13 -07:00
in_community_id = db . Column ( db . Integer , db . ForeignKey ( ' community.id ' ) )
2024-03-26 02:18:05 -07:00
source_instance_id = db . Column ( db . Integer , db . ForeignKey ( ' instance.id ' ) ) # the instance of the reporter. mostly used to distinguish between local (instance 1) and remote reports
2023-12-13 00:04:11 -08:00
created_at = db . Column ( db . DateTime , default = utcnow )
updated = db . Column ( db . DateTime , default = utcnow )
2024-01-01 19:07:41 -08:00
# textual representation of self.type
def type_text ( self ) :
2024-02-18 18:01:53 -08:00
types = ( ' User ' , ' Post ' , ' Comment ' , ' Community ' , ' Conversation ' )
2024-01-01 19:07:41 -08:00
if self . type is None :
return ' '
else :
return types [ self . type ]
def is_local ( self ) :
2024-03-26 14:42:36 -07:00
return self . source_instance_id == 1
2024-01-01 19:07:41 -08:00
2023-12-16 03:12:49 -08:00
2024-04-19 00:20:09 -07:00
class NotificationSubscription ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
name = db . Column ( db . String ( 256 ) ) # to avoid needing to look up the thing subscribed to via entity_id
type = db . Column ( db . Integer , default = 0 , index = True ) # see constants.py for possible values: NOTIF_*
entity_id = db . Column ( db . Integer , index = True ) # ID of the user, post, community, etc being subscribed to
user_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , index = True ) # To whom this subscription belongs
created_at = db . Column ( db . DateTime , default = utcnow ) # Perhaps very old subscriptions can be automatically deleted
2024-05-16 01:43:03 -07:00
class Poll ( db . Model ) :
post_id = db . Column ( db . Integer , db . ForeignKey ( ' post.id ' ) , primary_key = True )
end_poll = db . Column ( db . DateTime )
2024-05-16 02:53:38 -07:00
mode = db . Column ( db . String ( 10 ) ) # 'single' or 'multiple' determines whether people can vote for one or multiple options
2024-05-16 01:43:03 -07:00
local_only = db . Column ( db . Boolean )
latest_vote = db . Column ( db . DateTime )
2024-05-18 00:41:20 -07:00
def has_voted ( self , user_id ) :
existing_vote = PollChoiceVote . query . filter ( PollChoiceVote . user_id == user_id , PollChoiceVote . post_id == self . post_id ) . first ( )
return existing_vote is not None
def vote_for_choice ( self , choice_id , user_id ) :
existing_vote = PollChoiceVote . query . filter ( PollChoiceVote . user_id == user_id ,
PollChoiceVote . choice_id == choice_id ) . first ( )
if not existing_vote :
new_vote = PollChoiceVote ( choice_id = choice_id , user_id = user_id , post_id = self . post_id )
db . session . add ( new_vote )
choice = PollChoice . query . get ( choice_id )
choice . num_votes + = 1
self . latest_vote = datetime . utcnow ( )
db . session . commit ( )
def total_votes ( self ) :
return db . session . execute ( text ( ' SELECT SUM(num_votes) as s FROM " poll_choice " WHERE post_id = :post_id ' ) ,
{ ' post_id ' : self . post_id } ) . scalar ( )
2024-05-16 01:43:03 -07:00
class PollChoice ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
post_id = db . Column ( db . Integer , db . ForeignKey ( ' post.id ' ) , index = True )
choice_text = db . Column ( db . String ( 200 ) )
sort_order = db . Column ( db . Integer )
num_votes = db . Column ( db . Integer , default = 0 )
2024-05-18 00:41:20 -07:00
def percentage ( self , poll_total_votes ) :
return math . ceil ( self . num_votes / poll_total_votes * 100 )
2024-05-16 01:43:03 -07:00
class PollChoiceVote ( db . Model ) :
choice_id = db . Column ( db . Integer , db . ForeignKey ( ' poll_choice.id ' ) , primary_key = True )
user_id = db . Column ( db . Integer , db . ForeignKey ( ' user.id ' ) , primary_key = True )
post_id = db . Column ( db . Integer , db . ForeignKey ( ' post.id ' ) , index = True )
created_at = db . Column ( db . DateTime , default = utcnow )
2023-12-29 22:03:44 -08:00
class IpBan ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
ip_address = db . Column ( db . String ( 50 ) , index = True )
notes = db . Column ( db . String ( 150 ) )
created_at = db . Column ( db . DateTime , default = utcnow )
2023-12-16 03:12:49 -08:00
class Site ( db . Model ) :
id = db . Column ( db . Integer , primary_key = True )
name = db . Column ( db . String ( 256 ) )
description = db . Column ( db . String ( 256 ) )
icon_id = db . Column ( db . Integer , db . ForeignKey ( ' file.id ' ) )
sidebar = db . Column ( db . Text , default = ' ' )
legal_information = db . Column ( db . Text , default = ' ' )
public_key = db . Column ( db . Text )
private_key = db . Column ( db . Text )
enable_downvotes = db . Column ( db . Boolean , default = True )
allow_local_image_posts = db . Column ( db . Boolean , default = True )
remote_image_cache_days = db . Column ( db . Integer , default = 30 )
enable_nsfw = db . Column ( db . Boolean , default = False )
enable_nsfl = db . Column ( db . Boolean , default = False )
community_creation_admin_only = db . Column ( db . Boolean , default = False )
reports_email_admins = db . Column ( db . Boolean , default = True )
2024-02-01 18:30:03 -08:00
registration_mode = db . Column ( db . String ( 20 ) , default = ' Closed ' ) # possible values: Open, RequireApplication, Closed
2023-12-16 03:12:49 -08:00
application_question = db . Column ( db . Text , default = ' ' )
allow_or_block_list = db . Column ( db . Integer , default = 2 ) # 1 = allow list, 2 = block list
allowlist = db . Column ( db . Text , default = ' ' )
blocklist = db . Column ( db . Text , default = ' ' )
2024-03-21 16:22:19 -07:00
blocked_phrases = db . Column ( db . Text , default = ' ' ) # discard incoming content with these phrases
2024-03-21 18:35:51 -07:00
auto_decline_referrers = db . Column ( db . Text , default = ' rdrama.net \n ahrefs.com ' ) # automatically decline registration requests if the referrer is one of these
2023-12-16 03:12:49 -08:00
created_at = db . Column ( db . DateTime , default = utcnow )
updated = db . Column ( db . DateTime , default = utcnow )
last_active = db . Column ( db . DateTime , default = utcnow )
2024-01-12 14:12:31 -08:00
log_activitypub_json = db . Column ( db . Boolean , default = False )
2024-02-06 20:31:12 -08:00
default_theme = db . Column ( db . String ( 20 ) , default = ' ' )
2024-04-11 21:22:58 -07:00
contact_email = db . Column ( db . String ( 255 ) , default = ' ' )
2024-04-22 10:53:12 -07:00
about = db . Column ( db . Text , default = ' ' )
2023-12-16 03:12:49 -08:00
2023-12-29 14:36:24 -08:00
@staticmethod
def admins ( ) - > List [ User ] :
return User . query . filter_by ( deleted = False , banned = False ) . join ( user_role ) . filter ( user_role . c . role_id == 4 ) . all ( )
2023-12-16 03:12:49 -08:00
2023-08-05 02:26:24 -07:00
@login.user_loader
def load_user ( id ) :
return User . query . get ( int ( id ) )