microblog/app/models.py

194 lines
7.1 KiB
Python
Raw Normal View History

2017-09-11 21:30:21 +00:00
from datetime import datetime
from hashlib import md5
2017-11-13 07:53:18 +00:00
import json
2017-09-26 07:16:46 +00:00
from time import time
from flask import current_app
2017-09-13 06:31:39 +00:00
from flask_login import UserMixin
from werkzeug.security import generate_password_hash, check_password_hash
2017-09-26 07:16:46 +00:00
import jwt
from app import db, login
2017-09-20 19:51:53 +00:00
from app.search import add_to_index, remove_from_index, query_index
class SearchableMixin(object):
@classmethod
def search(cls, expression, page, per_page):
ids, total = query_index(cls.__tablename__, expression, page, per_page)
if total == 0:
return cls.query.filter_by(id=0), 0
when = []
for i in range(len(ids)):
when.append((ids[i], i))
return cls.query.filter(cls.id.in_(ids)).order_by(
db.case(*when, value=cls.id)), total
@classmethod
def before_commit(cls, session):
session._changes = {
'add': list(session.new),
'update': list(session.dirty),
'delete': list(session.deleted)
}
@classmethod
def after_commit(cls, session):
for obj in session._changes['add']:
if isinstance(obj, SearchableMixin):
add_to_index(obj.__tablename__, obj)
for obj in session._changes['update']:
if isinstance(obj, SearchableMixin):
add_to_index(obj.__tablename__, obj)
for obj in session._changes['delete']:
if isinstance(obj, SearchableMixin):
remove_from_index(obj.__tablename__, obj)
session._changes = None
@classmethod
def reindex(cls):
for obj in cls.query:
add_to_index(cls.__tablename__, obj)
db.event.listen(db.session, 'before_commit', SearchableMixin.before_commit)
db.event.listen(db.session, 'after_commit', SearchableMixin.after_commit)
2017-09-11 21:30:21 +00:00
2017-09-16 07:05:17 +00:00
followers = db.Table(
'followers',
db.Column('follower_id', db.Integer, db.ForeignKey('user.id'),
primary_key=True),
db.Column('followed_id', db.Integer, db.ForeignKey('user.id'),
primary_key=True)
)
2017-09-13 06:31:39 +00:00
class User(UserMixin, db.Model):
2017-09-11 21:30:21 +00:00
id = db.Column(db.Integer, primary_key=True)
username = db.Column(db.String(64), index=True, unique=True)
email = db.Column(db.String(120), index=True, unique=True)
password_hash = db.Column(db.String(128))
posts = db.relationship('Post', back_populates='author', lazy='dynamic')
about_me = db.Column(db.String(140))
last_seen = db.Column(db.DateTime, default=datetime.utcnow)
2017-09-16 07:05:17 +00:00
following = db.relationship(
'User', secondary=followers,
primaryjoin=(followers.c.follower_id == id),
secondaryjoin=(followers.c.followed_id == id),
lazy='dynamic', back_populates='followers')
followers = db.relationship(
'User', secondary=followers,
primaryjoin=(followers.c.followed_id == id),
secondaryjoin=(followers.c.follower_id == id),
lazy='dynamic', back_populates='following')
2017-11-13 07:53:18 +00:00
messages_sent = db.relationship('Message',
foreign_keys='Message.sender_id',
lazy='dynamic', back_populates='author')
messages_received = db.relationship('Message',
foreign_keys='Message.recipient_id',
lazy='dynamic',
back_populates='recipient')
last_message_read_time = db.Column(db.DateTime)
notifications = db.relationship('Notification', lazy='dynamic',
back_populates='user')
2017-09-11 21:30:21 +00:00
def __repr__(self):
return '<User {}>'.format(self.username)
2017-09-13 06:31:39 +00:00
def set_password(self, password):
self.password_hash = generate_password_hash(password)
def check_password(self, password):
return check_password_hash(self.password_hash, password)
def avatar(self, size):
digest = md5(self.email.lower().encode('utf-8')).hexdigest()
return f'https://www.gravatar.com/avatar/{digest}?d=identicon&s={size}'
2017-09-16 07:05:17 +00:00
def follow(self, user):
if not self.is_following(user):
self.following.append(user)
def unfollow(self, user):
if self.is_following(user):
self.following.remove(user)
def is_following(self, user):
return user in self.following
def following_posts(self):
following = Post.query.join(
followers, (followers.c.followed_id == Post.user_id)).filter(
followers.c.follower_id == self.id)
own = Post.query.filter_by(user_id=self.id)
return following.union(own).order_by(Post.timestamp.desc())
2017-09-26 07:16:46 +00: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'], algorithm='HS256')
2017-09-26 07:16:46 +00:00
@staticmethod
def verify_reset_password_token(token):
try:
id = jwt.decode(token, current_app.config['SECRET_KEY'],
2017-09-26 07:16:46 +00:00
algorithms=['HS256'])['reset_password']
except:
return
return User.query.get(id)
2017-11-13 07:53:18 +00:00
def new_messages(self):
last_read_time = self.last_message_read_time or datetime(1900, 1, 1)
return Message.query.filter_by(recipient=self).filter(
Message.timestamp > last_read_time).count()
def add_notification(self, name, data):
self.notifications.filter_by(name=name).delete()
n = Notification(name=name, payload_json=json.dumps(data), user=self)
db.session.add(n)
return n
2017-09-13 06:31:39 +00:00
@login.user_loader
def load_user(id):
return User.query.get(int(id))
2017-09-11 21:30:21 +00:00
2017-09-20 19:51:53 +00:00
class Post(SearchableMixin, db.Model):
__searchable__ = ['body']
2017-09-11 21:30:21 +00:00
id = db.Column(db.Integer, primary_key=True)
body = db.Column(db.String(140))
timestamp = db.Column(db.DateTime, index=True, default=datetime.utcnow)
user_id = db.Column(db.Integer, db.ForeignKey('user.id'), index=True)
2017-10-05 22:34:15 +00:00
language = db.Column(db.String(5))
2017-09-11 21:30:21 +00:00
author = db.relationship('User', back_populates='posts')
def __repr__(self):
return '<Post {}>'.format(self.body)
2017-11-13 07:53:18 +00:00
class Message(db.Model):
id = db.Column(db.Integer, primary_key=True)
sender_id = db.Column(db.Integer, db.ForeignKey('user.id'), index=True)
recipient_id = db.Column(db.Integer, db.ForeignKey('user.id'), index=True)
body = db.Column(db.String(140))
timestamp = db.Column(db.DateTime, index=True, default=datetime.utcnow)
author = db.relationship('User', foreign_keys='Message.sender_id',
back_populates='messages_sent')
recipient = db.relationship('User', foreign_keys='Message.recipient_id',
back_populates='messages_received')
def __repr__(self):
return '<Message {}>'.format(self.body)
class Notification(db.Model):
id = db.Column(db.Integer, primary_key=True)
name = db.Column(db.String(128), index=True)
user_id = db.Column(db.Integer, db.ForeignKey('user.id'), index=True)
timestamp = db.Column(db.Float, index=True, default=time)
payload_json = db.Column(db.Text)
user = db.relationship('User', back_populates='notifications')
def get_data(self):
return json.loads(str(self.payload_json))