2023-08-05 02:26:24 -07:00
|
|
|
from datetime import datetime, timedelta, date
|
|
|
|
from hashlib import md5
|
|
|
|
from time import time
|
2023-09-17 02:19:51 -07:00
|
|
|
from typing import List
|
|
|
|
|
2023-10-20 19:49:01 -07:00
|
|
|
from flask import current_app, escape, url_for
|
2023-11-29 09:36:08 -08:00
|
|
|
from flask_login import UserMixin, current_user
|
2023-10-20 19:49:01 -07:00
|
|
|
from sqlalchemy import or_, text
|
2023-08-05 02:26:24 -07:00
|
|
|
from werkzeug.security import generate_password_hash, check_password_hash
|
|
|
|
from flask_babel import _, lazy_gettext as _l
|
|
|
|
from sqlalchemy.orm import backref
|
|
|
|
from sqlalchemy_utils.types import TSVectorType # https://sqlalchemy-searchable.readthedocs.io/en/latest/installation.html
|
2023-10-03 02:29:13 -07:00
|
|
|
from flask_sqlalchemy import BaseQuery
|
|
|
|
from sqlalchemy_searchable import SearchQueryMixin
|
2023-11-29 10:12:17 -08:00
|
|
|
from app import db, login, cache
|
2023-08-05 02:26:24 -07:00
|
|
|
import jwt
|
2023-11-29 23:57:51 -08:00
|
|
|
import os
|
2023-08-05 02:26:24 -07:00
|
|
|
|
2023-09-08 01:04:01 -07:00
|
|
|
from app.constants import SUBSCRIPTION_NONMEMBER, SUBSCRIPTION_MEMBER, SUBSCRIPTION_MODERATOR, SUBSCRIPTION_OWNER, \
|
2023-12-03 01:41:15 -08:00
|
|
|
SUBSCRIPTION_BANNED, SUBSCRIPTION_PENDING
|
2023-08-29 03:01:06 -07:00
|
|
|
|
2023-08-05 02:26:24 -07:00
|
|
|
|
2023-10-03 02:29:13 -07:00
|
|
|
class FullTextSearchQuery(BaseQuery, SearchQueryMixin):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2023-08-05 02:26:24 -07:00
|
|
|
class File(db.Model):
|
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
|
|
file_path = db.Column(db.String(255))
|
|
|
|
file_name = db.Column(db.String(255))
|
|
|
|
width = db.Column(db.Integer)
|
|
|
|
height = db.Column(db.Integer)
|
|
|
|
alt_text = db.Column(db.String(256))
|
|
|
|
source_url = db.Column(db.String(256))
|
2023-11-27 01:05:35 -08:00
|
|
|
thumbnail_path = db.Column(db.String(255))
|
|
|
|
thumbnail_width = db.Column(db.Integer)
|
|
|
|
thumbnail_height = db.Column(db.Integer)
|
|
|
|
|
2023-11-29 10:12:17 -08:00
|
|
|
@cache.memoize(timeout=500)
|
2023-11-27 01:05:35 -08:00
|
|
|
def view_url(self):
|
|
|
|
if self.source_url:
|
|
|
|
return self.source_url
|
|
|
|
elif self.file_path:
|
|
|
|
file_path = self.file_path[4:] if self.file_path.startswith('app/') else self.file_path
|
|
|
|
return f"https://{current_app.config['SERVER_NAME']}/{file_path}"
|
|
|
|
else:
|
|
|
|
return ''
|
|
|
|
|
2023-11-29 10:12:17 -08:00
|
|
|
@cache.memoize(timeout=500)
|
2023-11-27 01:05:35 -08:00
|
|
|
def thumbnail_url(self):
|
|
|
|
thumbnail_path = self.thumbnail_path[4:] if self.thumbnail_path.startswith('app/') else self.thumbnail_path
|
|
|
|
return f"https://{current_app.config['SERVER_NAME']}/{thumbnail_path}"
|
2023-08-05 02:26:24 -07:00
|
|
|
|
2023-11-29 23:57:51 -08:00
|
|
|
def delete_from_disk(self):
|
|
|
|
if self.file_path and os.path.isfile(self.file_path):
|
|
|
|
os.unlink(self.file_path)
|
|
|
|
if self.thumbnail_path and os.path.isfile(self.thumbnail_path):
|
|
|
|
os.unlink(self.thumbnail_path)
|
|
|
|
|
2023-08-05 02:26:24 -07:00
|
|
|
|
|
|
|
class Community(db.Model):
|
2023-10-03 02:29:13 -07:00
|
|
|
query_class = FullTextSearchQuery
|
2023-08-05 02:26:24 -07:00
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
|
|
icon_id = db.Column(db.Integer, db.ForeignKey('file.id'))
|
2023-08-29 03:01:06 -07:00
|
|
|
image_id = db.Column(db.Integer, db.ForeignKey('file.id'))
|
2023-10-20 19:49:01 -07:00
|
|
|
user_id = db.Column(db.Integer, db.ForeignKey('user.id'))
|
2023-08-05 02:26:24 -07:00
|
|
|
name = db.Column(db.String(256), index=True)
|
|
|
|
title = db.Column(db.String(256))
|
|
|
|
description = db.Column(db.Text)
|
|
|
|
rules = db.Column(db.Text)
|
|
|
|
subscriptions_count = db.Column(db.Integer, default=0)
|
|
|
|
post_count = db.Column(db.Integer, default=0)
|
|
|
|
post_reply_count = db.Column(db.Integer, default=0)
|
|
|
|
nsfw = db.Column(db.Boolean, default=False)
|
|
|
|
nsfl = db.Column(db.Boolean, default=False)
|
|
|
|
created_at = db.Column(db.DateTime, default=datetime.utcnow)
|
|
|
|
last_active = db.Column(db.DateTime, default=datetime.utcnow)
|
|
|
|
public_key = db.Column(db.Text)
|
|
|
|
private_key = db.Column(db.Text)
|
|
|
|
|
|
|
|
ap_id = db.Column(db.String(255), index=True)
|
2023-08-22 02:24:11 -07:00
|
|
|
ap_profile_id = db.Column(db.String(255), index=True)
|
2023-08-05 02:26:24 -07:00
|
|
|
ap_followers_url = db.Column(db.String(255))
|
|
|
|
ap_preferred_username = db.Column(db.String(255))
|
|
|
|
ap_discoverable = db.Column(db.Boolean, default=False)
|
|
|
|
ap_public_url = db.Column(db.String(255))
|
|
|
|
ap_fetched_at = db.Column(db.DateTime)
|
|
|
|
ap_deleted_at = db.Column(db.DateTime)
|
|
|
|
ap_inbox_url = db.Column(db.String(255))
|
2023-12-03 01:41:15 -08:00
|
|
|
ap_moderators_url = db.Column(db.String(255))
|
2023-08-05 02:26:24 -07:00
|
|
|
ap_domain = db.Column(db.String(255))
|
|
|
|
|
|
|
|
banned = db.Column(db.Boolean, default=False)
|
2023-08-10 02:13:37 -07:00
|
|
|
restricted_to_mods = db.Column(db.Boolean, default=False)
|
2023-08-05 02:26:24 -07:00
|
|
|
searchable = db.Column(db.Boolean, default=True)
|
2023-09-05 01:25:02 -07:00
|
|
|
private_mods = db.Column(db.Boolean, default=False)
|
2023-08-05 02:26:24 -07:00
|
|
|
|
2023-08-29 03:01:06 -07:00
|
|
|
search_vector = db.Column(TSVectorType('name', 'title', 'description', 'rules'))
|
2023-08-05 02:26:24 -07:00
|
|
|
|
2023-08-10 02:13:37 -07:00
|
|
|
posts = db.relationship('Post', backref='community', lazy='dynamic', cascade="all, delete-orphan")
|
|
|
|
replies = db.relationship('PostReply', backref='community', lazy='dynamic', cascade="all, delete-orphan")
|
2023-08-29 03:01:06 -07:00
|
|
|
icon = db.relationship('File', foreign_keys=[icon_id], single_parent=True, backref='community', cascade="all, delete-orphan")
|
|
|
|
image = db.relationship('File', foreign_keys=[image_id], single_parent=True, cascade="all, delete-orphan")
|
|
|
|
|
2023-11-29 10:12:17 -08:00
|
|
|
@cache.memoize(timeout=500)
|
2023-12-07 20:13:38 -08:00
|
|
|
def icon_image(self, size='default') -> str:
|
2023-08-29 03:01:06 -07:00
|
|
|
if self.icon_id is not None:
|
2023-12-07 20:13:38 -08:00
|
|
|
if size == 'default':
|
|
|
|
if self.icon.file_path is not None:
|
|
|
|
if self.icon.file_path.startswith('app/'):
|
|
|
|
return self.icon.file_path.replace('app/', '/')
|
|
|
|
else:
|
|
|
|
return self.icon.file_path
|
|
|
|
if self.icon.source_url is not None:
|
|
|
|
if self.icon.source_url.startswith('app/'):
|
|
|
|
return self.icon.source_url.replace('app/', '/')
|
|
|
|
else:
|
|
|
|
return self.icon.source_url
|
|
|
|
elif size == 'tiny':
|
|
|
|
if self.icon.thumbnail_path is not None:
|
|
|
|
if self.icon.thumbnail_path.startswith('app/'):
|
|
|
|
return self.icon.thumbnail_path.replace('app/', '/')
|
|
|
|
else:
|
|
|
|
return self.icon.thumbnail_path
|
|
|
|
if self.icon.source_url is not None:
|
|
|
|
if self.icon.source_url.startswith('app/'):
|
|
|
|
return self.icon.source_url.replace('app/', '/')
|
|
|
|
else:
|
|
|
|
return self.icon.source_url
|
2023-08-29 03:01:06 -07:00
|
|
|
return ''
|
|
|
|
|
2023-11-29 10:12:17 -08:00
|
|
|
@cache.memoize(timeout=500)
|
2023-08-29 03:01:06 -07:00
|
|
|
def header_image(self) -> str:
|
|
|
|
if self.image_id is not None:
|
|
|
|
if self.image.file_path is not None:
|
2023-12-07 20:13:38 -08:00
|
|
|
if self.image.file_path.startswith('app/'):
|
|
|
|
return self.image.file_path.replace('app/', '/')
|
|
|
|
else:
|
|
|
|
return self.image.file_path
|
2023-08-29 03:01:06 -07:00
|
|
|
if self.image.source_url is not None:
|
2023-12-07 20:13:38 -08:00
|
|
|
if self.image.source_url.startswith('app/'):
|
|
|
|
return self.image.source_url.replace('app/', '/')
|
|
|
|
else:
|
|
|
|
return self.image.source_url
|
2023-08-29 03:01:06 -07:00
|
|
|
return ''
|
|
|
|
|
|
|
|
def display_name(self) -> str:
|
|
|
|
if self.ap_id is None:
|
|
|
|
return self.title
|
|
|
|
else:
|
|
|
|
return f"{self.title}@{self.ap_domain}"
|
|
|
|
|
|
|
|
def link(self) -> str:
|
|
|
|
if self.ap_id is None:
|
|
|
|
return self.name
|
|
|
|
else:
|
|
|
|
return self.ap_id
|
2023-08-10 02:13:37 -07:00
|
|
|
|
2023-09-17 02:19:51 -07:00
|
|
|
def moderators(self):
|
|
|
|
return CommunityMember.query.filter((CommunityMember.community_id == self.id) &
|
|
|
|
(or_(
|
|
|
|
CommunityMember.is_owner,
|
|
|
|
CommunityMember.is_moderator
|
|
|
|
))
|
|
|
|
).all()
|
|
|
|
|
2023-11-29 09:36:08 -08:00
|
|
|
def is_moderator(self):
|
|
|
|
return any(moderator.user_id == current_user.id for moderator in self.moderators())
|
|
|
|
|
2023-12-07 20:13:38 -08:00
|
|
|
def profile_id(self):
|
|
|
|
return self.ap_profile_id if self.ap_profile_id else f"https://{current_app.config['SERVER_NAME']}/c/{self.name}"
|
|
|
|
|
2023-12-09 01:14:16 -08:00
|
|
|
def is_local(self):
|
|
|
|
return self.ap_id is None or self.profile_id().startswith('https://' + current_app.config['SERVER_NAME'])
|
|
|
|
|
2023-08-05 02:26:24 -07:00
|
|
|
|
2023-10-18 02:23:59 -07:00
|
|
|
user_role = db.Table('user_role',
|
|
|
|
db.Column('user_id', db.Integer, db.ForeignKey('user.id')),
|
|
|
|
db.Column('role_id', db.Integer, db.ForeignKey('role.id')),
|
|
|
|
db.PrimaryKeyConstraint('user_id', 'role_id')
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2023-08-05 02:26:24 -07:00
|
|
|
class User(UserMixin, db.Model):
|
2023-10-03 02:29:13 -07:00
|
|
|
query_class = FullTextSearchQuery
|
2023-08-05 02:26:24 -07:00
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
2023-11-21 23:48:27 -08:00
|
|
|
user_name = db.Column(db.String(255), index=True)
|
2023-08-05 02:26:24 -07:00
|
|
|
email = db.Column(db.String(255), index=True)
|
|
|
|
password_hash = db.Column(db.String(128))
|
|
|
|
verified = db.Column(db.Boolean, default=False)
|
2023-08-25 20:41:11 -07:00
|
|
|
verification_token = db.Column(db.String(16), index=True)
|
2023-08-05 02:26:24 -07:00
|
|
|
banned = db.Column(db.Boolean, default=False)
|
|
|
|
deleted = db.Column(db.Boolean, default=False)
|
|
|
|
about = db.Column(db.Text)
|
|
|
|
keywords = db.Column(db.String(256))
|
|
|
|
show_nsfw = db.Column(db.Boolean, default=False)
|
|
|
|
show_nsfl = db.Column(db.Boolean, default=False)
|
|
|
|
created = db.Column(db.DateTime, default=datetime.utcnow)
|
|
|
|
last_seen = db.Column(db.DateTime, default=datetime.utcnow, index=True)
|
|
|
|
avatar_id = db.Column(db.Integer, db.ForeignKey('file.id'))
|
|
|
|
cover_id = db.Column(db.Integer, db.ForeignKey('file.id'))
|
|
|
|
public_key = db.Column(db.Text)
|
|
|
|
private_key = db.Column(db.Text)
|
|
|
|
newsletter = db.Column(db.Boolean, default=True)
|
|
|
|
bounces = db.Column(db.SmallInteger, default=0)
|
|
|
|
timezone = db.Column(db.String(20))
|
2023-09-10 01:20:53 -07:00
|
|
|
reputation = db.Column(db.Float, default=0.0)
|
2023-08-05 02:26:24 -07:00
|
|
|
stripe_customer_id = db.Column(db.String(50))
|
|
|
|
stripe_subscription_id = db.Column(db.String(50))
|
|
|
|
searchable = db.Column(db.Boolean, default=True)
|
2023-08-25 18:10:01 -07:00
|
|
|
indexable = db.Column(db.Boolean, default=False)
|
2023-10-07 01:32:19 -07:00
|
|
|
bot = db.Column(db.Boolean, default=False)
|
|
|
|
ignore_bots = db.Column(db.Boolean, default=False)
|
2023-11-30 02:21:37 -08:00
|
|
|
unread_notifications = db.Column(db.Integer, default=0)
|
2023-08-05 02:26:24 -07:00
|
|
|
|
2023-09-08 01:04:01 -07:00
|
|
|
avatar = db.relationship('File', foreign_keys=[avatar_id], single_parent=True, cascade="all, delete-orphan")
|
|
|
|
cover = db.relationship('File', foreign_keys=[cover_id], single_parent=True, cascade="all, delete-orphan")
|
|
|
|
|
|
|
|
ap_id = db.Column(db.String(255), index=True) # e.g. username@server
|
|
|
|
ap_profile_id = db.Column(db.String(255), index=True) # e.g. https://server/u/username
|
|
|
|
ap_public_url = db.Column(db.String(255)) # e.g. https://server/u/username
|
2023-08-05 02:26:24 -07:00
|
|
|
ap_fetched_at = db.Column(db.DateTime)
|
|
|
|
ap_followers_url = db.Column(db.String(255))
|
|
|
|
ap_preferred_username = db.Column(db.String(255))
|
|
|
|
ap_manually_approves_followers = db.Column(db.Boolean)
|
|
|
|
ap_deleted_at = db.Column(db.DateTime)
|
|
|
|
ap_inbox_url = db.Column(db.String(255))
|
|
|
|
ap_domain = db.Column(db.String(255))
|
|
|
|
|
|
|
|
search_vector = db.Column(TSVectorType('user_name', 'bio', 'keywords'))
|
|
|
|
activity = db.relationship('ActivityLog', backref='account', lazy='dynamic', cascade="all, delete-orphan")
|
2023-11-29 10:12:17 -08:00
|
|
|
posts = db.relationship('Post', lazy='dynamic', cascade="all, delete-orphan")
|
2023-08-10 02:13:37 -07:00
|
|
|
post_replies = db.relationship('PostReply', backref='author', lazy='dynamic', cascade="all, delete-orphan")
|
2023-08-05 02:26:24 -07:00
|
|
|
|
2023-10-18 02:23:59 -07:00
|
|
|
roles = db.relationship('Role', secondary=user_role, lazy='dynamic', cascade="all, delete")
|
|
|
|
|
2023-08-05 02:26:24 -07:00
|
|
|
def __repr__(self):
|
|
|
|
return '<User {}>'.format(self.user_name)
|
|
|
|
|
|
|
|
def set_password(self, password):
|
|
|
|
self.password_hash = generate_password_hash(password)
|
|
|
|
|
|
|
|
def check_password(self, password):
|
|
|
|
try:
|
|
|
|
result = check_password_hash(self.password_hash, password)
|
|
|
|
return result
|
|
|
|
except Exception:
|
|
|
|
return False
|
|
|
|
|
2023-10-20 19:49:01 -07:00
|
|
|
def display_name(self):
|
|
|
|
if self.deleted is False:
|
|
|
|
return self.user_name
|
|
|
|
else:
|
|
|
|
return '[deleted]'
|
|
|
|
|
2023-08-05 02:26:24 -07:00
|
|
|
def avatar(self, size):
|
|
|
|
digest = md5(self.email.lower().encode('utf-8')).hexdigest()
|
|
|
|
return 'https://www.gravatar.com/avatar/{}?d=identicon&s={}'.format(
|
|
|
|
digest, size)
|
|
|
|
|
2023-11-29 10:12:17 -08:00
|
|
|
@cache.memoize(timeout=500)
|
2023-10-07 01:32:19 -07:00
|
|
|
def avatar_image(self) -> str:
|
|
|
|
if self.avatar_id is not None:
|
|
|
|
if self.avatar.file_path is not None:
|
2023-12-07 20:13:38 -08:00
|
|
|
if self.avatar.file_path.startswith('app/'):
|
|
|
|
return self.avatar.file_path.replace('app/', '/')
|
|
|
|
else:
|
|
|
|
return self.avatar.file_path
|
2023-10-07 01:32:19 -07:00
|
|
|
if self.avatar.source_url is not None:
|
2023-12-07 20:13:38 -08:00
|
|
|
if self.avatar.source_url.startswith('app/'):
|
|
|
|
return self.avatar.source_url.replace('app/', '/')
|
|
|
|
else:
|
|
|
|
return self.avatar.source_url
|
2023-10-07 01:32:19 -07:00
|
|
|
return ''
|
|
|
|
|
2023-11-29 10:12:17 -08:00
|
|
|
@cache.memoize(timeout=500)
|
2023-10-07 01:32:19 -07:00
|
|
|
def cover_image(self) -> str:
|
|
|
|
if self.cover_id is not None:
|
|
|
|
if self.cover.file_path is not None:
|
2023-12-07 20:13:38 -08:00
|
|
|
if self.cover.file_path.startswith('app/'):
|
|
|
|
return self.cover.file_path.replace('app/', '/')
|
|
|
|
else:
|
|
|
|
return self.cover.file_path
|
2023-10-07 01:32:19 -07:00
|
|
|
if self.cover.source_url is not None:
|
2023-12-07 20:13:38 -08:00
|
|
|
if self.cover.source_url.startswith('app/'):
|
|
|
|
return self.cover.source_url.replace('app/', '/')
|
|
|
|
else:
|
|
|
|
return self.cover.source_url
|
2023-10-07 01:32:19 -07:00
|
|
|
return ''
|
|
|
|
|
2023-12-09 01:14:16 -08:00
|
|
|
def is_local(self):
|
|
|
|
return self.ap_id is None or self.ap_profile_id.startswith('https://' + current_app.config['SERVER_NAME'])
|
|
|
|
|
2023-10-10 02:25:37 -07:00
|
|
|
def link(self) -> str:
|
2023-12-09 01:14:16 -08:00
|
|
|
if self.is_local():
|
2023-10-10 02:25:37 -07:00
|
|
|
return self.user_name
|
|
|
|
else:
|
|
|
|
return self.ap_id
|
|
|
|
|
2023-12-09 01:14:16 -08:00
|
|
|
def followers_url(self):
|
|
|
|
if self.ap_followers_url:
|
|
|
|
return self.ap_followers_url
|
|
|
|
else:
|
|
|
|
return self.profile_id() + '/followers'
|
|
|
|
|
2023-08-05 02:26:24 -07:00
|
|
|
def get_reset_password_token(self, expires_in=600):
|
|
|
|
return jwt.encode(
|
|
|
|
{'reset_password': self.id, 'exp': time() + expires_in},
|
|
|
|
current_app.config['SECRET_KEY'],
|
2023-09-02 21:30:20 -07:00
|
|
|
algorithm='HS256')
|
2023-08-05 02:26:24 -07:00
|
|
|
|
|
|
|
def another_account_using_email(self, email):
|
|
|
|
another_account = User.query.filter(User.email == email, User.id != self.id).first()
|
|
|
|
return another_account is not None
|
|
|
|
|
|
|
|
def expires_soon(self):
|
|
|
|
if self.expires is None:
|
|
|
|
return False
|
|
|
|
return self.expires < datetime.utcnow() + timedelta(weeks=1)
|
|
|
|
|
|
|
|
def is_expired(self):
|
|
|
|
if self.expires is None:
|
|
|
|
return True
|
|
|
|
return self.expires < datetime.utcnow()
|
|
|
|
|
|
|
|
def expired_ages_ago(self):
|
|
|
|
if self.expires is None:
|
|
|
|
return True
|
|
|
|
return self.expires < datetime(2019, 9, 1)
|
|
|
|
|
2023-12-03 01:41:15 -08:00
|
|
|
def subscribed(self, community_id: int) -> int:
|
|
|
|
if community_id is None:
|
2023-08-29 03:01:06 -07:00
|
|
|
return False
|
2023-12-03 01:41:15 -08:00
|
|
|
subscription:CommunityMember = CommunityMember.query.filter_by(user_id=self.id, community_id=community_id).first()
|
2023-08-29 03:01:06 -07:00
|
|
|
if subscription:
|
2023-09-08 01:04:01 -07:00
|
|
|
if subscription.is_banned:
|
|
|
|
return SUBSCRIPTION_BANNED
|
|
|
|
elif subscription.is_owner:
|
2023-08-29 03:01:06 -07:00
|
|
|
return SUBSCRIPTION_OWNER
|
|
|
|
elif subscription.is_moderator:
|
|
|
|
return SUBSCRIPTION_MODERATOR
|
|
|
|
else:
|
|
|
|
return SUBSCRIPTION_MEMBER
|
|
|
|
else:
|
2023-12-03 01:41:15 -08:00
|
|
|
join_request = CommunityJoinRequest.query.filter_by(user_id=self.id, community_id=community_id).first()
|
|
|
|
if join_request:
|
|
|
|
return SUBSCRIPTION_PENDING
|
|
|
|
else:
|
|
|
|
return SUBSCRIPTION_NONMEMBER
|
2023-08-29 03:01:06 -07:00
|
|
|
|
2023-09-17 02:19:51 -07:00
|
|
|
def communities(self) -> List[Community]:
|
|
|
|
return Community.query.filter(Community.banned == False).\
|
|
|
|
join(CommunityMember).filter(CommunityMember.is_banned == False).all()
|
|
|
|
|
2023-11-17 01:02:44 -08:00
|
|
|
def profile_id(self):
|
2023-12-07 20:13:38 -08:00
|
|
|
return self.ap_profile_id if self.ap_profile_id else f"https://{current_app.config['SERVER_NAME']}/u/{self.user_name}"
|
2023-11-17 01:02:44 -08:00
|
|
|
|
2023-12-09 01:14:16 -08:00
|
|
|
|
|
|
|
|
2023-11-29 08:14:22 -08:00
|
|
|
def created_recently(self):
|
|
|
|
return self.created and self.created > datetime.utcnow() - timedelta(days=7)
|
|
|
|
|
2023-08-05 02:26:24 -07:00
|
|
|
@staticmethod
|
|
|
|
def verify_reset_password_token(token):
|
|
|
|
try:
|
|
|
|
id = jwt.decode(token, current_app.config['SECRET_KEY'],
|
|
|
|
algorithms=['HS256'])['reset_password']
|
|
|
|
except:
|
|
|
|
return
|
|
|
|
return User.query.get(id)
|
|
|
|
|
2023-10-20 19:49:01 -07:00
|
|
|
def purge_content(self):
|
2023-11-30 02:21:37 -08:00
|
|
|
files = File.query.join(Post).filter(Post.user_id == self.id).all()
|
|
|
|
for file in files:
|
|
|
|
file.delete_from_disk()
|
2023-10-20 19:49:01 -07:00
|
|
|
db.session.query(ActivityLog).filter(ActivityLog.user_id == self.id).delete()
|
|
|
|
db.session.query(PostVote).filter(PostVote.user_id == self.id).delete()
|
|
|
|
db.session.query(PostReplyVote).filter(PostReplyVote.user_id == self.id).delete()
|
|
|
|
db.session.query(PostReply).filter(PostReply.user_id == self.id).delete()
|
|
|
|
db.session.query(FilterKeyword).filter(FilterKeyword.user_id == self.id).delete()
|
|
|
|
db.session.query(Filter).filter(Filter.user_id == self.id).delete()
|
|
|
|
db.session.query(DomainBlock).filter(DomainBlock.user_id == self.id).delete()
|
|
|
|
db.session.query(CommunityJoinRequest).filter(CommunityJoinRequest.user_id == self.id).delete()
|
|
|
|
db.session.query(CommunityMember).filter(CommunityMember.user_id == self.id).delete()
|
|
|
|
db.session.query(CommunityBlock).filter(CommunityBlock.user_id == self.id).delete()
|
|
|
|
db.session.query(CommunityBan).filter(CommunityBan.user_id == self.id).delete()
|
|
|
|
db.session.query(Community).filter(Community.user_id == self.id).delete()
|
|
|
|
db.session.query(Post).filter(Post.user_id == self.id).delete()
|
|
|
|
db.session.query(UserNote).filter(UserNote.user_id == self.id).delete()
|
|
|
|
db.session.query(UserNote).filter(UserNote.target_id == self.id).delete()
|
|
|
|
db.session.query(UserFollowRequest).filter(UserFollowRequest.follow_id == self.id).delete()
|
|
|
|
db.session.query(UserFollowRequest).filter(UserFollowRequest.user_id == self.id).delete()
|
|
|
|
db.session.query(UserBlock).filter(UserBlock.blocked_id == self.id).delete()
|
|
|
|
db.session.query(UserBlock).filter(UserBlock.blocker_id == self.id).delete()
|
|
|
|
db.session.execute(text('DELETE FROM user_role WHERE user_id = :user_id'),
|
|
|
|
{'user_id': self.id})
|
|
|
|
|
|
|
|
|
2023-08-05 02:26:24 -07:00
|
|
|
class ActivityLog(db.Model):
|
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
|
|
user_id = db.Column(db.Integer, db.ForeignKey('user.id'), index=True)
|
|
|
|
activity_type = db.Column(db.String(64))
|
|
|
|
activity = db.Column(db.String(255))
|
|
|
|
timestamp = db.Column(db.DateTime, index=True, default=datetime.utcnow)
|
|
|
|
|
|
|
|
|
|
|
|
class Post(db.Model):
|
2023-10-03 02:29:13 -07:00
|
|
|
query_class = FullTextSearchQuery
|
2023-08-05 02:26:24 -07:00
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
|
|
user_id = db.Column(db.Integer, db.ForeignKey('user.id'), index=True)
|
|
|
|
community_id = db.Column(db.Integer, db.ForeignKey('community.id'), index=True)
|
|
|
|
image_id = db.Column(db.Integer, db.ForeignKey('file.id'), index=True)
|
|
|
|
domain_id = db.Column(db.Integer, db.ForeignKey('domain.id'), index=True)
|
|
|
|
slug = db.Column(db.String(255))
|
|
|
|
title = db.Column(db.String(255))
|
|
|
|
url = db.Column(db.String(2048))
|
|
|
|
body = db.Column(db.Text)
|
2023-08-10 02:13:37 -07:00
|
|
|
body_html = db.Column(db.Text)
|
2023-08-05 02:26:24 -07:00
|
|
|
type = db.Column(db.Integer)
|
2023-09-16 00:09:04 -07:00
|
|
|
comments_enabled = db.Column(db.Boolean, default=True)
|
2023-08-05 02:26:24 -07:00
|
|
|
has_embed = db.Column(db.Boolean, default=False)
|
|
|
|
reply_count = db.Column(db.Integer, default=0)
|
|
|
|
score = db.Column(db.Integer, default=0, index=True)
|
|
|
|
nsfw = db.Column(db.Boolean, default=False)
|
|
|
|
nsfl = db.Column(db.Boolean, default=False)
|
|
|
|
sticky = db.Column(db.Boolean, default=False)
|
2023-11-29 23:57:51 -08:00
|
|
|
notify_author = db.Column(db.Boolean, default=True)
|
2023-08-25 18:10:01 -07:00
|
|
|
indexable = db.Column(db.Boolean, default=False)
|
2023-10-07 01:32:19 -07:00
|
|
|
from_bot = db.Column(db.Boolean, default=False)
|
2023-10-02 02:16:44 -07:00
|
|
|
created_at = db.Column(db.DateTime, index=True, default=datetime.utcnow) # this is when the content arrived here
|
|
|
|
posted_at = db.Column(db.DateTime, index=True, default=datetime.utcnow) # this is when the original server created it
|
2023-08-05 02:26:24 -07:00
|
|
|
last_active = db.Column(db.DateTime, index=True, default=datetime.utcnow)
|
|
|
|
ip = db.Column(db.String(50))
|
|
|
|
up_votes = db.Column(db.Integer, default=0)
|
|
|
|
down_votes = db.Column(db.Integer, default=0)
|
|
|
|
ranking = db.Column(db.Integer, default=0)
|
|
|
|
language = db.Column(db.String(10))
|
|
|
|
edited_at = db.Column(db.DateTime)
|
|
|
|
|
|
|
|
ap_id = db.Column(db.String(255), index=True)
|
2023-08-10 02:13:37 -07:00
|
|
|
ap_create_id = db.Column(db.String(100))
|
|
|
|
ap_announce_id = db.Column(db.String(100))
|
2023-08-05 02:26:24 -07:00
|
|
|
|
|
|
|
search_vector = db.Column(TSVectorType('title', 'body'))
|
|
|
|
|
2023-11-29 10:12:17 -08:00
|
|
|
image = db.relationship(File, lazy='joined', foreign_keys=[image_id], cascade="all, delete")
|
|
|
|
domain = db.relationship('Domain', lazy='joined', foreign_keys=[domain_id])
|
2023-12-03 01:41:15 -08:00
|
|
|
author = db.relationship('User', lazy='joined', overlaps='posts', foreign_keys=[user_id])
|
2023-08-10 02:13:37 -07:00
|
|
|
|
2023-12-09 01:14:16 -08:00
|
|
|
def is_local(self):
|
|
|
|
return self.ap_id is None or self.ap_id.startswith('https://' + current_app.config['SERVER_NAME'])
|
|
|
|
|
2023-09-16 00:09:04 -07:00
|
|
|
@classmethod
|
|
|
|
def get_by_ap_id(cls, ap_id):
|
|
|
|
return cls.query.filter_by(ap_id=ap_id).first()
|
|
|
|
|
2023-11-21 02:05:07 -08:00
|
|
|
def delete_dependencies(self):
|
|
|
|
db.session.execute(text('DELETE FROM post_reply_vote WHERE post_reply_id IN (SELECT id FROM post_reply WHERE post_id = :post_id)'),
|
|
|
|
{'post_id': self.id})
|
|
|
|
db.session.execute(text('DELETE FROM post_reply WHERE post_id = :post_id'), {'post_id': self.id})
|
|
|
|
db.session.execute(text('DELETE FROM post_vote WHERE post_id = :post_id'), {'post_id': self.id})
|
2023-11-29 23:57:51 -08:00
|
|
|
if self.image_id:
|
|
|
|
file = File.query.get(self.image_id)
|
|
|
|
file.delete_from_disk()
|
2023-11-21 02:05:07 -08:00
|
|
|
|
2023-11-28 23:32:07 -08:00
|
|
|
def youtube_embed(self):
|
|
|
|
if self.url:
|
|
|
|
vpos = self.url.find('v=')
|
|
|
|
if vpos != -1:
|
|
|
|
return self.url[vpos + 2:vpos + 13]
|
|
|
|
|
2023-12-07 20:13:38 -08:00
|
|
|
def profile_id(self):
|
|
|
|
return f"https://{current_app.config['SERVER_NAME']}/post/{self.id}"
|
|
|
|
|
2023-08-05 02:26:24 -07:00
|
|
|
|
|
|
|
class PostReply(db.Model):
|
2023-10-03 02:29:13 -07:00
|
|
|
query_class = FullTextSearchQuery
|
2023-08-05 02:26:24 -07:00
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
|
|
user_id = db.Column(db.Integer, db.ForeignKey('user.id'), index=True)
|
|
|
|
post_id = db.Column(db.Integer, db.ForeignKey('post.id'), index=True)
|
|
|
|
community_id = db.Column(db.Integer, db.ForeignKey('community.id'), index=True)
|
2023-10-10 02:25:37 -07:00
|
|
|
domain_id = db.Column(db.Integer, db.ForeignKey('domain.id'), index=True)
|
2023-08-05 02:26:24 -07:00
|
|
|
image_id = db.Column(db.Integer, db.ForeignKey('file.id'), index=True)
|
|
|
|
parent_id = db.Column(db.Integer)
|
|
|
|
root_id = db.Column(db.Integer)
|
2023-10-10 02:25:37 -07:00
|
|
|
depth = db.Column(db.Integer, default=0)
|
2023-08-05 02:26:24 -07:00
|
|
|
body = db.Column(db.Text)
|
2023-08-10 02:13:37 -07:00
|
|
|
body_html = db.Column(db.Text)
|
2023-09-16 00:09:04 -07:00
|
|
|
body_html_safe = db.Column(db.Boolean, default=False)
|
2023-08-05 02:26:24 -07:00
|
|
|
score = db.Column(db.Integer, default=0, index=True)
|
|
|
|
nsfw = db.Column(db.Boolean, default=False)
|
|
|
|
nsfl = db.Column(db.Boolean, default=False)
|
2023-11-29 23:57:51 -08:00
|
|
|
notify_author = db.Column(db.Boolean, default=True)
|
2023-08-05 02:26:24 -07:00
|
|
|
created_at = db.Column(db.DateTime, index=True, default=datetime.utcnow)
|
|
|
|
posted_at = db.Column(db.DateTime, index=True, default=datetime.utcnow)
|
|
|
|
ip = db.Column(db.String(50))
|
2023-10-07 01:32:19 -07:00
|
|
|
from_bot = db.Column(db.Boolean, default=False)
|
2023-08-05 02:26:24 -07:00
|
|
|
up_votes = db.Column(db.Integer, default=0)
|
|
|
|
down_votes = db.Column(db.Integer, default=0)
|
2023-10-10 02:25:37 -07:00
|
|
|
ranking = db.Column(db.Integer, default=0, index=True)
|
2023-08-05 02:26:24 -07:00
|
|
|
language = db.Column(db.String(10))
|
|
|
|
edited_at = db.Column(db.DateTime)
|
|
|
|
|
|
|
|
ap_id = db.Column(db.String(255), index=True)
|
2023-09-16 00:09:04 -07:00
|
|
|
ap_create_id = db.Column(db.String(100))
|
|
|
|
ap_announce_id = db.Column(db.String(100))
|
2023-08-05 02:26:24 -07:00
|
|
|
|
|
|
|
search_vector = db.Column(TSVectorType('body'))
|
|
|
|
|
2023-12-09 01:14:16 -08:00
|
|
|
def is_local(self):
|
|
|
|
return self.ap_id is None or self.ap_id.startswith('https://' + current_app.config['SERVER_NAME'])
|
|
|
|
|
2023-09-16 00:09:04 -07:00
|
|
|
@classmethod
|
|
|
|
def get_by_ap_id(cls, ap_id):
|
|
|
|
return cls.query.filter_by(ap_id=ap_id).first()
|
|
|
|
|
2023-12-07 20:13:38 -08:00
|
|
|
def profile_id(self):
|
|
|
|
return f"https://{current_app.config['SERVER_NAME']}/comment/{self.id}"
|
|
|
|
|
2023-12-09 01:14:16 -08:00
|
|
|
# the ap_id of the parent object, whether it's another PostReply or a Post
|
|
|
|
def in_reply_to(self):
|
|
|
|
if self.parent_id is None:
|
|
|
|
return self.post.ap_id
|
|
|
|
else:
|
|
|
|
parent = PostReply.query.get(self.parent_id)
|
|
|
|
return parent.ap_id
|
|
|
|
|
|
|
|
# the AP profile of the person who wrote the parent object, which could be another PostReply or a Post
|
|
|
|
def to(self):
|
|
|
|
if self.parent_id is None:
|
|
|
|
return self.post.author.profile_id()
|
|
|
|
else:
|
|
|
|
parent = PostReply.query.get(self.parent_id)
|
|
|
|
return parent.author.profile_id()
|
|
|
|
|
2023-08-05 02:26:24 -07:00
|
|
|
|
|
|
|
class Domain(db.Model):
|
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
|
|
name = db.Column(db.String(255), index=True)
|
|
|
|
post_count = db.Column(db.Integer, default=0)
|
2023-09-16 00:09:04 -07:00
|
|
|
banned = db.Column(db.Boolean, default=False, index=True) # Domains can be banned site-wide (by admin) or DomainBlock'ed by users
|
2023-08-05 02:26:24 -07:00
|
|
|
|
|
|
|
|
|
|
|
class DomainBlock(db.Model):
|
|
|
|
user_id = db.Column(db.Integer, db.ForeignKey('user.id'), primary_key=True)
|
|
|
|
domain_id = db.Column(db.Integer, db.ForeignKey('domain.id'), primary_key=True)
|
|
|
|
created_at = db.Column(db.DateTime, default=datetime.utcnow)
|
|
|
|
|
|
|
|
|
|
|
|
class CommunityBlock(db.Model):
|
|
|
|
user_id = db.Column(db.Integer, db.ForeignKey('user.id'), primary_key=True)
|
|
|
|
community_id = db.Column(db.Integer, db.ForeignKey('community.id'), primary_key=True)
|
|
|
|
created_at = db.Column(db.DateTime, default=datetime.utcnow)
|
|
|
|
|
|
|
|
|
|
|
|
class CommunityMember(db.Model):
|
|
|
|
user_id = db.Column(db.Integer, db.ForeignKey('user.id'), primary_key=True)
|
|
|
|
community_id = db.Column(db.Integer, db.ForeignKey('community.id'), primary_key=True)
|
|
|
|
is_moderator = db.Column(db.Boolean, default=False)
|
|
|
|
is_owner = db.Column(db.Boolean, default=False)
|
2023-09-17 02:19:51 -07:00
|
|
|
is_banned = db.Column(db.Boolean, default=False)
|
2023-08-05 02:26:24 -07:00
|
|
|
created_at = db.Column(db.DateTime, default=datetime.utcnow)
|
|
|
|
|
|
|
|
|
2023-09-09 01:46:40 -07:00
|
|
|
# people banned from communities
|
2023-09-08 01:04:01 -07:00
|
|
|
class CommunityBan(db.Model):
|
|
|
|
user_id = db.Column(db.Integer, db.ForeignKey('user.id'), primary_key=True)
|
|
|
|
community_id = db.Column(db.Integer, db.ForeignKey('community.id'), primary_key=True)
|
|
|
|
banned_by = db.Column(db.Integer, db.ForeignKey('user.id'))
|
|
|
|
reason = db.Column(db.String(50))
|
|
|
|
created_at = db.Column(db.DateTime, default=datetime.utcnow)
|
|
|
|
ban_until = db.Column(db.DateTime)
|
|
|
|
|
|
|
|
|
2023-08-05 02:26:24 -07:00
|
|
|
class UserNote(db.Model):
|
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
|
|
user_id = db.Column(db.Integer, db.ForeignKey('user.id'))
|
|
|
|
target_id = db.Column(db.Integer, db.ForeignKey('user.id'))
|
|
|
|
body = db.Column(db.Text)
|
|
|
|
created_at = db.Column(db.DateTime, default=datetime.utcnow)
|
|
|
|
|
|
|
|
|
|
|
|
class UserBlock(db.Model):
|
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
|
|
blocker_id = db.Column(db.Integer, db.ForeignKey('user.id'))
|
|
|
|
blocked_id = db.Column(db.Integer, db.ForeignKey('user.id'))
|
|
|
|
created_at = db.Column(db.DateTime, default=datetime.utcnow)
|
|
|
|
|
|
|
|
|
|
|
|
class BannedInstances(db.Model):
|
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
2023-08-10 02:13:37 -07:00
|
|
|
domain = db.Column(db.String(256), index=True)
|
2023-08-05 02:26:24 -07:00
|
|
|
reason = db.Column(db.String(256))
|
|
|
|
initiator = db.Column(db.String(256))
|
|
|
|
created_at = db.Column(db.DateTime, default=datetime.utcnow)
|
|
|
|
|
|
|
|
|
2023-11-03 00:32:12 -07:00
|
|
|
class AllowedInstances(db.Model):
|
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
|
|
domain = db.Column(db.String(256), index=True)
|
|
|
|
created_at = db.Column(db.DateTime, default=datetime.utcnow)
|
|
|
|
|
|
|
|
|
2023-08-10 02:13:37 -07:00
|
|
|
class Instance(db.Model):
|
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
2023-08-29 03:01:06 -07:00
|
|
|
domain = db.Column(db.String(256), index=True)
|
2023-08-10 02:13:37 -07:00
|
|
|
inbox = db.Column(db.String(256))
|
|
|
|
shared_inbox = db.Column(db.String(256))
|
|
|
|
outbox = db.Column(db.String(256))
|
2023-09-10 01:20:53 -07:00
|
|
|
vote_weight = db.Column(db.Float, default=1.0)
|
2023-11-22 18:10:44 -08:00
|
|
|
software = db.Column(db.String(50))
|
|
|
|
version = db.Column(db.String(50))
|
|
|
|
created_at = db.Column(db.DateTime, default=datetime.utcnow)
|
|
|
|
updated_at = db.Column(db.DateTime, default=datetime.utcnow)
|
2023-08-10 02:13:37 -07:00
|
|
|
|
|
|
|
|
2023-08-05 02:26:24 -07:00
|
|
|
class Settings(db.Model):
|
|
|
|
name = db.Column(db.String(50), primary_key=True)
|
|
|
|
value = db.Column(db.String(1024))
|
|
|
|
|
|
|
|
|
2023-09-05 01:25:02 -07:00
|
|
|
class Interest(db.Model):
|
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
|
|
name = db.Column(db.String(50))
|
|
|
|
communities = db.Column(db.Text)
|
|
|
|
|
|
|
|
|
2023-09-08 01:04:01 -07:00
|
|
|
class CommunityJoinRequest(db.Model):
|
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
|
|
user_id = db.Column(db.Integer, db.ForeignKey('user.id'))
|
|
|
|
community_id = db.Column(db.Integer, db.ForeignKey('community.id'))
|
|
|
|
|
|
|
|
|
|
|
|
class UserFollowRequest(db.Model):
|
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
|
|
user_id = db.Column(db.Integer, db.ForeignKey('user.id'))
|
|
|
|
follow_id = db.Column(db.Integer, db.ForeignKey('user.id'))
|
|
|
|
|
|
|
|
|
2023-09-10 01:20:53 -07:00
|
|
|
class PostVote(db.Model):
|
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
|
|
user_id = db.Column(db.Integer, db.ForeignKey('user.id'))
|
|
|
|
author_id = db.Column(db.Integer, db.ForeignKey('user.id'))
|
|
|
|
post_id = db.Column(db.Integer, db.ForeignKey('post.id'))
|
|
|
|
effect = db.Column(db.Float)
|
|
|
|
created_at = db.Column(db.DateTime, default=datetime.utcnow)
|
2023-11-29 23:57:51 -08:00
|
|
|
post = db.relationship('Post', foreign_keys=[post_id])
|
2023-09-10 01:20:53 -07:00
|
|
|
|
|
|
|
|
|
|
|
class PostReplyVote(db.Model):
|
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
2023-11-24 01:52:42 -08:00
|
|
|
user_id = db.Column(db.Integer, db.ForeignKey('user.id')) # who voted
|
|
|
|
author_id = db.Column(db.Integer, db.ForeignKey('user.id')) # the author of the reply voted on - who's reputation is affected
|
2023-09-10 01:20:53 -07:00
|
|
|
post_reply_id = db.Column(db.Integer, db.ForeignKey('post_reply.id'))
|
|
|
|
effect = db.Column(db.Float)
|
|
|
|
created_at = db.Column(db.DateTime, default=datetime.utcnow)
|
|
|
|
|
|
|
|
|
2023-09-09 01:46:40 -07:00
|
|
|
# save every activity to a log, to aid debugging
|
|
|
|
class ActivityPubLog(db.Model):
|
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
|
|
direction = db.Column(db.String(3)) # 'in' or 'out'
|
2023-09-10 01:20:53 -07:00
|
|
|
activity_id = db.Column(db.String(100), index=True)
|
2023-09-09 01:46:40 -07:00
|
|
|
activity_type = db.Column(db.String(50)) # e.g. 'Follow', 'Accept', 'Like', etc
|
|
|
|
activity_json = db.Column(db.Text) # the full json of the activity
|
|
|
|
result = db.Column(db.String(10)) # 'success' or 'failure'
|
|
|
|
exception_message = db.Column(db.Text)
|
|
|
|
created_at = db.Column(db.DateTime, default=datetime.utcnow)
|
|
|
|
|
|
|
|
|
2023-10-02 02:16:44 -07:00
|
|
|
class Filter(db.Model):
|
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
|
|
title = db.Column(db.String(50))
|
|
|
|
filter_posts = db.Column(db.Boolean, default=True)
|
|
|
|
filter_replies = db.Column(db.Boolean, default=False)
|
|
|
|
hide_type = db.Column(db.Integer, default=0) # 0 = hide with warning, 1 = hide completely
|
|
|
|
user_id = db.Column(db.Integer, db.ForeignKey('user.id'))
|
|
|
|
|
|
|
|
|
|
|
|
class FilterKeyword(db.Model):
|
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
|
|
keyword = db.Column(db.String(100))
|
|
|
|
filter_id = db.Column(db.Integer, db.ForeignKey('filter.id'))
|
|
|
|
user_id = db.Column(db.Integer, db.ForeignKey('user.id'))
|
|
|
|
|
|
|
|
|
2023-10-18 02:23:59 -07:00
|
|
|
class Role(db.Model):
|
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
|
|
name = db.Column(db.String(50))
|
|
|
|
weight = db.Column(db.Integer, default=0)
|
|
|
|
permissions = db.relationship('RolePermission')
|
|
|
|
|
|
|
|
|
|
|
|
class RolePermission(db.Model):
|
|
|
|
role_id = db.Column(db.Integer, db.ForeignKey('role.id'), primary_key=True)
|
|
|
|
permission = db.Column(db.String, primary_key=True, index=True)
|
|
|
|
|
2023-10-02 02:16:44 -07:00
|
|
|
|
2023-11-29 23:57:51 -08:00
|
|
|
class Notification(db.Model):
|
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
|
|
title = db.Column(db.String(50))
|
|
|
|
url = db.Column(db.String(512))
|
|
|
|
read = db.Column(db.Boolean, default=False)
|
|
|
|
user_id = db.Column(db.Integer, db.ForeignKey('user.id'))
|
|
|
|
author_id = db.Column(db.Integer, db.ForeignKey('user.id'))
|
|
|
|
created_at = db.Column(db.DateTime, default=datetime.utcnow)
|
|
|
|
|
|
|
|
|
2023-08-05 02:26:24 -07:00
|
|
|
@login.user_loader
|
|
|
|
def load_user(id):
|
|
|
|
return User.query.get(int(id))
|