You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

256 lines
10 KiB

# Copyright (c) 2018 Tildes contributors <code@tildes.net>
# SPDX-License-Identifier: AGPL-3.0-or-later
"""Contains the CommentTree class."""
from datetime import datetime
from typing import Iterator, List, Optional, Sequence
from prometheus_client import Histogram
from tildes.enums import CommentSortOption
from tildes.metrics import get_histogram
from tildes.models.user import User
from .comment import Comment
class CommentTree:
"""Class representing the tree of comments on a particular topic.
The Comment objects held by this class have additional attributes added:
- `replies`: the list of all immediate children to that comment
- `has_visible_descendant`: whether the comment has any visible descendants (if
not, it can be pruned from the tree)
- `collapsed_state`: whether to display this comment in a collapsed state
- `num_children`: how many (visible) children the comment has
"""
def __init__(self, comments: Sequence[Comment], sort: CommentSortOption) -> None:
"""Create a sorted CommentTree from a flat list of Comments."""
self.tree: List[Comment] = []
self.sort = sort
# sort the comments by date, since replies will always be posted later this will
# ensure that parent comments are always processed first
self.comments = sorted(comments, key=lambda c: c.created_time)
for comment in self.comments:
comment.collapsed_state = None
self.comments_by_id = {comment.comment_id: comment for comment in comments}
self._build_tree()
# The method of building the tree already sorts it by posting time, so there's
# no need to sort again if that's the desired sorting. Note also that because
# _sort_tree() uses sorted() which is a stable sort, this means that the
# "secondary sort" will always be by posting time as well.
if self.tree and sort != CommentSortOption.POSTED:
with self._sorting_histogram().time():
self.tree = self._sort_tree(self.tree, self.sort)
self.tree = self._prune_empty_branches(self.tree)
self._count_children()
def _count_children(self) -> None:
"""Set the num_children attr for all comments."""
# work backwards through comments so that all children will have their count
# done first, and we can just sum all the counts from the replies
for comment in reversed(self.comments):
comment.num_children = 0
for reply in comment.replies:
comment.num_children += reply.num_children
if not (reply.is_deleted or reply.is_removed):
comment.num_children += 1
def _build_tree(self) -> None:
"""Build the initial tree from the flat list of Comments."""
for comment in self.comments:
comment.replies = []
comment.has_visible_descendant = False
if comment.parent_comment_id:
parent_comment = self.comments_by_id[comment.parent_comment_id]
parent_comment.replies.append(comment)
# if this comment isn't deleted, work back up towards the root,
# indicating to all parents they have a visible descendant
if not comment.is_deleted:
while not parent_comment.has_visible_descendant:
parent_comment.has_visible_descendant = True
if not parent_comment.parent_comment_id:
break
next_parent_id = parent_comment.parent_comment_id
parent_comment = self.comments_by_id[next_parent_id]
else:
self.tree.append(comment)
@staticmethod
def _sort_tree(tree: List[Comment], sort: CommentSortOption) -> List[Comment]:
"""Sort the tree by the desired ordering (recursively).
Because Python's sorted() function is stable, the ordering of any comments that
compare equal on the sorting method will be the same as the order that they were
originally in when passed to this function.
"""
if sort == CommentSortOption.NEWEST:
tree = sorted(tree, key=lambda c: c.created_time, reverse=True)
elif sort == CommentSortOption.POSTED:
tree = sorted(tree, key=lambda c: c.created_time)
elif sort == CommentSortOption.VOTES:
tree = sorted(tree, key=lambda c: c.num_votes, reverse=True)
for comment in tree:
if not comment.has_visible_descendant:
# no need to bother sorting replies if none will be visible
continue
comment.replies = CommentTree._sort_tree(comment.replies, sort)
return tree
@staticmethod
def _prune_empty_branches(tree: Sequence[Comment]) -> List[Comment]:
"""Remove branches from the tree with no visible comments."""
pruned_tree = []
for comment in tree:
if comment.is_deleted and not comment.has_visible_descendant:
# prune this branch from the tree
continue
# recursively prune the tree of replies to this comment
replies = CommentTree._prune_empty_branches(comment.replies)
comment.replies = replies
pruned_tree.append(comment)
return pruned_tree
def __iter__(self) -> Iterator[Comment]:
"""Iterate over the (top-level) Comments in the tree."""
for comment in self.tree:
yield comment
def __len__(self) -> int:
"""Return the number of comments in the tree (including deleted)."""
return len(self.comments)
@property
def num_top_level(self) -> int:
"""Return the number of top-level comments in the tree."""
return len(self.tree)
@property
def most_recent_comment(self) -> Optional[Comment]:
"""Return the most recent non-deleted Comment in the tree."""
for comment in reversed(self.comments):
if not comment.is_deleted:
return comment
return None
def _sorting_histogram(self) -> Histogram:
"""Return the (labeled) histogram to use for timing the sorting."""
num_comments = len(self.comments)
# make an "order of magnitude" label based on the number of comments
if num_comments == 0:
raise ValueError("Attempting to time an empty comment tree sort")
if num_comments < 10:
num_comments_range = "1 - 9"
elif num_comments < 100:
num_comments_range = "10 - 99"
elif num_comments < 1000:
num_comments_range = "100 - 999"
else:
num_comments_range = "1000+"
return get_histogram(
"comment_tree_sorting",
num_comments_range=num_comments_range,
order=self.sort.name,
)
def collapse_from_tags(self, viewer: Optional[User]) -> None:
"""Collapse comments based on how they've been tagged."""
for comment in self.comments:
# never affect the viewer's own comments
if viewer and comment.user == viewer:
continue
if comment.tag_counts["noise"] >= 2:
comment.collapsed_state = "full"
def collapse_old_comments(self, threshold: datetime) -> None:
"""Collapse old comments in the tree.
Any comments newer than the threshold will be uncollapsed, along with their
direct parents. All comments with any uncollapsed descendant will be collapsed
individually. Branches with no uncollapsed comments will be collapsed fully.
"""
# pylint: disable=too-many-branches
for comment in reversed(self.comments):
# as soon as we reach an old comment, we can stop
if comment.created_time <= threshold:
break
if comment.is_deleted or comment.is_removed:
continue
# don't override any other collapsing decisions
if comment.collapsed_state:
continue
# uncollapse the comment
comment.collapsed_state = "uncollapsed"
# fetch its direct parent and uncollapse it as well
if comment.parent_comment_id:
parent_comment = self.comments_by_id[comment.parent_comment_id]
parent_comment.collapsed_state = "uncollapsed"
@staticmethod
def _has_uncollapsed_descendant(comment: Comment) -> bool:
"""Recursively check if the comment has an uncollapsed descendant."""
for reply in comment.replies:
if reply.collapsed_state == "uncollapsed":
return True
if CommentTree._has_uncollapsed_descendant(reply):
return True
return False
@staticmethod
def _recursively_collapse(comment: Comment) -> None:
"""Recursively collapse a comment and its replies as much as possible."""
# stop processing this branch if we hit an uncollapsed/fully-collapsed comment
if comment.collapsed_state in ("uncollapsed", "full"):
return
# if it doesn't have any uncollapsed descendants, collapse the whole branch
# and stop looking any deeper into it
if not CommentTree._has_uncollapsed_descendant(comment):
comment.collapsed_state = "full"
return
# otherwise (does have uncollapsed descendant), collapse this comment
# individually and recurse into all branches underneath it
comment.collapsed_state = "individual"
for reply in comment.replies:
CommentTree._recursively_collapse(reply)
def finalize_collapsing_maximized(self) -> None:
"""Finish collapsing comments, collapsing as much as possible."""
for comment in self.tree:
self._recursively_collapse(comment)
# if all the top-level comments end up fully collapsed, uncollapse instead
if all([comment.collapsed_state == "full" for comment in self.tree]):
for comment in self.tree:
comment.collapsed_state = None