contentdb/app/models/__init__.py

207 lines
6.4 KiB
Python
Raw Permalink Normal View History

2020-12-10 12:37:15 +01:00
# ContentDB
2021-01-30 17:59:42 +01:00
# Copyright (C) 2018-21 rubenwardy
2020-12-10 12:37:15 +01:00
#
# This program is free software: you can redistribute it and/or modify
2021-01-30 17:59:42 +01:00
# it under the terms of the GNU Affero General Public License as published by
2020-12-10 12:37:15 +01:00
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
2021-01-30 17:59:42 +01:00
# GNU Affero General Public License for more details.
2020-12-10 12:37:15 +01:00
#
2021-01-30 17:59:42 +01:00
# You should have received a copy of the GNU Affero General Public License
2020-12-10 12:37:15 +01:00
# along with this program. If not, see <https://www.gnu.org/licenses/>.
from flask_migrate import Migrate
from flask_sqlalchemy import SQLAlchemy
from sqlalchemy_searchable import make_searchable
from app import app
# Initialise database
db = SQLAlchemy(app)
migrate = Migrate(app, db)
make_searchable(db.metadata)
from .packages import *
from .users import *
from .threads import *
2023-08-14 22:48:50 +02:00
from .collections import *
2020-12-10 12:37:15 +01:00
class APIToken(db.Model):
id = db.Column(db.Integer, primary_key=True)
2020-12-10 17:49:37 +01:00
access_token = db.Column(db.String(34), unique=True, nullable=False)
2020-12-10 12:37:15 +01:00
name = db.Column(db.String(100), nullable=False)
2020-12-10 17:49:37 +01:00
2020-12-10 12:37:15 +01:00
owner_id = db.Column(db.Integer, db.ForeignKey("user.id"), nullable=False)
2020-12-10 17:49:37 +01:00
owner = db.relationship("User", foreign_keys=[owner_id], back_populates="tokens")
2020-12-10 12:37:15 +01:00
created_at = db.Column(db.DateTime, nullable=False, default=datetime.datetime.utcnow)
package_id = db.Column(db.Integer, db.ForeignKey("package.id"), nullable=True)
2020-12-10 17:49:37 +01:00
package = db.relationship("Package", foreign_keys=[package_id], back_populates="tokens")
2020-12-10 12:37:15 +01:00
2023-10-31 19:40:01 +01:00
client_id = db.Column(db.String(24), db.ForeignKey("oauth_client.id"), nullable=True)
client = db.relationship("OAuthClient", foreign_keys=[client_id], back_populates="tokens")
auth_code = db.Column(db.String(34), unique=True, nullable=True)
2023-06-19 20:32:36 +02:00
def can_operate_on_package(self, package):
2023-10-31 19:40:01 +01:00
if self.client is not None:
return False
2020-12-10 12:37:15 +01:00
if self.package and self.package != package:
return False
return package.author == self.owner
class AuditSeverity(enum.Enum):
NORMAL = 0 # Normal user changes
USER = 1 # Security user changes
EDITOR = 2 # Editor changes
MODERATION = 3 # Destructive / moderator changes
def __str__(self):
return self.name
2024-06-08 13:27:05 +02:00
@property
def title(self):
2020-12-10 12:37:15 +01:00
return self.name.replace("_", " ").title()
@classmethod
def choices(cls):
2024-06-08 13:27:05 +02:00
return [(choice, choice.title) for choice in cls]
2020-12-10 12:37:15 +01:00
@classmethod
def coerce(cls, item):
return item if type(item) == AuditSeverity else AuditSeverity[item.upper()]
2020-12-10 12:37:15 +01:00
class AuditLogEntry(db.Model):
id = db.Column(db.Integer, primary_key=True)
created_at = db.Column(db.DateTime, nullable=False, default=datetime.datetime.utcnow)
causer_id = db.Column(db.Integer, db.ForeignKey("user.id"), nullable=True)
2020-12-10 17:49:37 +01:00
causer = db.relationship("User", foreign_keys=[causer_id], back_populates="audit_log_entries")
2020-12-10 12:37:15 +01:00
severity = db.Column(db.Enum(AuditSeverity), nullable=False)
title = db.Column(db.String(100), nullable=False)
url = db.Column(db.String(200), nullable=True)
package_id = db.Column(db.Integer, db.ForeignKey("package.id"), nullable=True)
2020-12-10 17:49:37 +01:00
package = db.relationship("Package", foreign_keys=[package_id], back_populates="audit_log_entries")
2020-12-10 12:37:15 +01:00
description = db.Column(db.Text, nullable=True, default=None)
def __init__(self, causer, severity, title, url, package=None, description=None):
if len(title) > 100:
2023-01-14 00:49:06 +01:00
if description is None:
description = title[99:]
2020-12-10 12:37:15 +01:00
title = title[:99] + ""
self.causer = causer
self.severity = severity
self.title = title
self.url = url
self.package = package
self.description = description
def check_perm(self, user, perm):
if not user.is_authenticated:
return False
if type(perm) == str:
perm = Permission[perm]
elif type(perm) != Permission:
raise Exception("Unknown permission given to AuditLogEntry.check_perm()")
if perm == Permission.VIEW_AUDIT_DESCRIPTION:
2023-06-19 22:27:49 +02:00
return user.rank.at_least(UserRank.APPROVER if self.package is not None else UserRank.MODERATOR)
else:
raise Exception("Permission {} is not related to audit log entries".format(perm.name))
2020-12-10 12:37:15 +01:00
REPO_BLACKLIST = [".zip", "mediafire.com", "dropbox.com", "weebly.com",
"minetest.net", "dropboxusercontent.com", "4shared.com",
"digitalaudioconcepts.com", "hg.intevation.org", "www.wtfpl.net",
"imageshack.com", "imgur.com"]
2020-12-10 17:49:37 +01:00
2020-12-10 12:37:15 +01:00
class ForumTopic(db.Model):
topic_id = db.Column(db.Integer, primary_key=True, autoincrement=False)
2020-12-10 17:49:37 +01:00
2020-12-10 12:37:15 +01:00
author_id = db.Column(db.Integer, db.ForeignKey("user.id"), nullable=False)
author = db.relationship("User", back_populates="forum_topics")
2020-12-10 12:37:15 +01:00
2022-01-22 21:47:43 +01:00
wip = db.Column(db.Boolean, default=False, nullable=False)
# TODO: remove
2022-01-22 21:47:43 +01:00
discarded = db.Column(db.Boolean, default=False, nullable=False)
2020-12-10 12:37:15 +01:00
type = db.Column(db.Enum(PackageType), nullable=False)
title = db.Column(db.String(200), nullable=False)
name = db.Column(db.String(30), nullable=True)
link = db.Column(db.String(200), nullable=True)
posts = db.Column(db.Integer, nullable=False)
views = db.Column(db.Integer, nullable=False)
created_at = db.Column(db.DateTime, nullable=False, default=datetime.datetime.utcnow)
2024-06-22 11:45:14 +02:00
@property
def url(self):
return "https://forum.minetest.net/viewtopic.php?t=" + str(self.topic_id)
2023-06-19 20:32:36 +02:00
def get_repo_url(self):
2020-12-10 12:37:15 +01:00
if self.link is None:
return None
for item in REPO_BLACKLIST:
if item in self.link:
return None
return self.link.replace("repo.or.cz/w/", "repo.or.cz/")
def as_dict(self):
2020-12-10 12:37:15 +01:00
return {
"author": self.author.username,
"name": self.name,
"type": self.type.to_name(),
2020-12-10 12:37:15 +01:00
"title": self.title,
"id": self.topic_id,
"link": self.link,
"posts": self.posts,
"views": self.views,
"is_wip": self.wip,
"created_at": self.created_at.isoformat(),
}
def check_perm(self, user, perm):
2020-12-10 12:37:15 +01:00
if not user.is_authenticated:
return False
if type(perm) == str:
perm = Permission[perm]
elif type(perm) != Permission:
raise Exception("Unknown permission given to ForumTopic.check_perm()")
2020-12-10 12:37:15 +01:00
if perm == Permission.TOPIC_DISCARD:
2023-06-19 22:27:49 +02:00
return self.author == user or user.rank.at_least(UserRank.EDITOR)
2020-12-10 12:37:15 +01:00
else:
raise Exception("Permission {} is not related to topics".format(perm.name))
if app.config.get("LOG_SQL"):
import logging
logging.basicConfig()
logging.getLogger('sqlalchemy.engine').setLevel(logging.INFO)