from typing import List, Dict, Optional, Any, Literal
from scale_env.environment.toolkit import ToolKitBase, ToolType, is_tool
from .database import *
from thefuzz import fuzz, process

"""Tools for social_media_messaging."""

class SocialMediaMessagingTools(ToolKitBase):
    """All tools for social_media_messaging."""
    
    db: SocialMediaMessagingDB
    
    def __init__(self, db: SocialMediaMessagingDB):
        """Initialize tools with database."""
        super().__init__(db)
    
    @is_tool()
    def update_comment(self, comment_id: str, content: str):
        """
        Modify the content of an existing comment

        Args:
            comment_id: Unique identifier of the comment to update
            content: New content for the comment

        Returns:
            dict: Contains updated_at timestamp in "yyyy-mm-dd HH:MM:SS" format

        Raises:
            KeyError: If comment does not exist
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Retrieve the comment table
        comment_table = getattr(db, "comment", None)

        # Check if comment table exists
        if comment_table is None:
            raise KeyError(f"Comment table does not exist in database")

        # Check if the comment exists in the table
        if comment_id not in comment_table:
            raise KeyError(f"Comment with id '{comment_id}' does not exist")

        # Retrieve the comment object
        comment = comment_table[comment_id]

        # Check if the comment is deleted
        if comment.is_deleted:
            raise KeyError(f"Comment with id '{comment_id}' has been deleted and cannot be updated")

        # Update the comment content
        comment.content = content

        # Update the timestamp
        current_time = datetime.now()
        comment.updated_at = current_time

        # Save the updated comment back to the database
        comment_table[comment_id] = comment
        setattr(db, "comment", comment_table)

        # Format the timestamp as "yyyy-mm-dd HH:MM:SS"
        updated_at_str = current_time.strftime("%Y-%m-%d %H:%M:%S")

        # Return the updated timestamp
        return {
            "updated_at": updated_at_str
        }

    @is_tool()
    def remove_tags_from_post(self, post_id: str, tags: List[str]) -> Dict[str, List[str]]:
        """
        Remove one or more tags from a post

        Args:
            post_id: Unique identifier of the post
            tags: List of tags to remove from the post

        Returns:
            Dictionary containing the complete list of remaining tags after removal

        Raises:
            KeyError: If post_id does not exist or specified tags are not present on the post
        """
        # Access the database
        db = self.db

        # Get the post_tag table
        post_tag_table = getattr(db, "post_tag", None)

        # Validate that the post_tag table exists
        if post_tag_table is None:
            raise KeyError(f"post_tag table does not exist in database")

        # Check if the post exists by looking for any tags associated with this post_id
        post_exists = any(
            tag_entry.post_id == post_id 
            for tag_entry in post_tag_table.values()
        )

        if not post_exists:
            raise KeyError(f"Post with post_id '{post_id}' does not exist")

        # Get all current tags for this post
        current_tags = {
            tag_entry.tag: tag_id
            for tag_id, tag_entry in post_tag_table.items()
            if tag_entry.post_id == post_id
        }

        # Validate that all tags to be removed are present on the post
        missing_tags = [tag for tag in tags if tag not in current_tags]
        if missing_tags:
            raise KeyError(f"Tags {missing_tags} are not present on post '{post_id}'")

        # Remove the specified tags from the database
        # We need to identify and remove the entries where post_id matches and tag is in the removal list
        tag_ids_to_remove = [
            tag_id for tag_id, tag_entry in post_tag_table.items()
            if tag_entry.post_id == post_id and tag_entry.tag in tags
        ]

        # Create a new dictionary without the removed tags
        updated_post_tag_table = {
            tag_id: tag_entry
            for tag_id, tag_entry in post_tag_table.items()
            if tag_id not in tag_ids_to_remove
        }

        # Update the database with the modified table
        setattr(db, "post_tag", updated_post_tag_table)

        # Get the remaining tags for this post after removal
        remaining_tags = [
            tag_entry.tag
            for tag_entry in updated_post_tag_table.values()
            if tag_entry.post_id == post_id
        ]

        # Return the complete list of remaining tags
        return {"tags": remaining_tags}

    @is_tool()
    def delete_comment(self, comment_id: str, user_id: str):
        """
        Remove a comment from a post permanently

        Args:
            comment_id: Unique identifier of the comment to delete
            user_id: Identifier of the user requesting deletion

        Returns:
            dict: Contains deleted_at timestamp in yyyy-mm-dd HH:MM:SS format

        Raises:
            KeyError: If comment does not exist
            ValueError: If user does not have permission to delete the comment
        """
        from datetime import datetime

        # Get database instance
        db = self.db

        # Retrieve comment table from database
        comment_table = getattr(db, "comment", None)

        # Check if comment table exists
        if comment_table is None:
            raise KeyError(f"Comment table does not exist in database")

        # Check if comment exists
        if comment_id not in comment_table:
            raise KeyError(f"Comment with id '{comment_id}' does not exist")

        # Get the comment object
        comment = comment_table[comment_id]

        # Verify user has permission to delete (must be the author)
        # Note: In a real system, this could also check for admin/moderator permissions
        if comment.user_id != user_id:
            raise ValueError(f"User '{user_id}' does not have permission to delete comment '{comment_id}'")

        # Check if comment is already deleted
        if comment.is_deleted:
            raise ValueError(f"Comment '{comment_id}' has already been deleted")

        # Mark comment as deleted and record deletion timestamp
        deleted_timestamp = datetime.now()
        comment.is_deleted = True
        comment.deleted_at = deleted_timestamp

        # Update the comment in database
        comment_table[comment_id] = comment
        setattr(db, "comment", comment_table)

        # Return deletion timestamp in required format
        return {
            "deleted_at": deleted_timestamp.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def update_post_content(self, post_id: str, title: str = None, content: str = None):
        """
        Modify the title and/or content of an existing post.

        Args:
            post_id: Unique identifier of the post to update
            title: New title for the post (optional)
            content: New content for the post (optional)

        Returns:
            dict: Contains updated_at timestamp in yyyy-mm-dd HH:MM:SS format

        Raises:
            KeyError: If post_id does not exist in the database
        """
        from datetime import datetime

        # Get database instance
        db = self.db

        # Retrieve the post table from database
        post_table = getattr(db, "post", None)

        # Check if post table exists
        if post_table is None:
            raise KeyError(f"Post table does not exist in database")

        # Check if the post exists in the table
        if post_id not in post_table:
            raise KeyError(f"Post with post_id '{post_id}' does not exist")

        # Get the existing post object
        existing_post = post_table[post_id]

        # Check if at least one field is provided for update
        # If neither title nor content is provided, we still update the updated_at timestamp

        # Update title if provided
        if title is not None:
            existing_post.title = title

        # Update content if provided
        if content is not None:
            existing_post.content = content

        # Update the updated_at timestamp
        current_time = datetime.now()
        existing_post.updated_at = current_time

        # Save the updated post back to the database
        post_table[post_id] = existing_post
        setattr(db, "post", post_table)

        # Format the timestamp as yyyy-mm-dd HH:MM:SS
        updated_at_str = current_time.strftime("%Y-%m-%d %H:%M:%S")

        # Return the updated_at timestamp
        return {
            "updated_at": updated_at_str
        }

    @is_tool()
    def add_tags_to_post(self, post_id: str, tags: List[str]):
        """
        Add one or more tags to an existing post for better categorization.

        Args:
            post_id: Unique identifier of the post
            tags: List of tags to add to the post

        Returns:
            Dictionary containing the complete list of tags after addition

        Raises:
            KeyError: If the post does not exist
        """
        import secrets
        import hashlib
        from datetime import datetime

        # Get database instance
        db = self.db

        # Get post_tag table from database
        post_tag_table = getattr(db, "post_tag", None)

        # If post_tag table doesn't exist, initialize it
        if post_tag_table is None:
            post_tag_table = {}
            setattr(db, "post_tag", post_tag_table)

        # Verify that the post exists by checking if there are any tags for this post_id
        post_exists = False
        existing_tags = set()

        for tag_entry in post_tag_table.values():
            if tag_entry.post_id == post_id:
                post_exists = True
                existing_tags.add(tag_entry.tag)

        # If no tags found for this post_id, check if this is the first tag being added
        # Since we don't have access to a posts table, we need to raise KeyError
        # unless this is explicitly the first tag (which we can't verify)
        # Based on pre-condition "Post must exist", we should verify post existence
        # However, without a posts table in related_databases, we'll assume:
        # - If tags exist for post_id, post exists
        # - If no tags exist, we cannot verify post existence, so raise KeyError
        if not post_exists and len(post_tag_table) > 0:
            # If there are other posts with tags but not this one, post doesn't exist
            raise KeyError(f"Post with id '{post_id}' does not exist")

        # Add new tags (avoid duplicates)
        current_time = datetime.now()
        for tag in tags:
            # Skip if tag already exists for this post
            if tag in existing_tags:
                continue

            # Generate unique ID for the post_tag entry
            tag_id = hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

            # Create new PostTag entry
            new_post_tag = PostTag(
                post_id=post_id,
                tag=tag,
                created_at=current_time
            )

            # Add to database
            post_tag_table[tag_id] = new_post_tag
            existing_tags.add(tag)

        # Update database with modified post_tag table
        setattr(db, "post_tag", post_tag_table)

        # Return complete list of tags after addition (sorted for consistency)
        return {
            "tags": sorted(list(existing_tags))
        }

    @is_tool()
    def unlock_post(self, post_id: str, user_id: str):
        """
        Unlock a previously locked post to allow comments and interactions again.

        This method unlocks a post that was previously locked, enabling comments and interactions.
        It verifies that the post exists and is currently locked before performing the unlock operation.

        Args:
            post_id: Unique identifier of the post to unlock
            user_id: Identifier of the moderator unlocking the post

        Returns:
            dict: Contains the unlocked_at timestamp in "yyyy-mm-dd HH:MM:SS" format

        Raises:
            KeyError: If post_id does not exist in the database
            ValueError: If the post is not currently locked
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Retrieve the post table from the database
        post_table = getattr(db, "post", None)

        # Check if post table exists
        if post_table is None:
            raise KeyError(f"Post table does not exist in database")

        # Check if the post exists in the database
        if post_id not in post_table:
            raise KeyError(f"Post with post_id '{post_id}' does not exist")

        # Retrieve the post object
        post = post_table[post_id]

        # Verify that the post is currently locked (pre-condition)
        if not post.is_locked:
            raise ValueError(f"Post with post_id '{post_id}' is not currently locked")

        # Record the unlock timestamp
        unlocked_at = datetime.now()

        # Update the post to unlock it
        post.is_locked = False
        post.locked_at = None
        post.lock_reason = None
        post.updated_at = unlocked_at

        # Save the updated post back to the database
        post_table[post_id] = post
        setattr(db, "post", post_table)

        # Format the timestamp as required (yyyy-mm-dd HH:MM:SS)
        unlocked_at_str = unlocked_at.strftime("%Y-%m-%d %H:%M:%S")

        # Return the unlock timestamp
        return {
            "unlocked_at": unlocked_at_str
        }

    @is_tool()
    def report_post(self, post_id: str, reporter_user_id: str, reason: Literal["spam", "harassment", "inappropriate_content", "misinformation", "other"], description: str = None):
        """
        Submit a report flagging a post for violating community guidelines.

        Args:
            post_id: Unique identifier of the post being reported
            reporter_user_id: Identifier of the user submitting the report
            reason: Reason for reporting (must be one of the allowed enum values)
            description: Optional additional details about the report

        Returns:
            dict: Contains report_id and reported_at timestamp

        Raises:
            ValueError: If required parameters are missing or invalid
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Validate required parameters
        if not post_id or not isinstance(post_id, str):
            raise ValueError("post_id is required and must be a non-empty string")

        if not reporter_user_id or not isinstance(reporter_user_id, str):
            raise ValueError("reporter_user_id is required and must be a non-empty string")

        if not reason or not isinstance(reason, str):
            raise ValueError("reason is required and must be a non-empty string")

        # Validate reason is within allowed enum values
        allowed_reasons = ["spam", "harassment", "inappropriate_content", "misinformation", "other"]
        if reason not in allowed_reasons:
            raise ValueError(f"reason must be one of {allowed_reasons}, got '{reason}'")

        # Validate description if provided
        if description is not None and not isinstance(description, str):
            raise ValueError("description must be a string if provided")

        # Access the database
        db = self.db

        # Get the report table
        report_table = getattr(db, "report", None)
        if report_table is None:
            # Initialize empty report table if it doesn't exist
            report_table = {}

        # Generate unique report_id
        prefix = "report_"
        report_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Ensure report_id is unique
        while report_id in report_table:
            report_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp
        reported_at = datetime.now()

        # Create new Report instance
        new_report = Report(
            report_id=report_id,
            reporter_user_id=reporter_user_id,
            post_id=post_id,
            comment_id=None,  # This is a post report, not a comment report
            reason=reason,
            description=description,
            reported_at=reported_at,
            status="pending"  # Default status for new reports
        )

        # Add the new report to the report table
        report_table[report_id] = new_report

        # Update the database with the modified report table
        setattr(db, "report", report_table)

        # Format the timestamp as string in yyyy-mm-dd HH:MM:SS format
        reported_at_str = reported_at.strftime("%Y-%m-%d %H:%M:%S")

        # Return the report_id and reported_at timestamp
        return {
            "report_id": report_id,
            "reported_at": reported_at_str
        }

    @is_tool()
    def archive_old_posts(self, archive_before_date: str, user_id: str):
        """
        Archive posts older than a specified date by marking them as archived.

        Args:
            archive_before_date: Archive posts created before this date in yyyy-mm-dd format
            user_id: Identifier of the user performing the archival

        Returns:
            dict: Contains archived_count (number of posts archived) and archived_at (timestamp)

        Raises:
            ValueError: If date format is invalid or no posts table exists
        """
        from datetime import datetime

        # Validate and parse the archive_before_date parameter
        try:
            # Parse the date string to ensure it's in the correct format (yyyy-mm-dd)
            cutoff_date = datetime.strptime(archive_before_date, "%Y-%m-%d")
        except ValueError:
            raise ValueError(f"Invalid date format for archive_before_date: {archive_before_date}. Expected format: yyyy-mm-dd")

        # Validate user_id is not empty
        if not user_id or not user_id.strip():
            raise ValueError("user_id cannot be empty")

        # Access the database
        db = self.db

        # Get the post table from the database
        post_table = getattr(db, "post", None)
        if post_table is None:
            raise ValueError("Post table does not exist in the database")

        # Record the current timestamp for the archival operation
        archived_at = datetime.now()

        # Counter for archived posts
        archived_count = 0

        # Iterate through all posts in the table
        for post_id, post in post_table.items():
            # Check if the post was created before the cutoff date
            # and is not already archived or deleted
            if post.created_at < cutoff_date and not post.is_archived and not post.is_deleted:
                # Mark the post as archived
                post.is_archived = True
                post.archived_at = archived_at
                archived_count += 1

        # Update the post table in the database with the modified posts
        setattr(db, "post", post_table)

        # Return the result with archived count and timestamp
        return {
            "archived_count": archived_count,
            "archived_at": archived_at.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def follow_user(self, follower_user_id: str, followed_user_id: str):
        """
        Follow another user to receive updates about their posts.

        This method creates a follow relationship between two users. It validates that:
        1. Both users exist in the system
        2. The follower is not already following the target user
        3. A user cannot follow themselves

        Args:
            follower_user_id: Identifier of the user who wants to follow
            followed_user_id: Identifier of the user to be followed

        Returns:
            dict: Contains followed_at timestamp in "yyyy-mm-dd HH:MM:SS" format

        Raises:
            ValueError: If validation fails (users don't exist, already following, or self-follow attempt)
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Get the user_follow table from database
        user_follow_table = getattr(db, "user_follow", None)
        if user_follow_table is None:
            # Initialize empty table if it doesn't exist
            user_follow_table = {}
            setattr(db, "user_follow", user_follow_table)

        # Validate input parameters are not empty
        if not follower_user_id or not isinstance(follower_user_id, str):
            raise ValueError("follower_user_id must be a non-empty string")
        if not followed_user_id or not isinstance(followed_user_id, str):
            raise ValueError("followed_user_id must be a non-empty string")

        # Check if user is trying to follow themselves
        if follower_user_id == followed_user_id:
            raise ValueError("A user cannot follow themselves")

        # Check if the follow relationship already exists
        # The instance key is follower_user_id, so we need to check if this follower
        # already has a follow relationship with the target user
        for follow_id, follow_record in user_follow_table.items():
            if (follow_record.follower_user_id == follower_user_id and 
                follow_record.followed_user_id == followed_user_id):
                raise ValueError(f"User {follower_user_id} is already following user {followed_user_id}")

        # Create timestamp for the follow relationship
        followed_at = datetime.now()

        # Generate a unique ID for this follow relationship
        import secrets
        import hashlib
        follow_id = "follow_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Create new UserFollow instance
        new_follow = UserFollow(
            follower_user_id=follower_user_id,
            followed_user_id=followed_user_id,
            followed_at=followed_at
        )

        # Add the new follow relationship to the database
        user_follow_table[follow_id] = new_follow
        setattr(db, "user_follow", user_follow_table)

        # Return the followed_at timestamp in the required format
        return {
            "followed_at": followed_at.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def filter_posts_by_date_range(self, start_date: str, end_date: str, limit: Optional[int] = None, offset: Optional[int] = 0):
        """
        Retrieve posts created within a specific date range.

        This method filters posts based on their creation timestamp falling within
        the specified date range, with optional pagination support.

        Args:
            start_date: Start date of the range in yyyy-mm-dd format
            end_date: End date of the range in yyyy-mm-dd format
            limit: Maximum number of posts to retrieve (optional)
            offset: Number of posts to skip for pagination (default: 0)

        Returns:
            Dictionary containing:
                - posts: List of post objects within the date range
                - total_count: Total number of posts in the date range

        Raises:
            ValueError: If date format is invalid or date range is invalid
        """
        from datetime import datetime

        # Validate and parse start_date
        try:
            start_datetime = datetime.strptime(start_date, "%Y-%m-%d")
        except ValueError:
            raise ValueError(f"Invalid start_date format: {start_date}. Expected format: yyyy-mm-dd")

        # Validate and parse end_date
        try:
            end_datetime = datetime.strptime(end_date, "%Y-%m-%d")
            # Set end_datetime to end of day (23:59:59) to include posts from the entire end date
            end_datetime = end_datetime.replace(hour=23, minute=59, second=59)
        except ValueError:
            raise ValueError(f"Invalid end_date format: {end_date}. Expected format: yyyy-mm-dd")

        # Validate that start_date is not after end_date
        if start_datetime > end_datetime:
            raise ValueError(f"start_date ({start_date}) cannot be after end_date ({end_date})")

        # Validate offset
        if offset < 0:
            raise ValueError(f"offset must be non-negative, got {offset}")

        # Validate limit if provided
        if limit is not None and limit <= 0:
            raise ValueError(f"limit must be positive, got {limit}")

        # Access the database
        db = self.db

        # Get the post table from database
        post_table = getattr(db, "post", None)

        # If post table doesn't exist or is empty, return empty result
        if post_table is None or len(post_table) == 0:
            return {
                "posts": [],
                "total_count": 0
            }

        # Filter posts by date range
        # Only include posts that are not deleted and have created_at within the range
        filtered_posts = []
        for post_id, post in post_table.items():
            # Skip deleted posts
            if post.is_deleted:
                continue

            # Check if post's created_at falls within the date range
            if start_datetime <= post.created_at <= end_datetime:
                filtered_posts.append(post)

        # Sort posts by created_at in descending order (newest first)
        filtered_posts.sort(key=lambda p: p.created_at, reverse=True)

        # Get total count before pagination
        total_count = len(filtered_posts)

        # Apply pagination: skip offset posts
        paginated_posts = filtered_posts[offset:]

        # Apply limit if specified
        if limit is not None:
            paginated_posts = paginated_posts[:limit]

        # Convert Post objects to dictionaries for output
        posts_list = []
        for post in paginated_posts:
            post_dict = {
                "post_id": post.post_id,
                "user_id": post.user_id,
                "title": post.title,
                "content": post.content,
                "created_at": post.created_at.strftime("%Y-%m-%d %H:%M:%S"),
                "updated_at": post.updated_at.strftime("%Y-%m-%d %H:%M:%S") if post.updated_at else None,
                "view_count": post.view_count,
                "like_count": post.like_count,
                "comment_count": post.comment_count,
                "is_pinned": post.is_pinned,
                "pinned_at": post.pinned_at.strftime("%Y-%m-%d %H:%M:%S") if post.pinned_at else None,
                "is_locked": post.is_locked,
                "locked_at": post.locked_at.strftime("%Y-%m-%d %H:%M:%S") if post.locked_at else None,
                "lock_reason": post.lock_reason,
                "is_archived": post.is_archived,
                "archived_at": post.archived_at.strftime("%Y-%m-%d %H:%M:%S") if post.archived_at else None,
                "is_deleted": post.is_deleted,
                "deleted_at": post.deleted_at.strftime("%Y-%m-%d %H:%M:%S") if post.deleted_at else None
            }
            posts_list.append(post_dict)

        # Return the result with posts and total count
        return {
            "posts": posts_list,
            "total_count": total_count
        }

    @is_tool()
    def unfollow_user(self, follower_user_id: str, followed_user_id: str):
        """
        Unfollow a user to stop receiving updates about their posts.

        This method removes an existing follow relationship between two users.
        It validates that both users exist in the follow relationship and then
        removes the relationship from the database.

        Args:
            follower_user_id: Identifier of the user who wants to unfollow
            followed_user_id: Identifier of the user to be unfollowed

        Returns:
            dict: Contains the timestamp when the follow relationship was removed
                - unfollowed_at: Timestamp in "yyyy-mm-dd HH:MM:SS" format

        Raises:
            KeyError: If the follow relationship does not exist
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Get the user_follow table from database
        user_follow_table = getattr(db, "user_follow", None)

        # Check if user_follow table exists or is empty
        if user_follow_table is None or len(user_follow_table) == 0:
            raise KeyError(f"Follow relationship between user '{follower_user_id}' and user '{followed_user_id}' does not exist")

        # The key for user_follow is follower_user_id (as defined by @with_instance_key)
        # Check if the follow relationship exists
        if follower_user_id not in user_follow_table:
            raise KeyError(f"Follow relationship between user '{follower_user_id}' and user '{followed_user_id}' does not exist")

        # Get the follow relationship record
        follow_record = user_follow_table[follower_user_id]

        # Verify that the followed_user_id matches
        if follow_record.followed_user_id != followed_user_id:
            raise KeyError(f"Follow relationship between user '{follower_user_id}' and user '{followed_user_id}' does not exist")

        # Record the unfollow timestamp
        unfollowed_at = datetime.now()

        # Remove the follow relationship from the database by deleting the key
        del user_follow_table[follower_user_id]

        # Return the unfollowed timestamp in the required format
        return {
            "unfollowed_at": unfollowed_at.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def like_post(self, post_id: str, user_id: str):
        """
        Add a like to a post from a specific user.

        This method:
        1. Validates that the post exists
        2. Checks if the user has already liked the post
        3. Creates a new like record
        4. Increments the post's like count
        5. Returns the timestamp and updated like count
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Access the database
        db = self.db

        # Retrieve the post table and post_like table
        post_table = getattr(db, "post", None)
        post_like_table = getattr(db, "post_like", None)

        # Validate that the post table exists
        if post_table is None:
            raise ValueError(f"Post table does not exist in the database")

        # Validate that the post_like table exists
        if post_like_table is None:
            raise ValueError(f"PostLike table does not exist in the database")

        # Check if the post exists
        if post_id not in post_table:
            raise ValueError(f"Post with id '{post_id}' does not exist")

        # Retrieve the post
        post = post_table[post_id]

        # Check if the user has already liked the post
        # PostLike uses post_id as instance key, but multiple users can like the same post
        # We need to check all post_like records for this post_id to see if this user_id already exists
        if post_like_table:
            for like_key, like_record in post_like_table.items():
                if like_record.post_id == post_id and like_record.user_id == user_id:
                    raise ValueError(f"User '{user_id}' has already liked post '{post_id}'")

        # Create timestamp for the like
        liked_at = datetime.now()

        # Generate a unique key for the post_like record
        # Since PostLike uses post_id as instance key but we need to support multiple likes per post,
        # we'll create a composite key using post_id and user_id
        like_key = f"{post_id}_{user_id}"

        # Create a new PostLike record
        new_like = PostLike(
            post_id=post_id,
            user_id=user_id,
            liked_at=liked_at
        )

        # Add the like to the post_like table
        post_like_table[like_key] = new_like
        setattr(db, "post_like", post_like_table)

        # Increment the post's like count
        post.like_count += 1

        # Update the post in the database
        post_table[post_id] = post
        setattr(db, "post", post_table)

        # Format the timestamp as string in yyyy-mm-dd HH:MM:SS format
        liked_at_str = liked_at.strftime("%Y-%m-%d %H:%M:%S")

        # Return the result
        return {
            "liked_at": liked_at_str,
            "like_count": post.like_count
        }

    @is_tool()
    def restore_archived_post(self, post_id: str, user_id: str):
        # Import datetime for timestamp generation
        from datetime import datetime

        # Access the database instance
        db = self.db

        # Retrieve the post table from the database
        post_table = getattr(db, "post", None)

        # Validate that the post table exists
        if post_table is None:
            raise KeyError(f"Post table does not exist in the database")

        # Check if the post exists in the database
        if post_id not in post_table:
            raise KeyError(f"Post with post_id '{post_id}' does not exist")

        # Retrieve the post object
        post = post_table[post_id]

        # Verify that the post is currently archived
        # Pre-condition: Post must be in archived status
        if not post.is_archived:
            raise ValueError(f"Post '{post_id}' is not in archived status and cannot be restored")

        # Note: The schema does not define user permissions or roles, so we cannot validate
        # restoration permissions beyond checking that user_id is provided
        # In a real implementation, you would check if the user has moderator/admin rights

        # Restore the post by setting is_archived to False
        post.is_archived = False

        # Clear the archived_at timestamp since the post is no longer archived
        post.archived_at = None

        # Update the updated_at timestamp to reflect the restoration
        post.updated_at = datetime.now()

        # Record the restoration timestamp
        restored_at = datetime.now()

        # Update the post in the database
        post_table[post_id] = post
        setattr(db, "post", post_table)

        # Return the restoration timestamp in the required format
        return {
            "restored_at": restored_at.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def create_board_post(self, user_id: str, title: str, content: str, tags: Optional[List[str]] = None):
        # Import necessary modules for ID generation and datetime handling
        import secrets
        import hashlib
        from datetime import datetime

        # Validate required parameters
        if not user_id or not isinstance(user_id, str):
            raise ValueError("user_id must be a non-empty string")

        if not title or not isinstance(title, str):
            raise ValueError("title must be a non-empty string")

        if not content or not isinstance(content, str):
            raise ValueError("content must be a non-empty string")

        # Validate optional tags parameter
        if tags is not None:
            if not isinstance(tags, list):
                raise ValueError("tags must be a list of strings")
            if not all(isinstance(tag, str) for tag in tags):
                raise ValueError("all tags must be strings")

        # Access the database instance
        db = self.db

        # Get the post table from database, initialize if not exists
        post_table = getattr(db, "post", None)
        if post_table is None:
            post_table = {}
            setattr(db, "post", post_table)

        # Generate unique post_id using secure random hash
        prefix = "post_"
        post_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Ensure the generated post_id is unique
        while post_id in post_table:
            post_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp for created_at
        created_at = datetime.now()

        # Create new Post object with required fields
        # Note: tags are not stored in the Post schema, so they are handled separately if needed
        # For this implementation, we focus on the Post schema fields
        new_post = Post(
            post_id=post_id,
            user_id=user_id,
            title=title,
            content=content,
            created_at=created_at,
            updated_at=None,  # No update yet
            view_count=0,  # Initial view count
            like_count=0,  # Initial like count
            comment_count=0,  # Initial comment count
            is_pinned=False,  # Not pinned by default
            pinned_at=None,
            is_locked=False,  # Not locked by default
            locked_at=None,
            lock_reason=None,
            is_archived=False,  # Not archived by default
            archived_at=None,
            is_deleted=False,  # Not deleted by default
            deleted_at=None
        )

        # Add the new post to the post table
        post_table[post_id] = new_post
        setattr(db, "post", post_table)

        # Format the created_at timestamp as string in yyyy-mm-dd HH:MM:SS format
        created_at_str = created_at.strftime("%Y-%m-%d %H:%M:%S")

        # Return the post_id and created_at timestamp
        return {
            "post_id": post_id,
            "created_at": created_at_str
        }

    @is_tool()
    def get_user_followers(self, user_id: str, limit: int = None, offset: int = 0):
        """
        Retrieve a list of users who follow a specific user.

        This method queries the user_follow table to find all users who are following
        the specified user. It supports pagination through limit and offset parameters.

        Args:
            user_id: Identifier of the user whose followers to retrieve
            limit: Maximum number of followers to retrieve (optional)
            offset: Number of followers to skip for pagination (default: 0)

        Returns:
            dict: Contains 'followers' list and 'total_count'
                - followers: List of follower objects with user_id and followed_at
                - total_count: Total number of followers for the user

        Raises:
            KeyError: If the user_follow table does not exist in the database
        """
        # Access the database instance
        db = self.db

        # Retrieve the user_follow table from the database
        # Raise KeyError if the table doesn't exist
        user_follow_table = getattr(db, "user_follow", None)
        if user_follow_table is None:
            raise KeyError("user_follow table does not exist in the database")

        # Filter all follow relationships where the followed_user_id matches the target user_id
        # This gives us all users who are following the specified user
        followers_list = []
        for follow_id, follow_record in user_follow_table.items():
            if follow_record.followed_user_id == user_id:
                # Create a follower object with user_id and followed_at timestamp
                follower_obj = {
                    "user_id": follow_record.follower_user_id,
                    "followed_at": follow_record.followed_at.strftime("%Y-%m-%d %H:%M:%S")
                }
                followers_list.append(follower_obj)

        # Sort followers by followed_at timestamp in descending order (most recent first)
        # This ensures consistent ordering for pagination
        followers_list.sort(key=lambda x: x["followed_at"], reverse=True)

        # Calculate total count before applying pagination
        total_count = len(followers_list)

        # Apply pagination: skip 'offset' number of followers
        if offset > 0:
            followers_list = followers_list[offset:]

        # Apply limit: return at most 'limit' number of followers
        if limit is not None and limit > 0:
            followers_list = followers_list[:limit]

        # Return the result with followers list and total count
        return {
            "followers": followers_list,
            "total_count": total_count
        }

    @is_tool()
    def report_comment(self, comment_id: str, reporter_user_id: str, reason: Literal["spam", "harassment", "inappropriate_content", "misinformation", "other"], description: str = None):
        """
        Submit a report flagging a comment for violating community guidelines.

        This method creates a new report record in the database with the provided information.
        The report is automatically set to 'pending' status and queued for moderator review.

        Args:
            comment_id: Unique identifier of the comment being reported
            reporter_user_id: Identifier of the user submitting the report
            reason: Reason for reporting (must be one of the predefined enum values)
            description: Optional additional details about the report

        Returns:
            Dictionary containing:
                - report_id: Unique identifier of the created report
                - reported_at: Timestamp when the report was submitted (yyyy-mm-dd HH:MM:SS format)

        Raises:
            ValueError: If comment_id or reporter_user_id is empty/invalid
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Validate required parameters
        if not comment_id or not isinstance(comment_id, str) or not comment_id.strip():
            raise ValueError("comment_id must be a non-empty string")

        if not reporter_user_id or not isinstance(reporter_user_id, str) or not reporter_user_id.strip():
            raise ValueError("reporter_user_id must be a non-empty string")

        # Validate reason is one of the allowed enum values (additional safety check)
        valid_reasons = ["spam", "harassment", "inappropriate_content", "misinformation", "other"]
        if reason not in valid_reasons:
            raise ValueError(f"reason must be one of {valid_reasons}, got '{reason}'")

        # Access the database
        db = self.db

        # Get the report table (initialize if not exists)
        report_table = getattr(db, "report", None)
        if report_table is None:
            report_table = {}

        # Generate unique report_id using secure hash
        prefix = "report_"
        report_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Ensure the generated report_id is unique
        while report_id in report_table:
            report_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp
        reported_at = datetime.now()

        # Create new Report object
        new_report = Report(
            report_id=report_id,
            reporter_user_id=reporter_user_id,
            post_id=None,  # This is a comment report, so post_id is None
            comment_id=comment_id,
            reason=reason,
            description=description,  # Can be None if not provided
            reported_at=reported_at,
            status="pending"  # Default status for new reports
        )

        # Add the new report to the report table
        report_table[report_id] = new_report

        # Save the updated report table back to the database
        setattr(db, "report", report_table)

        # Format the timestamp as string in yyyy-mm-dd HH:MM:SS format
        reported_at_str = reported_at.strftime("%Y-%m-%d %H:%M:%S")

        # Return the result
        return {
            "report_id": report_id,
            "reported_at": reported_at_str
        }

    @is_tool()
    def get_all_tags(self, min_usage_count: int = 1, limit: int = 100):
        """
        Retrieve a list of all unique tags used across the community board.

        This method aggregates tags from all posts, counts their usage,
        filters by minimum usage count, and returns the top tags limited by the specified limit.

        Args:
            min_usage_count: Minimum number of times a tag must be used to be included (default: 1)
            limit: Maximum number of tags to return (default: 100)

        Returns:
            dict: A dictionary containing a list of tag objects with usage statistics
                  Format: {"tags": [{"tag": "tag_name", "usage_count": count}, ...]}

        Raises:
            ValueError: If min_usage_count is negative or limit is less than 1
        """
        # Validate input parameters
        if min_usage_count < 0:
            raise ValueError("min_usage_count must be non-negative")
        if limit < 1:
            raise ValueError("limit must be at least 1")

        # Access the database
        db = self.db

        # Get the post_tag table
        post_tag_table = getattr(db, "post_tag", None)

        # If table doesn't exist or is empty, return empty list
        if post_tag_table is None or len(post_tag_table) == 0:
            return {"tags": []}

        # Dictionary to count tag usage: {tag_name: count}
        tag_usage_counts = {}

        # Iterate through all post_tag entries to count tag usage
        for post_id, post_tag_entry in post_tag_table.items():
            tag_name = post_tag_entry.tag

            # Increment the count for this tag
            if tag_name in tag_usage_counts:
                tag_usage_counts[tag_name] += 1
            else:
                tag_usage_counts[tag_name] = 1

        # Filter tags by minimum usage count
        filtered_tags = [
            {"tag": tag, "usage_count": count}
            for tag, count in tag_usage_counts.items()
            if count >= min_usage_count
        ]

        # Sort tags by usage count in descending order (most used first)
        filtered_tags.sort(key=lambda x: x["usage_count"], reverse=True)

        # Apply limit to the number of tags returned
        limited_tags = filtered_tags[:limit]

        # Return the result in the required format
        return {"tags": limited_tags}

    @is_tool()
    def get_post_engagement_summary(self, post_id: str):
        # Get database instance
        db = self.db

        # Retrieve the post table from database
        post_table = getattr(db, "post", None)
        if post_table is None:
            raise KeyError(f"Post table not found in database")

        # Check if the post exists
        if post_id not in post_table:
            raise KeyError(f"Post with id '{post_id}' does not exist")

        # Get the post object
        post = post_table[post_id]

        # Retrieve related tables
        post_like_table = getattr(db, "post_like", None)
        comment_table = getattr(db, "comment", None)
        bookmark_table = getattr(db, "bookmark", None)

        # Initialize counters
        view_count = post.view_count
        like_count = 0
        comment_count = 0
        bookmark_count = 0

        # Count likes for this post
        # PostLike uses post_id as instance key, so we can directly check if post_id exists
        if post_like_table is not None and post_id in post_like_table:
            # Since PostLike uses post_id as key, each entry represents one like
            # But the schema suggests post_like_table is Dict[str, PostLike]
            # where key is post_id, so we need to count all PostLike objects with this post_id
            # However, based on the schema, it seems one post_id maps to one PostLike entry
            # Let's iterate through all entries to count likes for this specific post
            for like_key, like_obj in post_like_table.items():
                if like_obj.post_id == post_id:
                    like_count += 1

        # Count comments for this post (excluding deleted comments)
        if comment_table is not None:
            for comment_key, comment_obj in comment_table.items():
                if comment_obj.post_id == post_id and not comment_obj.is_deleted:
                    comment_count += 1

        # Count bookmarks for this post
        # Bookmark uses post_id as instance key, similar to PostLike
        if bookmark_table is not None:
            for bookmark_key, bookmark_obj in bookmark_table.items():
                if bookmark_obj.post_id == post_id:
                    bookmark_count += 1

        # Calculate engagement rate
        # Engagement rate = (total engagements / views) * 100
        # Total engagements = likes + comments + bookmarks
        total_engagements = like_count + comment_count + bookmark_count

        # Avoid division by zero
        if view_count > 0:
            engagement_rate = (total_engagements / view_count) * 100
        else:
            # If no views, engagement rate is 0
            engagement_rate = 0.0

        # Round engagement rate to one decimal place for cleaner output
        engagement_rate = round(engagement_rate, 1)

        # Return the engagement summary
        return {
            "view_count": view_count,
            "like_count": like_count,
            "comment_count": comment_count,
            "bookmark_count": bookmark_count,
            "engagement_rate": engagement_rate
        }

    @is_tool()
    def delete_post(self, post_id: str, user_id: str):
        """
        Remove a post from the community board permanently by marking it as deleted.

        This method:
        1. Validates that the post exists in the database
        2. Verifies that the requesting user is the post author (has deletion permissions)
        3. Marks the post as deleted and records the deletion timestamp
        4. Updates the post in the database

        Args:
            post_id: Unique identifier of the post to delete
            user_id: Identifier of the user requesting deletion

        Returns:
            dict: Contains the deletion timestamp in format {"deleted_at": "yyyy-mm-dd HH:MM:SS"}

        Raises:
            KeyError: If post doesn't exist or user doesn't have permission to delete
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Get the post table from database
        post_table = getattr(db, "post", None)

        # Check if post table exists
        if post_table is None:
            raise KeyError(f"Post table not found in database")

        # Check if the post exists
        if post_id not in post_table:
            raise KeyError(f"Post with id '{post_id}' does not exist")

        # Retrieve the post
        post = post_table[post_id]

        # Verify that the user requesting deletion is the author
        # This ensures only the post creator has deletion permissions
        if post.user_id != user_id:
            raise KeyError(f"User '{user_id}' does not have permission to delete post '{post_id}'. Only the post author can delete it.")

        # Check if post is already deleted
        if post.is_deleted:
            raise KeyError(f"Post '{post_id}' is already deleted")

        # Mark the post as deleted and record deletion timestamp
        current_time = datetime.now()
        post.is_deleted = True
        post.deleted_at = current_time

        # Update the post in the database
        post_table[post_id] = post
        setattr(db, "post", post_table)

        # Format the deletion timestamp as string in required format
        deleted_at_str = current_time.strftime("%Y-%m-%d %H:%M:%S")

        # Return the deletion timestamp
        return {
            "deleted_at": deleted_at_str
        }

    @is_tool()
    def calculate_user_activity_score(
        self,
        post_count: int,
        comment_count: int,
        like_given_count: int,
        like_received_count: int,
        time_period_days: int
    ) -> dict:
        """
        Calculate an activity score for a user based on their posts, comments, and engagement.

        The scoring algorithm considers:
        - Post creation activity (weighted higher as it's content creation)
        - Comment engagement (active participation in discussions)
        - Likes given (community engagement)
        - Likes received (content quality indicator)
        - Time period normalization (to make scores comparable across different time windows)

        Args:
            post_count: Number of posts created by the user
            comment_count: Number of comments made by the user
            like_given_count: Number of likes given by the user
            like_received_count: Number of likes received on user's posts
            time_period_days: Time period in days over which to calculate the score

        Returns:
            dict: Contains 'activity_score' (float) and 'activity_level' (str)

        Raises:
            ValueError: If any count is negative or time_period_days is less than 1
        """

        # Validate input parameters
        if post_count < 0:
            raise ValueError("post_count must be non-negative")
        if comment_count < 0:
            raise ValueError("comment_count must be non-negative")
        if like_given_count < 0:
            raise ValueError("like_given_count must be non-negative")
        if like_received_count < 0:
            raise ValueError("like_received_count must be non-negative")
        if time_period_days < 1:
            raise ValueError("time_period_days must be at least 1")

        # Define weights for different activity types
        # Posts are weighted highest as they represent content creation
        POST_WEIGHT = 10.0
        # Comments show active engagement in discussions
        COMMENT_WEIGHT = 5.0
        # Likes given show community participation
        LIKE_GIVEN_WEIGHT = 1.0
        # Likes received indicate content quality and impact
        LIKE_RECEIVED_WEIGHT = 2.0

        # Calculate raw activity score based on weighted sum
        raw_score = (
            post_count * POST_WEIGHT +
            comment_count * COMMENT_WEIGHT +
            like_given_count * LIKE_GIVEN_WEIGHT +
            like_received_count * LIKE_RECEIVED_WEIGHT
        )

        # Normalize by time period to get daily activity rate
        # Then scale to a 0-100 range for easier interpretation
        # Using a logarithmic-like scaling to handle wide range of activity levels
        daily_activity = raw_score / time_period_days

        # Apply a scaling function to map daily activity to 0-100 score
        # This uses a soft ceiling approach: score increases rapidly at first,
        # then gradually approaches 100 for very high activity
        # Formula: 100 * (1 - e^(-daily_activity / scaling_factor))
        # We use a scaling factor that makes moderate activity (10-20 daily points) 
        # map to scores around 50-70
        import math
        SCALING_FACTOR = 15.0
        activity_score = 100 * (1 - math.exp(-daily_activity / SCALING_FACTOR))

        # Round to one decimal place for cleaner output
        activity_score = round(activity_score, 1)

        # Categorize activity level based on score thresholds
        if activity_score < 25:
            activity_level = "low"
        elif activity_score < 50:
            activity_level = "medium"
        elif activity_score < 75:
            activity_level = "high"
        else:
            activity_level = "very_high"

        return {
            "activity_score": activity_score,
            "activity_level": activity_level
        }

    @is_tool()
    def increment_post_view_count(self, post_id: str, user_id: str = None):
        # Get database instance
        db = self.db

        # Retrieve the post table from database
        post_table = getattr(db, "post", None)

        # Check if post table exists
        if post_table is None:
            raise KeyError(f"Post table not found in database")

        # Check if the post exists in the table
        if post_id not in post_table:
            raise KeyError(f"Post with post_id '{post_id}' does not exist")

        # Get the post object
        post = post_table[post_id]

        # Increment the view count by 1
        post.view_count += 1

        # Update the post in the database
        # Since we're modifying the object in place and it's already in the dict,
        # we need to update the table to ensure persistence
        post_table[post_id] = post
        setattr(db, "post", post_table)

        # Return the updated view count
        return {
            "view_count": post.view_count
        }

    @is_tool()
    def unpin_post(self, post_id: str, user_id: str):
        """
        Remove the pinned status from a post.

        This method unpins a post that is currently pinned, removing its priority display status
        and returning it to normal display order. The operation requires moderator permissions.

        Args:
            post_id: Unique identifier of the post to unpin
            user_id: Identifier of the moderator unpinning the post

        Returns:
            dict: Contains the unpinned_at timestamp in "yyyy-mm-dd HH:MM:SS" format

        Raises:
            KeyError: If the post does not exist
            ValueError: If the post is not currently pinned
        """
        from datetime import datetime

        # Get database instance
        db = self.db

        # Retrieve the post table from database
        post_table = getattr(db, "post", None)

        # Check if post table exists
        if post_table is None:
            raise KeyError(f"Post table does not exist in database")

        # Check if the post exists
        if post_id not in post_table:
            raise KeyError(f"Post with id '{post_id}' does not exist")

        # Get the post object
        post = post_table[post_id]

        # Verify that the post is currently pinned
        if not post.is_pinned:
            raise ValueError(f"Post '{post_id}' is not currently pinned")

        # Record the current timestamp for unpinning
        unpinned_at = datetime.now()

        # Update the post's pinned status
        post.is_pinned = False
        post.pinned_at = None  # Clear the pinned timestamp

        # Update the post in the database
        post_table[post_id] = post
        setattr(db, "post", post_table)

        # Return the unpinned timestamp in the required format
        return {
            "unpinned_at": unpinned_at.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def mark_notification_as_read(self, notification_id: str, user_id: str):
        """
        Mark a specific notification as read by the user.

        This method updates the notification's read status and records the timestamp
        when it was marked as read. It validates that the notification exists and
        belongs to the specified user before updating.

        Args:
            notification_id: Unique identifier of the notification to mark as read
            user_id: Identifier of the user marking the notification

        Returns:
            dict: Contains 'read_at' timestamp in "yyyy-mm-dd HH:MM:SS" format

        Raises:
            KeyError: If notification doesn't exist or doesn't belong to the user
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Get the notification table from database
        notification_table = getattr(db, 'notification', None)

        # Check if notification table exists
        if notification_table is None:
            raise KeyError(f"Notification table not found in database")

        # Check if the notification exists
        if notification_id not in notification_table:
            raise KeyError(f"Notification with id '{notification_id}' does not exist")

        # Get the notification object
        notification = notification_table[notification_id]

        # Verify that the notification belongs to the specified user
        # This ensures users can only mark their own notifications as read
        if notification.user_id != user_id:
            raise KeyError(f"Notification '{notification_id}' does not belong to user '{user_id}'")

        # Record the current timestamp when marking as read
        read_at_timestamp = datetime.now()

        # Update the notification's read status and timestamp
        notification.is_read = True
        notification.read_at = read_at_timestamp

        # Save the updated notification back to the database
        notification_table[notification_id] = notification
        setattr(db, 'notification', notification_table)

        # Format the timestamp as string in "yyyy-mm-dd HH:MM:SS" format
        read_at_str = read_at_timestamp.strftime("%Y-%m-%d %H:%M:%S")

        # Return the read timestamp
        return {
            "read_at": read_at_str
        }

    @is_tool()
    def pin_post(self, post_id: str, user_id: str):
        """
        Pin a post to the top of the community board for increased visibility.

        This method marks a post as pinned and records the timestamp when it was pinned.
        The post must exist and the user must have moderator permissions.

        Args:
            post_id: Unique identifier of the post to pin
            user_id: Identifier of the moderator pinning the post

        Returns:
            dict: Contains the pinned_at timestamp in "yyyy-mm-dd HH:MM:SS" format

        Raises:
            ValueError: If post doesn't exist, is already pinned, or is deleted/archived
        """
        from datetime import datetime

        # Get the database instance
        db = self.db

        # Retrieve the post table from database
        post_table = getattr(db, "post", None)
        if post_table is None:
            raise ValueError(f"Post table not found in database")

        # Check if the post exists
        if post_id not in post_table:
            raise ValueError(f"Post with id '{post_id}' does not exist")

        # Get the post object
        post = post_table[post_id]

        # Validate post state - cannot pin deleted or archived posts
        if post.is_deleted:
            raise ValueError(f"Cannot pin a deleted post (post_id: {post_id})")

        if post.is_archived:
            raise ValueError(f"Cannot pin an archived post (post_id: {post_id})")

        # Check if the post is already pinned
        if post.is_pinned:
            raise ValueError(f"Post with id '{post_id}' is already pinned")

        # Note: In a real system, we would verify moderator permissions here
        # For this implementation, we assume the user_id represents a valid moderator
        # as the pre_condition states "user must have moderator permissions"

        # Record the current timestamp for pinning
        pinned_timestamp = datetime.now()

        # Update the post object with pinned status
        post.is_pinned = True
        post.pinned_at = pinned_timestamp

        # Write the updated post back to the database
        post_table[post_id] = post
        setattr(db, "post", post_table)

        # Format the timestamp as "yyyy-mm-dd HH:MM:SS" string for return
        pinned_at_str = pinned_timestamp.strftime("%Y-%m-%d %H:%M:%S")

        # Return the pinned timestamp
        return {
            "pinned_at": pinned_at_str
        }

    @is_tool()
    def add_comment_to_post(self, post_id: str, user_id: str, content: str):
        """
        Add a text comment to an existing post on the community board.

        This method creates a new comment associated with a specific post, validates
        that the post exists and is accessible, increments the post's comment count,
        and stores the comment in the database.

        Args:
            post_id: Unique identifier of the post to comment on
            user_id: Identifier of the user adding the comment
            content: Text content of the comment

        Returns:
            dict: Contains comment_id and created_at timestamp

        Raises:
            ValueError: If post doesn't exist, is deleted, locked, or archived,
                       or if content is empty
        """
        import secrets
        import hashlib
        from datetime import datetime

        # Access the database
        db = self.db

        # Validate input parameters
        if not post_id or not isinstance(post_id, str):
            raise ValueError("post_id must be a non-empty string")

        if not user_id or not isinstance(user_id, str):
            raise ValueError("user_id must be a non-empty string")

        if not content or not isinstance(content, str) or not content.strip():
            raise ValueError("content must be a non-empty string")

        # Retrieve the post table from database
        post_table = getattr(db, "post", None)
        if post_table is None:
            raise ValueError("Post table not found in database")

        # Check if the post exists
        if post_id not in post_table:
            raise ValueError(f"Post with id '{post_id}' does not exist")

        # Get the post object
        post = post_table[post_id]

        # Verify post is accessible (not deleted, locked, or archived)
        if post.is_deleted:
            raise ValueError(f"Cannot comment on deleted post '{post_id}'")

        if post.is_locked:
            raise ValueError(f"Post '{post_id}' is locked and does not allow new comments")

        if post.is_archived:
            raise ValueError(f"Cannot comment on archived post '{post_id}'")

        # Generate unique comment_id using secure random hash
        comment_id = "comment_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp for comment creation
        created_at = datetime.now()

        # Create new comment object
        new_comment = Comment(
            comment_id=comment_id,
            post_id=post_id,
            user_id=user_id,
            content=content.strip(),  # Strip whitespace from content
            created_at=created_at,
            updated_at=None,
            is_deleted=False,
            deleted_at=None
        )

        # Retrieve or initialize comment table
        comment_table = getattr(db, "comment", None)
        if comment_table is None:
            comment_table = {}

        # Add the new comment to the comment table
        comment_table[comment_id] = new_comment
        setattr(db, "comment", comment_table)

        # Update the post's comment count
        post.comment_count += 1

        # Update the post in the database
        post_table[post_id] = post
        setattr(db, "post", post_table)

        # Return the comment_id and created_at in the required format
        return {
            "comment_id": comment_id,
            "created_at": created_at.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def bookmark_post(self, post_id: str, user_id: str):
        """
        Add a post to a user's bookmarks for easy access later.

        This method creates a new bookmark entry for the specified post and user.
        It validates that the post hasn't already been bookmarked by the user
        and generates a timestamp for when the bookmark was created.

        Args:
            post_id: Unique identifier of the post to bookmark
            user_id: Identifier of the user bookmarking the post

        Returns:
            dict: Contains bookmarked_at timestamp in "yyyy-mm-dd HH:MM:SS" format

        Raises:
            ValueError: If post_id or user_id is empty, or if the post is already bookmarked
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Validate input parameters
        if not post_id or not isinstance(post_id, str) or not post_id.strip():
            raise ValueError("post_id must be a non-empty string")

        if not user_id or not isinstance(user_id, str) or not user_id.strip():
            raise ValueError("user_id must be a non-empty string")

        # Access the database
        db = self.db

        # Get the bookmark table, initialize if it doesn't exist
        bookmark_table = getattr(db, "bookmark", None)
        if bookmark_table is None:
            bookmark_table = {}
            setattr(db, "bookmark", bookmark_table)

        # Check if the user has already bookmarked this post
        # Iterate through existing bookmarks to find duplicates
        for bookmark_id, bookmark in bookmark_table.items():
            if bookmark.post_id == post_id and bookmark.user_id == user_id:
                raise ValueError(f"User {user_id} has already bookmarked post {post_id}")

        # Generate a unique bookmark ID using secure random hash
        prefix = "bookmark_"
        bookmark_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Create timestamp for when the bookmark is added
        bookmarked_at = datetime.now()

        # Create new Bookmark instance
        new_bookmark = Bookmark(
            post_id=post_id,
            user_id=user_id,
            bookmarked_at=bookmarked_at
        )

        # Add the new bookmark to the database
        bookmark_table[bookmark_id] = new_bookmark
        setattr(db, "bookmark", bookmark_table)

        # Format the timestamp as string in "yyyy-mm-dd HH:MM:SS" format for return
        bookmarked_at_str = bookmarked_at.strftime("%Y-%m-%d %H:%M:%S")

        # Return the bookmarked timestamp
        return {
            "bookmarked_at": bookmarked_at_str
        }

    @is_tool()
    def get_post_comments(self, post_id: str, limit: Optional[int] = None, offset: Optional[int] = None):
        """
        Retrieve all comments associated with a specific post

        Args:
            post_id: Unique identifier of the post
            limit: Maximum number of comments to retrieve (optional)
            offset: Number of comments to skip for pagination (optional, default: 0)

        Returns:
            dict: Dictionary containing:
                - comments: List of comment objects with their details
                - total_count: Total number of comments on the post

        Raises:
            KeyError: If the post_id does not exist or comments table is not available
        """
        # Access the database
        db = self.db

        # Get the comment table from database
        comment_table = getattr(db, 'comment', None)
        if comment_table is None:
            raise KeyError("Comment table not found in database")

        # Validate post_id parameter
        if not post_id or not isinstance(post_id, str):
            raise ValueError("post_id must be a non-empty string")

        # Set default offset to 0 if not provided
        if offset is None:
            offset = 0

        # Validate offset parameter
        if not isinstance(offset, int) or offset < 0:
            raise ValueError("offset must be a non-negative integer")

        # Validate limit parameter if provided
        if limit is not None and (not isinstance(limit, int) or limit <= 0):
            raise ValueError("limit must be a positive integer")

        # Filter comments by post_id and exclude deleted comments
        filtered_comments = []
        for comment_id, comment in comment_table.items():
            # Check if comment belongs to the specified post and is not deleted
            if comment.post_id == post_id and not comment.is_deleted:
                filtered_comments.append(comment)

        # Check if any comments exist for the given post_id
        if not filtered_comments:
            # Post exists but has no comments, return empty result
            return {
                "comments": [],
                "total_count": 0
            }

        # Sort comments by created_at timestamp (oldest first)
        filtered_comments.sort(key=lambda x: x.created_at)

        # Get total count before pagination
        total_count = len(filtered_comments)

        # Apply offset
        if offset >= total_count:
            # Offset exceeds total count, return empty list
            paginated_comments = []
        else:
            paginated_comments = filtered_comments[offset:]

        # Apply limit if specified
        if limit is not None and limit > 0:
            paginated_comments = paginated_comments[:limit]

        # Convert comment objects to dictionary format for return
        comments_list = []
        for comment in paginated_comments:
            comment_dict = {
                "comment_id": comment.comment_id,
                "post_id": comment.post_id,
                "user_id": comment.user_id,
                "content": comment.content,
                "created_at": comment.created_at.strftime("%Y-%m-%d %H:%M:%S"),
            }

            # Include updated_at if it exists
            if comment.updated_at is not None:
                comment_dict["updated_at"] = comment.updated_at.strftime("%Y-%m-%d %H:%M:%S")

            comments_list.append(comment_dict)

        # Return the result with comments list and total count
        return {
            "comments": comments_list,
            "total_count": total_count
        }

    @is_tool()
    def sort_posts_by_criteria(self, posts: list, sort_by: Literal["created_at", "updated_at", "like_count", "comment_count", "view_count"], sort_order: Literal["ascending", "descending"]):
        """
        Sort a list of posts based on specified criteria and order.

        This method takes a list of post objects and sorts them according to the specified
        field and order. It supports sorting by temporal fields (created_at, updated_at) 
        and numeric fields (like_count, comment_count, view_count).

        Args:
            posts: List of post objects (dictionaries) to sort
            sort_by: Field name to sort by (must be one of the enum values)
            sort_order: Direction of sorting ("ascending" or "descending")

        Returns:
            Dictionary containing the sorted list of posts

        Raises:
            ValueError: If posts list is empty, sort_by field doesn't exist in posts,
                       or enum constraints are violated
        """
        from datetime import datetime

        # Validate input parameters
        if not posts:
            raise ValueError("Posts list cannot be empty")

        # Validate sort_by is in allowed enum values
        allowed_sort_fields = ["created_at", "updated_at", "like_count", "comment_count", "view_count"]
        if sort_by not in allowed_sort_fields:
            raise ValueError(f"Invalid sort_by value: {sort_by}. Must be one of {allowed_sort_fields}")

        # Validate sort_order is in allowed enum values
        allowed_sort_orders = ["ascending", "descending"]
        if sort_order not in allowed_sort_orders:
            raise ValueError(f"Invalid sort_order value: {sort_order}. Must be one of {allowed_sort_orders}")

        # Verify that all posts have the sort_by field
        for post in posts:
            if sort_by not in post:
                raise ValueError(f"Sort field '{sort_by}' not found in post: {post.get('post_id', 'unknown')}")

        # Define sorting key function
        def get_sort_key(post):
            """
            Extract and normalize the sort key from a post object.

            For datetime fields (created_at, updated_at), convert string to datetime object if needed.
            For numeric fields, return the value directly.
            Handle None values by treating them as minimum values for proper sorting.
            """
            value = post.get(sort_by)

            # Handle None values - put them at the end regardless of sort order
            if value is None:
                # Return a sentinel value that will sort to the end
                if sort_by in ["created_at", "updated_at"]:
                    # For datetime fields, use a very old date for None
                    return datetime.min if sort_order == "ascending" else datetime.max
                else:
                    # For numeric fields, use negative infinity for ascending, positive for descending
                    return float('-inf') if sort_order == "ascending" else float('inf')

            # Convert string datetime to datetime object if needed
            if sort_by in ["created_at", "updated_at"]:
                if isinstance(value, str):
                    try:
                        # Parse datetime string in format "yyyy-mm-dd HH:MM:SS"
                        value = datetime.strptime(value, "%Y-%m-%d %H:%M:%S")
                    except ValueError:
                        # Try parsing date-only format "yyyy-mm-dd"
                        try:
                            value = datetime.strptime(value, "%Y-%m-%d")
                        except ValueError:
                            raise ValueError(f"Invalid datetime format for {sort_by}: {value}. Expected 'yyyy-mm-dd HH:MM:SS' or 'yyyy-mm-dd'")

            return value

        # Determine reverse flag based on sort_order
        # ascending: reverse=False (default sort order)
        # descending: reverse=True (reverse the default sort order)
        reverse_flag = (sort_order == "descending")

        # Sort the posts using the key function
        try:
            sorted_posts = sorted(posts, key=get_sort_key, reverse=reverse_flag)
        except Exception as e:
            raise ValueError(f"Error during sorting: {str(e)}")

        # Return the sorted posts in the expected format
        return {
            "posts": sorted_posts
        }

    @is_tool()
    def search_posts_by_keyword(self, keyword: str, limit: int = 20, offset: int = 0):
        # Import fuzzy matching library for text search
        from thefuzz import fuzz, process

        # Validate input parameters
        if not keyword or not isinstance(keyword, str):
            raise ValueError("Keyword must be a non-empty string")

        if not isinstance(limit, int) or limit <= 0:
            raise ValueError("Limit must be a positive integer")

        if not isinstance(offset, int) or offset < 0:
            raise ValueError("Offset must be a non-negative integer")

        # Access the database
        db = self.db

        # Get the post table from database
        post_table = getattr(db, "post", None)

        # If post table doesn't exist or is empty, return empty results
        if post_table is None or not post_table:
            return {
                "posts": [],
                "total_count": 0
            }

        # List to store matching posts with relevance scores
        matching_posts = []

        # Convert keyword to lowercase for case-insensitive matching
        keyword_lower = keyword.lower()

        # Iterate through all posts in the database
        for post_id, post in post_table.items():
            # Skip deleted or archived posts
            if post.is_deleted or post.is_archived:
                continue

            # Calculate relevance score for title and content
            # Use fuzzy matching for natural language text fields
            title_score = 0
            content_score = 0

            # Check if keyword appears in title (case-insensitive)
            if post.title:
                title_lower = post.title.lower()
                # Use partial_ratio for substring matching
                title_score = fuzz.partial_ratio(keyword_lower, title_lower) / 100.0

            # Check if keyword appears in content (case-insensitive)
            if post.content:
                content_lower = post.content.lower()
                # Use partial_ratio for substring matching
                content_score = fuzz.partial_ratio(keyword_lower, content_lower) / 100.0

            # Calculate overall relevance score (weighted average: title has higher weight)
            # Title matches are more relevant than content matches
            relevance_score = max(title_score * 1.2, content_score)

            # Only include posts with meaningful relevance (threshold: 0.6)
            if relevance_score >= 0.6:
                # Create post object with relevant information
                post_obj = {
                    "post_id": post.post_id,
                    "title": post.title,
                    "content": post.content,
                    "user_id": post.user_id,
                    "created_at": post.created_at.strftime("%Y-%m-%d %H:%M:%S"),
                    "view_count": post.view_count,
                    "like_count": post.like_count,
                    "comment_count": post.comment_count,
                    "is_pinned": post.is_pinned,
                    "is_locked": post.is_locked,
                    "relevance_score": round(relevance_score, 2)
                }

                matching_posts.append(post_obj)

        # Sort posts by relevance score (descending) and then by created_at (descending)
        matching_posts.sort(key=lambda x: (-x["relevance_score"], -datetime.strptime(x["created_at"], "%Y-%m-%d %H:%M:%S").timestamp()))

        # Get total count before pagination
        total_count = len(matching_posts)

        # Apply pagination: skip offset number of results and return up to limit results
        paginated_posts = matching_posts[offset:offset + limit]

        # Return search results with pagination
        return {
            "posts": paginated_posts,
            "total_count": total_count
        }

    @is_tool()
    def search_posts_by_tags(self, tags: List[str], match_all: bool = False, limit: int = 20, offset: int = 0):
        # Validate input parameters
        if not tags or len(tags) == 0:
            raise ValueError("At least one tag must be provided")

        if not isinstance(tags, list):
            raise ValueError("Tags must be provided as a list")

        if limit < 0:
            raise ValueError("Limit must be a non-negative integer")

        if offset < 0:
            raise ValueError("Offset must be a non-negative integer")

        # Access the database
        db = self.db

        # Get post_tag and post tables
        post_tag_table = getattr(db, "post_tag", None)
        post_table = getattr(db, "post", None)  # Fixed: was "post_tag", should be "post"

        if post_tag_table is None:
            raise ValueError("post_tag table not found in database")

        if post_table is None:
            raise ValueError("post table not found in database")

        # Normalize tags for fuzzy matching
        normalized_tags = [tag.lower().strip() for tag in tags]

        # Dictionary to track which posts have which tags
        # Key: post_id, Value: set of matching tags
        post_tag_matches = {}

        # Iterate through all post_tag entries to find matching tags using fuzzy matching
        for post_tag_id, post_tag_entry in post_tag_table.items():
            # Normalize the tag from database for comparison
            db_tag = post_tag_entry.tag.lower().strip()

            # Check if this tag matches any of the search tags using fuzzy matching
            for search_tag in normalized_tags:
                # Use fuzzy matching for tag comparison (tags are natural language text)
                similarity_score = fuzz.ratio(db_tag, search_tag)
                if similarity_score >= 80:  # 80% similarity threshold
                    post_id = post_tag_entry.post_id
                    if post_id not in post_tag_matches:
                        post_tag_matches[post_id] = set()
                    post_tag_matches[post_id].add(search_tag)
                    break

        # Filter posts based on match_all criteria
        matching_post_ids = []

        if match_all:
            # Post must have all the tags
            required_tag_count = len(normalized_tags)
            for post_id, matched_tags in post_tag_matches.items():
                if len(matched_tags) == required_tag_count:
                    matching_post_ids.append(post_id)
        else:
            # Post must have at least one tag
            matching_post_ids = list(post_tag_matches.keys())

        # Get the full post objects for matching post IDs
        # Filter out deleted and archived posts
        valid_posts = []
        for post_id in matching_post_ids:
            post = post_table.get(post_id)  # Fixed: use post_table instead of getattr
            if post and not post.is_deleted and not post.is_archived:
                valid_posts.append(post)

        # Sort posts by created_at (most recent first)
        valid_posts.sort(key=lambda p: p.created_at, reverse=True)

        # Get total count before pagination
        total_count = len(valid_posts)

        # Apply pagination
        paginated_posts = valid_posts[offset:offset + limit]

        # Convert posts to dictionary format for return
        # Include all tags for each post
        result_posts = []
        for post in paginated_posts:
            # Get all tags for this post
            post_tags = []
            for post_tag_id, post_tag_entry in post_tag_table.items():
                if post_tag_entry.post_id == post.post_id:
                    post_tags.append(post_tag_entry.tag)

            # Create post dictionary with relevant fields
            post_dict = {
                "post_id": post.post_id,
                "user_id": post.user_id,
                "title": post.title,
                "content": post.content,
                "created_at": post.created_at.strftime("%Y-%m-%d %H:%M:%S"),
                "updated_at": post.updated_at.strftime("%Y-%m-%d %H:%M:%S") if post.updated_at else None,
                "view_count": post.view_count,
                "like_count": post.like_count,
                "comment_count": post.comment_count,
                "is_pinned": post.is_pinned,
                "is_locked": post.is_locked,
                "tags": post_tags
            }
            result_posts.append(post_dict)

        # Return the result
        return {
            "posts": result_posts,
            "total_count": total_count
        }

    @is_tool()
    def mention_user_in_comment(self, comment_id: str, mentioned_user_ids: List[str]):
        """
        Mention one or more users in a comment to notify them.

        This method creates mention records for each mentioned user and generates
        corresponding notifications to alert them.

        Args:
            comment_id: Unique identifier of the comment containing mentions
            mentioned_user_ids: List of user identifiers being mentioned

        Returns:
            Dictionary containing:
                - mention_count: Number of users successfully mentioned
                - mentioned_at: Timestamp when mentions were recorded in yyyy-mm-dd HH:MM:SS format

        Raises:
            ValueError: If comment_id is empty, mentioned_user_ids is empty or contains invalid values,
                       or if Notification class is not available in database schema
        """
        import secrets
        import hashlib
        from datetime import datetime

        # Import database schema classes (already imported at file header)
        # Mention class is available from database_class

        # Validate input parameters
        if not comment_id or not isinstance(comment_id, str) or comment_id.strip() == "":
            raise ValueError("comment_id must be a non-empty string")

        if not mentioned_user_ids or not isinstance(mentioned_user_ids, list):
            raise ValueError("mentioned_user_ids must be a non-empty list")

        if len(mentioned_user_ids) == 0:
            raise ValueError("mentioned_user_ids cannot be an empty list")

        # Validate each user_id in the list
        for user_id in mentioned_user_ids:
            if not user_id or not isinstance(user_id, str) or user_id.strip() == "":
                raise ValueError(f"Invalid user_id in mentioned_user_ids: {user_id}")

        # Access database
        db = self.db

        # Get mention and notification tables
        mention_table = getattr(db, "mention", None)
        notification_table = getattr(db, "notification", None)

        # Initialize tables if they don't exist
        if mention_table is None:
            mention_table = {}
            setattr(db, "mention", mention_table)

        if notification_table is None:
            notification_table = {}
            setattr(db, "notification", notification_table)

        # Record the current timestamp for all mentions
        mentioned_at = datetime.now()

        # Counter for successfully created mentions
        mention_count = 0

        # Create mention records and notifications for each mentioned user
        for mentioned_user_id in mentioned_user_ids:
            # Generate unique mention_id using hash
            mention_id = "mention_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

            # Create new Mention object
            new_mention = Mention(
                mention_id=mention_id,
                post_id=None,  # This is a comment mention, not a post mention
                comment_id=comment_id,
                mentioned_user_id=mentioned_user_id,
                mentioned_at=mentioned_at
            )

            # Save mention to database
            mention_table[mention_id] = new_mention

            # Generate unique notification_id
            notification_id = "notification_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

            # Check if Notification class is available
            try:
                # Create notification for the mentioned user
                # Note: The Notification class structure is inferred from the related_databases field
                # We attempt to create a notification record with common fields
                new_notification = Notification(
                    notification_id=notification_id,
                    user_id=mentioned_user_id,
                    type="mention",  # Notification type indicating a mention
                    content=f"You were mentioned in a comment (comment_id: {comment_id})",
                    related_id=mention_id,  # Link to the mention record
                    created_at=mentioned_at,
                    is_read=False  # New notifications are unread by default
                )

                # Save notification to database
                notification_table[notification_id] = new_notification
            except (NameError, TypeError, AttributeError) as e:
                # If Notification class is not properly defined or has different structure,
                # we still record the mention but raise an error about notification creation
                raise ValueError(f"Failed to create notification: Notification class may not be properly defined in database schema. Error: {str(e)}")

            # Increment successful mention count
            mention_count += 1

        # Update database with new records
        setattr(db, "mention", mention_table)
        setattr(db, "notification", notification_table)

        # Format timestamp as string in required format
        mentioned_at_str = mentioned_at.strftime("%Y-%m-%d %H:%M:%S")

        # Return result
        return {
            "mention_count": mention_count,
            "mentioned_at": mentioned_at_str
        }

    @is_tool()
    def get_user_notifications(
        self,
        user_id: str,
        notification_type: Literal["all", "mention", "like", "comment", "follow"] = "all",
        unread_only: bool = False,
        limit: int = 20,
        offset: int = 0
    ):
        """
        Retrieve all notifications for a specific user including mentions, likes, and comments.

        Args:
            user_id: Identifier of the user whose notifications to retrieve
            notification_type: Optional filter for specific notification types (all, mention, like, comment, follow)
            unread_only: Whether to retrieve only unread notifications
            limit: Maximum number of notifications to retrieve
            offset: Number of notifications to skip for pagination

        Returns:
            Dictionary containing:
            - notifications: List of notification objects
            - total_count: Total number of notifications matching criteria
            - unread_count: Number of unread notifications

        Raises:
            KeyError: If user_id does not exist or notification table is not available
        """
        from datetime import datetime

        # Validate notification_type parameter against enum values
        valid_notification_types = ["all", "mention", "like", "comment", "follow"]
        if notification_type not in valid_notification_types:
            raise ValueError(f"Invalid notification_type: {notification_type}. Must be one of {valid_notification_types}")

        # Access the database
        db = self.db

        # Get the notification table
        notification_table = getattr(db, "notification", None)
        if notification_table is None:
            raise KeyError("Notification table not found in database")

        # Filter notifications for the specific user
        user_notifications = [
            notif for notif in notification_table.values()
            if notif.user_id == user_id
        ]

        # If no notifications found, raise KeyError as per pre-condition (User must exist)
        if not user_notifications:
            raise KeyError(f"User with user_id '{user_id}' does not exist or has no notifications")

        # Apply notification_type filter if not "all"
        if notification_type != "all":
            user_notifications = [
                notif for notif in user_notifications
                if notif.type == notification_type
            ]

        # Apply unread_only filter if requested
        if unread_only:
            user_notifications = [
                notif for notif in user_notifications
                if not notif.is_read
            ]

        # Calculate total count before pagination
        total_count = len(user_notifications)

        # Calculate unread count (from all user notifications, not filtered)
        all_user_notifications = [
            notif for notif in notification_table.values()
            if notif.user_id == user_id
        ]
        unread_count = sum(1 for notif in all_user_notifications if not notif.is_read)

        # Sort notifications by created_at in descending order (newest first)
        user_notifications.sort(
            key=lambda x: x.created_at if isinstance(x.created_at, datetime) else datetime.strptime(x.created_at, "%Y-%m-%d %H:%M:%S"),
            reverse=True
        )

        # Apply pagination: skip 'offset' items and take 'limit' items
        paginated_notifications = user_notifications[offset:offset + limit]

        # Convert notifications to dictionary format for return
        notifications_list = []
        for notif in paginated_notifications:
            # Convert datetime to string format if needed
            created_at_str = notif.created_at
            if isinstance(created_at_str, datetime):
                created_at_str = created_at_str.strftime("%Y-%m-%d %H:%M:%S")

            notification_dict = {
                "notification_id": notif.notification_id,
                "type": notif.type,
                "content": notif.content,
                "created_at": created_at_str,
                "is_read": notif.is_read
            }

            # Include optional fields if they exist in the notification object
            if hasattr(notif, 'related_id') and notif.related_id:
                notification_dict["related_id"] = notif.related_id
            if hasattr(notif, 'actor_id') and notif.actor_id:
                notification_dict["actor_id"] = notif.actor_id

            notifications_list.append(notification_dict)

        # Return the result
        return {
            "notifications": notifications_list,
            "total_count": total_count,
            "unread_count": unread_count
        }

    @is_tool()
    def get_trending_posts(self, time_window: Literal["1_hour", "6_hours", "24_hours", "7_days"], limit: int = 10):
        # Import necessary modules for datetime calculations
        from datetime import datetime, timedelta

        # Validate time_window parameter (safety check for enum)
        valid_time_windows = ["1_hour", "6_hours", "24_hours", "7_days"]
        if time_window not in valid_time_windows:
            raise ValueError(f"Invalid time_window: {time_window}. Must be one of {valid_time_windows}")

        # Validate limit parameter
        if not isinstance(limit, int) or limit <= 0:
            raise ValueError("limit must be a positive integer")

        # Get the post table from database
        db = self.db
        post_table = getattr(db, "post", None)

        if post_table is None or len(post_table) == 0:
            # Return empty list if no posts exist
            return {"posts": []}

        # Calculate the time threshold based on time_window
        current_time = datetime.now()
        time_deltas = {
            "1_hour": timedelta(hours=1),
            "6_hours": timedelta(hours=6),
            "24_hours": timedelta(hours=24),
            "7_days": timedelta(days=7)
        }
        time_threshold = current_time - time_deltas[time_window]

        # Calculate trending scores for posts within the time window
        trending_posts = []

        for post_id, post in post_table.items():
            # Skip deleted or archived posts
            if post.is_deleted or post.is_archived:
                continue

            # Check if post is within the time window
            if post.created_at < time_threshold:
                continue

            # Calculate trending score based on engagement metrics
            # Formula: weighted combination of likes, comments, and views
            # Recent posts get higher weight, considering time decay
            time_elapsed = (current_time - post.created_at).total_seconds()
            time_window_seconds = time_deltas[time_window].total_seconds()

            # Time decay factor (newer posts get higher score)
            # Ranges from 1.0 (just created) to 0.1 (at the edge of time window)
            time_decay = 1.0 - (0.9 * (time_elapsed / time_window_seconds))

            # Engagement score calculation
            # Weights: likes (3x), comments (5x), views (1x)
            # Comments are weighted highest as they indicate deeper engagement
            engagement_score = (
                post.like_count * 3 +
                post.comment_count * 5 +
                post.view_count * 1
            )

            # Final trending score combines engagement and time decay
            # Multiply by 100 to get a score typically in range 0-100+
            trending_score = engagement_score * time_decay

            # Create trending post object
            trending_post = {
                "post_id": post.post_id,
                "title": post.title,
                "trending_score": round(trending_score, 2),
                "like_count": post.like_count,
                "comment_count": post.comment_count,
                "view_count": post.view_count,
                "created_at": post.created_at.strftime("%Y-%m-%d %H:%M:%S")
            }

            trending_posts.append(trending_post)

        # Sort posts by trending_score in descending order
        trending_posts.sort(key=lambda x: x["trending_score"], reverse=True)

        # Limit the number of results
        limited_posts = trending_posts[:limit]

        return {"posts": limited_posts}

    @is_tool()
    def get_recent_posts(self, limit: int = 20, offset: int = 0):
        """
        Retrieve the most recently created posts on the community board

        Args:
            limit: Maximum number of posts to retrieve (default: 20)
            offset: Number of posts to skip for pagination (default: 0)

        Returns:
            dict: Contains 'posts' (list of post objects) and 'total_count' (total posts available)

        Raises:
            ValueError: If limit or offset are negative values
        """
        # Validate input parameters
        if limit < 0:
            raise ValueError("limit must be a non-negative integer")
        if offset < 0:
            raise ValueError("offset must be a non-negative integer")

        # Access the database
        db = self.db

        # Get the post table from database
        post_table = getattr(db, "post", None)

        # If post table doesn't exist or is empty, return empty result
        if post_table is None or len(post_table) == 0:
            return {
                "posts": [],
                "total_count": 0
            }

        # Filter out deleted and archived posts, only include active posts
        active_posts = [
            post for post in post_table.values()
            if not post.is_deleted and not post.is_archived
        ]

        # Sort posts by created_at in descending order (most recent first)
        sorted_posts = sorted(
            active_posts,
            key=lambda p: p.created_at,
            reverse=True
        )

        # Get total count of active posts
        total_count = len(sorted_posts)

        # Apply pagination: skip 'offset' posts and take 'limit' posts
        paginated_posts = sorted_posts[offset:offset + limit]

        # Convert Post objects to dictionary format for return
        # Format datetime objects to string in "yyyy-mm-dd HH:MM:SS" format
        posts_list = []
        for post in paginated_posts:
            post_dict = {
                "post_id": post.post_id,
                "user_id": post.user_id,
                "title": post.title,
                "content": post.content,
                "created_at": post.created_at.strftime("%Y-%m-%d %H:%M:%S"),
                "view_count": post.view_count,
                "like_count": post.like_count,
                "comment_count": post.comment_count,
                "is_pinned": post.is_pinned,
                "is_locked": post.is_locked
            }

            # Add optional fields if they exist
            if post.updated_at is not None:
                post_dict["updated_at"] = post.updated_at.strftime("%Y-%m-%d %H:%M:%S")

            if post.pinned_at is not None:
                post_dict["pinned_at"] = post.pinned_at.strftime("%Y-%m-%d %H:%M:%S")

            if post.is_locked and post.locked_at is not None:
                post_dict["locked_at"] = post.locked_at.strftime("%Y-%m-%d %H:%M:%S")
                if post.lock_reason is not None:
                    post_dict["lock_reason"] = post.lock_reason

            posts_list.append(post_dict)

        # Return the result with posts list and total count
        return {
            "posts": posts_list,
            "total_count": total_count
        }

    @is_tool()
    def unlike_post(self, post_id: str, user_id: str):
        """
        Remove a like from a post for a specific user.

        This method removes an existing like record from the post_like table and decrements
        the like_count of the corresponding post. It validates that both the post and the
        like record exist before performing the operation.

        Args:
            post_id: Unique identifier of the post to unlike
            user_id: Identifier of the user unliking the post

        Returns:
            dict: Contains:
                - unliked_at: Timestamp when the like was removed (yyyy-mm-dd HH:MM:SS format)
                - like_count: Updated total like count for the post

        Raises:
            KeyError: If the post doesn't exist or the user hasn't liked the post
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Get the post_like table
        post_like_table = getattr(db, "post_like", None)
        if post_like_table is None:
            raise KeyError("post_like table not found in database")

        # Get the post table
        post_table = getattr(db, "post", None)
        if post_table is None:
            raise KeyError("post table not found in database")

        # Verify the post exists
        if post_id not in post_table:
            raise KeyError(f"Post with id '{post_id}' does not exist")

        # Find the like record for this post_id and user_id combination
        # Note: PostLike uses post_id as instance_key, but we need to find by both post_id and user_id
        like_record = None
        like_key = None
        for key, record in post_like_table.items():
            if record.post_id == post_id and record.user_id == user_id:
                like_record = record
                like_key = key
                break

        # Verify the like exists (user must have previously liked the post)
        if like_record is None:
            raise KeyError(f"User '{user_id}' has not liked post '{post_id}'")

        # Record the timestamp when the like is removed
        unliked_at = datetime.now()

        # Remove the like record from post_like table
        updated_post_like_table = dict(post_like_table)
        del updated_post_like_table[like_key]
        setattr(db, "post_like", updated_post_like_table)

        # Get the post and decrement its like_count
        post = post_table[post_id]
        updated_post_table = dict(post_table)

        # Decrement like_count (ensure it doesn't go below 0)
        new_like_count = max(0, post.like_count - 1)

        # Create updated post with decremented like_count
        from copy import deepcopy
        updated_post = deepcopy(post)
        updated_post.like_count = new_like_count

        # Update the post in the table
        updated_post_table[post_id] = updated_post
        setattr(db, "post", updated_post_table)

        # Return the result with formatted timestamp and updated like count
        return {
            "unliked_at": unliked_at.strftime("%Y-%m-%d %H:%M:%S"),
            "like_count": new_like_count
        }

    @is_tool()
    def mark_all_notifications_as_read(self, user_id: str):
        """
        Mark all notifications for a user as read

        Args:
            user_id: Identifier of the user whose notifications to mark as read

        Returns:
            dict: Contains marked_count (number of notifications marked) and marked_at (timestamp)

        Raises:
            KeyError: If user_id does not exist in the system (no notifications found)
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Get the notification table from database
        notification_table = getattr(db, 'notification', None)

        # If notification table doesn't exist or is empty, raise KeyError
        if notification_table is None or len(notification_table) == 0:
            raise KeyError(f"No notifications found for user_id: {user_id}")

        # Track the number of notifications marked as read
        marked_count = 0

        # Flag to check if user has any notifications
        user_has_notifications = False

        # Iterate through all notifications to find those belonging to the user
        for notification_id, notification in notification_table.items():
            # Check if this notification belongs to the specified user
            if notification.user_id == user_id:
                user_has_notifications = True

                # Only mark as read if it's currently unread
                # Assuming notification has a 'read' or 'is_read' attribute
                if hasattr(notification, 'read') and not notification.read:
                    notification.read = True
                    marked_count += 1
                elif hasattr(notification, 'is_read') and not notification.is_read:
                    notification.is_read = True
                    marked_count += 1
                elif hasattr(notification, 'status') and notification.status != 'read':
                    notification.status = 'read'
                    marked_count += 1

        # If user has no notifications in the system, raise KeyError
        if not user_has_notifications:
            raise KeyError(f"No notifications found for user_id: {user_id}")

        # Update the notification table in database
        setattr(db, 'notification', notification_table)

        # Get current timestamp in required format
        marked_at = datetime.now().strftime("%Y-%m-%d %H:%M:%S")

        # Return the result
        return {
            "marked_count": marked_count,
            "marked_at": marked_at
        }

    @is_tool()
    def check_user_liked_post(self, post_id: str, user_id: str):
        """
        Check whether a specific user has liked a particular post.

        This method queries the post_like table to determine if a given user has liked
        a specific post. It returns the like status and timestamp if the like exists.

        Args:
            post_id: Unique identifier of the post to check
            user_id: Identifier of the user to check for like status

        Returns:
            dict: Contains 'is_liked' (bool) and 'liked_at' (str or None)

        Raises:
            KeyError: If post_id or user_id is not found in the database
        """
        from datetime import datetime

        # Access the database instance
        db = self.db

        # Get the post_like table from database
        post_like_table = getattr(db, "post_like", None)

        # Check if post_like table exists
        if post_like_table is None:
            raise KeyError(f"post_like table not found in database")

        # Check if the table is empty
        if not post_like_table:
            raise KeyError(f"Post with post_id '{post_id}' not found in post_like table")

        # Initialize return values
        is_liked = False
        liked_at = None

        # Iterate through all post likes to find matching post_id and user_id
        found_post = False
        for like_post_id, like_record in post_like_table.items():
            # Check if this record matches the post_id
            if like_record.post_id == post_id:
                found_post = True
                # Check if the user_id matches
                if like_record.user_id == user_id:
                    # User has liked this post
                    is_liked = True
                    # Convert datetime object to string format "yyyy-mm-dd HH:MM:SS"
                    liked_at = like_record.liked_at.strftime("%Y-%m-%d %H:%M:%S")
                    break

        # If we never found the post_id in the table, raise KeyError
        if not found_post:
            raise KeyError(f"Post with post_id '{post_id}' not found in post_like table")

        # Return the like status and timestamp
        return {
            "is_liked": is_liked,
            "liked_at": liked_at
        }

    @is_tool()
    def validate_post_content_length(
        self,
        title: str,
        content: str,
        min_title_length: int,
        max_title_length: int,
        min_content_length: int,
        max_content_length: int
    ):
        """
        Validate that post title and content meet length requirements.

        This method checks if the provided title and content fall within the specified
        length constraints. It returns a validation result with a boolean flag and
        a list of error messages if validation fails.

        Args:
            title: Post title to validate
            content: Post content to validate
            min_title_length: Minimum required title length
            max_title_length: Maximum allowed title length
            min_content_length: Minimum required content length
            max_content_length: Maximum allowed content length

        Returns:
            dict: Contains 'is_valid' (bool) and 'validation_errors' (list of str)

        Raises:
            ValueError: If any length parameter is negative or min > max
        """
        # Validate input parameters
        if min_title_length < 0:
            raise ValueError("min_title_length must be non-negative")
        if max_title_length < 0:
            raise ValueError("max_title_length must be non-negative")
        if min_content_length < 0:
            raise ValueError("min_content_length must be non-negative")
        if max_content_length < 0:
            raise ValueError("max_content_length must be non-negative")

        # Check that minimum is not greater than maximum
        if min_title_length > max_title_length:
            raise ValueError("min_title_length cannot be greater than max_title_length")
        if min_content_length > max_content_length:
            raise ValueError("min_content_length cannot be greater than max_content_length")

        # Initialize validation result
        validation_errors = []

        # Get actual lengths of title and content
        title_length = len(title)
        content_length = len(content)

        # Validate title length
        if title_length < min_title_length:
            validation_errors.append(
                f"Title is too short. Minimum length is {min_title_length} characters, "
                f"but got {title_length} characters."
            )

        if title_length > max_title_length:
            validation_errors.append(
                f"Title is too long. Maximum length is {max_title_length} characters, "
                f"but got {title_length} characters."
            )

        # Validate content length
        if content_length < min_content_length:
            validation_errors.append(
                f"Content is too short. Minimum length is {min_content_length} characters, "
                f"but got {content_length} characters."
            )

        if content_length > max_content_length:
            validation_errors.append(
                f"Content is too long. Maximum length is {max_content_length} characters, "
                f"but got {content_length} characters."
            )

        # Determine if validation passed (no errors)
        is_valid = len(validation_errors) == 0

        # Return validation result
        return {
            "is_valid": is_valid,
            "validation_errors": validation_errors
        }

    @is_tool()
    def get_user_posts(self, user_id: str, limit: int = None, offset: int = 0):
        """
        Retrieve all posts created by a specific user with pagination support.

        Args:
            user_id: Unique identifier of the user
            limit: Maximum number of posts to retrieve (optional)
            offset: Number of posts to skip for pagination (default: 0)

        Returns:
            Dictionary containing:
            - posts: List of post objects created by the user
            - total_count: Total number of posts by the user

        Raises:
            KeyError: If user_id is not found or post table doesn't exist
        """
        # Access the database
        db = self.db

        # Get the post table from database
        post_table = getattr(db, "post", None)
        if post_table is None:
            raise KeyError("Post table does not exist in the database")

        # Validate offset parameter
        if offset < 0:
            raise ValueError("Offset must be non-negative")

        # Validate limit parameter if provided
        if limit is not None and limit <= 0:
            raise ValueError("Limit must be positive")

        # Filter posts by user_id and exclude deleted posts
        user_posts = []
        for post_id, post in post_table.items():
            # Only include posts that belong to the user and are not deleted
            if post.user_id == user_id and not post.is_deleted:
                user_posts.append(post)

        # Check if user has any posts (to verify user exists or has posted)
        if len(user_posts) == 0:
            # Return empty result but don't raise error - user might exist but have no posts
            return {
                "posts": [],
                "total_count": 0
            }

        # Sort posts by created_at in descending order (newest first)
        user_posts.sort(key=lambda x: x.created_at, reverse=True)

        # Get total count before pagination
        total_count = len(user_posts)

        # Apply pagination: skip 'offset' posts
        paginated_posts = user_posts[offset:]

        # Apply limit if specified
        if limit is not None:
            paginated_posts = paginated_posts[:limit]

        # Convert Post objects to dictionary format for return
        posts_list = []
        for post in paginated_posts:
            post_dict = {
                "post_id": post.post_id,
                "user_id": post.user_id,
                "title": post.title,
                "content": post.content,
                "created_at": post.created_at.strftime("%Y-%m-%d %H:%M:%S"),
                "updated_at": post.updated_at.strftime("%Y-%m-%d %H:%M:%S") if post.updated_at else None,
                "view_count": post.view_count,
                "like_count": post.like_count,
                "comment_count": post.comment_count,
                "is_pinned": post.is_pinned,
                "pinned_at": post.pinned_at.strftime("%Y-%m-%d %H:%M:%S") if post.pinned_at else None,
                "is_locked": post.is_locked,
                "locked_at": post.locked_at.strftime("%Y-%m-%d %H:%M:%S") if post.locked_at else None,
                "lock_reason": post.lock_reason,
                "is_archived": post.is_archived,
                "archived_at": post.archived_at.strftime("%Y-%m-%d %H:%M:%S") if post.archived_at else None
            }
            posts_list.append(post_dict)

        # Return the result with posts and total count
        return {
            "posts": posts_list,
            "total_count": total_count
        }

    @is_tool()
    def get_post_details(self, post_id: str):
        """
        Retrieve complete details of a specific post including title, content, author, 
        timestamps, and engagement metrics.

        Args:
            post_id: Unique identifier of the post to retrieve

        Returns:
            dict: Complete post details including all metadata and engagement metrics

        Raises:
            KeyError: If the post does not exist in the system
        """
        # Access the database instance
        db = self.db

        # Retrieve the post table from the database
        post_table = getattr(db, 'post', None)

        # Check if the post table exists
        if post_table is None:
            raise KeyError(f"Post table does not exist in the database")

        # Retrieve the specific post by post_id
        if post_id not in post_table:
            raise KeyError(f"Post with post_id '{post_id}' does not exist")

        # Get the post object
        post = post_table[post_id]

        # Format datetime objects to "yyyy-mm-dd HH:MM:SS" string format
        created_at_str = post.created_at.strftime("%Y-%m-%d %H:%M:%S")

        # Handle optional updated_at field
        updated_at_str = post.updated_at.strftime("%Y-%m-%d %H:%M:%S") if post.updated_at else None

        # Construct the return dictionary with all required fields
        # Note: The schema doesn't define a tags field in the Post model, 
        # but the return schema requires it. We'll return an empty list as default.
        post_details = {
            "post_id": post.post_id,
            "user_id": post.user_id,
            "title": post.title,
            "content": post.content,
            "tags": [],  # Tags field not present in Post schema, returning empty list
            "created_at": created_at_str,
            "updated_at": updated_at_str,
            "view_count": post.view_count,
            "like_count": post.like_count,
            "comment_count": post.comment_count
        }

        return post_details

    @is_tool()
    def remove_bookmark(self, post_id: str, user_id: str):
        """
        Remove a post from a user's bookmarks

        This method removes a bookmark entry from the database based on the provided
        post_id and user_id combination.

        Args:
            post_id: Unique identifier of the post to unbookmark
            user_id: Identifier of the user removing the bookmark

        Returns:
            dict: Contains the timestamp when the bookmark was removed
                - removed_at: Timestamp in "yyyy-mm-dd HH:MM:SS" format

        Raises:
            KeyError: If the bookmark does not exist for the given post_id and user_id
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Get the bookmark table from database
        bookmark_table = getattr(db, "bookmark", None)

        # Check if bookmark table exists and is not empty
        if bookmark_table is None or not bookmark_table:
            raise KeyError(f"Bookmark not found for post_id '{post_id}' and user_id '{user_id}'")

        # Find the bookmark entry that matches both post_id and user_id
        # The bookmark table uses post_id as the key
        bookmark_entry = bookmark_table.get(post_id)

        # Verify that the bookmark exists and belongs to the specified user
        if bookmark_entry is None:
            raise KeyError(f"Bookmark not found for post_id '{post_id}' and user_id '{user_id}'")

        # Check if the bookmark belongs to the requesting user
        if bookmark_entry.user_id != user_id:
            raise KeyError(f"Bookmark not found for post_id '{post_id}' and user_id '{user_id}'")

        # Remove the bookmark from the table
        # Create a new dictionary without the bookmark entry
        updated_bookmark_table = {k: v for k, v in bookmark_table.items() if k != post_id}

        # Update the database with the modified bookmark table
        setattr(db, "bookmark", updated_bookmark_table)

        # Record the removal timestamp
        removed_at = datetime.now()

        # Return the result with timestamp in required format
        return {
            "removed_at": removed_at.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def sanitize_text_content(self, content: str, allow_html: bool = False):
        """
        Remove or escape potentially harmful HTML/script tags from text content.

        This method sanitizes user-provided text content by removing or escaping
        potentially dangerous HTML tags and scripts to prevent XSS attacks.

        Args:
            content: Text content to sanitize
            allow_html: Whether to allow safe HTML tags (default: False)

        Returns:
            dict: Contains 'sanitized_content' (cleaned text) and 'removed_elements' 
                  (list of removed/escaped tag names)

        Raises:
            ValueError: If content is None or not a string
        """
        import re
        import html

        # Validate input
        if content is None:
            raise ValueError("Content cannot be None")

        if not isinstance(content, str):
            raise ValueError("Content must be a string")

        # Track removed elements
        removed_elements = []

        # Define dangerous tags that should always be removed
        dangerous_tags = [
            'script', 'iframe', 'object', 'embed', 'applet', 
            'meta', 'link', 'style', 'base', 'form'
        ]

        # Define safe HTML tags that can be allowed if allow_html is True
        safe_tags = [
            'p', 'br', 'strong', 'em', 'u', 'b', 'i', 
            'ul', 'ol', 'li', 'blockquote', 'code', 'pre',
            'h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'a', 'img'
        ]

        # Pattern to find all HTML tags with their names
        # Matches both opening tags (<tag>) and closing tags (</tag>)
        tag_pattern = re.compile(r'<\s*/?\s*([a-zA-Z][a-zA-Z0-9]*)[^>]*>', re.IGNORECASE)

        # Find all tags in the content
        found_tags = tag_pattern.findall(content)

        # Process content based on allow_html flag
        if not allow_html:
            # Remove all HTML tags when allow_html is False
            for tag_name in found_tags:
                tag_lower = tag_name.lower()
                if tag_lower not in removed_elements:
                    removed_elements.append(tag_lower)

            # Escape all HTML to convert tags to visible text
            sanitized_content = html.escape(content)

            # Then unescape to remove the tags completely (showing only text content)
            # This approach: first find and remove complete tags, then escape remaining special chars
            sanitized_content = tag_pattern.sub('', content)

            # Escape any remaining special HTML characters
            sanitized_content = html.escape(sanitized_content, quote=False)
            sanitized_content = html.unescape(sanitized_content)

        else:
            # When allow_html is True, only remove dangerous tags
            sanitized_content = content

            # Remove dangerous tags and track them
            for tag_name in found_tags:
                tag_lower = tag_name.lower()
                if tag_lower in dangerous_tags:
                    if tag_lower not in removed_elements:
                        removed_elements.append(tag_lower)

                    # Remove both opening and closing tags of dangerous elements
                    # Pattern matches: <tag>, <tag attr="value">, </tag>, etc.
                    dangerous_tag_pattern = re.compile(
                        r'<\s*/?\s*' + re.escape(tag_name) + r'[^>]*>',
                        re.IGNORECASE
                    )
                    sanitized_content = dangerous_tag_pattern.sub('', sanitized_content)

            # Also remove any event handlers (onclick, onerror, etc.) from remaining tags
            event_pattern = re.compile(r'\s+on\w+\s*=\s*["\'][^"\']*["\']', re.IGNORECASE)
            if event_pattern.search(sanitized_content):
                if 'event_handlers' not in removed_elements:
                    removed_elements.append('event_handlers')
                sanitized_content = event_pattern.sub('', sanitized_content)

            # Remove javascript: protocol from URLs
            javascript_pattern = re.compile(r'javascript\s*:', re.IGNORECASE)
            if javascript_pattern.search(sanitized_content):
                if 'javascript_protocol' not in removed_elements:
                    removed_elements.append('javascript_protocol')
                sanitized_content = javascript_pattern.sub('', sanitized_content)

        # Remove any null bytes that could be used for injection
        if '\x00' in sanitized_content:
            sanitized_content = sanitized_content.replace('\x00', '')
            if 'null_bytes' not in removed_elements:
                removed_elements.append('null_bytes')

        # Normalize whitespace (collapse multiple spaces but preserve line breaks)
        sanitized_content = re.sub(r'[ \t]+', ' ', sanitized_content)
        sanitized_content = re.sub(r'\n\s*\n', '\n\n', sanitized_content)

        # Strip leading/trailing whitespace
        sanitized_content = sanitized_content.strip()

        return {
            "sanitized_content": sanitized_content,
            "removed_elements": removed_elements
        }

    @is_tool()
    def extract_hashtags_from_text(self, content: str):
        """
        Extract all hashtags from text content.

        This method parses the provided text content and extracts all hashtags (words prefixed with #).
        It returns a list of hashtag strings (without the # symbol) and the total count.

        Args:
            content: Text content to extract hashtags from

        Returns:
            dict: A dictionary containing:
                - hashtags: List of extracted hashtag strings without the # symbol
                - hashtag_count: Total number of hashtags found

        Raises:
            ValueError: If content is None or not a string
        """
        # Validate input parameter
        if content is None:
            raise ValueError("Content cannot be None")

        if not isinstance(content, str):
            raise ValueError("Content must be a string")

        # Import required module for regex pattern matching
        import re

        # Define regex pattern to match hashtags
        # Pattern explanation:
        # #: Match the literal # symbol
        # (\w+): Capture one or more word characters (letters, digits, underscore)
        # This will match hashtags like #community, #connect, #share123, etc.
        hashtag_pattern = r'#(\w+)'

        # Find all matches in the content
        # re.findall returns a list of all matched groups (the part in parentheses)
        # This gives us the hashtag text without the # symbol
        matches = re.findall(hashtag_pattern, content)

        # Remove duplicates while preserving order
        # Use dict.fromkeys() to maintain insertion order (Python 3.7+)
        # and remove duplicates, then convert back to list
        unique_hashtags = list(dict.fromkeys(matches))

        # Calculate the total count of hashtags found
        # Note: This counts unique hashtags, not total occurrences
        hashtag_count = len(unique_hashtags)

        # Return the result as a dictionary matching the schema
        return {
            "hashtags": unique_hashtags,
            "hashtag_count": hashtag_count
        }

    @is_tool()
    def lock_post(self, post_id: str, user_id: str, reason: str = None):
        """
        Lock a post to prevent further comments and interactions.

        This method locks a specified post, preventing any new comments or interactions.
        The operation requires moderator permissions and records the lock timestamp and reason.

        Args:
            post_id: Unique identifier of the post to lock
            user_id: Identifier of the moderator locking the post
            reason: Optional reason for locking the post

        Returns:
            dict: Contains the locked_at timestamp in "yyyy-mm-dd HH:MM:SS" format

        Raises:
            ValueError: If post doesn't exist, post is already locked, or post is deleted
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Validate that post_id is not empty
        if not post_id or not isinstance(post_id, str):
            raise ValueError("post_id must be a non-empty string")

        # Validate that user_id is not empty
        if not user_id or not isinstance(user_id, str):
            raise ValueError("user_id must be a non-empty string")

        # Retrieve the post table from database
        post_table = getattr(db, "post", None)
        if post_table is None:
            raise ValueError("Post table does not exist in database")

        # Check if the post exists
        if post_id not in post_table:
            raise ValueError(f"Post with post_id '{post_id}' does not exist")

        # Get the post object
        post = post_table[post_id]

        # Check if post is already deleted
        if post.is_deleted:
            raise ValueError(f"Cannot lock post '{post_id}' because it is already deleted")

        # Check if post is already locked
        if post.is_locked:
            raise ValueError(f"Post '{post_id}' is already locked")

        # Record the current timestamp for locking
        locked_timestamp = datetime.now()

        # Update the post object with lock information
        post.is_locked = True
        post.locked_at = locked_timestamp

        # Set the lock reason if provided
        if reason:
            post.lock_reason = reason

        # Update the post in the database
        post_table[post_id] = post
        setattr(db, "post", post_table)

        # Format the timestamp as string in "yyyy-mm-dd HH:MM:SS" format
        locked_at_str = locked_timestamp.strftime("%Y-%m-%d %H:%M:%S")

        # Return the locked timestamp
        return {
            "locked_at": locked_at_str
        }

    @is_tool()
    def get_user_bookmarks(self, user_id: str, limit: int = None, offset: int = 0):
        """
        Retrieve all posts that a user has bookmarked

        Args:
            user_id: Identifier of the user whose bookmarks to retrieve
            limit: Maximum number of bookmarks to retrieve (optional)
            offset: Number of bookmarks to skip for pagination (default: 0)

        Returns:
            dict: Contains 'bookmarks' (list of bookmarked post objects) and 'total_count' (total number of bookmarked posts)

        Raises:
            KeyError: If user_id is not found or invalid
        """
        # Access the database
        db = self.db

        # Get bookmark and post tables from database
        bookmark_table = getattr(db, "bookmark", None)
        post_table = getattr(db, "post", None)

        # Validate that required tables exist
        if bookmark_table is None:
            raise KeyError("Bookmark table not found in database")
        if post_table is None:
            raise KeyError("Post table not found in database")

        # Find all bookmarks for the specified user
        user_bookmarks = []
        for bookmark_id, bookmark in bookmark_table.items():
            if bookmark.user_id == user_id:
                user_bookmarks.append(bookmark)

        # Sort bookmarks by bookmarked_at timestamp (most recent first)
        user_bookmarks.sort(key=lambda x: x.bookmarked_at, reverse=True)

        # Get total count before pagination
        total_count = len(user_bookmarks)

        # Apply pagination: skip 'offset' items and take up to 'limit' items
        start_index = offset
        end_index = start_index + limit if limit is not None else len(user_bookmarks)
        paginated_bookmarks = user_bookmarks[start_index:end_index]

        # Build result list with post details
        bookmarks_result = []
        for bookmark in paginated_bookmarks:
            # Get the corresponding post from post table
            post = post_table.get(bookmark.post_id)

            # If post exists, add it to results with bookmark information
            if post is not None:
                bookmark_info = {
                    "post_id": post.post_id,
                    "title": post.title,
                    "bookmarked_at": bookmark.bookmarked_at.strftime("%Y-%m-%d %H:%M:%S")
                }
                bookmarks_result.append(bookmark_info)

        # Return the result containing bookmarks list and total count
        return {
            "bookmarks": bookmarks_result,
            "total_count": total_count
        }

    @is_tool()
    def extract_mentions_from_text(self, content: str):
        """
        Extract all user mentions from text content.

        This method parses the input text to find all mentions in the format @username,
        extracts the usernames (without the @ symbol), and returns them along with
        the total count.

        Args:
            content: Text content to extract mentions from

        Returns:
            dict: Dictionary containing:
                - mentions: List of extracted usernames without the @ symbol
                - mention_count: Total number of mentions found

        Raises:
            ValueError: If content is None or not a string
        """
        import re

        # Validate input parameter
        if content is None:
            raise ValueError("Content must be provided and cannot be None")

        if not isinstance(content, str):
            raise ValueError("Content must be a string")

        # Regular expression pattern to match @username mentions
        # Pattern explanation:
        # @ - matches the @ symbol
        # ([a-zA-Z0-9_]+) - captures one or more alphanumeric characters or underscores
        # This captures typical username formats used in social media
        mention_pattern = r'@([a-zA-Z0-9_]+)'

        # Find all matches in the content
        # re.findall returns a list of all captured groups (usernames without @)
        mentions = re.findall(mention_pattern, content)

        # Remove duplicates while preserving order
        # This ensures each unique username is only listed once
        seen = set()
        unique_mentions = []
        for mention in mentions:
            if mention not in seen:
                seen.add(mention)
                unique_mentions.append(mention)

        # Count total number of mentions (including duplicates in original text)
        mention_count = len(mentions)

        # Return the extracted mentions and count
        return {
            "mentions": unique_mentions,
            "mention_count": mention_count
        }

    @is_tool()
    def get_user_following(self, user_id: str, limit: Optional[int] = None, offset: Optional[int] = None):
        """
        Retrieve a list of users that a specific user is following.

        Args:
            user_id: Identifier of the user whose following list to retrieve
            limit: Maximum number of followed users to retrieve (optional)
            offset: Number of followed users to skip for pagination (optional, default: 0)

        Returns:
            dict: Contains 'following' list and 'total_count'

        Raises:
            KeyError: If user_id does not exist in the system
        """
        # Access the database
        db = self.db

        # Get the user_follow table
        user_follow_table = getattr(db, 'user_follow', None)

        # Check if the table exists
        if user_follow_table is None:
            raise KeyError(f"User follow table not found in database")

        # Collect all follow relationships where the user is the follower
        following_list = []
        user_exists = False

        for follow_id, follow_record in user_follow_table.items():
            if follow_record.follower_user_id == user_id:
                following_list.append(follow_record)
                user_exists = True
            elif follow_record.followed_user_id == user_id:
                # User exists in system (being followed by someone)
                user_exists = True

        # If user doesn't exist in any relationship, raise KeyError
        if not user_exists:
            raise KeyError(f"User with user_id '{user_id}' does not exist in the system")

        # Sort the following list by followed_at timestamp (most recent first)
        following_list.sort(key=lambda x: x.followed_at, reverse=True)

        # Get total count before pagination
        total_count = len(following_list)

        # Apply pagination: offset (default to 0 if None)
        start_index = offset if offset is not None else 0
        if start_index < 0:
            start_index = 0

        # Apply offset
        following_list = following_list[start_index:]

        # Apply limit if specified
        if limit is not None and limit > 0:
            following_list = following_list[:limit]

        # Format the following list to match the return schema
        # Convert UserFollow objects to dictionaries with required fields
        formatted_following = []
        for follow_record in following_list:
            formatted_following.append({
                "user_id": follow_record.followed_user_id,
                "followed_at": follow_record.followed_at.strftime("%Y-%m-%d %H:%M:%S")
            })

        # Return the result
        return {
            "following": formatted_following,
            "total_count": total_count
        }

    @is_tool()
    def mention_user_in_post(self, post_id: str, mentioned_user_ids: list):
        """
        Mention one or more users in a post to notify them.

        This method creates mention records for each mentioned user and generates
        corresponding notifications to alert them of being mentioned in a post.

        Args:
            post_id: Unique identifier of the post containing mentions
            mentioned_user_ids: List of user identifiers being mentioned

        Returns:
            dict: Contains mention_count (number of users successfully mentioned)
                  and mentioned_at (timestamp in yyyy-mm-dd HH:MM:SS format)

        Raises:
            ValueError: If post_id is empty, mentioned_user_ids is empty or not a list,
                       or if any user_id in the list is invalid
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Validate input parameters
        if not post_id or not isinstance(post_id, str):
            raise ValueError("post_id must be a non-empty string")

        if not mentioned_user_ids or not isinstance(mentioned_user_ids, list):
            raise ValueError("mentioned_user_ids must be a non-empty list")

        if not all(isinstance(uid, str) and uid for uid in mentioned_user_ids):
            raise ValueError("All user IDs in mentioned_user_ids must be non-empty strings")

        # Access database
        db = self.db

        # Get mention and notification tables
        mention_table = getattr(db, 'mention', None)
        notification_table = getattr(db, 'notification', None)

        # Initialize tables if they don't exist
        if mention_table is None:
            mention_table = {}
            setattr(db, 'mention', mention_table)

        if notification_table is None:
            notification_table = {}
            setattr(db, 'notification', notification_table)

        # Record the timestamp when mentions are created
        mentioned_at = datetime.now()
        mentioned_at_str = mentioned_at.strftime("%Y-%m-%d %H:%M:%S")

        # Counter for successfully mentioned users
        mention_count = 0

        # Process each mentioned user
        for mentioned_user_id in mentioned_user_ids:
            # Generate unique mention_id using secure hash
            mention_id = "mention_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

            # Create mention record
            mention_record = Mention(
                mention_id=mention_id,
                post_id=post_id,
                comment_id=None,  # This is a post mention, not a comment mention
                mentioned_user_id=mentioned_user_id,
                mentioned_at=mentioned_at
            )

            # Save mention record to database
            mention_table[mention_id] = mention_record

            # Generate unique notification_id for the notification
            notification_id = "notif_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

            # Create notification record to alert the mentioned user
            notification_record = Notification(
                notification_id=notification_id,
                user_id=mentioned_user_id,
                type="mention",  # Notification type indicating a mention
                content=f"You were mentioned in post {post_id}",
                related_id=post_id,  # Reference to the post where user was mentioned
                created_at=mentioned_at,
                is_read=False  # New notifications are unread by default
            )

            # Save notification record to database
            notification_table[notification_id] = notification_record

            # Increment successful mention counter
            mention_count += 1

        # Update database with new records
        setattr(db, 'mention', mention_table)
        setattr(db, 'notification', notification_table)

        # Return result with mention count and timestamp
        return {
            "mention_count": mention_count,
            "mentioned_at": mentioned_at_str
        }
