open-law/app/models/user.py

87 lines
2.6 KiB
Python

from uuid import uuid4
from flask_login import UserMixin, AnonymousUserMixin
from sqlalchemy import func
from sqlalchemy.ext.hybrid import hybrid_property
from werkzeug.security import generate_password_hash, check_password_hash
from app import db
from app.models.utils import BaseModel
from app.logger import log
from app import schema as s, models as m
def gen_uniq_id() -> str:
return str(uuid4())
class User(BaseModel, UserMixin):
__tablename__ = "users"
username = db.Column(db.String(64), unique=True, default=gen_uniq_id)
password_hash = db.Column(db.String(256), default="")
is_activated = db.Column(db.Boolean, default=False)
wallet_id = db.Column(db.String(64), nullable=True)
avatar_img = db.Column(db.Text, nullable=True)
is_super_user = db.Column(db.Boolean, default=False)
# Relationships
access_groups = db.relationship(
"AccessGroup", secondary="users_access_groups", back_populates="users"
)
stars = db.relationship("Book", secondary="books_stars", back_populates="stars")
books = db.relationship("Book")
notifications = db.relationship("Notification")
@hybrid_property
def password(self):
return self.password_hash
@password.setter
def password(self, password):
self.password_hash = generate_password_hash(password)
@classmethod
def authenticate(cls, user_id, password):
user = cls.query.filter(
func.lower(cls.username) == func.lower(user_id),
).first()
if not user:
log(log.WARNING, "user:[%s] not found", user_id)
if user is not None and check_password_hash(user.password, password):
return user
def __repr__(self):
return f"<{self.id}: {self.username}>"
@property
def json(self):
u = s.User.from_orm(self)
return u.json()
@property
def contributions(self):
contributions = m.Interpretation.query.filter_by(user_id=self.id).all()
comments = m.Comment.query.filter_by(user_id=self.id).all()
for comment in comments:
if comment.parent:
contributions.append(comment.parent.interpretation)
else:
contributions.append(comment.interpretation)
return contributions
@property
def active_notifications(self):
items = [
notification
for notification in self.notifications
if not notification.is_read
]
items.sort(key=lambda x: x.created_at)
return items
class AnonymousUser(AnonymousUserMixin):
pass