2024-06-14 03:03:47 -07:00
import os
2024-12-02 14:25:16 -08:00
import re
2024-09-15 00:30:45 -07:00
from datetime import timedelta
2023-12-31 14:38:24 -08:00
from time import sleep
2024-09-25 13:50:47 -07:00
from io import BytesIO
import json as python_json
2023-11-26 02:21:04 -08:00
2024-09-25 13:50:47 -07:00
from flask import request , flash , json , url_for , current_app , redirect , g , abort , send_file
2023-11-03 01:59:48 -07:00
from flask_login import login_required , current_user
from flask_babel import _
2024-04-07 00:40:52 -07:00
from slugify import slugify
2024-03-27 00:20:08 -07:00
from sqlalchemy import text , desc , or_
2024-06-14 03:03:47 -07:00
from PIL import Image
2024-12-02 14:25:16 -08:00
from urllib . parse import urlparse
2023-11-03 01:59:48 -07:00
2024-03-21 16:22:19 -07:00
from app import db , celery , cache
2024-11-24 14:01:11 -08:00
from app . activitypub . routes import process_inbox_request , process_delete_request , replay_inbox_request
2024-05-04 13:16:55 -07:00
from app . activitypub . signature import post_request , default_context
2024-09-12 11:45:38 -07:00
from app . activitypub . util import instance_allowed , instance_blocked , extract_domain_and_actor
2024-01-03 19:00:19 -08:00
from app . admin . forms import FederationForm , SiteMiscForm , SiteProfileForm , EditCommunityForm , EditUserForm , \
2024-10-08 04:17:20 -07:00
EditTopicForm , SendNewsletterForm , AddUserForm , PreLoadCommunitiesForm , ImportExportBannedListsForm , \
2024-12-02 11:29:11 -08:00
EditInstanceForm , RemoteInstanceScanForm
2024-03-03 15:13:14 -08:00
from app . admin . util import unsubscribe_from_everything_then_delete , unsubscribe_from_community , send_newsletter , \
2024-04-08 00:48:25 -07:00
topics_for_form
2024-09-12 11:45:38 -07:00
from app . community . util import save_icon_file , save_banner_file , search_for_community
from app . community . routes import do_subscribe
2024-03-27 00:20:08 -07:00
from app . constants import REPORT_STATE_NEW , REPORT_STATE_ESCALATED
2024-06-26 04:29:49 -07:00
from app . email import send_welcome_email
2023-12-31 17:49:15 -08:00
from app . models import AllowedInstances , BannedInstances , ActivityPubLog , utcnow , Site , Community , CommunityMember , \
2024-09-25 13:50:47 -07:00
User , Instance , File , Report , Topic , UserRegistration , Role , Post , PostReply , Language , RolePermission , Domain , \
Tag
2024-01-11 15:34:08 -08:00
from app . utils import render_template , permission_required , set_setting , get_setting , gibberish , markdown_to_html , \
2024-04-08 00:48:25 -07:00
moderating_communities , joined_communities , finalize_user_setup , theme_list , blocked_phrases , blocked_referrers , \
2024-09-26 11:46:04 -07:00
topic_tree , languages_for_form , menu_topics , ensure_directory_exists , add_to_modlog , get_request , file_get_contents
2023-11-03 01:59:48 -07:00
from app . admin import bp
@bp.route ( ' / ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
@permission_required ( ' change instance settings ' )
def admin_home ( ) :
2024-01-11 15:34:08 -08:00
return render_template ( ' admin/home.html ' , title = _ ( ' Admin ' ) , moderating_communities = moderating_communities ( current_user . get_id ( ) ) ,
2024-02-01 18:30:03 -08:00
joined_communities = joined_communities ( current_user . get_id ( ) ) ,
2024-05-30 02:54:25 -07:00
menu_topics = menu_topics ( ) ,
2024-02-01 18:30:03 -08:00
site = g . site )
2023-12-16 03:12:49 -08:00
@bp.route ( ' /site ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
@permission_required ( ' change instance settings ' )
def admin_site ( ) :
form = SiteProfileForm ( )
site = Site . query . get ( 1 )
if site is None :
site = Site ( )
if form . validate_on_submit ( ) :
site . name = form . name . data
2024-04-22 10:23:26 -07:00
site . description = form . description . data #tagline
site . about = form . about . data
2023-12-16 03:12:49 -08:00
site . sidebar = form . sidebar . data
site . legal_information = form . legal_information . data
site . updated = utcnow ( )
2024-04-16 03:10:24 -07:00
site . contact_email = form . contact_email . data
2023-12-16 03:12:49 -08:00
if site . id is None :
db . session . add ( site )
2024-06-14 03:03:47 -07:00
# Save site icon
uploaded_icon = request . files [ ' icon ' ]
if uploaded_icon and uploaded_icon . filename != ' ' :
2024-08-07 01:00:55 -07:00
allowed_extensions = [ ' .gif ' , ' .jpg ' , ' .jpeg ' , ' .png ' , ' .webp ' , ' .svg ' ]
2024-06-14 03:03:47 -07:00
file_ext = os . path . splitext ( uploaded_icon . filename ) [ 1 ]
if file_ext . lower ( ) not in allowed_extensions :
abort ( 400 )
directory = ' app/static/media '
ensure_directory_exists ( directory )
# Remove existing logo files
if os . path . isfile ( f ' app { site . logo } ' ) :
os . unlink ( f ' app { site . logo } ' )
if os . path . isfile ( f ' app { site . logo_152 } ' ) :
os . unlink ( f ' app { site . logo_152 } ' )
if os . path . isfile ( f ' app { site . logo_32 } ' ) :
os . unlink ( f ' app { site . logo_32 } ' )
if os . path . isfile ( f ' app { site . logo_16 } ' ) :
os . unlink ( f ' app { site . logo_16 } ' )
# Save logo file
base_filename = f ' logo_ { gibberish ( 5 ) } '
uploaded_icon . save ( f ' { directory } / { base_filename } { file_ext } ' )
2024-08-07 01:00:55 -07:00
if file_ext == ' .svg ' :
2024-06-14 03:03:47 -07:00
delete_original = False
2024-08-07 01:00:55 -07:00
site . logo = site . logo_152 = site . logo_32 = site . logo_16 = f ' /static/media/ { base_filename } { file_ext } '
else :
img = Image . open ( f ' { directory } / { base_filename } { file_ext } ' )
if img . width > 100 :
img . thumbnail ( ( 100 , 100 ) )
img . save ( f ' { directory } / { base_filename } _100 { file_ext } ' )
site . logo = f ' /static/media/ { base_filename } _100 { file_ext } '
delete_original = True
else :
site . logo = f ' /static/media/ { base_filename } { file_ext } '
delete_original = False
2024-06-14 03:03:47 -07:00
2024-08-07 01:00:55 -07:00
# Save multiple copies of the logo - different sizes
img = Image . open ( f ' { directory } / { base_filename } { file_ext } ' )
img . thumbnail ( ( 152 , 152 ) )
img . save ( f ' { directory } / { base_filename } _152 { file_ext } ' )
site . logo_152 = f ' /static/media/ { base_filename } _152 { file_ext } '
2024-06-14 03:03:47 -07:00
2024-08-07 01:00:55 -07:00
img = Image . open ( f ' { directory } / { base_filename } { file_ext } ' )
img . thumbnail ( ( 32 , 32 ) )
img . save ( f ' { directory } / { base_filename } _32 { file_ext } ' )
site . logo_32 = f ' /static/media/ { base_filename } _32 { file_ext } '
2024-06-14 03:03:47 -07:00
2024-08-07 01:00:55 -07:00
img = Image . open ( f ' { directory } / { base_filename } { file_ext } ' )
img . thumbnail ( ( 16 , 16 ) )
img . save ( f ' { directory } / { base_filename } _16 { file_ext } ' )
site . logo_16 = f ' /static/media/ { base_filename } _16 { file_ext } '
2024-06-14 03:03:47 -07:00
if delete_original :
os . unlink ( f ' app/static/media/ { base_filename } { file_ext } ' )
2023-12-16 03:12:49 -08:00
db . session . commit ( )
2024-06-10 04:07:15 -07:00
set_setting ( ' announcement ' , form . announcement . data )
2023-12-16 03:12:49 -08:00
flash ( ' Settings saved. ' )
elif request . method == ' GET ' :
form . name . data = site . name
form . description . data = site . description
2024-04-22 10:23:26 -07:00
form . about . data = site . about
2023-12-16 03:12:49 -08:00
form . sidebar . data = site . sidebar
form . legal_information . data = site . legal_information
2024-04-22 10:23:26 -07:00
form . contact_email . data = site . contact_email
2024-06-10 04:07:15 -07:00
form . announcement . data = get_setting ( ' announcement ' , ' ' )
2024-01-11 15:34:08 -08:00
return render_template ( ' admin/site.html ' , title = _ ( ' Site profile ' ) , form = form ,
moderating_communities = moderating_communities ( current_user . get_id ( ) ) ,
2024-02-01 18:30:03 -08:00
joined_communities = joined_communities ( current_user . get_id ( ) ) ,
2024-05-30 02:54:25 -07:00
menu_topics = menu_topics ( ) ,
2024-02-01 18:30:03 -08:00
site = g . site
2024-01-11 15:34:08 -08:00
)
2023-12-16 03:12:49 -08:00
@bp.route ( ' /misc ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
@permission_required ( ' change instance settings ' )
def admin_misc ( ) :
form = SiteMiscForm ( )
site = Site . query . get ( 1 )
if site is None :
site = Site ( )
2024-02-06 21:33:25 -08:00
form . default_theme . choices = theme_list ( )
2023-12-16 03:12:49 -08:00
if form . validate_on_submit ( ) :
site . enable_downvotes = form . enable_downvotes . data
site . allow_local_image_posts = form . allow_local_image_posts . data
site . remote_image_cache_days = form . remote_image_cache_days . data
site . enable_nsfw = form . enable_nsfw . data
site . enable_nsfl = form . enable_nsfl . data
site . community_creation_admin_only = form . community_creation_admin_only . data
site . reports_email_admins = form . reports_email_admins . data
site . registration_mode = form . registration_mode . data
site . application_question = form . application_question . data
2024-03-21 18:35:51 -07:00
site . auto_decline_referrers = form . auto_decline_referrers . data
2024-01-12 14:12:31 -08:00
site . log_activitypub_json = form . log_activitypub_json . data
2024-07-12 04:56:57 -07:00
site . show_inoculation_block = form . show_inoculation_block . data
2023-12-16 03:12:49 -08:00
site . updated = utcnow ( )
2024-02-06 20:31:12 -08:00
site . default_theme = form . default_theme . data
2023-12-16 03:12:49 -08:00
if site . id is None :
db . session . add ( site )
db . session . commit ( )
2024-03-21 18:35:51 -07:00
cache . delete_memoized ( blocked_referrers )
2024-07-07 01:03:35 -07:00
set_setting ( ' public_modlog ' , form . public_modlog . data )
2023-12-16 03:12:49 -08:00
flash ( ' Settings saved. ' )
elif request . method == ' GET ' :
form . enable_downvotes . data = site . enable_downvotes
form . allow_local_image_posts . data = site . allow_local_image_posts
form . remote_image_cache_days . data = site . remote_image_cache_days
form . enable_nsfw . data = site . enable_nsfw
form . enable_nsfl . data = site . enable_nsfl
form . community_creation_admin_only . data = site . community_creation_admin_only
form . reports_email_admins . data = site . reports_email_admins
form . registration_mode . data = site . registration_mode
form . application_question . data = site . application_question
2024-03-21 18:35:51 -07:00
form . auto_decline_referrers . data = site . auto_decline_referrers
2024-01-12 14:12:31 -08:00
form . log_activitypub_json . data = site . log_activitypub_json
2024-07-12 04:56:57 -07:00
form . show_inoculation_block . data = site . show_inoculation_block
2024-02-06 20:31:12 -08:00
form . default_theme . data = site . default_theme if site . default_theme is not None else ' '
2024-07-07 01:03:35 -07:00
form . public_modlog . data = get_setting ( ' public_modlog ' , False )
2024-01-11 15:34:08 -08:00
return render_template ( ' admin/misc.html ' , title = _ ( ' Misc settings ' ) , form = form ,
moderating_communities = moderating_communities ( current_user . get_id ( ) ) ,
2024-02-01 18:30:03 -08:00
joined_communities = joined_communities ( current_user . get_id ( ) ) ,
2024-05-30 02:54:25 -07:00
menu_topics = menu_topics ( ) ,
2024-02-01 18:30:03 -08:00
site = g . site
2024-01-11 15:34:08 -08:00
)
2023-12-16 03:12:49 -08:00
@bp.route ( ' /federation ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
@permission_required ( ' change instance settings ' )
def admin_federation ( ) :
form = FederationForm ( )
2024-09-11 16:14:12 -07:00
preload_form = PreLoadCommunitiesForm ( )
2024-09-16 16:41:48 -07:00
ban_lists_form = ImportExportBannedListsForm ( )
2024-12-02 11:29:11 -08:00
remote_scan_form = RemoteInstanceScanForm ( )
2024-09-12 09:56:09 -07:00
# this is the pre-load communities button
2024-09-11 16:14:12 -07:00
if preload_form . pre_load_submit . data and preload_form . validate ( ) :
2024-09-12 16:01:37 -07:00
# how many communities to add
if preload_form . communities_num . data :
communities_to_add = preload_form . communities_num . data
else :
communities_to_add = 25
2024-09-12 09:56:09 -07:00
# pull down the community.full.json
2024-09-13 20:50:23 -07:00
resp = get_request ( ' https://data.lemmyverse.net/data/community.full.json ' )
community_json = resp . json ( )
resp . close ( )
2024-09-12 09:56:09 -07:00
2024-10-31 14:29:17 -07:00
already_known = list ( db . session . execute ( text ( ' SELECT ap_public_url FROM " community " ' ) ) . scalars ( ) )
banned_urls = list ( db . session . execute ( text ( ' SELECT domain FROM " banned_instances " ' ) ) . scalars ( ) )
seven_things_plus = [
' shit ' , ' piss ' , ' fuck ' ,
' cunt ' , ' cocksucker ' , ' motherfucker ' , ' tits ' ,
' memes ' , ' piracy ' , ' 196 ' , ' greentext ' , ' usauthoritarianism ' ,
' enoughmuskspam ' , ' political_weirdos ' , ' 4chan '
]
total_count = already_known_count = nsfw_count = low_content_count = low_active_users_count = banned_count = bad_words_count = 0
candidate_communities = [ ]
2024-09-13 20:50:23 -07:00
for community in community_json :
2024-10-31 14:29:17 -07:00
total_count + = 1
# sort out already known communities
if community [ ' url ' ] in already_known :
already_known_count + = 1
2024-09-12 09:56:09 -07:00
continue
2024-10-31 14:29:17 -07:00
# sort out the nsfw communities
elif community [ ' nsfw ' ] :
nsfw_count + = 1
2024-09-12 09:56:09 -07:00
continue
2024-10-31 14:29:17 -07:00
# sort out any that have less than 100 posts
elif community [ ' counts ' ] [ ' posts ' ] < 100 :
low_content_count + = 1
2024-09-12 09:56:09 -07:00
continue
2024-10-31 14:29:17 -07:00
# sort out any that do not have greater than 500 active users over the past week
elif community [ ' counts ' ] [ ' users_active_week ' ] < 500 :
low_active_users_count + = 1
continue
# sort out any instances we have already banned
elif community [ ' baseurl ' ] in banned_urls :
banned_count + = 1
continue
# sort out the 'seven things you can't say on tv' names (cursewords), plus some
# "low effort" communities
if any ( badword in community [ ' name ' ] . lower ( ) for badword in seven_things_plus ) :
bad_words_count + = 1
2024-09-12 09:56:09 -07:00
continue
2024-10-31 14:29:17 -07:00
2024-09-12 09:56:09 -07:00
else :
2024-10-31 14:29:17 -07:00
candidate_communities . append ( community )
2024-09-12 09:56:09 -07:00
2024-10-31 14:29:17 -07:00
filtered_count = already_known_count + nsfw_count + low_content_count + low_active_users_count + banned_count + bad_words_count
flash ( _ ( ' %d out of %d communities were excluded using current filters ' % ( filtered_count , total_count ) ) )
2024-09-12 09:56:09 -07:00
# sort the list based on the users_active_week key
2024-10-31 14:29:17 -07:00
parsed_communities_sorted = sorted ( candidate_communities , key = lambda c : c [ ' counts ' ] [ ' users_active_week ' ] , reverse = True )
2024-09-12 09:56:09 -07:00
2024-09-12 11:45:38 -07:00
# get the community urls to join
community_urls_to_join = [ ]
2024-10-31 14:29:17 -07:00
2024-09-13 11:04:04 -07:00
# if the admin user wants more added than we have, then just add all of them
if communities_to_add > len ( parsed_communities_sorted ) :
2024-10-31 14:29:17 -07:00
communities_to_add = len ( parsed_communities_sorted )
2024-09-13 11:04:04 -07:00
# make the list of urls
2024-09-12 09:56:09 -07:00
for i in range ( communities_to_add ) :
2024-10-31 14:29:17 -07:00
community_urls_to_join . append ( parsed_communities_sorted [ i ] [ ' url ' ] . lower ( ) )
2024-09-12 11:45:38 -07:00
# loop through the list and send off the follow requests
2024-09-13 08:26:00 -07:00
# use User #1, the first instance admin
2024-10-31 14:29:17 -07:00
# NOTE: Subscribing using the admin's alt_profile causes problems:
# 1. 'Leave' will use the main user name so unsubscribe won't succeed.
# 2. De-selecting and re-selecting 'vote privately' generates a new alt_user_name every time,
# so the username needed for a successful unsubscribe might get lost
# 3. If the admin doesn't have 'vote privately' selected, the federation JSON will populate
# with a blank space for the name, so the subscription won't succeed.
# 4. Membership is based on user id, so using the alt_profile doesn't decrease the admin's joined communities
#
# Therefore, 'main_user_name=False' has been changed to 'admin_preload=True' below
2024-09-13 08:26:00 -07:00
user = User . query . get ( 1 )
2024-09-12 11:45:38 -07:00
pre_load_messages = [ ]
2024-09-13 20:50:23 -07:00
for community in community_urls_to_join :
2024-10-31 14:29:17 -07:00
# get the relevant url bits
2024-09-13 20:50:23 -07:00
server , community = extract_domain_and_actor ( community )
2024-10-31 14:29:17 -07:00
2024-09-12 11:45:38 -07:00
# find the community
new_community = search_for_community ( ' ! ' + community + ' @ ' + server )
2024-10-31 14:29:17 -07:00
# subscribe to the community
2024-09-13 08:10:10 -07:00
# capture the messages returned by do_subscibe
# and show to user if instance is in debug mode
if current_app . debug :
2024-10-31 14:29:17 -07:00
message = do_subscribe ( new_community . ap_id , user . id , admin_preload = True )
2024-09-13 08:10:10 -07:00
pre_load_messages . append ( message )
else :
2024-10-31 14:29:17 -07:00
message_we_wont_do_anything_with = do_subscribe . delay ( new_community . ap_id , user . id , admin_preload = True )
2024-09-12 11:45:38 -07:00
2024-09-13 08:10:10 -07:00
if current_app . debug :
2024-09-13 20:50:23 -07:00
flash ( _ ( ' Results: %(results)s ' , results = str ( pre_load_messages ) ) )
2024-09-13 08:10:10 -07:00
else :
2024-09-13 20:50:23 -07:00
flash (
_ ( ' Subscription process for %(communities_to_add)d of %(parsed_communities_sorted)d communities launched in background, check admin/activities for details ' ,
communities_to_add = communities_to_add , parsed_communities_sorted = len ( parsed_communities_sorted ) ) )
2024-09-12 09:56:09 -07:00
return redirect ( url_for ( ' admin.admin_federation ' ) )
2024-10-31 14:29:17 -07:00
2024-12-02 11:29:11 -08:00
# this is the remote server scan
elif remote_scan_form . remote_scan_submit . data and remote_scan_form . validate ( ) :
2024-12-02 14:25:16 -08:00
# filters to be used later
already_known = list ( db . session . execute ( text ( ' SELECT ap_public_url FROM " community " ' ) ) . scalars ( ) )
banned_urls = list ( db . session . execute ( text ( ' SELECT domain FROM " banned_instances " ' ) ) . scalars ( ) )
seven_things_plus = [
' shit ' , ' piss ' , ' fuck ' ,
' cunt ' , ' cocksucker ' , ' motherfucker ' , ' tits ' ,
' memes ' , ' piracy ' , ' 196 ' , ' greentext ' , ' usauthoritarianism ' ,
' enoughmuskspam ' , ' political_weirdos ' , ' 4chan '
]
2024-12-04 15:36:36 -08:00
is_lemmy = False
is_mbin = False
2024-12-02 11:29:11 -08:00
# get the remote_url data
remote_url = remote_scan_form . remote_url . data
2024-12-02 14:25:16 -08:00
# test to make sure its a valid fqdn
regex_pattern = ' ^(https: \ / \ /)(?=. { 1,255}$)((. { 1,63} \ .) { 1,127}(?![0-9]*$)[a-z0-9-]+ \ .?)$ '
result = re . match ( regex_pattern , remote_url )
if result is None :
2024-12-02 15:00:22 -08:00
flash ( _ ( f ' { remote_url } does not appear to be a valid url. Make sure input is in the form " https://server-name.tld " without trailing slashes or paths. ' ) )
2024-12-02 14:25:16 -08:00
return redirect ( url_for ( ' admin.admin_federation ' ) )
2024-12-02 15:00:22 -08:00
# check if it's a banned instance
2024-12-02 14:25:16 -08:00
# Parse the URL
parsed_url = urlparse ( remote_url )
# Extract the server domain name
server_domain = parsed_url . netloc
if server_domain in banned_urls :
flash ( _ ( f ' { remote_url } is a banned instance. ' ) )
return redirect ( url_for ( ' admin.admin_federation ' ) )
2024-12-02 11:29:11 -08:00
# get dry run
dry_run = remote_scan_form . dry_run . data
# get the number of follows requested
2024-12-04 13:44:19 -08:00
communities_requested = remote_scan_form . communities_requested . data
2024-12-02 11:29:11 -08:00
# get the minimums
min_posts = remote_scan_form . minimum_posts . data
min_users = remote_scan_form . minimum_active_users . data
# get the nodeinfo
resp = get_request ( f ' { remote_url } /.well-known/nodeinfo ' )
nodeinfo_dict = json . loads ( resp . text )
# check the ['links'] for instanceinfo url
schema2p0 = " http://nodeinfo.diaspora.software/ns/schema/2.0 "
schema2p1 = " http://nodeinfo.diaspora.software/ns/schema/2.1 "
for e in nodeinfo_dict [ ' links ' ] :
if e [ ' rel ' ] == schema2p0 or e [ ' rel ' ] == schema2p1 :
remote_instanceinfo_url = e [ " href " ]
# get the instanceinfo
resp = get_request ( remote_instanceinfo_url )
instanceinfo_dict = json . loads ( resp . text )
# determine the instance software
instance_software_name = instanceinfo_dict [ ' software ' ] [ ' name ' ]
# instance_software_version = instanceinfo_dict['software']['version']
2024-12-04 15:36:36 -08:00
# if the instance is not running lemmy or mbin break for now as
2024-12-02 11:29:11 -08:00
# we dont yet support others for scanning
2024-12-04 15:36:36 -08:00
if instance_software_name == " lemmy " :
is_lemmy = True
elif instance_software_name == " mbin " :
is_mbin = True
else :
flash ( _ ( f " { remote_url } does not appear to be a lemmy or mbin instance. " ) )
2024-12-02 11:29:11 -08:00
return redirect ( url_for ( ' admin.admin_federation ' ) )
2024-12-04 15:36:36 -08:00
if is_lemmy :
# lemmy has a hard-coded upper limit of 50 commnities
# in their api response
# loop through and send off requests to the remote endpoint for 50 communities at a time
comms_list = [ ]
page = 1
get_more_communities = True
while get_more_communities :
params = { " sort " : " Active " , " type_ " : " Local " , " limit " : " 50 " , " page " : f " { page } " , " show_nsfw " : " false " }
resp = get_request ( f " { remote_url } /api/v3/community/list " , params = params )
page_dict = json . loads ( resp . text )
# get the individual communities out of the communities[] list in the response and
# add them to a holding list[] of our own
for c in page_dict [ " communities " ] :
comms_list . append ( c )
# check the amount of items in the page_dict['communities'] list
# if it's lesss than 50 then we know its the last page of communities
# so we break the loop
if len ( page_dict [ ' communities ' ] ) < 50 :
get_more_communities = False
else :
page + = 1
# filter out the communities
already_known_count = nsfw_count = low_content_count = low_active_users_count = bad_words_count = 0
candidate_communities = [ ]
for community in comms_list :
# sort out already known communities
if community [ ' community ' ] [ ' actor_id ' ] in already_known :
already_known_count + = 1
continue
# sort out any that have less than minimum posts
elif community [ ' counts ' ] [ ' posts ' ] < min_posts :
low_content_count + = 1
continue
# sort out any that do not have greater than the requested active users over the past week
elif community [ ' counts ' ] [ ' users_active_week ' ] < min_users :
low_active_users_count + = 1
continue
# sort out the 'seven things you can't say on tv' names (cursewords), plus some
# "low effort" communities
if any ( badword in community [ ' community ' ] [ ' name ' ] . lower ( ) for badword in seven_things_plus ) :
bad_words_count + = 1
continue
else :
candidate_communities . append ( community )
2024-12-02 11:29:11 -08:00
2024-12-04 15:36:36 -08:00
# get the community urls to join
community_urls_to_join = [ ]
# if the admin user wants more added than we have, then just add all of them
if communities_requested > len ( candidate_communities ) :
communities_to_add = len ( candidate_communities )
2024-12-02 11:29:11 -08:00
else :
2024-12-04 15:36:36 -08:00
communities_to_add = communities_requested
# make the list of urls
for i in range ( communities_to_add ) :
community_urls_to_join . append ( candidate_communities [ i ] [ ' community ' ] [ ' actor_id ' ] . lower ( ) )
# if its a dry run, just return the stats
if dry_run :
message = f " Dry-Run for { remote_url } : \
Local Communities on the server : { len ( comms_list ) } , \
Communities we already have : { already_known_count } , \
Communities below minimum posts : { low_content_count } , \
Communities below minimum users : { low_active_users_count } , \
Candidate Communities based on filters : { len ( candidate_communities ) } , \
Communities to join request : { communities_requested } , \
Communities to join based on current filters : { len ( community_urls_to_join ) } . "
flash ( _ ( message ) )
return redirect ( url_for ( ' admin.admin_federation ' ) )
if is_mbin :
2024-12-04 16:01:08 -08:00
# loop through and send the right number of requests to the remote endpoint for mbin
2024-12-04 15:36:36 -08:00
# mbin does not have the hard-coded limit, but lets stick with 50 to match lemmy
mags_list = [ ]
page = 1
get_more_magazines = True
while get_more_magazines :
params = { " p " : f " { page } " , " perPage " : " 50 " , " sort " : " active " , " federation " : " local " , " hide_adult " : " hide " }
resp = get_request ( f " { remote_url } /api/magazines " , params = params )
page_dict = json . loads ( resp . text )
# get the individual magazines out of the items[] list in the response and
# add them to a holding list[] of our own
for m in page_dict [ ' items ' ] :
mags_list . append ( m )
# check the amount of items in the page_dict['items'] list
# if it's lesss than 50 then we know its the last page of magazines
# so we break the loop
if len ( page_dict [ ' items ' ] ) < 50 :
get_more_magazines = False
else :
page + = 1
# filter out the magazines
already_known_count = low_content_count = low_subscribed_users_count = bad_words_count = 0
2024-12-04 15:54:22 -08:00
candidate_communities = [ ]
2024-12-04 15:36:36 -08:00
for magazine in mags_list :
# sort out already known communities
if magazine [ ' apProfileId ' ] in already_known :
already_known_count + = 1
continue
# sort out any that have less than minimum posts
elif magazine [ ' entryCount ' ] < min_posts :
low_content_count + = 1
continue
# sort out any that do not have greater than the requested users over the past week
# mbin does not show active users here, so its based on subscriber count
elif magazine [ ' subscriptionsCount ' ] < min_users :
low_subscribed_users_count + = 1
continue
# sort out the 'seven things you can't say on tv' names (cursewords), plus some
# "low effort" communities
if any ( badword in magazine [ ' name ' ] . lower ( ) for badword in seven_things_plus ) :
bad_words_count + = 1
continue
else :
2024-12-04 15:54:22 -08:00
candidate_communities . append ( magazine )
2024-12-02 11:29:11 -08:00
2024-12-04 15:36:36 -08:00
# get the community urls to join
community_urls_to_join = [ ]
# if the admin user wants more added than we have, then just add all of them
2024-12-04 15:54:22 -08:00
if communities_requested > len ( candidate_communities ) :
magazines_to_add = len ( candidate_communities )
2024-12-04 15:36:36 -08:00
else :
magazines_to_add = communities_requested
# make the list of urls
for i in range ( magazines_to_add ) :
2024-12-04 15:54:22 -08:00
community_urls_to_join . append ( candidate_communities [ i ] [ ' apProfileId ' ] . lower ( ) )
2024-12-04 15:36:36 -08:00
# if its a dry run, just return the stats
if dry_run :
message = f " Dry-Run for { remote_url } : \
Local Magazines on the server : { len ( mags_list ) } , \
Magazines we already have : { already_known_count } , \
Magazines below minimum posts : { low_content_count } , \
Magazines below minimum users : { low_subscribed_users_count } , \
2024-12-04 15:54:22 -08:00
Candidate Magazines based on filters : { len ( candidate_communities ) } , \
2024-12-04 15:36:36 -08:00
Magazines to join request : { communities_requested } , \
Magazines to join based on current filters : { len ( community_urls_to_join ) } . "
flash ( _ ( message ) )
return redirect ( url_for ( ' admin.admin_federation ' ) )
2024-12-02 11:29:11 -08:00
user = User . query . get ( 1 )
remote_scan_messages = [ ]
for community in community_urls_to_join :
# get the relevant url bits
server , community = extract_domain_and_actor ( community )
# find the community
new_community = search_for_community ( ' ! ' + community + ' @ ' + server )
# subscribe to the community
2024-12-04 15:54:22 -08:00
# capture the messages returned by do_subscribe
2024-12-02 11:29:11 -08:00
# and show to user if instance is in debug mode
if current_app . debug :
message = do_subscribe ( new_community . ap_id , user . id , admin_preload = True )
remote_scan_messages . append ( message )
else :
message_we_wont_do_anything_with = do_subscribe . delay ( new_community . ap_id , user . id , admin_preload = True )
if current_app . debug :
flash ( _ ( ' Results: %(results)s ' , results = str ( remote_scan_messages ) ) )
else :
flash (
2024-12-04 14:17:18 -08:00
_ ( ' Based on current filters, the subscription process for %(communities_to_join)d of %(candidate_communities)d communities launched in background, check admin/activities for details ' ,
communities_to_join = len ( community_urls_to_join ) , candidate_communities = len ( candidate_communities ) ) )
2024-12-02 11:29:11 -08:00
return redirect ( url_for ( ' admin.admin_federation ' ) )
2024-09-16 16:41:48 -07:00
# this is the import bans button
elif ban_lists_form . import_submit . data and ban_lists_form . validate ( ) :
2024-09-26 11:46:04 -07:00
import_file = request . files [ ' import_file ' ]
if import_file and import_file . filename != ' ' :
file_ext = os . path . splitext ( import_file . filename ) [ 1 ]
if file_ext . lower ( ) != ' .json ' :
abort ( 400 )
new_filename = gibberish ( 15 ) + ' .json '
directory = f ' app/static/media/ '
# save the file
final_place = os . path . join ( directory , new_filename + file_ext )
import_file . save ( final_place )
# import bans in background task
2024-09-26 16:08:06 -07:00
if current_app . debug :
import_bans_task ( final_place )
return redirect ( url_for ( ' admin.admin_federation ' ) )
else :
import_bans_task . delay ( final_place )
flash ( _ ( f ' Ban imports started in a background process. ' ) )
return redirect ( url_for ( ' admin.admin_federation ' ) )
2024-09-26 16:36:49 -07:00
else :
flash ( _ ( f ' Ban imports requested, but no json provided. ' ) )
return redirect ( url_for ( ' admin.admin_federation ' ) )
2024-09-16 16:41:48 -07:00
# this is the export bans button
elif ban_lists_form . export_submit . data and ban_lists_form . validate ( ) :
2024-09-29 13:56:25 -07:00
2024-09-25 13:50:47 -07:00
ban_lists_dict = { }
2024-09-26 16:08:06 -07:00
if get_setting ( ' use_allowlist ' ) :
# get the allowed_instances info
allowed_instances = [ ]
already_allowed = AllowedInstances . query . all ( )
if len ( already_allowed ) > 0 :
for aa in already_allowed :
allowed_instances . append ( aa . domain )
2024-09-29 13:56:25 -07:00
ban_lists_dict [ ' allowed_instances ' ] = allowed_instances
2024-09-26 16:08:06 -07:00
else :
# get banned_instances info
banned_instances = [ ]
instance_bans = BannedInstances . query . all ( )
if len ( instance_bans ) > 0 :
for bi in instance_bans :
banned_instances . append ( bi . domain )
ban_lists_dict [ ' banned_instances ' ] = banned_instances
2024-09-25 13:50:47 -07:00
# get banned_domains info
banned_domains = [ ]
domain_bans = Domain . query . filter_by ( banned = True ) . all ( )
2024-09-26 11:46:04 -07:00
if len ( domain_bans ) > 0 :
2024-09-29 13:56:25 -07:00
for domain in domain_bans :
banned_domains . append ( domain . name )
2024-09-25 13:50:47 -07:00
ban_lists_dict [ ' banned_domains ' ] = banned_domains
# get banned_tags info
banned_tags = [ ]
tag_bans = Tag . query . filter_by ( banned = True ) . all ( )
2024-09-26 11:46:04 -07:00
if len ( tag_bans ) > 0 :
2024-09-29 14:31:30 -07:00
for tag_ban in tag_bans :
2024-09-26 11:46:04 -07:00
tag_dict = { }
2024-09-29 14:31:30 -07:00
tag_dict [ ' name ' ] = tag_ban . name
tag_dict [ ' display_as ' ] = tag_ban . display_as
2024-09-26 11:46:04 -07:00
banned_tags . append ( tag_dict )
2024-09-25 13:50:47 -07:00
ban_lists_dict [ ' banned_tags ' ] = banned_tags
# get banned_users info
banned_users = [ ]
user_bans = User . query . filter_by ( banned = True ) . all ( )
2024-09-26 11:46:04 -07:00
if len ( user_bans ) > 0 :
2024-09-29 14:31:30 -07:00
for user_ban in user_bans :
banned_users . append ( user_ban . ap_id )
2024-09-25 13:50:47 -07:00
ban_lists_dict [ ' banned_users ' ] = banned_users
# setup the BytesIO buffer
buffer = BytesIO ( )
buffer . write ( str ( python_json . dumps ( ban_lists_dict ) ) . encode ( ' utf-8 ' ) )
buffer . seek ( 0 )
# send the file to the user as a download
# the as_attachment=True results in flask
# redirecting to the current page, so no
# url_for needed here
return send_file ( buffer , download_name = f ' { current_app . config [ " SERVER_NAME " ] } _bans.json ' , as_attachment = True ,
mimetype = ' application/json ' )
2024-09-16 16:41:48 -07:00
2024-09-12 09:56:09 -07:00
# this is the main settings form
2024-09-11 16:14:12 -07:00
elif form . validate_on_submit ( ) :
2023-11-03 01:59:48 -07:00
if form . use_allowlist . data :
set_setting ( ' use_allowlist ' , True )
db . session . execute ( text ( ' DELETE FROM allowed_instances ' ) )
for allow in form . allowlist . data . split ( ' \n ' ) :
if allow . strip ( ) :
db . session . add ( AllowedInstances ( domain = allow . strip ( ) ) )
2024-03-21 16:22:19 -07:00
cache . delete_memoized ( instance_allowed , allow . strip ( ) )
2023-11-03 01:59:48 -07:00
if form . use_blocklist . data :
set_setting ( ' use_allowlist ' , False )
db . session . execute ( text ( ' DELETE FROM banned_instances ' ) )
for banned in form . blocklist . data . split ( ' \n ' ) :
if banned . strip ( ) :
db . session . add ( BannedInstances ( domain = banned . strip ( ) ) )
2024-03-21 16:22:19 -07:00
cache . delete_memoized ( instance_blocked , banned . strip ( ) )
2024-09-29 13:56:25 -07:00
g . site . blocked_phrases = form . blocked_phrases . data
2024-05-15 20:44:42 -07:00
set_setting ( ' actor_blocked_words ' , form . blocked_actors . data )
2024-03-21 16:22:19 -07:00
cache . delete_memoized ( blocked_phrases )
2024-05-15 20:44:42 -07:00
cache . delete_memoized ( get_setting , ' actor_blocked_words ' )
2023-11-03 01:59:48 -07:00
db . session . commit ( )
2024-03-21 16:22:19 -07:00
2023-11-03 01:59:48 -07:00
flash ( _ ( ' Admin settings saved ' ) )
2024-09-12 09:56:09 -07:00
# this is just the regular page load
2023-11-03 01:59:48 -07:00
elif request . method == ' GET ' :
form . use_allowlist . data = get_setting ( ' use_allowlist ' , False )
form . use_blocklist . data = not form . use_allowlist . data
instances = BannedInstances . query . all ( )
form . blocklist . data = ' \n ' . join ( [ instance . domain for instance in instances ] )
instances = AllowedInstances . query . all ( )
form . allowlist . data = ' \n ' . join ( [ instance . domain for instance in instances ] )
2024-09-29 13:56:25 -07:00
form . blocked_phrases . data = g . site . blocked_phrases
2024-05-15 20:44:42 -07:00
form . blocked_actors . data = get_setting ( ' actor_blocked_words ' , ' 88 ' )
2023-11-03 01:59:48 -07:00
2024-09-11 16:14:12 -07:00
return render_template ( ' admin/federation.html ' , title = _ ( ' Federation settings ' ) ,
2024-09-16 16:41:48 -07:00
form = form , preload_form = preload_form , ban_lists_form = ban_lists_form ,
2024-12-02 11:29:11 -08:00
remote_scan_form = remote_scan_form , current_app_debug = current_app . debug ,
2024-01-11 15:34:08 -08:00
moderating_communities = moderating_communities ( current_user . get_id ( ) ) ,
2024-02-01 18:30:03 -08:00
joined_communities = joined_communities ( current_user . get_id ( ) ) ,
2024-05-30 02:54:25 -07:00
menu_topics = menu_topics ( ) ,
2024-02-01 18:30:03 -08:00
site = g . site
2024-01-11 15:34:08 -08:00
)
2023-11-03 01:59:48 -07:00
2024-09-29 14:31:30 -07:00
2024-09-26 11:46:04 -07:00
@celery.task
def import_bans_task ( filename ) :
contents = file_get_contents ( filename )
contents_json = json . loads ( contents )
# import allowed_instances
if get_setting ( ' use_allowlist ' ) :
# check for allowed_instances existing and being more than 0 entries
instances_allowed = contents_json [ ' allowed_instances ' ]
2024-09-26 16:36:49 -07:00
if isinstance ( instance_allowed , list ) and len ( instance_allowed ) > 0 :
2024-09-26 11:46:04 -07:00
# get the existing allows and their domains
already_allowed_instances = [ ]
already_allowed = AllowedInstances . query . all ( )
if len ( already_allowed ) > 0 :
2024-09-29 14:31:30 -07:00
for already_allowed in already_allowed :
already_allowed_instances . append ( already_allowed . domain )
2024-09-26 11:46:04 -07:00
# loop through the instances_allowed
2024-09-29 14:31:30 -07:00
for allowed_instance in instances_allowed :
2024-09-26 11:46:04 -07:00
# check if we have already allowed this instance
2024-09-29 14:31:30 -07:00
if allowed_instance in already_allowed_instances :
2024-09-26 11:46:04 -07:00
continue
else :
# allow the instance
2024-09-29 14:31:30 -07:00
db . session . add ( AllowedInstances ( domain = allowed_instance ) )
2024-09-26 15:28:27 -07:00
# commit to the db
db . session . commit ( )
2024-09-26 11:46:04 -07:00
# import banned_instances
else :
# check for banned_instances existing and being more than 0 entries
instance_bans = contents_json [ ' banned_instances ' ]
2024-09-26 16:36:49 -07:00
if isinstance ( instance_bans , list ) and len ( instance_bans ) > 0 :
2024-09-26 11:46:04 -07:00
# get the existing bans and their domains
already_banned_instances = [ ]
already_banned = BannedInstances . query . all ( )
if len ( already_banned ) > 0 :
for ab in already_banned :
already_banned_instances . append ( ab . domain )
# loop through the instance_bans
2024-09-29 14:31:30 -07:00
for instance_ban in instance_bans :
2024-09-26 11:46:04 -07:00
# check if we have already banned this instance
2024-09-29 14:31:30 -07:00
if instance_ban in already_banned_instances :
2024-09-26 11:46:04 -07:00
continue
else :
# ban the domain
2024-09-29 14:31:30 -07:00
db . session . add ( BannedInstances ( domain = instance_ban ) )
2024-09-26 15:28:27 -07:00
# commit to the db
db . session . commit ( )
2024-09-26 11:46:04 -07:00
# import banned_domains
# check for banned_domains existing and being more than 0 entries
domain_bans = contents_json [ ' banned_domains ' ]
2024-09-26 16:36:49 -07:00
if isinstance ( domain_bans , list ) and len ( domain_bans ) > 0 :
2024-09-26 11:46:04 -07:00
# get the existing bans and their domains
already_banned_domains = [ ]
already_banned = Domain . query . filter_by ( banned = True ) . all ( )
if len ( already_banned ) > 0 :
for ab in already_banned :
already_banned_domains . append ( ab . name )
# loop through the domain_bans
2024-09-29 14:31:30 -07:00
for domain_ban in domain_bans :
2024-09-26 11:46:04 -07:00
# check if we have already banned this domain
2024-09-29 14:31:30 -07:00
if domain_ban in already_banned_domains :
2024-09-26 11:46:04 -07:00
continue
else :
# ban the domain
2024-09-29 14:31:30 -07:00
db . session . add ( Domain ( name = domain_ban , banned = True ) )
2024-09-26 15:28:27 -07:00
# commit to the db
db . session . commit ( )
2024-09-26 11:46:04 -07:00
# import banned_tags
# check for banned_tags existing and being more than 0 entries
tag_bans = contents_json [ ' banned_tags ' ]
2024-09-26 16:36:49 -07:00
if isinstance ( tag_bans , list ) and len ( tag_bans ) > 0 :
2024-09-26 11:46:04 -07:00
# get the existing bans and their domains
already_banned_tags = [ ]
already_banned = Tag . query . filter_by ( banned = True ) . all ( )
if len ( already_banned ) > 0 :
for ab in already_banned :
already_banned_tags . append ( ab . name )
# loop through the tag_bans
2024-09-29 14:31:30 -07:00
for tag_ban in tag_bans :
2024-09-26 11:46:04 -07:00
# check if we have already banned this tag
2024-09-29 14:31:30 -07:00
if tag_ban [ ' name ' ] in already_banned_tags :
2024-09-26 11:46:04 -07:00
continue
else :
# ban the domain
2024-09-29 14:31:30 -07:00
db . session . add ( Tag ( name = tag_ban [ ' name ' ] , display_as = tag_ban [ ' display_as ' ] , banned = True ) )
2024-09-26 15:28:27 -07:00
# commit to the db
db . session . commit ( )
2024-09-26 11:46:04 -07:00
# import banned_users
# check for banned_users existing and being more than 0 entries
user_bans = contents_json [ ' banned_users ' ]
2024-09-26 16:36:49 -07:00
if isinstance ( user_bans , list ) and len ( user_bans ) > 0 :
2024-09-26 11:46:04 -07:00
# get the existing bans and their domains
already_banned_users = [ ]
already_banned = User . query . filter_by ( banned = True ) . all ( )
if len ( already_banned ) > 0 :
for ab in already_banned :
already_banned_users . append ( ab . ap_id )
# loop through the user_bans
2024-09-29 14:31:30 -07:00
for user_ban in user_bans :
2024-09-26 11:46:04 -07:00
# check if we have already banned this user
2024-09-29 14:31:30 -07:00
if user_ban in already_banned_users :
2024-09-26 11:46:04 -07:00
continue
else :
# ban the user
2024-09-29 14:31:30 -07:00
db . session . add ( User ( user_name = user_ban . split ( ' @ ' ) [ 0 ] , ap_id = user_ban , banned = True ) )
2024-09-26 15:28:27 -07:00
# commit to the db
db . session . commit ( )
2024-09-26 11:46:04 -07:00
2024-09-29 14:31:30 -07:00
2023-11-24 01:32:46 -08:00
@bp.route ( ' /activities ' , methods = [ ' GET ' ] )
@login_required
@permission_required ( ' change instance settings ' )
def admin_activities ( ) :
2023-11-26 02:21:04 -08:00
db . session . query ( ActivityPubLog ) . filter (
2023-12-16 03:12:49 -08:00
ActivityPubLog . created_at < utcnow ( ) - timedelta ( days = 3 ) ) . delete ( )
2023-11-26 02:21:04 -08:00
db . session . commit ( )
2023-12-22 14:32:22 -08:00
page = request . args . get ( ' page ' , 1 , type = int )
2024-06-07 07:34:47 -07:00
result_filter = request . args . get ( ' result ' , type = str )
2024-08-19 12:36:39 -07:00
direction_filter = request . args . get ( ' direction ' , type = str )
2023-12-22 14:32:22 -08:00
2024-08-19 12:36:39 -07:00
activities = ActivityPubLog . query . order_by ( desc ( ActivityPubLog . created_at ) )
2024-06-07 07:34:47 -07:00
if result_filter :
2024-08-19 12:36:39 -07:00
activities = activities . filter ( ActivityPubLog . result == result_filter )
if direction_filter :
activities = activities . filter ( ActivityPubLog . direction == direction_filter )
2023-12-22 14:32:22 -08:00
2024-08-19 12:36:39 -07:00
activities = activities . paginate ( page = page , per_page = 1000 , error_out = False )
next_url = url_for ( ' admin.admin_activities ' , page = activities . next_num , result = result_filter , direction = direction_filter ) if activities . has_next else None
prev_url = url_for ( ' admin.admin_activities ' , page = activities . prev_num , result = result_filter , direction = direction_filter ) if activities . has_prev and page != 1 else None
2023-12-22 14:32:22 -08:00
return render_template ( ' admin/activities.html ' , title = _ ( ' ActivityPub Log ' ) , next_url = next_url , prev_url = prev_url ,
2024-02-01 18:30:03 -08:00
activities = activities ,
site = g . site )
2023-12-22 14:32:22 -08:00
@bp.route ( ' /activity_json/<int:activity_id> ' )
@login_required
@permission_required ( ' change instance settings ' )
def activity_json ( activity_id ) :
activity = ActivityPubLog . query . get_or_404 ( activity_id )
return render_template ( ' admin/activity_json.html ' , title = _ ( ' Activity JSON ' ) ,
2024-10-02 00:29:09 -07:00
activity_json_data = activity . activity_json , activity = activity ,
2024-02-01 18:30:03 -08:00
current_app = current_app ,
site = g . site )
2023-12-25 15:36:20 -08:00
@bp.route ( ' /activity_json/<int:activity_id>/replay ' )
@login_required
@permission_required ( ' change instance settings ' )
def activity_replay ( activity_id ) :
activity = ActivityPubLog . query . get_or_404 ( activity_id )
2023-12-28 20:32:35 -08:00
request_json = json . loads ( activity . activity_json )
2024-11-24 14:01:11 -08:00
replay_inbox_request ( request_json )
2023-12-25 15:36:20 -08:00
return ' Ok '
2023-12-30 15:09:20 -08:00
@bp.route ( ' /communities ' , methods = [ ' GET ' ] )
@login_required
@permission_required ( ' administer all communities ' )
def admin_communities ( ) :
page = request . args . get ( ' page ' , 1 , type = int )
2024-01-06 16:45:02 -08:00
search = request . args . get ( ' search ' , ' ' )
2023-12-30 15:09:20 -08:00
2024-03-12 20:40:20 -07:00
communities = Community . query
2024-01-06 16:45:02 -08:00
if search :
communities = communities . filter ( Community . title . ilike ( f " % { search } % " ) )
communities = communities . order_by ( Community . title ) . paginate ( page = page , per_page = 1000 , error_out = False )
2023-12-30 15:09:20 -08:00
next_url = url_for ( ' admin.admin_communities ' , page = communities . next_num ) if communities . has_next else None
prev_url = url_for ( ' admin.admin_communities ' , page = communities . prev_num ) if communities . has_prev and page != 1 else None
return render_template ( ' admin/communities.html ' , title = _ ( ' Communities ' ) , next_url = next_url , prev_url = prev_url ,
2024-01-11 15:34:08 -08:00
communities = communities , moderating_communities = moderating_communities ( current_user . get_id ( ) ) ,
2024-02-01 18:30:03 -08:00
joined_communities = joined_communities ( current_user . get_id ( ) ) ,
2024-05-30 02:54:25 -07:00
menu_topics = menu_topics ( ) ,
2024-02-01 18:30:03 -08:00
site = g . site )
2023-12-30 15:09:20 -08:00
2024-05-18 16:41:02 -07:00
@bp.route ( ' /communities/no-topic ' , methods = [ ' GET ' ] )
@login_required
@permission_required ( ' administer all communities ' )
def admin_communities_no_topic ( ) :
page = request . args . get ( ' page ' , 1 , type = int )
search = request . args . get ( ' search ' , ' ' )
communities = Community . query . filter ( Community . topic_id == None )
if search :
communities = communities . filter ( Community . title . ilike ( f " % { search } % " ) )
2024-05-18 16:59:41 -07:00
communities = communities . order_by ( - Community . post_count ) . paginate ( page = page , per_page = 1000 , error_out = False )
2024-05-18 16:41:02 -07:00
next_url = url_for ( ' admin.admin_communities_no_topic ' , page = communities . next_num ) if communities . has_next else None
prev_url = url_for ( ' admin.admin_communities_no_topic ' , page = communities . prev_num ) if communities . has_prev and page != 1 else None
return render_template ( ' admin/communities.html ' , title = _ ( ' Communities with no topic ' ) , next_url = next_url , prev_url = prev_url ,
communities = communities , moderating_communities = moderating_communities ( current_user . get_id ( ) ) ,
joined_communities = joined_communities ( current_user . get_id ( ) ) ,
2024-05-30 02:54:25 -07:00
menu_topics = menu_topics ( ) ,
2024-05-18 16:41:02 -07:00
site = g . site )
2023-12-30 15:09:20 -08:00
@bp.route ( ' /community/<int:community_id>/edit ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
@permission_required ( ' administer all communities ' )
def admin_community_edit ( community_id ) :
form = EditCommunityForm ( )
community = Community . query . get_or_404 ( community_id )
2024-10-06 19:21:24 -07:00
old_topic_id = community . topic_id if community . topic_id else None
2024-03-03 15:13:14 -08:00
form . topic . choices = topics_for_form ( 0 )
2024-05-08 02:07:22 -07:00
form . languages . choices = languages_for_form ( )
2023-12-30 15:09:20 -08:00
if form . validate_on_submit ( ) :
community . name = form . url . data
community . title = form . title . data
community . description = form . description . data
2024-01-04 01:08:32 -08:00
community . description_html = markdown_to_html ( form . description . data )
2023-12-30 15:09:20 -08:00
community . rules = form . rules . data
2024-01-04 01:08:32 -08:00
community . rules_html = markdown_to_html ( form . rules . data )
2023-12-30 15:09:20 -08:00
community . nsfw = form . nsfw . data
2024-03-12 20:40:20 -07:00
community . banned = form . banned . data
2024-01-01 22:41:00 -08:00
community . local_only = form . local_only . data
community . restricted_to_mods = form . restricted_to_mods . data
community . new_mods_wanted = form . new_mods_wanted . data
2023-12-30 15:09:20 -08:00
community . show_popular = form . show_popular . data
community . show_all = form . show_all . data
community . low_quality = form . low_quality . data
community . content_retention = form . content_retention . data
2024-01-03 19:00:19 -08:00
community . topic_id = form . topic . data if form . topic . data != 0 else None
2024-01-20 18:44:13 -08:00
community . default_layout = form . default_layout . data
2024-04-18 01:51:08 -07:00
community . posting_warning = form . posting_warning . data
2024-05-08 02:07:22 -07:00
community . ignore_remote_language = form . ignore_remote_language . data
2024-03-03 15:13:14 -08:00
2023-12-30 15:09:20 -08:00
icon_file = request . files [ ' icon_file ' ]
if icon_file and icon_file . filename != ' ' :
if community . icon_id :
community . icon . delete_from_disk ( )
file = save_icon_file ( icon_file )
if file :
community . icon = file
banner_file = request . files [ ' banner_file ' ]
if banner_file and banner_file . filename != ' ' :
if community . image_id :
community . image . delete_from_disk ( )
file = save_banner_file ( banner_file )
if file :
community . image = file
2024-01-03 19:00:19 -08:00
2024-05-08 02:07:22 -07:00
# Languages of the community
db . session . execute ( text ( ' DELETE FROM " community_language " WHERE community_id = :community_id ' ) ,
{ ' community_id ' : community_id } )
for language_choice in form . languages . data :
community . languages . append ( Language . query . get ( language_choice ) )
# Always include the undetermined language, so posts with no language will be accepted
community . languages . append ( Language . query . filter ( Language . code == ' und ' ) . first ( ) )
2023-12-30 15:09:20 -08:00
db . session . commit ( )
2024-10-06 19:21:24 -07:00
if community . topic_id != old_topic_id :
if community . topic_id :
community . topic . num_communities = community . topic . communities . count ( )
if old_topic_id :
topic = Topic . query . get ( old_topic_id )
if topic :
topic . num_communities = topic . communities . count ( )
db . session . commit ( )
2023-12-30 15:09:20 -08:00
flash ( _ ( ' Saved ' ) )
return redirect ( url_for ( ' admin.admin_communities ' ) )
else :
if not community . is_local ( ) :
flash ( _ ( ' This is a remote community - most settings here will be regularly overwritten with data from the original server. ' ) , ' warning ' )
form . url . data = community . name
form . title . data = community . title
form . description . data = community . description
form . rules . data = community . rules
form . nsfw . data = community . nsfw
2024-03-12 20:40:20 -07:00
form . banned . data = community . banned
2024-01-01 22:41:00 -08:00
form . local_only . data = community . local_only
form . new_mods_wanted . data = community . new_mods_wanted
form . restricted_to_mods . data = community . restricted_to_mods
2023-12-30 15:09:20 -08:00
form . show_popular . data = community . show_popular
form . show_all . data = community . show_all
form . low_quality . data = community . low_quality
form . content_retention . data = community . content_retention
2024-01-03 19:00:19 -08:00
form . topic . data = community . topic_id if community . topic_id else None
2024-01-20 18:44:13 -08:00
form . default_layout . data = community . default_layout
2024-04-18 01:51:08 -07:00
form . posting_warning . data = community . posting_warning
2024-05-08 02:07:22 -07:00
form . languages . data = community . language_ids ( )
form . ignore_remote_language . data = community . ignore_remote_language
2024-01-11 15:34:08 -08:00
return render_template ( ' admin/edit_community.html ' , title = _ ( ' Edit community ' ) , form = form , community = community ,
moderating_communities = moderating_communities ( current_user . get_id ( ) ) ,
2024-02-01 18:30:03 -08:00
joined_communities = joined_communities ( current_user . get_id ( ) ) ,
2024-05-30 02:54:25 -07:00
menu_topics = menu_topics ( ) ,
2024-02-01 18:30:03 -08:00
site = g . site
2024-01-11 15:34:08 -08:00
)
2023-12-30 15:09:20 -08:00
@bp.route ( ' /community/<int:community_id>/delete ' , methods = [ ' GET ' ] )
@login_required
@permission_required ( ' administer all communities ' )
def admin_community_delete ( community_id ) :
2023-12-31 14:38:24 -08:00
community = Community . query . get_or_404 ( community_id )
2024-08-09 12:27:10 -07:00
2023-12-31 14:38:24 -08:00
community . banned = True # Unsubscribing everyone could take a long time so until that is completed hide this community from the UI by banning it.
community . last_active = utcnow ( )
db . session . commit ( )
unsubscribe_everyone_then_delete ( community . id )
2024-08-09 12:27:10 -07:00
reason = f " Community { community . name } deleted by { current_user . user_name } "
add_to_modlog ( ' delete_community ' , reason = reason )
2023-12-31 14:38:24 -08:00
flash ( _ ( ' Community deleted ' ) )
return redirect ( url_for ( ' admin.admin_communities ' ) )
def unsubscribe_everyone_then_delete ( community_id ) :
if current_app . debug :
unsubscribe_everyone_then_delete_task ( community_id )
else :
unsubscribe_everyone_then_delete_task . delay ( community_id )
@celery.task
def unsubscribe_everyone_then_delete_task ( community_id ) :
community = Community . query . get_or_404 ( community_id )
if not community . is_local ( ) :
members = CommunityMember . query . filter_by ( community_id = community_id ) . all ( )
for member in members :
user = User . query . get ( member . user_id )
2023-12-31 17:49:15 -08:00
unsubscribe_from_community ( community , user )
else :
# todo: federate delete of local community out to all following instances
. . .
2023-12-31 14:38:24 -08:00
sleep ( 5 )
community . delete_dependencies ( )
db . session . delete ( community ) # todo: when a remote community is deleted it will be able to be re-created by using the 'Add remote' function. Not ideal. Consider soft-delete.
2024-01-06 12:29:36 -08:00
db . session . commit ( )
2023-12-31 17:49:15 -08:00
2024-01-03 19:00:19 -08:00
@bp.route ( ' /topics ' , methods = [ ' GET ' ] )
@login_required
@permission_required ( ' administer all communities ' )
def admin_topics ( ) :
2024-03-03 15:13:14 -08:00
topics = topic_tree ( )
2024-01-11 15:34:08 -08:00
return render_template ( ' admin/topics.html ' , title = _ ( ' Topics ' ) , topics = topics ,
moderating_communities = moderating_communities ( current_user . get_id ( ) ) ,
2024-02-01 18:30:03 -08:00
joined_communities = joined_communities ( current_user . get_id ( ) ) ,
2024-05-30 02:54:25 -07:00
menu_topics = menu_topics ( ) ,
2024-02-01 18:30:03 -08:00
site = g . site
2024-01-11 15:34:08 -08:00
)
2024-01-03 19:00:19 -08:00
@bp.route ( ' /topic/add ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
@permission_required ( ' administer all communities ' )
def admin_topic_add ( ) :
form = EditTopicForm ( )
2024-03-03 15:13:14 -08:00
form . parent_id . choices = topics_for_form ( 0 )
2024-01-03 19:00:19 -08:00
if form . validate_on_submit ( ) :
2024-11-07 18:09:24 -08:00
topic = Topic ( name = form . name . data , machine_name = slugify ( form . machine_name . data . strip ( ) ) , num_communities = 0 ,
show_posts_in_children = form . show_posts_in_children . data )
2024-03-03 15:13:14 -08:00
if form . parent_id . data :
topic . parent_id = form . parent_id . data
else :
topic . parent_id = None
2024-01-03 19:00:19 -08:00
db . session . add ( topic )
db . session . commit ( )
2024-05-31 02:55:55 -07:00
cache . delete_memoized ( menu_topics )
2024-03-03 15:13:14 -08:00
2024-01-03 19:00:19 -08:00
flash ( _ ( ' Saved ' ) )
return redirect ( url_for ( ' admin.admin_topics ' ) )
2024-01-11 15:34:08 -08:00
return render_template ( ' admin/edit_topic.html ' , title = _ ( ' Add topic ' ) , form = form ,
moderating_communities = moderating_communities ( current_user . get_id ( ) ) ,
2024-02-01 18:30:03 -08:00
joined_communities = joined_communities ( current_user . get_id ( ) ) ,
2024-05-30 02:54:25 -07:00
menu_topics = menu_topics ( ) ,
2024-02-01 18:30:03 -08:00
site = g . site
2024-01-11 15:34:08 -08:00
)
2024-01-03 19:00:19 -08:00
@bp.route ( ' /topic/<int:topic_id>/edit ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
@permission_required ( ' administer all communities ' )
def admin_topic_edit ( topic_id ) :
form = EditTopicForm ( )
topic = Topic . query . get_or_404 ( topic_id )
2024-03-03 15:13:14 -08:00
form . parent_id . choices = topics_for_form ( topic_id )
2024-01-03 19:00:19 -08:00
if form . validate_on_submit ( ) :
topic . name = form . name . data
2024-03-03 15:13:14 -08:00
topic . num_communities = topic . communities . count ( )
2024-01-29 01:18:06 -08:00
topic . machine_name = form . machine_name . data
2024-11-07 18:09:24 -08:00
topic . show_posts_in_children = form . show_posts_in_children . data
2024-03-03 15:13:14 -08:00
if form . parent_id . data :
topic . parent_id = form . parent_id . data
else :
topic . parent_id = None
2024-01-03 19:00:19 -08:00
db . session . commit ( )
2024-05-31 02:55:55 -07:00
cache . delete_memoized ( menu_topics )
2024-01-03 19:00:19 -08:00
flash ( _ ( ' Saved ' ) )
return redirect ( url_for ( ' admin.admin_topics ' ) )
else :
form . name . data = topic . name
2024-01-29 01:18:06 -08:00
form . machine_name . data = topic . machine_name
2024-03-03 15:13:14 -08:00
form . parent_id . data = topic . parent_id
2024-11-07 18:09:24 -08:00
form . show_posts_in_children . data = topic . show_posts_in_children
2024-01-11 15:34:08 -08:00
return render_template ( ' admin/edit_topic.html ' , title = _ ( ' Edit topic ' ) , form = form , topic = topic ,
moderating_communities = moderating_communities ( current_user . get_id ( ) ) ,
2024-02-01 18:30:03 -08:00
joined_communities = joined_communities ( current_user . get_id ( ) ) ,
2024-05-30 02:54:25 -07:00
menu_topics = menu_topics ( ) ,
2024-02-01 18:30:03 -08:00
site = g . site
2024-01-11 15:34:08 -08:00
)
2024-01-03 19:00:19 -08:00
@bp.route ( ' /topic/<int:topic_id>/delete ' , methods = [ ' GET ' ] )
@login_required
@permission_required ( ' administer all communities ' )
def admin_topic_delete ( topic_id ) :
topic = Topic . query . get_or_404 ( topic_id )
topic . num_communities = topic . communities . count ( )
if topic . num_communities == 0 :
db . session . delete ( topic )
flash ( _ ( ' Topic deleted ' ) )
else :
flash ( _ ( ' Cannot delete topic with communities assigned to it. ' , ' error ' ) )
db . session . commit ( )
2024-05-31 02:55:55 -07:00
cache . delete_memoized ( menu_topics )
2024-01-03 19:00:19 -08:00
return redirect ( url_for ( ' admin.admin_topics ' ) )
2023-12-31 17:49:15 -08:00
@bp.route ( ' /users ' , methods = [ ' GET ' ] )
@login_required
@permission_required ( ' administer all users ' )
def admin_users ( ) :
page = request . args . get ( ' page ' , 1 , type = int )
2024-01-01 19:07:41 -08:00
search = request . args . get ( ' search ' , ' ' )
local_remote = request . args . get ( ' local_remote ' , ' ' )
2023-12-31 17:49:15 -08:00
2024-01-01 19:07:41 -08:00
users = User . query . filter_by ( deleted = False )
if local_remote == ' local ' :
users = users . filter_by ( ap_id = None )
if local_remote == ' remote ' :
users = users . filter ( User . ap_id != None )
2024-01-06 17:45:16 -08:00
if search :
users = users . filter ( User . email . ilike ( f " % { search } % " ) )
2024-01-01 19:07:41 -08:00
users = users . order_by ( User . user_name ) . paginate ( page = page , per_page = 1000 , error_out = False )
2023-12-31 17:49:15 -08:00
next_url = url_for ( ' admin.admin_users ' , page = users . next_num ) if users . has_next else None
prev_url = url_for ( ' admin.admin_users ' , page = users . prev_num ) if users . has_prev and page != 1 else None
2024-01-01 19:07:41 -08:00
return render_template ( ' admin/users.html ' , title = _ ( ' Users ' ) , next_url = next_url , prev_url = prev_url , users = users ,
2024-01-11 15:34:08 -08:00
local_remote = local_remote , search = search ,
moderating_communities = moderating_communities ( current_user . get_id ( ) ) ,
2024-02-26 09:03:08 -08:00
joined_communities = joined_communities ( current_user . get_id ( ) ) ,
2024-05-30 02:54:25 -07:00
menu_topics = menu_topics ( ) ,
2024-02-26 09:03:08 -08:00
site = g . site
)
@bp.route ( ' /users/trash ' , methods = [ ' GET ' ] )
@login_required
@permission_required ( ' administer all users ' )
def admin_users_trash ( ) :
page = request . args . get ( ' page ' , 1 , type = int )
search = request . args . get ( ' search ' , ' ' )
local_remote = request . args . get ( ' local_remote ' , ' ' )
2024-04-11 14:59:06 -07:00
type = request . args . get ( ' type ' , ' bad_rep ' )
2024-02-26 09:03:08 -08:00
users = User . query . filter_by ( deleted = False )
if local_remote == ' local ' :
users = users . filter_by ( ap_id = None )
if local_remote == ' remote ' :
users = users . filter ( User . ap_id != None )
if search :
users = users . filter ( User . email . ilike ( f " % { search } % " ) )
2024-04-11 14:45:14 -07:00
2024-04-11 14:59:06 -07:00
if type == ' ' or type == ' bad_rep ' :
2024-06-11 16:47:04 -07:00
users = users . filter ( User . last_seen > utcnow ( ) - timedelta ( days = 7 ) )
2024-04-11 14:59:06 -07:00
users = users . filter ( User . reputation < - 10 )
users = users . order_by ( User . reputation ) . paginate ( page = page , per_page = 1000 , error_out = False )
elif type == ' bad_attitude ' :
2024-06-11 16:47:04 -07:00
users = users . filter ( User . last_seen > utcnow ( ) - timedelta ( days = 7 ) )
2024-06-11 17:12:44 -07:00
users = users . filter ( User . attitude < 0.0 ) . filter ( User . reputation < - 10 )
2024-06-11 16:30:43 -07:00
users = users . order_by ( User . attitude ) . paginate ( page = page , per_page = 1000 , error_out = False )
2024-02-26 09:03:08 -08:00
2024-06-11 16:30:43 -07:00
next_url = url_for ( ' admin.admin_users_trash ' , page = users . next_num , search = search , local_remote = local_remote , type = type ) if users . has_next else None
prev_url = url_for ( ' admin.admin_users_trash ' , page = users . prev_num , search = search , local_remote = local_remote , type = type ) if users . has_prev and page != 1 else None
2024-02-26 09:03:08 -08:00
2024-06-11 16:47:04 -07:00
return render_template ( ' admin/users_trash.html ' , title = _ ( ' Problematic users ' ) , next_url = next_url , prev_url = prev_url , users = users ,
2024-04-11 14:59:06 -07:00
local_remote = local_remote , search = search , type = type ,
2024-02-26 09:03:08 -08:00
moderating_communities = moderating_communities ( current_user . get_id ( ) ) ,
2024-02-01 18:30:03 -08:00
joined_communities = joined_communities ( current_user . get_id ( ) ) ,
2024-05-30 02:54:25 -07:00
menu_topics = menu_topics ( ) ,
2024-02-01 18:30:03 -08:00
site = g . site
2024-01-11 15:34:08 -08:00
)
2023-12-31 17:49:15 -08:00
2024-02-28 21:20:12 -08:00
@bp.route ( ' /content/trash ' , methods = [ ' GET ' ] )
@login_required
@permission_required ( ' administer all users ' )
def admin_content_trash ( ) :
page = request . args . get ( ' page ' , 1 , type = int )
2024-11-25 05:06:56 -08:00
posts = Post . query . filter ( Post . posted_at > utcnow ( ) - timedelta ( days = 3 ) , Post . deleted == False , Post . down_votes > 0 ) . order_by ( Post . score )
2024-02-28 21:20:12 -08:00
posts = posts . paginate ( page = page , per_page = 100 , error_out = False )
next_url = url_for ( ' admin.admin_content_trash ' , page = posts . next_num ) if posts . has_next else None
prev_url = url_for ( ' admin.admin_content_trash ' , page = posts . prev_num ) if posts . has_prev and page != 1 else None
return render_template ( ' admin/posts.html ' , title = _ ( ' Bad posts ' ) , next_url = next_url , prev_url = prev_url , posts = posts ,
moderating_communities = moderating_communities ( current_user . get_id ( ) ) ,
joined_communities = joined_communities ( current_user . get_id ( ) ) ,
2024-05-30 02:54:25 -07:00
menu_topics = menu_topics ( ) ,
2024-02-28 21:20:12 -08:00
site = g . site
)
2024-05-07 00:44:39 -07:00
@bp.route ( ' /content/spam ' , methods = [ ' GET ' ] )
@login_required
@permission_required ( ' administer all users ' )
def admin_content_spam ( ) :
# Essentially the same as admin_content_trash() except only shows heavily downvoted posts by new users - who are usually spammers
page = request . args . get ( ' page ' , 1 , type = int )
replies_page = request . args . get ( ' replies_page ' , 1 , type = int )
posts = Post . query . join ( User , User . id == Post . user_id ) . \
filter ( User . created > utcnow ( ) - timedelta ( days = 3 ) ) . \
filter ( Post . posted_at > utcnow ( ) - timedelta ( days = 3 ) ) . \
2024-06-01 21:45:21 -07:00
filter ( Post . deleted == False ) . \
2024-05-07 00:44:39 -07:00
filter ( Post . score < = 0 ) . order_by ( Post . score )
posts = posts . paginate ( page = page , per_page = 100 , error_out = False )
post_replies = PostReply . query . join ( User , User . id == PostReply . user_id ) . \
filter ( User . created > utcnow ( ) - timedelta ( days = 3 ) ) . \
filter ( PostReply . posted_at > utcnow ( ) - timedelta ( days = 3 ) ) . \
2024-06-01 21:45:21 -07:00
filter ( PostReply . deleted == False ) . \
2024-05-07 00:44:39 -07:00
filter ( PostReply . score < = 0 ) . order_by ( PostReply . score )
post_replies = post_replies . paginate ( page = replies_page , per_page = 100 , error_out = False )
next_url = url_for ( ' admin.admin_content_spam ' , page = posts . next_num ) if posts . has_next else None
prev_url = url_for ( ' admin.admin_content_spam ' , page = posts . prev_num ) if posts . has_prev and page != 1 else None
next_url_replies = url_for ( ' admin.admin_content_spam ' , replies_page = post_replies . next_num ) if post_replies . has_next else None
prev_url_replies = url_for ( ' admin.admin_content_spam ' , replies_page = post_replies . prev_num ) if post_replies . has_prev and replies_page != 1 else None
return render_template ( ' admin/spam_posts.html ' , title = _ ( ' Likely spam ' ) ,
next_url = next_url , prev_url = prev_url ,
next_url_replies = next_url_replies , prev_url_replies = prev_url_replies ,
posts = posts , post_replies = post_replies ,
moderating_communities = moderating_communities ( current_user . get_id ( ) ) ,
joined_communities = joined_communities ( current_user . get_id ( ) ) ,
2024-05-30 02:54:25 -07:00
menu_topics = menu_topics ( ) ,
2024-06-01 21:45:21 -07:00
site = g . site
)
@bp.route ( ' /content/deleted ' , methods = [ ' GET ' ] )
@login_required
@permission_required ( ' administer all users ' )
def admin_content_deleted ( ) :
# Shows all soft deleted posts
page = request . args . get ( ' page ' , 1 , type = int )
replies_page = request . args . get ( ' replies_page ' , 1 , type = int )
posts = Post . query . \
filter ( Post . deleted == True ) . \
2024-11-25 04:53:00 -08:00
order_by ( desc ( Post . posted_at ) )
2024-06-01 21:45:21 -07:00
posts = posts . paginate ( page = page , per_page = 100 , error_out = False )
post_replies = PostReply . query . \
filter ( PostReply . deleted == True ) . \
2024-11-25 04:53:00 -08:00
order_by ( desc ( PostReply . posted_at ) )
2024-06-01 21:45:21 -07:00
post_replies = post_replies . paginate ( page = replies_page , per_page = 100 , error_out = False )
next_url = url_for ( ' admin.admin_content_deleted ' , page = posts . next_num ) if posts . has_next else None
prev_url = url_for ( ' admin.admin_content_deleted ' , page = posts . prev_num ) if posts . has_prev and page != 1 else None
next_url_replies = url_for ( ' admin.admin_content_deleted ' , replies_page = post_replies . next_num ) if post_replies . has_next else None
prev_url_replies = url_for ( ' admin.admin_content_deleted ' , replies_page = post_replies . prev_num ) if post_replies . has_prev and replies_page != 1 else None
return render_template ( ' admin/deleted_posts.html ' , title = _ ( ' Deleted content ' ) ,
next_url = next_url , prev_url = prev_url ,
next_url_replies = next_url_replies , prev_url_replies = prev_url_replies ,
posts = posts , post_replies = post_replies ,
moderating_communities = moderating_communities ( current_user . get_id ( ) ) ,
joined_communities = joined_communities ( current_user . get_id ( ) ) ,
menu_topics = menu_topics ( ) ,
2024-05-07 00:44:39 -07:00
site = g . site
)
2024-02-01 18:30:03 -08:00
@bp.route ( ' /approve_registrations ' , methods = [ ' GET ' ] )
@login_required
@permission_required ( ' approve registrations ' )
def admin_approve_registrations ( ) :
registrations = UserRegistration . query . filter_by ( status = 0 ) . order_by ( UserRegistration . created_at ) . all ( )
recently_approved = UserRegistration . query . filter_by ( status = 1 ) . order_by ( desc ( UserRegistration . approved_at ) ) . limit ( 30 )
return render_template ( ' admin/approve_registrations.html ' ,
registrations = registrations ,
recently_approved = recently_approved ,
site = g . site )
@bp.route ( ' /approve_registrations/<int:user_id>/approve ' , methods = [ ' GET ' ] )
@login_required
@permission_required ( ' approve registrations ' )
def admin_approve_registrations_approve ( user_id ) :
user = User . query . get_or_404 ( user_id )
registration = UserRegistration . query . filter_by ( status = 0 , user_id = user_id ) . first ( )
if registration :
registration . status = 1
registration . approved_at = utcnow ( )
registration . approved_by = current_user . id
db . session . commit ( )
if user . verified :
2024-06-26 04:29:49 -07:00
finalize_user_setup ( user )
send_welcome_email ( user , True )
2024-02-01 18:30:03 -08:00
flash ( _ ( ' Registration approved. ' ) )
return redirect ( url_for ( ' admin.admin_approve_registrations ' ) )
2023-12-31 17:49:15 -08:00
@bp.route ( ' /user/<int:user_id>/edit ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
@permission_required ( ' administer all users ' )
def admin_user_edit ( user_id ) :
form = EditUserForm ( )
user = User . query . get_or_404 ( user_id )
if form . validate_on_submit ( ) :
user . bot = form . bot . data
2024-02-18 18:01:53 -08:00
user . banned = form . banned . data
2024-12-01 18:03:16 -08:00
user . ban_posts = form . ban_posts . data
user . ban_comments = form . ban_comments . data
2024-06-28 08:04:06 -07:00
user . hide_nsfw = form . hide_nsfw . data
user . hide_nsfl = form . hide_nsfl . data
2024-06-26 04:29:49 -07:00
if form . verified . data and not user . verified :
finalize_user_setup ( user )
user . verified = form . verified . data
2024-03-22 11:08:28 -07:00
if form . remove_avatar . data and user . avatar_id :
file = File . query . get ( user . avatar_id )
file . delete_from_disk ( )
user . avatar_id = None
db . session . delete ( file )
if form . remove_banner . data and user . cover_id :
file = File . query . get ( user . cover_id )
file . delete_from_disk ( )
user . cover_id = None
db . session . delete ( file )
2024-02-27 13:27:31 -08:00
# Update user roles. The UI only lets the user choose 1 role but the DB structure allows for multiple roles per user.
2024-03-03 15:21:06 -08:00
db . session . execute ( text ( ' DELETE FROM user_role WHERE user_id = :user_id ' ) , { ' user_id ' : user . id } )
2024-02-27 13:27:31 -08:00
user . roles . append ( Role . query . get ( form . role . data ) )
2024-02-27 13:38:44 -08:00
if form . role . data == 4 :
flash ( _ ( " Permissions are cached for 50 seconds so new admin roles won ' t take effect immediately. " ) )
2024-02-27 13:27:31 -08:00
2023-12-31 17:49:15 -08:00
db . session . commit ( )
2024-04-19 21:26:33 -07:00
2023-12-31 17:49:15 -08:00
flash ( _ ( ' Saved ' ) )
2024-03-03 15:21:06 -08:00
return redirect ( url_for ( ' admin.admin_users ' , local_remote = ' local ' if user . is_local ( ) else ' remote ' ) )
2023-12-31 17:49:15 -08:00
else :
if not user . is_local ( ) :
flash ( _ ( ' This is a remote user - most settings here will be regularly overwritten with data from the original server. ' ) , ' warning ' )
form . bot . data = user . bot
2024-02-18 18:01:53 -08:00
form . verified . data = user . verified
form . banned . data = user . banned
2024-12-01 18:03:16 -08:00
form . ban_posts . data = user . ban_posts
form . ban_comments . data = user . ban_comments
2024-06-28 08:04:06 -07:00
form . hide_nsfw . data = user . hide_nsfw
form . hide_nsfl . data = user . hide_nsfl
2024-03-03 15:21:06 -08:00
if user . roles and user . roles . count ( ) > 0 :
2024-02-27 13:27:31 -08:00
form . role . data = user . roles [ 0 ] . id
2023-12-31 17:49:15 -08:00
2024-01-11 15:34:08 -08:00
return render_template ( ' admin/edit_user.html ' , title = _ ( ' Edit user ' ) , form = form , user = user ,
moderating_communities = moderating_communities ( current_user . get_id ( ) ) ,
2024-02-01 18:30:03 -08:00
joined_communities = joined_communities ( current_user . get_id ( ) ) ,
2024-05-30 02:54:25 -07:00
menu_topics = menu_topics ( ) ,
2024-02-01 18:30:03 -08:00
site = g . site
2024-01-11 15:34:08 -08:00
)
2023-12-31 17:49:15 -08:00
2024-02-27 08:04:07 -08:00
@bp.route ( ' /users/add ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
@permission_required ( ' administer all users ' )
def admin_users_add ( ) :
form = AddUserForm ( )
user = User ( )
if form . validate_on_submit ( ) :
user . user_name = form . user_name . data
2024-06-03 14:44:10 -07:00
user . title = form . user_name . data
2024-02-27 08:04:07 -08:00
user . set_password ( form . password . data )
user . about = form . about . data
user . email = form . email . data
user . about_html = markdown_to_html ( form . about . data )
user . matrix_user_id = form . matrix_user_id . data
user . bot = form . bot . data
profile_file = request . files [ ' profile_file ' ]
if profile_file and profile_file . filename != ' ' :
# remove old avatar
if user . avatar_id :
file = File . query . get ( user . avatar_id )
file . delete_from_disk ( )
user . avatar_id = None
db . session . delete ( file )
# add new avatar
file = save_icon_file ( profile_file , ' users ' )
if file :
user . avatar = file
banner_file = request . files [ ' banner_file ' ]
if banner_file and banner_file . filename != ' ' :
# remove old cover
if user . cover_id :
file = File . query . get ( user . cover_id )
file . delete_from_disk ( )
user . cover_id = None
db . session . delete ( file )
# add new cover
file = save_banner_file ( banner_file , ' users ' )
if file :
user . cover = file
user . newsletter = form . newsletter . data
user . ignore_bots = form . ignore_bots . data
2024-06-28 08:04:06 -07:00
user . hide_nsfw = form . hide_nsfw . data
user . hide_nsfl = form . hide_nsfl . data
2024-02-27 08:04:07 -08:00
2024-05-11 05:22:23 -07:00
user . instance_id = 1
2024-02-27 13:27:31 -08:00
user . roles . append ( Role . query . get ( form . role . data ) )
2024-02-27 08:04:07 -08:00
db . session . add ( user )
db . session . commit ( )
2024-06-26 04:29:49 -07:00
finalize_user_setup ( user )
2024-02-27 08:04:07 -08:00
flash ( _ ( ' User added ' ) )
return redirect ( url_for ( ' admin.admin_users ' , local_remote = ' local ' ) )
return render_template ( ' admin/add_user.html ' , title = _ ( ' Add user ' ) , form = form , user = user ,
moderating_communities = moderating_communities ( current_user . get_id ( ) ) ,
joined_communities = joined_communities ( current_user . get_id ( ) ) ,
2024-05-30 02:54:25 -07:00
menu_topics = menu_topics ( ) ,
2024-02-27 08:04:07 -08:00
site = g . site
)
2023-12-31 17:49:15 -08:00
@bp.route ( ' /user/<int:user_id>/delete ' , methods = [ ' GET ' ] )
@login_required
@permission_required ( ' administer all users ' )
def admin_user_delete ( user_id ) :
user = User . query . get_or_404 ( user_id )
user . banned = True # Unsubscribing everyone could take a long time so until that is completed hide this user from the UI by banning it.
user . last_active = utcnow ( )
2024-10-13 16:35:24 -07:00
user . deleted_by = current_user . id
2023-12-31 17:49:15 -08:00
db . session . commit ( )
if user . is_local ( ) :
unsubscribe_from_everything_then_delete ( user . id )
else :
user . deleted = True
user . delete_dependencies ( )
db . session . commit ( )
2024-07-07 00:01:52 -07:00
add_to_modlog ( ' delete_user ' , link_text = user . display_name ( ) , link = user . link ( ) )
2023-12-31 17:49:15 -08:00
flash ( _ ( ' User deleted ' ) )
return redirect ( url_for ( ' admin.admin_users ' ) )
2024-01-01 19:07:41 -08:00
@bp.route ( ' /reports ' , methods = [ ' GET ' ] )
@login_required
@permission_required ( ' administer all users ' )
def admin_reports ( ) :
2023-12-31 17:49:15 -08:00
2024-01-01 19:07:41 -08:00
page = request . args . get ( ' page ' , 1 , type = int )
search = request . args . get ( ' search ' , ' ' )
local_remote = request . args . get ( ' local_remote ' , ' ' )
2023-12-31 17:49:15 -08:00
2024-03-27 00:20:08 -07:00
reports = Report . query . filter ( or_ ( Report . status == REPORT_STATE_NEW , Report . status == REPORT_STATE_ESCALATED ) )
2024-01-01 19:07:41 -08:00
if local_remote == ' local ' :
2024-06-27 00:38:10 -07:00
reports = reports . filter_by ( source_instance_id = 1 )
2024-01-01 19:07:41 -08:00
if local_remote == ' remote ' :
2024-06-27 00:38:10 -07:00
reports = reports . filter ( Report . source_instance_id != 1 )
2024-01-01 19:07:41 -08:00
reports = reports . order_by ( desc ( Report . created_at ) ) . paginate ( page = page , per_page = 1000 , error_out = False )
2023-12-31 17:49:15 -08:00
2024-01-01 19:07:41 -08:00
next_url = url_for ( ' admin.admin_reports ' , page = reports . next_num ) if reports . has_next else None
prev_url = url_for ( ' admin.admin_reports ' , page = reports . prev_num ) if reports . has_prev and page != 1 else None
2023-12-31 17:49:15 -08:00
2024-01-01 19:07:41 -08:00
return render_template ( ' admin/reports.html ' , title = _ ( ' Reports ' ) , next_url = next_url , prev_url = prev_url , reports = reports ,
2024-01-11 15:34:08 -08:00
local_remote = local_remote , search = search ,
moderating_communities = moderating_communities ( current_user . get_id ( ) ) ,
2024-02-01 18:30:03 -08:00
joined_communities = joined_communities ( current_user . get_id ( ) ) ,
2024-05-30 02:54:25 -07:00
menu_topics = menu_topics ( ) ,
2024-02-01 18:30:03 -08:00
site = g . site
2024-02-23 23:01:38 -08:00
)
@bp.route ( ' /newsletter ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
@permission_required ( ' administer all users ' )
def newsletter ( ) :
form = SendNewsletterForm ( )
if form . validate_on_submit ( ) :
send_newsletter ( form )
flash ( ' Newsletter sent ' )
return redirect ( url_for ( ' admin.newsletter ' ) )
return render_template ( " admin/newsletter.html " , form = form , title = _ ( ' Send newsletter ' ) ,
moderating_communities = moderating_communities ( current_user . get_id ( ) ) ,
joined_communities = joined_communities ( current_user . get_id ( ) ) ,
2024-05-30 02:54:25 -07:00
menu_topics = menu_topics ( ) ,
2024-02-23 23:01:38 -08:00
site = g . site
)
2024-07-16 06:29:06 -07:00
@bp.route ( ' /permissions ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
@permission_required ( ' change instance settings ' )
def admin_permissions ( ) :
if request . method == ' POST ' :
permissions = db . session . execute ( text ( ' SELECT DISTINCT permission FROM " role_permission " ' ) ) . fetchall ( )
db . session . execute ( text ( ' DELETE FROM " role_permission " ' ) )
roles = [ 3 , 4 ] # 3 = Staff, 4 = Admin
for permission in permissions :
for role in roles :
if request . form . get ( f ' role_ { role } _ { permission [ 0 ] } ' ) :
db . session . add ( RolePermission ( role_id = role , permission = permission [ 0 ] ) )
db . session . commit ( )
flash ( _ ( ' Settings saved ' ) )
roles = Role . query . filter ( Role . id > 2 ) . order_by ( Role . weight ) . all ( )
permissions = db . session . execute ( text ( ' SELECT DISTINCT permission FROM " role_permission " ' ) ) . fetchall ( )
return render_template ( ' admin/permissions.html ' , title = _ ( ' Role permissions ' ) , roles = roles ,
permissions = permissions ,
moderating_communities = moderating_communities ( current_user . get_id ( ) ) ,
joined_communities = joined_communities ( current_user . get_id ( ) ) ,
menu_topics = menu_topics ( ) ,
site = g . site
2024-09-04 16:45:44 -07:00
)
2024-09-23 14:36:19 -07:00
2024-09-04 16:45:44 -07:00
@bp.route ( ' /instances ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
@permission_required ( ' change instance settings ' )
def admin_instances ( ) :
page = request . args . get ( ' page ' , 1 , type = int )
search = request . args . get ( ' search ' , ' ' )
2024-09-23 14:36:19 -07:00
filter = request . args . get ( ' filter ' , ' ' )
2024-09-04 16:45:44 -07:00
low_bandwidth = request . cookies . get ( ' low_bandwidth ' , ' 0 ' ) == ' 1 '
instances = Instance . query . order_by ( Instance . domain )
2024-09-04 19:45:17 -07:00
if search :
instances = instances . filter ( Instance . domain . ilike ( f " % { search } % " ) )
2024-09-23 14:36:19 -07:00
title = ' Instances '
if filter :
if filter == ' trusted ' :
instances = instances . filter ( Instance . trusted == True )
title = ' Trusted instances '
elif filter == ' online ' :
instances = instances . filter ( Instance . dormant == False , Instance . gone_forever == False )
title = ' Online instances '
elif filter == ' dormant ' :
instances = instances . filter ( Instance . dormant == True , Instance . gone_forever == False )
title = ' Dormant instances '
elif filter == ' gone_forever ' :
instances = instances . filter ( Instance . gone_forever == True )
title = ' Gone forever instances '
2024-09-04 19:45:17 -07:00
2024-09-04 16:45:44 -07:00
# Pagination
instances = instances . paginate ( page = page ,
per_page = 250 if current_user . is_authenticated and not low_bandwidth else 50 ,
error_out = False )
next_url = url_for ( ' admin.admin_instances ' , page = instances . next_num ) if instances . has_next else None
prev_url = url_for ( ' admin.admin_instances ' , page = instances . prev_num ) if instances . has_prev and page != 1 else None
return render_template ( ' admin/instances.html ' , instances = instances ,
2024-09-23 14:36:19 -07:00
title = _ ( title ) , search = search ,
2024-09-04 16:45:44 -07:00
next_url = next_url , prev_url = prev_url ,
2024-09-04 20:15:24 -07:00
low_bandwidth = low_bandwidth ,
2024-09-04 16:45:44 -07:00
moderating_communities = moderating_communities ( current_user . get_id ( ) ) ,
joined_communities = joined_communities ( current_user . get_id ( ) ) ,
menu_topics = menu_topics ( ) , site = g . site )
2024-09-23 16:33:49 -07:00
@bp.route ( ' /instance/<int:instance_id>/edit ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
@permission_required ( ' administer all communities ' )
def admin_instance_edit ( instance_id ) :
form = EditInstanceForm ( )
instance = Instance . query . get_or_404 ( instance_id )
if form . validate_on_submit ( ) :
instance . vote_weight = form . vote_weight . data
instance . dormant = form . dormant . data
instance . gone_forever = form . gone_forever . data
instance . trusted = form . trusted . data
instance . posting_warning = form . posting_warning . data
db . session . commit ( )
flash ( _ ( ' Saved ' ) )
return redirect ( url_for ( ' admin.admin_instances ' ) )
else :
form . vote_weight . data = instance . vote_weight
form . dormant . data = instance . dormant
form . gone_forever . data = instance . gone_forever
form . trusted . data = instance . trusted
form . posting_warning . data = instance . posting_warning
return render_template ( ' admin/edit_instance.html ' , title = _ ( ' Edit instance ' ) , form = form , instance = instance ,
moderating_communities = moderating_communities ( current_user . get_id ( ) ) ,
joined_communities = joined_communities ( current_user . get_id ( ) ) ,
menu_topics = menu_topics ( ) ,
site = g . site
2024-11-24 14:01:11 -08:00
)