open-law/app/models/section.py

133 lines
3.8 KiB
Python
Raw Normal View History

2023-05-18 16:40:18 +03:00
from sqlalchemy import func, text
2023-05-18 16:35:50 +03:00
2023-04-21 15:58:47 +03:00
from app import db
from app.models.utils import BaseModel
2023-05-12 09:55:10 +03:00
from app.controllers import create_breadcrumbs
from .interpretation import Interpretation
from .comment import Comment
2023-05-18 16:35:50 +03:00
from .interpretation_vote import InterpretationVote
2023-04-21 15:58:47 +03:00
class Section(BaseModel):
__tablename__ = "sections"
2023-04-27 16:17:25 +03:00
label = db.Column(db.String(256), unique=False, nullable=False)
2023-04-21 15:58:47 +03:00
# Foreign keys
collection_id = db.Column(db.ForeignKey("collections.id"))
user_id = db.Column(db.ForeignKey("users.id"))
2023-04-21 17:20:22 +03:00
version_id = db.Column(db.ForeignKey("book_versions.id"))
selected_interpretation_id = db.Column(db.Integer, nullable=True)
2023-04-21 15:58:47 +03:00
# Relationships
collection = db.relationship("Collection", viewonly=True)
2023-04-21 17:20:22 +03:00
user = db.relationship("User", viewonly=True)
version = db.relationship("BookVersion", viewonly=True)
2023-05-11 18:02:14 +03:00
interpretations = db.relationship(
"Interpretation", viewonly=True, order_by="desc(Interpretation.id)"
)
2023-05-17 18:39:37 +03:00
tags = db.relationship(
"Tag",
secondary="section_tags",
back_populates="sections",
)
2023-04-21 15:58:47 +03:00
2023-04-27 15:40:48 +03:00
@property
def path(self):
parent = self.collection
grand_parent = parent.parent
path = f"{self.version.book.label} / "
if grand_parent.is_root:
path += f"{parent.label} / "
else:
path += f"{grand_parent.label} / {parent.label} / "
path += self.label
return path
2023-05-12 09:55:10 +03:00
@property
def breadcrumbs_path(self):
parent = self.collection
grand_parent = parent.parent
if grand_parent.is_root:
collection_path = (parent.id,)
else:
collection_path = (
grand_parent.id,
parent.id,
)
breadcrumbs_path = create_breadcrumbs(self.book_id, collection_path)
return breadcrumbs_path
2023-04-28 12:12:46 +03:00
@property
def book_id(self):
_book_id = self.version.book_id
return _book_id
@property
def sub_collection_id(self):
parent = self.collection
grand_parent = parent.parent
if grand_parent.is_root:
_sub_collection_id = parent.id
else:
_sub_collection_id = grand_parent.id
return _sub_collection_id
2023-05-05 12:02:35 +03:00
@property
def active_interpretations(self):
return [
interpretation
for interpretation in self.interpretations
if not interpretation.is_deleted
]
@property
def approved_interpretation(self):
interpretation = Interpretation.query.filter_by(
2023-05-18 17:50:07 +03:00
approved=True, section_id=self.id, is_deleted=False
).first()
2023-05-18 16:35:50 +03:00
if interpretation:
return interpretation
# most upvoted
result = (
db.session.query(
Interpretation, func.count(Interpretation.votes).label("total_votes")
)
.join(InterpretationVote)
2023-05-18 17:50:07 +03:00
.filter(
Interpretation.section_id == self.id, Interpretation.is_deleted is False
)
2023-05-18 16:35:50 +03:00
.group_by(Interpretation.id)
.order_by(text("total_votes DESC"))
).first()
if result:
return result[0]
# oldest
interpretation = (
2023-05-18 17:50:07 +03:00
Interpretation.query.filter_by(section_id=self.id, is_deleted=False)
2023-05-18 16:35:50 +03:00
.order_by(Interpretation.created_at)
.first()
)
if interpretation:
return interpretation
@property
def approved_comments(self):
interpretation_ids = [
interpretation.id for interpretation in self.interpretations
]
comments = (
Comment.query.filter_by(approved=True)
.filter(Comment.interpretation_id.in_(interpretation_ids))
.all()
)
return comments
2023-04-21 15:58:47 +03:00
def __repr__(self):
return f"<{self.id}: {self.label}>"