mirror of
https://codeberg.org/rimu/pyfedi
synced 2025-01-23 19:36:56 -08:00
1646 lines
75 KiB
Python
1646 lines
75 KiB
Python
import os
|
|
import re
|
|
from datetime import timedelta
|
|
from time import sleep
|
|
from io import BytesIO
|
|
import json as python_json
|
|
|
|
from flask import request, flash, json, url_for, current_app, redirect, g, abort, send_file
|
|
from flask_login import login_required, current_user
|
|
from flask_babel import _
|
|
from slugify import slugify
|
|
from sqlalchemy import text, desc, or_
|
|
from PIL import Image
|
|
from urllib.parse import urlparse
|
|
|
|
from app import db, celery, cache
|
|
from app.activitypub.routes import process_inbox_request, process_delete_request, replay_inbox_request
|
|
from app.activitypub.signature import post_request, default_context
|
|
from app.activitypub.util import instance_allowed, instance_blocked, extract_domain_and_actor
|
|
from app.admin.forms import FederationForm, SiteMiscForm, SiteProfileForm, EditCommunityForm, EditUserForm, \
|
|
EditTopicForm, SendNewsletterForm, AddUserForm, PreLoadCommunitiesForm, ImportExportBannedListsForm, \
|
|
EditInstanceForm, RemoteInstanceScanForm
|
|
from app.admin.util import unsubscribe_from_everything_then_delete, unsubscribe_from_community, send_newsletter, \
|
|
topics_for_form
|
|
from app.community.util import save_icon_file, save_banner_file, search_for_community
|
|
from app.community.routes import do_subscribe
|
|
from app.constants import REPORT_STATE_NEW, REPORT_STATE_ESCALATED
|
|
from app.email import send_welcome_email
|
|
from app.models import AllowedInstances, BannedInstances, ActivityPubLog, utcnow, Site, Community, CommunityMember, \
|
|
User, Instance, File, Report, Topic, UserRegistration, Role, Post, PostReply, Language, RolePermission, Domain, \
|
|
Tag
|
|
from app.utils import render_template, permission_required, set_setting, get_setting, gibberish, markdown_to_html, \
|
|
moderating_communities, joined_communities, finalize_user_setup, theme_list, blocked_phrases, blocked_referrers, \
|
|
topic_tree, languages_for_form, menu_topics, ensure_directory_exists, add_to_modlog, get_request, file_get_contents
|
|
from app.admin import bp
|
|
|
|
|
|
@bp.route('/', methods=['GET', 'POST'])
|
|
@login_required
|
|
@permission_required('change instance settings')
|
|
def admin_home():
|
|
return render_template('admin/home.html', title=_('Admin'), moderating_communities=moderating_communities(current_user.get_id()),
|
|
joined_communities=joined_communities(current_user.get_id()),
|
|
menu_topics=menu_topics(),
|
|
site=g.site)
|
|
|
|
|
|
@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
|
|
site.description = form.description.data #tagline
|
|
site.about = form.about.data
|
|
site.sidebar = form.sidebar.data
|
|
site.legal_information = form.legal_information.data
|
|
site.updated = utcnow()
|
|
site.contact_email = form.contact_email.data
|
|
if site.id is None:
|
|
db.session.add(site)
|
|
# Save site icon
|
|
uploaded_icon = request.files['icon']
|
|
if uploaded_icon and uploaded_icon.filename != '':
|
|
allowed_extensions = ['.gif', '.jpg', '.jpeg', '.png', '.webp', '.svg']
|
|
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}')
|
|
if file_ext == '.svg':
|
|
delete_original = False
|
|
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
|
|
|
|
# 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}'
|
|
|
|
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}'
|
|
|
|
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}'
|
|
|
|
if delete_original:
|
|
os.unlink(f'app/static/media/{base_filename}{file_ext}')
|
|
|
|
db.session.commit()
|
|
set_setting('announcement', form.announcement.data)
|
|
flash('Settings saved.')
|
|
elif request.method == 'GET':
|
|
form.name.data = site.name
|
|
form.description.data = site.description
|
|
form.about.data = site.about
|
|
form.sidebar.data = site.sidebar
|
|
form.legal_information.data = site.legal_information
|
|
form.contact_email.data = site.contact_email
|
|
form.announcement.data = get_setting('announcement', '')
|
|
return render_template('admin/site.html', title=_('Site profile'), form=form,
|
|
moderating_communities=moderating_communities(current_user.get_id()),
|
|
joined_communities=joined_communities(current_user.get_id()),
|
|
menu_topics=menu_topics(),
|
|
site=g.site
|
|
)
|
|
|
|
|
|
@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()
|
|
form.default_theme.choices = theme_list()
|
|
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
|
|
site.auto_decline_referrers = form.auto_decline_referrers.data
|
|
site.log_activitypub_json = form.log_activitypub_json.data
|
|
site.show_inoculation_block = form.show_inoculation_block.data
|
|
site.updated = utcnow()
|
|
site.default_theme = form.default_theme.data
|
|
if site.id is None:
|
|
db.session.add(site)
|
|
db.session.commit()
|
|
cache.delete_memoized(blocked_referrers)
|
|
set_setting('public_modlog', form.public_modlog.data)
|
|
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
|
|
form.auto_decline_referrers.data = site.auto_decline_referrers
|
|
form.log_activitypub_json.data = site.log_activitypub_json
|
|
form.show_inoculation_block.data = site.show_inoculation_block
|
|
form.default_theme.data = site.default_theme if site.default_theme is not None else ''
|
|
form.public_modlog.data = get_setting('public_modlog', False)
|
|
return render_template('admin/misc.html', title=_('Misc settings'), form=form,
|
|
moderating_communities=moderating_communities(current_user.get_id()),
|
|
joined_communities=joined_communities(current_user.get_id()),
|
|
menu_topics=menu_topics(),
|
|
site=g.site
|
|
)
|
|
|
|
|
|
@bp.route('/federation', methods=['GET', 'POST'])
|
|
@login_required
|
|
@permission_required('change instance settings')
|
|
def admin_federation():
|
|
form = FederationForm()
|
|
preload_form = PreLoadCommunitiesForm()
|
|
ban_lists_form = ImportExportBannedListsForm()
|
|
remote_scan_form = RemoteInstanceScanForm()
|
|
|
|
# this is the pre-load communities button
|
|
if preload_form.pre_load_submit.data and preload_form.validate():
|
|
# how many communities to add
|
|
if preload_form.communities_num.data:
|
|
communities_to_add = preload_form.communities_num.data
|
|
else:
|
|
communities_to_add = 25
|
|
|
|
# pull down the community.full.json
|
|
resp = get_request('https://data.lemmyverse.net/data/community.full.json')
|
|
community_json = resp.json()
|
|
resp.close()
|
|
|
|
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 = []
|
|
|
|
for community in community_json:
|
|
total_count += 1
|
|
|
|
# sort out already known communities
|
|
if community['url'] in already_known:
|
|
already_known_count += 1
|
|
continue
|
|
|
|
# sort out the nsfw communities
|
|
elif community['nsfw']:
|
|
nsfw_count += 1
|
|
continue
|
|
|
|
# sort out any that have less than 100 posts
|
|
elif community['counts']['posts'] < 100:
|
|
low_content_count += 1
|
|
continue
|
|
|
|
# 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
|
|
continue
|
|
|
|
else:
|
|
candidate_communities.append(community)
|
|
|
|
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)))
|
|
|
|
# sort the list based on the users_active_week key
|
|
parsed_communities_sorted = sorted(candidate_communities, key=lambda c: c['counts']['users_active_week'], reverse=True)
|
|
|
|
# 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_to_add > len(parsed_communities_sorted):
|
|
communities_to_add = len(parsed_communities_sorted)
|
|
|
|
# make the list of urls
|
|
for i in range(communities_to_add):
|
|
community_urls_to_join.append(parsed_communities_sorted[i]['url'].lower())
|
|
|
|
# loop through the list and send off the follow requests
|
|
# use User #1, the first instance admin
|
|
|
|
# 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
|
|
|
|
user = User.query.get(1)
|
|
pre_load_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
|
|
# capture the messages returned by do_subscibe
|
|
# 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)
|
|
pre_load_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(pre_load_messages)))
|
|
else:
|
|
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)))
|
|
|
|
return redirect(url_for('admin.admin_federation'))
|
|
|
|
# this is the remote server scan
|
|
elif remote_scan_form.remote_scan_submit.data and remote_scan_form.validate():
|
|
# 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'
|
|
]
|
|
is_lemmy = False
|
|
is_mbin = False
|
|
|
|
|
|
# get the remote_url data
|
|
remote_url = remote_scan_form.remote_url.data
|
|
|
|
# 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:
|
|
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.'))
|
|
return redirect(url_for('admin.admin_federation'))
|
|
|
|
# check if it's a banned instance
|
|
# 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'))
|
|
|
|
# get dry run
|
|
dry_run = remote_scan_form.dry_run.data
|
|
|
|
# get the number of follows requested
|
|
communities_requested = remote_scan_form.communities_requested.data
|
|
|
|
# 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']
|
|
|
|
# if the instance is not running lemmy or mbin break for now as
|
|
# we dont yet support others for scanning
|
|
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."))
|
|
return redirect(url_for('admin.admin_federation'))
|
|
|
|
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)
|
|
|
|
# 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)
|
|
else:
|
|
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:
|
|
# hit the /api/magazines with a single call to get the stats for counts and num_requests
|
|
# mbin does not have the hard-coded limit, but lets stick with 50 to match lemmy
|
|
params = {"p":"1","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 number of requests to send
|
|
# num_requests = page_dict['pagination']['maxPage']
|
|
|
|
# loop through and send the right number of requests to the remote endpoint
|
|
# local_on_remote_instance = []
|
|
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
|
|
candidate_magazines = []
|
|
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:
|
|
candidate_magazines.append(magazine)
|
|
|
|
# testing
|
|
# flash(_(f"testing: candidate mags {len(candidate_magazines)}, m0: {candidate_magazines[0]}"))
|
|
# return redirect(url_for('admin.admin_federation'))
|
|
|
|
# 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_magazines):
|
|
magazines_to_add = len(candidate_magazines)
|
|
else:
|
|
magazines_to_add = communities_requested
|
|
|
|
# make the list of urls
|
|
for i in range(magazines_to_add):
|
|
community_urls_to_join.append(candidate_magazines[i]['apProfileId'].lower())
|
|
|
|
# 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}, \
|
|
Candidate Magazines based on filters: {len(candidate_magazines)}, \
|
|
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'))
|
|
|
|
|
|
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
|
|
# capture the messages returned by do_subscibe
|
|
# 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(
|
|
_('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)))
|
|
|
|
return redirect(url_for('admin.admin_federation'))
|
|
|
|
# this is the import bans button
|
|
elif ban_lists_form.import_submit.data and ban_lists_form.validate():
|
|
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
|
|
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'))
|
|
else:
|
|
flash(_(f'Ban imports requested, but no json provided.'))
|
|
return redirect(url_for('admin.admin_federation'))
|
|
|
|
# this is the export bans button
|
|
elif ban_lists_form.export_submit.data and ban_lists_form.validate():
|
|
|
|
ban_lists_dict = {}
|
|
|
|
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)
|
|
ban_lists_dict['allowed_instances'] = allowed_instances
|
|
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
|
|
|
|
# get banned_domains info
|
|
banned_domains = []
|
|
domain_bans = Domain.query.filter_by(banned=True).all()
|
|
if len(domain_bans) > 0:
|
|
for domain in domain_bans:
|
|
banned_domains.append(domain.name)
|
|
ban_lists_dict['banned_domains'] = banned_domains
|
|
|
|
# get banned_tags info
|
|
banned_tags = []
|
|
tag_bans = Tag.query.filter_by(banned=True).all()
|
|
if len(tag_bans) > 0:
|
|
for tag_ban in tag_bans:
|
|
tag_dict = {}
|
|
tag_dict['name'] = tag_ban.name
|
|
tag_dict['display_as'] = tag_ban.display_as
|
|
banned_tags.append(tag_dict)
|
|
ban_lists_dict['banned_tags'] = banned_tags
|
|
|
|
# get banned_users info
|
|
banned_users = []
|
|
user_bans = User.query.filter_by(banned=True).all()
|
|
if len(user_bans) > 0:
|
|
for user_ban in user_bans:
|
|
banned_users.append(user_ban.ap_id)
|
|
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')
|
|
|
|
# this is the main settings form
|
|
elif form.validate_on_submit():
|
|
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()))
|
|
cache.delete_memoized(instance_allowed, allow.strip())
|
|
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()))
|
|
cache.delete_memoized(instance_blocked, banned.strip())
|
|
g.site.blocked_phrases = form.blocked_phrases.data
|
|
set_setting('actor_blocked_words', form.blocked_actors.data)
|
|
cache.delete_memoized(blocked_phrases)
|
|
cache.delete_memoized(get_setting, 'actor_blocked_words')
|
|
db.session.commit()
|
|
|
|
flash(_('Admin settings saved'))
|
|
|
|
# this is just the regular page load
|
|
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])
|
|
form.blocked_phrases.data = g.site.blocked_phrases
|
|
form.blocked_actors.data = get_setting('actor_blocked_words', '88')
|
|
|
|
return render_template('admin/federation.html', title=_('Federation settings'),
|
|
form=form, preload_form=preload_form, ban_lists_form=ban_lists_form,
|
|
remote_scan_form=remote_scan_form, current_app_debug=current_app.debug,
|
|
moderating_communities=moderating_communities(current_user.get_id()),
|
|
joined_communities=joined_communities(current_user.get_id()),
|
|
menu_topics=menu_topics(),
|
|
site=g.site
|
|
)
|
|
|
|
|
|
@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']
|
|
if isinstance(instance_allowed, list) and len(instance_allowed) > 0:
|
|
# get the existing allows and their domains
|
|
already_allowed_instances = []
|
|
already_allowed = AllowedInstances.query.all()
|
|
if len(already_allowed) > 0:
|
|
for already_allowed in already_allowed:
|
|
already_allowed_instances.append(already_allowed.domain)
|
|
|
|
# loop through the instances_allowed
|
|
for allowed_instance in instances_allowed:
|
|
# check if we have already allowed this instance
|
|
if allowed_instance in already_allowed_instances:
|
|
continue
|
|
else:
|
|
# allow the instance
|
|
db.session.add(AllowedInstances(domain=allowed_instance))
|
|
# commit to the db
|
|
db.session.commit()
|
|
|
|
# import banned_instances
|
|
else:
|
|
# check for banned_instances existing and being more than 0 entries
|
|
instance_bans = contents_json['banned_instances']
|
|
if isinstance(instance_bans, list) and len(instance_bans) > 0:
|
|
# 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
|
|
for instance_ban in instance_bans:
|
|
# check if we have already banned this instance
|
|
if instance_ban in already_banned_instances:
|
|
continue
|
|
else:
|
|
# ban the domain
|
|
db.session.add(BannedInstances(domain=instance_ban))
|
|
# commit to the db
|
|
db.session.commit()
|
|
|
|
# import banned_domains
|
|
# check for banned_domains existing and being more than 0 entries
|
|
domain_bans = contents_json['banned_domains']
|
|
if isinstance(domain_bans, list) and len(domain_bans) > 0:
|
|
# 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
|
|
for domain_ban in domain_bans:
|
|
# check if we have already banned this domain
|
|
if domain_ban in already_banned_domains:
|
|
continue
|
|
else:
|
|
# ban the domain
|
|
db.session.add(Domain(name=domain_ban, banned=True))
|
|
# commit to the db
|
|
db.session.commit()
|
|
|
|
# import banned_tags
|
|
# check for banned_tags existing and being more than 0 entries
|
|
tag_bans = contents_json['banned_tags']
|
|
if isinstance(tag_bans, list) and len(tag_bans) > 0:
|
|
# 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
|
|
for tag_ban in tag_bans:
|
|
# check if we have already banned this tag
|
|
if tag_ban['name'] in already_banned_tags:
|
|
continue
|
|
else:
|
|
# ban the domain
|
|
db.session.add(Tag(name=tag_ban['name'], display_as=tag_ban['display_as'], banned=True))
|
|
# commit to the db
|
|
db.session.commit()
|
|
|
|
# import banned_users
|
|
# check for banned_users existing and being more than 0 entries
|
|
user_bans = contents_json['banned_users']
|
|
if isinstance(user_bans, list) and len(user_bans) > 0:
|
|
# 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
|
|
for user_ban in user_bans:
|
|
# check if we have already banned this user
|
|
if user_ban in already_banned_users:
|
|
continue
|
|
else:
|
|
# ban the user
|
|
db.session.add(User(user_name=user_ban.split('@')[0], ap_id=user_ban, banned=True))
|
|
# commit to the db
|
|
db.session.commit()
|
|
|
|
|
|
@bp.route('/activities', methods=['GET'])
|
|
@login_required
|
|
@permission_required('change instance settings')
|
|
def admin_activities():
|
|
db.session.query(ActivityPubLog).filter(
|
|
ActivityPubLog.created_at < utcnow() - timedelta(days=3)).delete()
|
|
db.session.commit()
|
|
|
|
page = request.args.get('page', 1, type=int)
|
|
result_filter = request.args.get('result', type=str)
|
|
direction_filter = request.args.get('direction', type=str)
|
|
|
|
activities = ActivityPubLog.query.order_by(desc(ActivityPubLog.created_at))
|
|
if result_filter:
|
|
activities = activities.filter(ActivityPubLog.result == result_filter)
|
|
if direction_filter:
|
|
activities = activities.filter(ActivityPubLog.direction == direction_filter)
|
|
|
|
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
|
|
|
|
return render_template('admin/activities.html', title=_('ActivityPub Log'), next_url=next_url, prev_url=prev_url,
|
|
activities=activities,
|
|
site=g.site)
|
|
|
|
|
|
@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'),
|
|
activity_json_data=activity.activity_json, activity=activity,
|
|
current_app=current_app,
|
|
site=g.site)
|
|
|
|
|
|
@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)
|
|
request_json = json.loads(activity.activity_json)
|
|
replay_inbox_request(request_json)
|
|
|
|
return 'Ok'
|
|
|
|
|
|
@bp.route('/communities', methods=['GET'])
|
|
@login_required
|
|
@permission_required('administer all communities')
|
|
def admin_communities():
|
|
|
|
page = request.args.get('page', 1, type=int)
|
|
search = request.args.get('search', '')
|
|
|
|
communities = Community.query
|
|
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)
|
|
|
|
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,
|
|
communities=communities, moderating_communities=moderating_communities(current_user.get_id()),
|
|
joined_communities=joined_communities(current_user.get_id()),
|
|
menu_topics=menu_topics(),
|
|
site=g.site)
|
|
|
|
|
|
@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}%"))
|
|
communities = communities.order_by(-Community.post_count).paginate(page=page, per_page=1000, error_out=False)
|
|
|
|
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()),
|
|
menu_topics=menu_topics(),
|
|
site=g.site)
|
|
|
|
|
|
@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)
|
|
old_topic_id = community.topic_id if community.topic_id else None
|
|
form.topic.choices = topics_for_form(0)
|
|
form.languages.choices = languages_for_form()
|
|
if form.validate_on_submit():
|
|
community.name = form.url.data
|
|
community.title = form.title.data
|
|
community.description = form.description.data
|
|
community.description_html = markdown_to_html(form.description.data)
|
|
community.rules = form.rules.data
|
|
community.rules_html = markdown_to_html(form.rules.data)
|
|
community.nsfw = form.nsfw.data
|
|
community.banned = form.banned.data
|
|
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
|
|
community.show_home = form.show_home.data
|
|
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
|
|
community.topic_id = form.topic.data if form.topic.data != 0 else None
|
|
community.default_layout = form.default_layout.data
|
|
community.posting_warning = form.posting_warning.data
|
|
community.ignore_remote_language = form.ignore_remote_language.data
|
|
|
|
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
|
|
|
|
# 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())
|
|
db.session.commit()
|
|
|
|
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()
|
|
|
|
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
|
|
form.banned.data = community.banned
|
|
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
|
|
form.show_home.data = community.show_home
|
|
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
|
|
form.topic.data = community.topic_id if community.topic_id else None
|
|
form.default_layout.data = community.default_layout
|
|
form.posting_warning.data = community.posting_warning
|
|
form.languages.data = community.language_ids()
|
|
form.ignore_remote_language.data = community.ignore_remote_language
|
|
return render_template('admin/edit_community.html', title=_('Edit community'), form=form, community=community,
|
|
moderating_communities=moderating_communities(current_user.get_id()),
|
|
joined_communities=joined_communities(current_user.get_id()),
|
|
menu_topics=menu_topics(),
|
|
site=g.site
|
|
)
|
|
|
|
|
|
@bp.route('/community/<int:community_id>/delete', methods=['GET'])
|
|
@login_required
|
|
@permission_required('administer all communities')
|
|
def admin_community_delete(community_id):
|
|
community = Community.query.get_or_404(community_id)
|
|
|
|
|
|
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)
|
|
|
|
reason = f"Community {community.name} deleted by {current_user.user_name}"
|
|
add_to_modlog('delete_community', reason=reason)
|
|
|
|
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)
|
|
unsubscribe_from_community(community, user)
|
|
else:
|
|
# todo: federate delete of local community out to all following instances
|
|
...
|
|
|
|
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.
|
|
db.session.commit()
|
|
|
|
|
|
@bp.route('/topics', methods=['GET'])
|
|
@login_required
|
|
@permission_required('administer all communities')
|
|
def admin_topics():
|
|
topics = topic_tree()
|
|
return render_template('admin/topics.html', title=_('Topics'), topics=topics,
|
|
moderating_communities=moderating_communities(current_user.get_id()),
|
|
joined_communities=joined_communities(current_user.get_id()),
|
|
menu_topics=menu_topics(),
|
|
site=g.site
|
|
)
|
|
|
|
|
|
@bp.route('/topic/add', methods=['GET', 'POST'])
|
|
@login_required
|
|
@permission_required('administer all communities')
|
|
def admin_topic_add():
|
|
form = EditTopicForm()
|
|
form.parent_id.choices = topics_for_form(0)
|
|
if form.validate_on_submit():
|
|
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)
|
|
if form.parent_id.data:
|
|
topic.parent_id = form.parent_id.data
|
|
else:
|
|
topic.parent_id = None
|
|
db.session.add(topic)
|
|
db.session.commit()
|
|
cache.delete_memoized(menu_topics)
|
|
|
|
flash(_('Saved'))
|
|
return redirect(url_for('admin.admin_topics'))
|
|
|
|
return render_template('admin/edit_topic.html', title=_('Add topic'), form=form,
|
|
moderating_communities=moderating_communities(current_user.get_id()),
|
|
joined_communities=joined_communities(current_user.get_id()),
|
|
menu_topics=menu_topics(),
|
|
site=g.site
|
|
)
|
|
|
|
@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)
|
|
form.parent_id.choices = topics_for_form(topic_id)
|
|
if form.validate_on_submit():
|
|
topic.name = form.name.data
|
|
topic.num_communities = topic.communities.count()
|
|
topic.machine_name = form.machine_name.data
|
|
topic.show_posts_in_children = form.show_posts_in_children.data
|
|
if form.parent_id.data:
|
|
topic.parent_id = form.parent_id.data
|
|
else:
|
|
topic.parent_id = None
|
|
db.session.commit()
|
|
cache.delete_memoized(menu_topics)
|
|
flash(_('Saved'))
|
|
return redirect(url_for('admin.admin_topics'))
|
|
else:
|
|
form.name.data = topic.name
|
|
form.machine_name.data = topic.machine_name
|
|
form.parent_id.data = topic.parent_id
|
|
form.show_posts_in_children.data = topic.show_posts_in_children
|
|
return render_template('admin/edit_topic.html', title=_('Edit topic'), form=form, topic=topic,
|
|
moderating_communities=moderating_communities(current_user.get_id()),
|
|
joined_communities=joined_communities(current_user.get_id()),
|
|
menu_topics=menu_topics(),
|
|
site=g.site
|
|
)
|
|
|
|
|
|
@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()
|
|
|
|
cache.delete_memoized(menu_topics)
|
|
|
|
return redirect(url_for('admin.admin_topics'))
|
|
|
|
|
|
@bp.route('/users', methods=['GET'])
|
|
@login_required
|
|
@permission_required('administer all users')
|
|
def admin_users():
|
|
|
|
page = request.args.get('page', 1, type=int)
|
|
search = request.args.get('search', '')
|
|
local_remote = request.args.get('local_remote', '')
|
|
|
|
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}%"))
|
|
users = users.order_by(User.user_name).paginate(page=page, per_page=1000, error_out=False)
|
|
|
|
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
|
|
|
|
return render_template('admin/users.html', title=_('Users'), next_url=next_url, prev_url=prev_url, users=users,
|
|
local_remote=local_remote, search=search,
|
|
moderating_communities=moderating_communities(current_user.get_id()),
|
|
joined_communities=joined_communities(current_user.get_id()),
|
|
menu_topics=menu_topics(),
|
|
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', '')
|
|
type = request.args.get('type', 'bad_rep')
|
|
|
|
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}%"))
|
|
|
|
if type == '' or type == 'bad_rep':
|
|
users = users.filter(User.last_seen > utcnow() - timedelta(days=7))
|
|
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':
|
|
users = users.filter(User.last_seen > utcnow() - timedelta(days=7))
|
|
users = users.filter(User.attitude < 0.0).filter(User.reputation < -10)
|
|
users = users.order_by(User.attitude).paginate(page=page, per_page=1000, error_out=False)
|
|
|
|
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
|
|
|
|
return render_template('admin/users_trash.html', title=_('Problematic users'), next_url=next_url, prev_url=prev_url, users=users,
|
|
local_remote=local_remote, search=search, type=type,
|
|
moderating_communities=moderating_communities(current_user.get_id()),
|
|
joined_communities=joined_communities(current_user.get_id()),
|
|
menu_topics=menu_topics(),
|
|
site=g.site
|
|
)
|
|
|
|
|
|
@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)
|
|
|
|
posts = Post.query.filter(Post.posted_at > utcnow() - timedelta(days=3), Post.deleted == False, Post.down_votes > 0).order_by(Post.score)
|
|
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()),
|
|
menu_topics=menu_topics(),
|
|
site=g.site
|
|
)
|
|
|
|
|
|
@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)).\
|
|
filter(Post.deleted == False).\
|
|
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)). \
|
|
filter(PostReply.deleted == False). \
|
|
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()),
|
|
menu_topics=menu_topics(),
|
|
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).\
|
|
order_by(desc(Post.posted_at))
|
|
posts = posts.paginate(page=page, per_page=100, error_out=False)
|
|
|
|
post_replies = PostReply.query. \
|
|
filter(PostReply.deleted == True). \
|
|
order_by(desc(PostReply.posted_at))
|
|
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(),
|
|
site=g.site
|
|
)
|
|
|
|
|
|
@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:
|
|
finalize_user_setup(user)
|
|
send_welcome_email(user, True)
|
|
|
|
flash(_('Registration approved.'))
|
|
|
|
return redirect(url_for('admin.admin_approve_registrations'))
|
|
|
|
|
|
@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
|
|
user.banned = form.banned.data
|
|
user.ban_posts = form.ban_posts.data
|
|
user.ban_comments = form.ban_comments.data
|
|
user.hide_nsfw = form.hide_nsfw.data
|
|
user.hide_nsfl = form.hide_nsfl.data
|
|
if form.verified.data and not user.verified:
|
|
finalize_user_setup(user)
|
|
user.verified = form.verified.data
|
|
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)
|
|
|
|
# Update user roles. The UI only lets the user choose 1 role but the DB structure allows for multiple roles per user.
|
|
db.session.execute(text('DELETE FROM user_role WHERE user_id = :user_id'), {'user_id': user.id})
|
|
user.roles.append(Role.query.get(form.role.data))
|
|
if form.role.data == 4:
|
|
flash(_("Permissions are cached for 50 seconds so new admin roles won't take effect immediately."))
|
|
|
|
db.session.commit()
|
|
|
|
flash(_('Saved'))
|
|
return redirect(url_for('admin.admin_users', local_remote='local' if user.is_local() else 'remote'))
|
|
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
|
|
form.verified.data = user.verified
|
|
form.banned.data = user.banned
|
|
form.ban_posts.data = user.ban_posts
|
|
form.ban_comments.data = user.ban_comments
|
|
form.hide_nsfw.data = user.hide_nsfw
|
|
form.hide_nsfl.data = user.hide_nsfl
|
|
if user.roles and user.roles.count() > 0:
|
|
form.role.data = user.roles[0].id
|
|
|
|
return render_template('admin/edit_user.html', title=_('Edit user'), form=form, user=user,
|
|
moderating_communities=moderating_communities(current_user.get_id()),
|
|
joined_communities=joined_communities(current_user.get_id()),
|
|
menu_topics=menu_topics(),
|
|
site=g.site
|
|
)
|
|
|
|
|
|
@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
|
|
user.title = form.user_name.data
|
|
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
|
|
user.hide_nsfw = form.hide_nsfw.data
|
|
user.hide_nsfl = form.hide_nsfl.data
|
|
|
|
user.instance_id = 1
|
|
user.roles.append(Role.query.get(form.role.data))
|
|
db.session.add(user)
|
|
db.session.commit()
|
|
finalize_user_setup(user)
|
|
|
|
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()),
|
|
menu_topics=menu_topics(),
|
|
site=g.site
|
|
)
|
|
|
|
|
|
@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()
|
|
user.deleted_by = current_user.id
|
|
db.session.commit()
|
|
|
|
if user.is_local():
|
|
unsubscribe_from_everything_then_delete(user.id)
|
|
else:
|
|
user.deleted = True
|
|
user.delete_dependencies()
|
|
db.session.commit()
|
|
|
|
add_to_modlog('delete_user', link_text=user.display_name(), link=user.link())
|
|
|
|
flash(_('User deleted'))
|
|
return redirect(url_for('admin.admin_users'))
|
|
|
|
|
|
@bp.route('/reports', methods=['GET'])
|
|
@login_required
|
|
@permission_required('administer all users')
|
|
def admin_reports():
|
|
|
|
page = request.args.get('page', 1, type=int)
|
|
search = request.args.get('search', '')
|
|
local_remote = request.args.get('local_remote', '')
|
|
|
|
reports = Report.query.filter(or_(Report.status == REPORT_STATE_NEW, Report.status == REPORT_STATE_ESCALATED))
|
|
if local_remote == 'local':
|
|
reports = reports.filter_by(source_instance_id=1)
|
|
if local_remote == 'remote':
|
|
reports = reports.filter(Report.source_instance_id != 1)
|
|
reports = reports.order_by(desc(Report.created_at)).paginate(page=page, per_page=1000, error_out=False)
|
|
|
|
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
|
|
|
|
return render_template('admin/reports.html', title=_('Reports'), next_url=next_url, prev_url=prev_url, reports=reports,
|
|
local_remote=local_remote, search=search,
|
|
moderating_communities=moderating_communities(current_user.get_id()),
|
|
joined_communities=joined_communities(current_user.get_id()),
|
|
menu_topics=menu_topics(),
|
|
site=g.site
|
|
)
|
|
|
|
|
|
@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()),
|
|
menu_topics=menu_topics(),
|
|
site=g.site
|
|
)
|
|
|
|
|
|
@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
|
|
)
|
|
|
|
|
|
@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', '')
|
|
filter = request.args.get('filter', '')
|
|
low_bandwidth = request.cookies.get('low_bandwidth', '0') == '1'
|
|
|
|
instances = Instance.query.order_by(Instance.domain)
|
|
|
|
if search:
|
|
instances = instances.filter(Instance.domain.ilike(f"%{search}%"))
|
|
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'
|
|
|
|
# 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,
|
|
title=_(title), search=search,
|
|
next_url=next_url, prev_url=prev_url,
|
|
low_bandwidth=low_bandwidth,
|
|
moderating_communities=moderating_communities(current_user.get_id()),
|
|
joined_communities=joined_communities(current_user.get_id()),
|
|
menu_topics=menu_topics(), site=g.site)
|
|
|
|
|
|
@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
|
|
)
|