146 lines
6.6 KiB
Python
146 lines
6.6 KiB
Python
from datetime import datetime
|
|
from flask_login import UserMixin
|
|
from werkzeug.security import generate_password_hash, check_password_hash
|
|
from app import db, login_manager
|
|
|
|
class PartnerSite(db.Model):
|
|
"""合作PT站点模型"""
|
|
__tablename__ = 'partner_sites'
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
name = db.Column(db.String(100), unique=True, nullable=False, index=True)
|
|
url = db.Column(db.String(255), nullable=True)
|
|
is_active = db.Column(db.Boolean, default=True, nullable=False, index=True)
|
|
created_at = db.Column(db.DateTime, default=datetime.utcnow)
|
|
updated_at = db.Column(db.DateTime, default=datetime.utcnow, onupdate=datetime.utcnow)
|
|
|
|
def __repr__(self):
|
|
return f'<PartnerSite {self.name}>'
|
|
|
|
class User(UserMixin, db.Model):
|
|
"""用户模型"""
|
|
__tablename__ = 'users'
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
username = db.Column(db.String(64), unique=True, index=True, nullable=False)
|
|
email = db.Column(db.String(120), unique=True, index=True, nullable=False)
|
|
password_hash = db.Column(db.String(256))
|
|
role = db.Column(db.String(16), default='user', index=True)
|
|
status = db.Column(db.String(16), default='pending', index=True)
|
|
created_at = db.Column(db.DateTime, default=datetime.utcnow)
|
|
updated_at = db.Column(db.DateTime, default=datetime.utcnow, onupdate=datetime.utcnow)
|
|
pt_site = db.Column(db.String(100))
|
|
uid = db.Column(db.String(50))
|
|
reports = db.relationship('Report', backref='reporter', lazy='dynamic')
|
|
comments = db.relationship('Comment', back_populates='author', lazy='dynamic')
|
|
|
|
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 __repr__(self):
|
|
return f'<User {self.username}>'
|
|
|
|
@login_manager.user_loader
|
|
def load_user(user_id):
|
|
return User.query.get(int(user_id))
|
|
|
|
class Report(db.Model):
|
|
"""举报模型"""
|
|
__tablename__ = 'reports'
|
|
__table_args__ = (
|
|
db.Index('idx_report_status_created', 'status', 'created_at'),
|
|
)
|
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
reporter_id = db.Column(db.Integer, db.ForeignKey('users.id', ondelete='CASCADE'), nullable=False)
|
|
reported_pt_site = db.Column(db.String(100), nullable=False)
|
|
reported_username = db.Column(db.String(50))
|
|
reported_email = db.Column(db.String(120), index=True, nullable=False)
|
|
reason_category = db.Column(db.String(16), nullable=False)
|
|
description = db.Column(db.Text, nullable=False)
|
|
status = db.Column(db.String(16), index=True, default='pending')
|
|
created_at = db.Column(db.DateTime, index=True, default=datetime.utcnow)
|
|
updated_at = db.Column(db.DateTime, default=datetime.utcnow, onupdate=datetime.utcnow)
|
|
comments = db.relationship('Comment', backref='report', lazy='dynamic', cascade='all, delete-orphan')
|
|
evidences = db.relationship('Evidence', backref='report', lazy='dynamic', cascade='all, delete-orphan')
|
|
blacklist_entry = db.relationship('Blacklist', backref='report', uselist=False)
|
|
|
|
def __repr__(self):
|
|
return f'<Report {self.id}>'
|
|
|
|
class Evidence(db.Model):
|
|
"""证据模型"""
|
|
__tablename__ = 'evidences'
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
report_id = db.Column(db.Integer, db.ForeignKey('reports.id', ondelete='CASCADE'), nullable=False)
|
|
file_url = db.Column(db.String(1024), nullable=False)
|
|
file_type = db.Column(db.String(16))
|
|
created_at = db.Column(db.DateTime, default=datetime.utcnow)
|
|
updated_at = db.Column(db.DateTime, default=datetime.utcnow, onupdate=datetime.utcnow)
|
|
|
|
def __repr__(self):
|
|
return f'<Evidence {self.id} for Report {self.report_id}>'
|
|
|
|
class Blacklist(db.Model):
|
|
"""黑名单模型"""
|
|
__tablename__ = 'blacklist'
|
|
__table_args__ = (
|
|
db.Index('idx_blacklist_email_status', 'normalized_email', 'status'),
|
|
db.Index('idx_blacklist_username_status', 'username', 'status'),
|
|
)
|
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
username = db.Column(db.String(64), index=True)
|
|
email = db.Column(db.String(120), index=True)
|
|
normalized_email = db.Column(db.String(120), index=True)
|
|
pt_site = db.Column(db.String(100), index=True)
|
|
uid = db.Column(db.String(50))
|
|
report_id = db.Column(db.Integer, db.ForeignKey('reports.id'), unique=True)
|
|
status = db.Column(db.String(16), default='active', index=True)
|
|
created_at = db.Column(db.DateTime, index=True, default=datetime.utcnow)
|
|
updated_at = db.Column(db.DateTime, default=datetime.utcnow, onupdate=datetime.utcnow)
|
|
appeals = db.relationship('Appeal', backref='blacklist_entry', lazy='dynamic')
|
|
|
|
def __repr__(self):
|
|
return f'<Blacklist {self.normalized_email} on {self.pt_site}>'
|
|
|
|
class Comment(db.Model):
|
|
"""评论模型"""
|
|
__tablename__ = 'comments'
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
body = db.Column(db.Text)
|
|
created_at = db.Column(db.DateTime, index=True, default=datetime.utcnow)
|
|
author_id = db.Column(db.Integer, db.ForeignKey('users.id', ondelete='CASCADE'))
|
|
report_id = db.Column(db.Integer, db.ForeignKey('reports.id', ondelete='CASCADE'))
|
|
author = db.relationship('User', back_populates='comments')
|
|
|
|
class Appeal(db.Model):
|
|
"""申诉模型"""
|
|
__tablename__ = 'appeals'
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
reason = db.Column(db.Text, nullable=False)
|
|
status = db.Column(db.String(32), nullable=False, default='awaiting_admin_reply', index=True)
|
|
created_at = db.Column(db.DateTime, index=True, default=datetime.utcnow)
|
|
updated_at = db.Column(db.DateTime, default=datetime.utcnow, onupdate=datetime.utcnow)
|
|
appealer_id = db.Column(db.Integer, db.ForeignKey('users.id', ondelete='CASCADE'))
|
|
blacklist_entry_id = db.Column(db.Integer, db.ForeignKey('blacklist.id', ondelete='SET NULL'))
|
|
messages = db.relationship('AppealMessage', backref='appeal', lazy='dynamic', cascade='all, delete-orphan')
|
|
appealer = db.relationship('User', backref='appeals')
|
|
|
|
def __repr__(self):
|
|
return f'<Appeal {self.id}>'
|
|
|
|
class AppealMessage(db.Model):
|
|
"""申诉消息模型"""
|
|
__tablename__ = 'appeal_messages'
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
body = db.Column(db.Text, nullable=False)
|
|
created_at = db.Column(db.DateTime, index=True, default=datetime.utcnow)
|
|
author_id = db.Column(db.Integer, db.ForeignKey('users.id', ondelete='CASCADE'))
|
|
appeal_id = db.Column(db.Integer, db.ForeignKey('appeals.id', ondelete='CASCADE'))
|
|
author = db.relationship('User', backref='appeal_messages')
|
|
|
|
def __repr__(self):
|
|
return f'<AppealMessage {self.id}>'
|