mirror of
https://codeberg.org/rimu/pyfedi
synced 2025-01-23 11:26:56 -08:00
551 lines
29 KiB
Python
551 lines
29 KiB
Python
# if commands in this file are not working (e.g. 'flask translate') make sure you set the FLASK_APP environment variable.
|
|
# e.g. export FLASK_APP=pyfedi.py
|
|
import imaplib
|
|
import re
|
|
from datetime import datetime, timedelta
|
|
from random import randint
|
|
from time import sleep
|
|
|
|
import flask
|
|
from flask import json, current_app
|
|
from flask_babel import _
|
|
from sqlalchemy import or_, desc, text
|
|
|
|
from app import db
|
|
import click
|
|
import os
|
|
|
|
from app.activitypub.signature import RsaKeys
|
|
from app.activitypub.util import find_actor_or_create
|
|
from app.auth.util import random_token
|
|
from app.constants import NOTIF_COMMUNITY, NOTIF_POST, NOTIF_REPLY
|
|
from app.email import send_email
|
|
from app.models import Settings, BannedInstances, Role, User, RolePermission, Domain, ActivityPubLog, \
|
|
utcnow, Site, Instance, File, Notification, Post, CommunityMember, NotificationSubscription, PostReply, Language, \
|
|
Tag, InstanceRole, Community, DefederationSubscription
|
|
from app.post.routes import post_delete_post
|
|
from app.utils import retrieve_block_list, blocked_domains, retrieve_peertube_block_list, \
|
|
shorten_string, get_request, blocked_communities, gibberish, get_request_instance, \
|
|
instance_banned, recently_upvoted_post_replies, recently_upvoted_posts, jaccard_similarity, download_defeds
|
|
|
|
|
|
def register(app):
|
|
@app.cli.group()
|
|
def translate():
|
|
"""Translation and localization commands."""
|
|
pass
|
|
|
|
@translate.command()
|
|
@click.argument('lang')
|
|
def init(lang):
|
|
"""Initialize a new language."""
|
|
if os.system('pybabel extract -F babel.cfg -k _l -o messages.pot .'):
|
|
raise RuntimeError('extract command failed')
|
|
if os.system(
|
|
'pybabel init -i messages.pot -d app/translations -l ' + lang):
|
|
raise RuntimeError('init command failed')
|
|
os.remove('messages.pot')
|
|
|
|
@translate.command()
|
|
def update():
|
|
"""Update all languages."""
|
|
if os.system('pybabel extract -F babel.cfg -k _l -o messages.pot .'):
|
|
raise RuntimeError('extract command failed')
|
|
if os.system('pybabel update -i messages.pot -d app/translations'):
|
|
raise RuntimeError('update command failed')
|
|
os.remove('messages.pot')
|
|
|
|
@translate.command()
|
|
def compile():
|
|
"""Compile all languages."""
|
|
if os.system('pybabel compile -d app/translations'):
|
|
raise RuntimeError('compile command failed')
|
|
|
|
@app.cli.command("keys")
|
|
def keys():
|
|
private_key, public_key = RsaKeys.generate_keypair()
|
|
print(private_key)
|
|
print(public_key)
|
|
|
|
@app.cli.command("admin-keys")
|
|
def keys():
|
|
private_key, public_key = RsaKeys.generate_keypair()
|
|
u: User = User.query.get(1)
|
|
u.private_key = private_key
|
|
u.public_key = public_key
|
|
db.session.commit()
|
|
print('Admin keys have been reset')
|
|
|
|
@app.cli.command("init-db")
|
|
def init_db():
|
|
with app.app_context():
|
|
db.drop_all()
|
|
db.configure_mappers()
|
|
db.create_all()
|
|
private_key, public_key = RsaKeys.generate_keypair()
|
|
db.session.add(Site(name="PieFed", description='Explore Anything, Discuss Everything.', public_key=public_key, private_key=private_key))
|
|
db.session.add(Instance(domain=app.config['SERVER_NAME'], software='PieFed')) # Instance 1 is always the local instance
|
|
db.session.add(Settings(name='allow_nsfw', value=json.dumps(False)))
|
|
db.session.add(Settings(name='allow_nsfl', value=json.dumps(False)))
|
|
db.session.add(Settings(name='allow_dislike', value=json.dumps(True)))
|
|
db.session.add(Settings(name='allow_local_image_posts', value=json.dumps(True)))
|
|
db.session.add(Settings(name='allow_remote_image_posts', value=json.dumps(True)))
|
|
db.session.add(Settings(name='federation', value=json.dumps(True)))
|
|
db.session.add(Language(name='Undetermined', code='und'))
|
|
db.session.add(Language(name='English', code='en'))
|
|
banned_instances = ['anonib.al','lemmygrad.ml', 'gab.com', 'rqd2.net', 'exploding-heads.com', 'hexbear.net',
|
|
'threads.net', 'noauthority.social', 'pieville.net', 'links.hackliberty.org',
|
|
'poa.st', 'freespeechextremist.com', 'bae.st', 'nicecrew.digital', 'detroitriotcity.com',
|
|
'pawoo.net', 'shitposter.club', 'spinster.xyz', 'catgirl.life', 'gameliberty.club',
|
|
'yggdrasil.social', 'beefyboys.win', 'brighteon.social', 'cum.salon', 'wizard.casa']
|
|
for bi in banned_instances:
|
|
db.session.add(BannedInstances(domain=bi))
|
|
print("Added banned instance", bi)
|
|
|
|
# Load initial domain block list
|
|
block_list = retrieve_block_list()
|
|
if block_list:
|
|
for domain in block_list.split('\n'):
|
|
db.session.add(Domain(name=domain.strip(), banned=True))
|
|
print("Added 'No-QAnon' blocklist, see https://github.com/rimu/no-qanon")
|
|
|
|
# Load peertube domain block list
|
|
block_list = retrieve_peertube_block_list()
|
|
if block_list:
|
|
for domain in block_list.split('\n'):
|
|
db.session.add(Domain(name=domain.strip(), banned=True))
|
|
db.session.add(BannedInstances(domain=domain.strip()))
|
|
print("Added 'Peertube Isolation' blocklist, see https://peertube_isolation.frama.io/")
|
|
|
|
# Initial roles
|
|
# These roles will create rows in the 'role' table with IDs of 1,2,3,4. There are some constants (ROLE_*) in
|
|
# constants.py that will need to be updated if the role IDs ever change.
|
|
anon_role = Role(name='Anonymous user', weight=0)
|
|
anon_role.permissions.append(RolePermission(permission='register'))
|
|
db.session.add(anon_role)
|
|
|
|
auth_role = Role(name='Authenticated user', weight=1)
|
|
db.session.add(auth_role)
|
|
|
|
staff_role = Role(name='Staff', weight=2)
|
|
staff_role.permissions.append(RolePermission(permission='approve registrations'))
|
|
staff_role.permissions.append(RolePermission(permission='ban users'))
|
|
staff_role.permissions.append(RolePermission(permission='administer all communities'))
|
|
staff_role.permissions.append(RolePermission(permission='administer all users'))
|
|
db.session.add(staff_role)
|
|
|
|
admin_role = Role(name='Admin', weight=3)
|
|
admin_role.permissions.append(RolePermission(permission='approve registrations'))
|
|
admin_role.permissions.append(RolePermission(permission='change user roles'))
|
|
admin_role.permissions.append(RolePermission(permission='ban users'))
|
|
admin_role.permissions.append(RolePermission(permission='manage users'))
|
|
admin_role.permissions.append(RolePermission(permission='change instance settings'))
|
|
admin_role.permissions.append(RolePermission(permission='administer all communities'))
|
|
admin_role.permissions.append(RolePermission(permission='administer all users'))
|
|
db.session.add(admin_role)
|
|
|
|
# Admin user
|
|
user_name = input("Admin user name (ideally not 'admin'): ")
|
|
email = input("Admin email address: ")
|
|
password = input("Admin password: ")
|
|
while '@' in user_name or ' ' in user_name:
|
|
print('User name cannot be an email address or have spaces.')
|
|
user_name = input("Admin user name (ideally not 'admin'): ")
|
|
verification_token = random_token(16)
|
|
private_key, public_key = RsaKeys.generate_keypair()
|
|
admin_user = User(user_name=user_name, title=user_name,
|
|
email=email, verification_token=verification_token,
|
|
instance_id=1, email_unread_sent=False,
|
|
private_key=private_key, public_key=public_key,
|
|
alt_user_name=gibberish(randint(8, 20)))
|
|
admin_user.set_password(password)
|
|
admin_user.roles.append(admin_role)
|
|
admin_user.verified = True
|
|
admin_user.last_seen = utcnow()
|
|
admin_user.ap_profile_id = f"https://{current_app.config['SERVER_NAME']}/u/{admin_user.user_name.lower()}"
|
|
admin_user.ap_public_url = f"https://{current_app.config['SERVER_NAME']}/u/{admin_user.user_name}"
|
|
admin_user.ap_inbox_url = f"https://{current_app.config['SERVER_NAME']}/u/{admin_user.user_name.lower()}/inbox"
|
|
db.session.add(admin_user)
|
|
|
|
db.session.commit()
|
|
print("Initial setup is finished.")
|
|
|
|
@app.cli.command('daily-maintenance')
|
|
def daily_maintenance():
|
|
with app.app_context():
|
|
# Remove old content from communities
|
|
communities = Community.query.filter(Community.content_retention > 0).all()
|
|
for community in communities:
|
|
cut_off = utcnow() - timedelta(days=community.content_retention)
|
|
old_posts = Post.query.filter_by(deleted=False, sticky=False, community_id=community.id).filter(Post.posted_at < cut_off).all()
|
|
for post in old_posts:
|
|
post_delete_post(community, post, post.user_id, federate_all_communities=False)
|
|
community.post_count -= 1
|
|
|
|
# Ensure accurate count of posts associated with each hashtag
|
|
for tag in Tag.query.all():
|
|
post_count = db.session.execute(text('SELECT COUNT(post_id) as c FROM "post_tag" WHERE tag_id = :tag_id'),
|
|
{ 'tag_id': tag.id}).scalar()
|
|
tag.post_count = post_count
|
|
db.session.commit()
|
|
|
|
# Delete soft-deleted content after 7 days
|
|
for post_reply in PostReply.query.filter(PostReply.deleted == True,
|
|
PostReply.posted_at < utcnow() - timedelta(days=7)).all():
|
|
post_reply.delete_dependencies()
|
|
if not post_reply.has_replies():
|
|
db.session.delete(post_reply)
|
|
db.session.commit()
|
|
|
|
for post in Post.query.filter(Post.deleted == True,
|
|
Post.posted_at < utcnow() - timedelta(days=7)).all():
|
|
post.delete_dependencies()
|
|
db.session.delete(post)
|
|
db.session.commit()
|
|
|
|
# Ensure accurate community stats
|
|
for community in Community.query.filter(Community.banned == False).all():
|
|
community.subscriptions_count = db.session.execute(text('SELECT COUNT(user_id) as c FROM community_member WHERE community_id = :community_id AND is_banned = false'),
|
|
{'community_id': community.id}).scalar()
|
|
community.post_count = db.session.execute(text('SELECT COUNT(id) as c FROM post WHERE deleted is false and community_id = :community_id'),
|
|
{'community_id': community.id}).scalar()
|
|
community.post_reply_count = db.session.execute(text('SELECT COUNT(id) as c FROM post_reply WHERE deleted is false and community_id = :community_id'),
|
|
{'community_id': community.id}).scalar()
|
|
db.session.commit()
|
|
|
|
# Delete voting data after 6 months
|
|
db.session.execute(text('DELETE FROM "post_vote" WHERE created_at < :cutoff'), {'cutoff': utcnow() - timedelta(days=28 * 6)})
|
|
db.session.execute(text('DELETE FROM "post_reply_vote" WHERE created_at < :cutoff'), {'cutoff': utcnow() - timedelta(days=28 * 6)})
|
|
db.session.commit()
|
|
|
|
# Un-ban after ban expires
|
|
db.session.execute(text('UPDATE "user" SET banned = false WHERE banned is true AND banned_until < :cutoff AND banned_until is not null'),
|
|
{'cutoff': utcnow()})
|
|
db.session.commit()
|
|
|
|
# update and sync defederation subscriptions
|
|
db.session.execute(text('DELETE FROM banned_instances WHERE subscription_id is not null'))
|
|
for defederation_sub in DefederationSubscription.query.all():
|
|
download_defeds(defederation_sub.id, defederation_sub.domain)
|
|
|
|
# Check for dormant or dead instances
|
|
try:
|
|
# Check for dormant or dead instances
|
|
instances = Instance.query.filter(Instance.gone_forever == False, Instance.id != 1).all()
|
|
HEADERS = {'Accept': 'application/activity+json'}
|
|
|
|
for instance in instances:
|
|
if instance_banned(instance.domain) or instance.domain == 'flipboard.com':
|
|
continue
|
|
nodeinfo_href = instance.nodeinfo_href
|
|
if instance.software == 'lemmy' and instance.version is not None and instance.version >= '0.19.4' and \
|
|
instance.nodeinfo_href and instance.nodeinfo_href.endswith('nodeinfo/2.0.json'):
|
|
nodeinfo_href = None
|
|
|
|
if not nodeinfo_href:
|
|
try:
|
|
nodeinfo = get_request_instance(f"https://{instance.domain}/.well-known/nodeinfo",
|
|
headers=HEADERS, instance=instance)
|
|
|
|
if nodeinfo.status_code == 200:
|
|
nodeinfo_json = nodeinfo.json()
|
|
for links in nodeinfo_json['links']:
|
|
if isinstance(links, dict) and 'rel' in links and links['rel'] in [
|
|
'http://nodeinfo.diaspora.software/ns/schema/2.0',
|
|
'https://nodeinfo.diaspora.software/ns/schema/2.0',
|
|
'http://nodeinfo.diaspora.software/ns/schema/2.1']:
|
|
instance.nodeinfo_href = links['href']
|
|
instance.failures = 0
|
|
instance.dormant = False
|
|
break
|
|
else:
|
|
instance.failures += 1
|
|
elif nodeinfo.status_code >= 400:
|
|
current_app.logger.info(f"{instance.domain} has no well-known/nodeinfo response")
|
|
except Exception as e:
|
|
db.session.rollback()
|
|
instance.failures += 1
|
|
finally:
|
|
nodeinfo.close()
|
|
db.session.commit()
|
|
|
|
if instance.nodeinfo_href:
|
|
try:
|
|
node = get_request_instance(instance.nodeinfo_href, headers=HEADERS, instance=instance)
|
|
if node.status_code == 200:
|
|
node_json = node.json()
|
|
if 'software' in node_json:
|
|
instance.software = node_json['software']['name'].lower()[:50]
|
|
instance.version = node_json['software']['version'][:50]
|
|
instance.failures = 0
|
|
instance.dormant = False
|
|
elif node.status_code >= 400:
|
|
instance.nodeinfo_href = None
|
|
except Exception as e:
|
|
db.session.rollback()
|
|
instance.failures += 1
|
|
finally:
|
|
node.close()
|
|
db.session.commit()
|
|
|
|
# Handle admin roles
|
|
if instance.online() and (instance.software == 'lemmy' or instance.software == 'piefed'):
|
|
try:
|
|
response = get_request(f'https://{instance.domain}/api/v3/site')
|
|
if response and response.status_code == 200:
|
|
instance_data = response.json()
|
|
admin_profile_ids = []
|
|
for admin in instance_data['admins']:
|
|
profile_id = admin['person']['actor_id']
|
|
if profile_id.startswith('https://'):
|
|
admin_profile_ids.append(profile_id.lower())
|
|
user = find_actor_or_create(profile_id)
|
|
if user and not instance.user_is_admin(user.id):
|
|
new_instance_role = InstanceRole(instance_id=instance.id, user_id=user.id,
|
|
role='admin')
|
|
db.session.add(new_instance_role)
|
|
# remove any InstanceRoles that are no longer part of instance-data['admins']
|
|
for instance_admin in InstanceRole.query.filter_by(instance_id=instance.id):
|
|
if instance_admin.user.profile_id() not in admin_profile_ids:
|
|
db.session.query(InstanceRole).filter(
|
|
InstanceRole.user_id == instance_admin.user.id,
|
|
InstanceRole.instance_id == instance.id,
|
|
InstanceRole.role == 'admin').delete()
|
|
except Exception as e:
|
|
db.session.rollback()
|
|
instance.failures += 1
|
|
finally:
|
|
if response:
|
|
response.close()
|
|
db.session.commit()
|
|
|
|
except Exception as e:
|
|
db.session.rollback()
|
|
current_app.logger.error(f"Error in daily maintenance: {e}")
|
|
|
|
# remove any InstanceRoles that are no longer part of instance-data['admins']
|
|
#for instance_admin in InstanceRole.query.filter_by(instance_id=instance.id):
|
|
# if instance_admin.user.profile_id() not in admin_profile_ids:
|
|
# db.session.query(InstanceRole).filter(
|
|
# InstanceRole.user_id == instance_admin.user.id,
|
|
# InstanceRole.instance_id == instance.id,
|
|
# InstanceRole.role == 'admin').delete()
|
|
# db.session.commit()
|
|
|
|
@app.cli.command("spaceusage")
|
|
def spaceusage():
|
|
with app.app_context():
|
|
for user in User.query.all():
|
|
filesize = user.filesize()
|
|
num_content = user.num_content()
|
|
if filesize > 0 and num_content > 0:
|
|
print(f'{user.id},"{user.ap_id}",{filesize},{num_content}')
|
|
|
|
def list_files(directory):
|
|
for root, dirs, files in os.walk(directory):
|
|
for file in files:
|
|
yield os.path.join(root, file)
|
|
|
|
@app.cli.command("remove_orphan_files")
|
|
def remove_orphan_files():
|
|
""" Any user-uploaded file that does not have a corresponding entry in the File table should be deleted """
|
|
with app.app_context():
|
|
for file_path in list_files('app/static/media/users'):
|
|
if 'thumbnail' in file_path:
|
|
f = File.query.filter(File.thumbnail_path == file_path).first()
|
|
else:
|
|
f = File.query.filter(File.file_path == file_path).first()
|
|
if f is None:
|
|
os.unlink(file_path)
|
|
|
|
@app.cli.command("send_missed_notifs")
|
|
def send_missed_notifs():
|
|
with app.app_context():
|
|
site = Site.query.get(1)
|
|
users_to_notify = User.query.join(Notification, User.id == Notification.user_id).filter(
|
|
User.ap_id == None,
|
|
Notification.created_at > User.last_seen,
|
|
Notification.read == False,
|
|
User.email_unread_sent == False, # they have not been emailed since last activity
|
|
User.email_unread == True # they want to be emailed
|
|
).all()
|
|
|
|
for user in users_to_notify:
|
|
notifications = Notification.query.filter(Notification.user_id == user.id, Notification.read == False,
|
|
Notification.created_at > user.last_seen).all()
|
|
if notifications:
|
|
# Also get the top 20 posts since their last login
|
|
posts = Post.query.join(CommunityMember, Post.community_id == CommunityMember.community_id).filter(
|
|
CommunityMember.is_banned == False)
|
|
posts = posts.filter(CommunityMember.user_id == user.id)
|
|
if user.ignore_bots == 1:
|
|
posts = posts.filter(Post.from_bot == False)
|
|
if user.hide_nsfl == 1:
|
|
posts = posts.filter(Post.nsfl == False)
|
|
if user.hide_nsfw == 1:
|
|
posts = posts.filter(Post.nsfw == False)
|
|
domains_ids = blocked_domains(user.id)
|
|
if domains_ids:
|
|
posts = posts.filter(or_(Post.domain_id.not_in(domains_ids), Post.domain_id == None))
|
|
community_ids = blocked_communities(user.id)
|
|
if community_ids:
|
|
posts = posts.filter(Post.community_id.not_in(community_ids))
|
|
posts = posts.filter(Post.posted_at > user.last_seen).order_by(desc(Post.score))
|
|
posts = posts.limit(20).all()
|
|
|
|
# Send email!
|
|
send_email(_('[PieFed] You have unread notifications'),
|
|
sender=f'{site.name} <{current_app.config["MAIL_FROM"]}>',
|
|
recipients=[user.email],
|
|
text_body=flask.render_template('email/unread_notifications.txt', user=user,
|
|
notifications=notifications),
|
|
html_body=flask.render_template('email/unread_notifications.html', user=user,
|
|
notifications=notifications,
|
|
posts=posts,
|
|
domain=current_app.config['SERVER_NAME']))
|
|
user.email_unread_sent = True
|
|
db.session.commit()
|
|
|
|
@app.cli.command("process_email_bounces")
|
|
def process_email_bounces():
|
|
with app.app_context():
|
|
import email
|
|
|
|
imap_host = current_app.config['BOUNCE_HOST']
|
|
imap_user = current_app.config['BOUNCE_USERNAME']
|
|
imap_pass = current_app.config['BOUNCE_PASSWORD']
|
|
something_deleted = False
|
|
|
|
if imap_host:
|
|
|
|
# connect to host using SSL
|
|
imap = imaplib.IMAP4_SSL(imap_host, port=993)
|
|
|
|
## login to server
|
|
imap.login(imap_user, imap_pass)
|
|
|
|
imap.select('Inbox')
|
|
|
|
tmp, data = imap.search(None, 'ALL')
|
|
rgx = r'[\w\.-]+@[\w\.-]+'
|
|
|
|
emails = set()
|
|
|
|
for num in data[0].split():
|
|
tmp, data = imap.fetch(num, '(RFC822)')
|
|
email_message = email.message_from_bytes(data[0][1])
|
|
match = []
|
|
if not isinstance(email_message._payload, str):
|
|
if isinstance(email_message._payload[0]._payload, str):
|
|
payload = email_message._payload[0]._payload.replace("\n", " ").replace("\r", " ")
|
|
match = re.findall(rgx, payload)
|
|
elif isinstance(email_message._payload[0]._payload, list):
|
|
if isinstance(email_message._payload[0]._payload[0]._payload, str):
|
|
payload = email_message._payload[0]._payload[0]._payload.replace("\n", " ").replace("\r", " ")
|
|
match = re.findall(rgx, payload)
|
|
|
|
for m in match:
|
|
if current_app.config['SERVER_NAME'] not in m and current_app.config['SERVER_NAME'].upper() not in m:
|
|
emails.add(m)
|
|
print(str(num) + ' ' + m)
|
|
|
|
imap.store(num, '+FLAGS', '\\Deleted')
|
|
something_deleted = True
|
|
|
|
if something_deleted:
|
|
imap.expunge()
|
|
pass
|
|
|
|
imap.close()
|
|
|
|
# Keep track of how many times email to an account has bounced. After 2 bounces, disable email sending to them
|
|
for bounced_email in emails:
|
|
bounced_accounts = User.query.filter_by(email=bounced_email).all()
|
|
for account in bounced_accounts:
|
|
if account.bounces is None:
|
|
account.bounces = 0
|
|
if account.bounces > 2:
|
|
account.newsletter = False
|
|
account.email_unread = False
|
|
else:
|
|
account.bounces += 1
|
|
db.session.commit()
|
|
|
|
@app.cli.command("clean_up_old_activities")
|
|
def clean_up_old_activities():
|
|
with app.app_context():
|
|
db.session.query(ActivityPubLog).filter(ActivityPubLog.created_at < utcnow() - timedelta(days=3)).delete()
|
|
db.session.commit()
|
|
|
|
@app.cli.command("detect_vote_manipulation")
|
|
def detect_vote_manipulation():
|
|
with app.app_context():
|
|
print('Getting user ids...')
|
|
all_user_ids = [user.id for user in User.query.filter(User.last_seen > datetime.utcnow() - timedelta(days=7))]
|
|
print('Checking...')
|
|
for i, first_user_id in enumerate(all_user_ids):
|
|
current_user_upvoted_posts = ['post/' + str(id) for id in recently_upvoted_posts(first_user_id)]
|
|
current_user_upvoted_replies = ['reply/' + str(id) for id in recently_upvoted_post_replies(first_user_id)]
|
|
|
|
current_user_upvotes = set(current_user_upvoted_posts + current_user_upvoted_replies)
|
|
if len(current_user_upvotes) > 12:
|
|
print(i)
|
|
for j in range(i + 1, len(all_user_ids)):
|
|
other_user_id = all_user_ids[j]
|
|
if jaccard_similarity(current_user_upvotes, other_user_id) >= 95:
|
|
first_user = User.query.get(first_user_id)
|
|
other_user = User.query.get(other_user_id)
|
|
print(f'{first_user.link()} votes the same as {other_user.link()}')
|
|
|
|
@app.cli.command("migrate_community_notifs")
|
|
def migrate_community_notifs():
|
|
with app.app_context():
|
|
member_infos = CommunityMember.query.filter(CommunityMember.notify_new_posts == True,
|
|
CommunityMember.is_banned == False).all()
|
|
for member_info in member_infos:
|
|
new_notification = NotificationSubscription(user_id=member_info.user_id, entity_id=member_info.community_id,
|
|
type=NOTIF_COMMUNITY)
|
|
db.session.add(new_notification)
|
|
db.session.commit()
|
|
print('Done')
|
|
|
|
@app.cli.command("migrate_post_notifs")
|
|
def migrate_post_notifs():
|
|
with app.app_context():
|
|
posts = Post.query.filter(Post.notify_author == True).all()
|
|
for post in posts:
|
|
new_notification = NotificationSubscription(name=shorten_string(_('Replies to my post %(post_title)s',
|
|
post_title=post.title)),
|
|
user_id=post.user_id, entity_id=post.id,
|
|
type=NOTIF_POST)
|
|
db.session.add(new_notification)
|
|
db.session.commit()
|
|
|
|
post_replies = PostReply.query.filter(PostReply.notify_author == True).all()
|
|
for reply in post_replies:
|
|
new_notification = NotificationSubscription(name=shorten_string(_('Replies to my comment on %(post_title)s',
|
|
post_title=reply.post.title)),
|
|
user_id=post.user_id, entity_id=reply.id,
|
|
type=NOTIF_REPLY)
|
|
db.session.add(new_notification)
|
|
db.session.commit()
|
|
|
|
print('Done')
|
|
|
|
|
|
def parse_communities(interests_source, segment):
|
|
lines = interests_source.split("\n")
|
|
include_in_output = False
|
|
output = []
|
|
|
|
for line in lines:
|
|
line = line.strip()
|
|
if line == segment:
|
|
include_in_output = True
|
|
continue
|
|
elif line == '':
|
|
include_in_output = False
|
|
if include_in_output:
|
|
output.append(line)
|
|
|
|
return "\n".join(output)
|