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 knowledge_management."""

class KnowledgeManagementTools(ToolKitBase):
    """All tools for knowledge_management."""
    
    db: KnowledgeManagementDB
    
    def __init__(self, db: KnowledgeManagementDB):
        """Initialize tools with database."""
        super().__init__(db)
    
    @is_tool()
    def check_document_permission(self, document_id: str, user_id: str, permission_level: Literal["read", "write", "admin"]):
        """
        Check if a user has specific permission for a document

        This method verifies whether a given user has the specified permission level
        for a particular document by looking up the document_permission table.

        Args:
            document_id: The unique identifier of the document
            user_id: The identifier of the user
            permission_level: The permission level to check (must be "read", "write", or "admin")

        Returns:
            dict: A dictionary with 'has_permission' key indicating whether the user
                  has the specified permission (True/False)

        Raises:
            KeyError: If the document_permission table does not exist in the database
        """
        from scale_env.environment.db import DB

        # Access the database instance
        db = self.db

        # Get the document_permission table from the database
        # This will raise KeyError if the table doesn't exist
        document_permission_table = getattr(db, "document_permission")

        # Check if document_permission_table is None or empty
        if document_permission_table is None:
            # No permissions exist, so user doesn't have permission
            return {"has_permission": False}

        # Initialize permission status as False
        has_permission = False

        # Define permission hierarchy: admin > write > read
        permission_hierarchy = {"read": 1, "write": 2, "admin": 3}
        required_level = permission_hierarchy.get(permission_level, 0)

        # Iterate through all permission records in the table
        for permission_id, permission_record in document_permission_table.items():
            # Check if this record matches the document_id and user_id
            if permission_record.document_id == document_id and permission_record.user_id == user_id:
                # Get the user's current permission level
                user_permission_level = permission_hierarchy.get(permission_record.permission_level, 0)

                # Check if user's permission level is sufficient
                # A user with higher permission (e.g., "admin") also has lower permissions (e.g., "read", "write")
                if user_permission_level >= required_level:
                    has_permission = True
                    break

        # Return the permission check result
        return {"has_permission": has_permission}

    @is_tool()
    def remove_tags_from_document(self, document_id: str, tags: List[str]) -> dict:
        """
        Remove specific tags from a document

        Args:
            document_id: The unique identifier of the document
            tags: List of tags to remove from the document

        Returns:
            dict: Contains 'success' (bool) and 'tags' (list of remaining tags)

        Raises:
            KeyError: If document_id does not exist in the database
        """
        # Import necessary class
        from typing import List

        # Get database instance
        db = self.db

        # Get document_tag table
        document_tag_table = getattr(db, "document_tag", None)

        # Check if document_tag table exists
        if document_tag_table is None:
            raise KeyError(f"Document with ID '{document_id}' not found in database")

        # Check if document exists by looking for any tags associated with this document_id
        document_tags = [
            tag_entry for tag_id, tag_entry in document_tag_table.items()
            if tag_entry.document_id == document_id
        ]

        # If no tags found for this document, raise KeyError
        if not document_tags:
            raise KeyError(f"Document with ID '{document_id}' not found in database")

        # Convert tags to remove to a set for efficient lookup
        tags_to_remove_set = set(tags)

        # Track which tag entries to remove from the database
        tag_ids_to_remove = []

        # Iterate through all document tags and identify which ones to remove
        for tag_id, tag_entry in list(document_tag_table.items()):
            # Only process tags for the specified document
            if tag_entry.document_id == document_id:
                # If this tag is in the removal list, mark it for deletion
                if tag_entry.tag in tags_to_remove_set:
                    tag_ids_to_remove.append(tag_id)

        # Remove the identified tags from the database
        for tag_id in tag_ids_to_remove:
            del document_tag_table[tag_id]

        # Update the database table
        setattr(db, "document_tag", document_tag_table)

        # Collect remaining tags for this document after removal
        remaining_tags = [
            tag_entry.tag for tag_id, tag_entry in document_tag_table.items()
            if tag_entry.document_id == document_id
        ]

        # Return success response with remaining tags
        return {
            "success": True,
            "tags": remaining_tags
        }

    @is_tool()
    def create_document_link(self, source_document_id: str, target_document_id: str, link_type: Literal["references", "related", "supersedes", "depends_on"]):
        """
        Create a link between two related documents.

        This method creates a new document link relationship in the database after validating
        that both source and target documents exist and the link type is valid.

        Args:
            source_document_id: The identifier of the source document
            target_document_id: The identifier of the target document
            link_type: The type of relationship between documents (must be one of: 
                       "references", "related", "supersedes", "depends_on")

        Returns:
            dict: A dictionary containing:
                - link_id: The unique identifier of the created link
                - created_at: The timestamp when the link was created in "yyyy-mm-dd HH:MM:SS" format

        Raises:
            KeyError: If source_document_id or target_document_id does not exist in the database
            ValueError: If link_type is not one of the valid enum values or if parameters are invalid
        """
        import secrets
        import hashlib
        from datetime import datetime

        # Import the DocumentLink class (already imported at file header)
        # from knowledge_management_Document_Library_System import DocumentLink

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

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

        # Validate link_type against enum values (additional safety check)
        valid_link_types = ["references", "related", "supersedes", "depends_on"]
        if link_type not in valid_link_types:
            raise ValueError(f"link_type must be one of {valid_link_types}, got: {link_type}")

        # Access the database
        db = self.db

        # Check if document_link table exists, if not initialize it
        document_link_table = getattr(db, "document_link", None)
        if document_link_table is None:
            document_link_table = {}
            setattr(db, "document_link", document_link_table)

        # Note: The tool schema specifies pre-condition "Both documents exist"
        # However, the related_databases only includes "document_link", not the actual document table
        # According to requirements: "禁止验证或查询 schema 中不存在的数据结构"
        # Therefore, we assume the pre-condition is satisfied and do not validate document existence

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

        # Ensure link_id is unique (very unlikely collision, but good practice)
        while link_id in document_link_table:
            link_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp
        created_at = datetime.now()

        # Create new DocumentLink instance
        new_link = DocumentLink(
            link_id=link_id,
            source_document_id=source_document_id,
            target_document_id=target_document_id,
            link_type=link_type,
            created_at=created_at
        )

        # Add the new link to the database
        document_link_table[link_id] = new_link
        setattr(db, "document_link", document_link_table)

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

        # Return the result
        return {
            "link_id": link_id,
            "created_at": created_at_str
        }

    @is_tool()
    def get_all_templates(self, category: Optional[str] = None):
        """
        Retrieve all available document templates with optional category filtering.

        This method fetches all templates from the document_template database table.
        If a category filter is provided, only templates matching that category are returned.
        Uses fuzzy matching for category filtering to improve robustness.

        Args:
            category: Optional category filter for templates (e.g., "technical")

        Returns:
            dict: A dictionary containing:
                - templates: List of template dictionaries with template_id and template_name
                - total_count: Total number of templates returned

        Raises:
            Exception: If database access fails or other unexpected errors occur
        """
        from thefuzz import fuzz

        try:
            # Access the database instance
            db = self.db

            # Get the document_template table data
            template_table = getattr(db, "document_template", None)

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

            # Collect all templates
            all_templates = []

            # Iterate through all templates in the table
            for template_id, template_obj in template_table.items():
                # If category filter is provided, apply fuzzy matching
                if category is not None:
                    # Get the template's category (may be None)
                    template_category = template_obj.category

                    # Skip templates without a category if filtering is requested
                    if template_category is None:
                        continue

                    # Use fuzzy matching to compare categories (case-insensitive)
                    # Using a threshold of 80 for similarity to allow some flexibility
                    similarity_score = fuzz.ratio(category.lower(), template_category.lower())

                    # Only include templates with sufficient similarity
                    if similarity_score < 80:
                        continue

                # Create template dictionary with required fields
                template_dict = {
                    "template_id": template_obj.template_id,
                    "template_name": template_obj.template_name
                }

                all_templates.append(template_dict)

            # Return the results with total count
            return {
                "templates": all_templates,
                "total_count": len(all_templates)
            }

        except AttributeError as e:
            # Handle cases where database structure is unexpected
            raise Exception(f"Failed to access document_template table: {str(e)}")
        except Exception as e:
            # Handle any other unexpected errors
            raise Exception(f"Error retrieving templates: {str(e)}")

    @is_tool()
    def create_document_from_template(self, template_id: str, title: str, author: str):
        # Import necessary modules for ID generation and datetime handling
        import secrets
        import hashlib
        from datetime import datetime

        # Access the database instance
        db = self.db

        # Retrieve the document_template table from the database
        document_template_table = getattr(db, "document_template", None)

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

        # Check if the specified template exists in the database
        if template_id not in document_template_table:
            raise KeyError(f"Template with template_id '{template_id}' does not exist")

        # Retrieve the template object
        template = document_template_table[template_id]

        # Generate a unique document_id using SHA256 hash
        # Prefix with "doc_" to follow the example format
        document_id = "doc_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get the current timestamp for created_at and updated_at fields
        current_time = datetime.now()

        # Create a new Document instance based on the template
        # Use the template's content_structure as the initial content
        # Use the template's category if available
        new_document = Document(
            document_id=document_id,
            title=title,
            content=template.content_structure,  # Initialize content from template structure
            author=author,
            category=template.category,  # Inherit category from template (may be None)
            folder_id=None,  # No folder specified at creation
            is_archived=False,  # New documents are not archived by default
            file_size=None,  # File size not calculated at creation
            created_at=current_time,
            updated_at=current_time,
            archived_at=None  # Not archived, so no archived timestamp
        )

        # Retrieve the document table from the database
        document_table = getattr(db, "document", None)

        # Initialize the document table if it doesn't exist
        if document_table is None:
            document_table = {}

        # Add the new document to the document table
        document_table[document_id] = new_document

        # Update the database with the modified document table
        setattr(db, "document", document_table)

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

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

    @is_tool()
    def create_document_version(self, document_id: str, content: str, version_note: str = None):
        """
        Create a new version of an existing document.

        This method creates a new version entry for an existing document, automatically
        increments the version number, and stores the version with metadata.

        Args:
            document_id: The unique identifier of the document
            content: The content of the new version
            version_note: Optional note describing the changes in this version

        Returns:
            dict: Contains version_id, version_number, and created_at

        Raises:
            KeyError: If the document does not exist (no previous versions found)
        """
        import secrets
        import hashlib
        from datetime import datetime

        # Access the database
        db = self.db

        # Get the document_version table
        document_version_table = getattr(db, "document_version", None)

        # Initialize the table if it doesn't exist
        if document_version_table is None:
            document_version_table = {}
            setattr(db, "document_version", document_version_table)

        # Check if the document exists by looking for any existing versions
        existing_versions = [
            version for version in document_version_table.values()
            if version.document_id == document_id
        ]

        # If no versions exist for this document, raise KeyError
        if not existing_versions:
            raise KeyError(f"Document with id '{document_id}' does not exist")

        # Calculate the next version number
        # Extract version numbers and find the maximum
        version_numbers = []
        for version in existing_versions:
            try:
                # Parse version number (support formats like "1.0", "2", "1.5.3")
                # Take the first numeric part for comparison
                parts = version.version_number.split(".")
                major_version = float(parts[0]) if parts else 0.0
                minor_version = float(parts[1]) if len(parts) > 1 else 0.0
                version_numbers.append((major_version, minor_version))
            except (ValueError, IndexError):
                # If parsing fails, skip this version
                continue

        # Determine the next version number
        if version_numbers:
            max_major, max_minor = max(version_numbers)
            # Increment minor version
            next_version_number = f"{int(max_major)}.{int(max_minor) + 1}"
        else:
            # If no valid version numbers found, start with 1.0
            next_version_number = "1.0"

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

        # Create timestamp
        created_at = datetime.now()

        # Create the new DocumentVersion object
        new_version = DocumentVersion(
            version_id=version_id,
            document_id=document_id,
            version_number=next_version_number,
            content=content,
            version_note=version_note,
            created_at=created_at
        )

        # Add the new version to the database
        document_version_table[version_id] = new_version
        setattr(db, "document_version", document_version_table)

        # Return the result in the specified format
        return {
            "version_id": version_id,
            "version_number": next_version_number,
            "created_at": created_at.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def update_document_content(self, document_id: str, content: str):
        """
        Update the content of an existing document in the database.

        This method retrieves the document by its ID, updates its content field,
        and records the timestamp of the update. If the document doesn't exist,
        it raises a KeyError.

        Args:
            document_id: The unique identifier of the document to update
            content: The new content to replace the existing document content

        Returns:
            dict: A dictionary containing:
                - success (bool): True if update was successful
                - updated_at (str): Timestamp of update in "yyyy-mm-dd HH:MM:SS" format

        Raises:
            KeyError: If the document with the given document_id does not exist
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Retrieve the document table from the database
        document_table = getattr(db, "document", None)

        # Check if the document table exists
        if document_table is None:
            raise KeyError(f"Document table does not exist in the database")

        # Check if the document exists in the table
        if document_id not in document_table:
            raise KeyError(f"Document with ID '{document_id}' does not exist")

        # Retrieve the document object
        document = document_table[document_id]

        # Update the document content
        document.content = content

        # Update the timestamp to current time
        current_time = datetime.now()
        document.updated_at = current_time

        # Save the updated document back to the database
        document_table[document_id] = document
        setattr(db, "document", document_table)

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

        # Return success response with updated timestamp
        return {
            "success": True,
            "updated_at": updated_at_str
        }

    @is_tool()
    def set_document_permission(self, document_id: str, user_id: str, permission_level: Literal["read", "write", "admin"]):
        """
        Set access permissions for a document by creating or updating permission records.

        This method grants or updates the specified permission level for a user on a document.
        If a permission record already exists for the user-document pair, it updates the permission level.
        Otherwise, it creates a new permission record.

        Args:
            document_id: The unique identifier of the document
            user_id: The identifier of the user to grant permission
            permission_level: The level of permission to grant (must be "read", "write", or "admin")

        Returns:
            dict: Contains success status and timestamp
                - success (bool): True if permission was set successfully
                - updated_at (str): Timestamp in "yyyy-mm-dd HH:MM:SS" format

        Raises:
            KeyError: If document_permission table doesn't exist in database
            ValueError: If permission_level is not one of the allowed values
        """
        from datetime import datetime

        # Import the DocumentPermission class (already imported at file header)
        # Validate permission_level is one of the allowed enum values
        allowed_permissions = ["read", "write", "admin"]
        if permission_level not in allowed_permissions:
            raise ValueError(f"Invalid permission_level '{permission_level}'. Must be one of {allowed_permissions}")

        # Validate required parameters are not empty
        if not document_id or not isinstance(document_id, str):
            raise ValueError("document_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")

        # Access the database
        db = self.db

        # Get the document_permission table, raise KeyError if it doesn't exist
        document_permission_table = getattr(db, "document_permission", None)
        if document_permission_table is None:
            raise KeyError("document_permission table not found in database")

        # Generate current timestamp
        current_time = datetime.now()

        # Check if a permission record already exists for this document-user pair
        # Iterate through existing permissions to find matching document_id and user_id
        existing_permission_key = None
        for perm_key, perm_obj in document_permission_table.items():
            if perm_obj.document_id == document_id and perm_obj.user_id == user_id:
                existing_permission_key = perm_key
                break

        if existing_permission_key:
            # Update existing permission record
            existing_permission = document_permission_table[existing_permission_key]
            existing_permission.permission_level = permission_level
            existing_permission.updated_at = current_time

            # Write back to database
            setattr(db, "document_permission", document_permission_table)
        else:
            # Create new permission record
            # Generate unique permission ID using secure hash
            import secrets, hashlib
            permission_id = "perm_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

            # Create new DocumentPermission object
            new_permission = DocumentPermission(
                document_id=document_id,
                user_id=user_id,
                permission_level=permission_level,
                updated_at=current_time
            )

            # Add to table
            document_permission_table[permission_id] = new_permission

            # Write back to database
            setattr(db, "document_permission", document_permission_table)

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

        # Return success response
        return {
            "success": True,
            "updated_at": updated_at_str
        }

    @is_tool()
    def search_documents_by_tags(self, tags: List[str], match_mode: Literal["all", "any"] = "any") -> Dict[str, Any]:
        """
        Search documents by one or more tags.

        Args:
            tags: List of tags to search for
            match_mode: Whether to match all tags ("all") or any tag ("any")

        Returns:
            Dictionary containing:
            - documents: List of documents matching the tag criteria
            - total_count: Total number of matching documents

        Raises:
            ValueError: If tags list is empty or match_mode is invalid
        """
        # Import necessary classes (already imported at file header, but showing for clarity)
        from thefuzz import fuzz, process

        # Validate input parameters
        if not tags or len(tags) == 0:
            raise ValueError("Tags list cannot be empty")

        # Ensure match_mode is valid (safety check for enum parameter)
        if match_mode not in ["all", "any"]:
            raise ValueError(f"Invalid match_mode: {match_mode}. Must be 'all' or 'any'")

        # Get database instance
        db = self.db

        # Get document and document_tag tables
        document_table = getattr(db, "document", None)
        document_tag_table = getattr(db, "document_tag", None)

        # Check if tables exist and have data
        if document_table is None or document_tag_table is None:
            return {
                "documents": [],
                "total_count": 0
            }

        # Normalize input tags to lowercase for case-insensitive matching
        normalized_tags = [tag.lower().strip() for tag in tags]

        # Build a mapping of document_id to set of tags for that document
        # This allows efficient tag matching logic
        document_tags_map: Dict[str, set] = {}

        # Iterate through all document_tag entries
        for tag_entry_key, tag_entry in document_tag_table.items():
            doc_id = tag_entry.document_id
            tag_name = tag_entry.tag.lower().strip()

            # Initialize set for this document if not exists
            if doc_id not in document_tags_map:
                document_tags_map[doc_id] = set()

            # Use fuzzy matching to check if this tag matches any of the search tags
            # For tags, we use partial_ratio to allow partial matches
            for search_tag in normalized_tags:
                similarity = fuzz.partial_ratio(search_tag, tag_name)
                # Use threshold of 80 for fuzzy matching
                if similarity >= 80:
                    document_tags_map[doc_id].add(search_tag)
                    break

        # Find matching documents based on match_mode
        matching_document_ids = set()

        if match_mode == "all":
            # Document must have all tags from the search list
            for doc_id, doc_tags in document_tags_map.items():
                # Check if all search tags are present in document's tags
                if all(search_tag in doc_tags for search_tag in normalized_tags):
                    matching_document_ids.add(doc_id)
        else:  # match_mode == "any"
            # Document must have at least one tag from the search list
            for doc_id, doc_tags in document_tags_map.items():
                # Check if any search tag is present in document's tags
                if any(search_tag in doc_tags for search_tag in normalized_tags):
                    matching_document_ids.add(doc_id)

        # Retrieve full document objects for matching document IDs
        matching_documents = []
        for doc_id in matching_document_ids:
            doc = document_table.get(doc_id)
            if doc:
                # Convert document to dictionary format for return
                matching_documents.append({
                    "document_id": doc.document_id,
                    "title": doc.title,
                    "content": doc.content,
                    "author": doc.author,
                    "category": doc.category,
                    "folder_id": doc.folder_id,
                    "is_archived": doc.is_archived,
                    "file_size": doc.file_size,
                    "created_at": doc.created_at.strftime("%Y-%m-%d %H:%M:%S"),
                    "updated_at": doc.updated_at.strftime("%Y-%m-%d %H:%M:%S"),
                    "archived_at": doc.archived_at.strftime("%Y-%m-%d %H:%M:%S") if doc.archived_at else None
                })

        # Sort documents by updated_at timestamp (most recent first) for consistent ordering
        matching_documents.sort(key=lambda x: x["updated_at"], reverse=True)

        # Return results
        return {
            "documents": matching_documents,
            "total_count": len(matching_documents)
        }

    @is_tool()
    def get_document_metadata(self, document_id: str):
        """
        Retrieve metadata information of a document without content.

        This method fetches all metadata fields of a document (excluding the content field)
        and returns them in a structured format. It retrieves document information from
        the document table and associated tags from the document_tag table.

        Args:
            document_id: The unique identifier of the document to retrieve metadata for

        Returns:
            A dictionary containing document metadata with the following fields:
            - document_id: The unique identifier of the document
            - title: The title of the document
            - author: The author name of the document
            - tags: List of tags associated with the document
            - created_at: The creation timestamp in yyyy-mm-dd HH:MM:SS format
            - updated_at: The last update timestamp in yyyy-mm-dd HH:MM:SS format
            - file_size: The size of the document in bytes

        Raises:
            KeyError: If the document with the given document_id does not exist
        """
        # Access the database instance
        db = self.db

        # Retrieve the document table from database
        document_table = getattr(db, "document", None)

        # Check if document table exists and contains data
        if document_table is None:
            raise KeyError(f"Document table not found in database")

        # Check if the document exists in the table
        if document_id not in document_table:
            raise KeyError(f"Document with document_id '{document_id}' does not exist")

        # Retrieve the document object
        document = document_table[document_id]

        # Retrieve the document_tag table to get associated tags
        document_tag_table = getattr(db, "document_tag", None)

        # Collect all tags associated with this document
        tags = []
        if document_tag_table is not None:
            # Iterate through all document_tag entries to find matching document_id
            for tag_entry in document_tag_table.values():
                if tag_entry.document_id == document_id:
                    tags.append(tag_entry.tag)

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

        # Handle updated_at timestamp
        updated_at_str = document.updated_at.strftime("%Y-%m-%d %H:%M:%S")

        # Construct and return the metadata dictionary (excluding content field)
        metadata = {
            "document_id": document.document_id,
            "title": document.title,
            "author": document.author,
            "tags": tags,
            "created_at": created_at_str,
            "updated_at": updated_at_str,
            "file_size": document.file_size
        }

        return metadata

    @is_tool()
    def validate_document_format(self, content: str, format: Literal["markdown", "html", "plain_text"]) -> dict:
        """
        Validate if document content follows a specific format or structure.

        Args:
            content: The document content to validate
            format: The expected format type (markdown, html, or plain_text)

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

        Raises:
            ValueError: If parameters are invalid
        """
        # Import necessary libraries for format validation
        import re
        from html.parser import HTMLParser

        # Initialize validation result
        validation_errors = []
        is_valid = True

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

        if not isinstance(format, str):
            raise ValueError("Format must be a string")

        # Validate format parameter against enum values (safety protection)
        valid_formats = ["markdown", "html", "plain_text"]
        if format not in valid_formats:
            raise ValueError(f"Format must be one of {valid_formats}, got '{format}'")

        # Handle empty content
        if not content or content.strip() == "":
            validation_errors.append("Content cannot be empty")
            return {
                "is_valid": False,
                "validation_errors": validation_errors
            }

        # Format-specific validation
        if format == "markdown":
            # Validate markdown format
            validation_errors = self._validate_markdown(content)

        elif format == "html":
            # Validate HTML format
            validation_errors = self._validate_html(content)

        elif format == "plain_text":
            # Validate plain text format
            validation_errors = self._validate_plain_text(content)

        # Determine if content is valid based on errors
        is_valid = len(validation_errors) == 0

        return {
            "is_valid": is_valid,
            "validation_errors": validation_errors
        }

    def _validate_markdown(self, content: str) -> list:
        """
        Validate markdown format content.

        Args:
            content: Markdown content to validate

        Returns:
            list: List of validation error messages
        """
        import re

        errors = []
        lines = content.split('\n')

        # Check for at least one heading (# Title)
        has_heading = False
        heading_pattern = re.compile(r'^#{1,6}\s+.+')

        for line in lines:
            if heading_pattern.match(line.strip()):
                has_heading = True
                break

        if not has_heading:
            errors.append("Missing required heading (markdown should contain at least one heading starting with #)")

        # Check for malformed markdown syntax
        # Check for unmatched bold/italic markers
        bold_pattern = re.compile(r'\*\*')
        italic_pattern = re.compile(r'(?<!\*)\*(?!\*)')

        bold_count = len(bold_pattern.findall(content))
        if bold_count % 2 != 0:
            errors.append("Unmatched bold markers (**) detected")

        # Check for malformed links [text](url)
        link_pattern = re.compile(r'\[([^\]]+)\]\(([^\)]+)\)')
        malformed_link_pattern = re.compile(r'\[[^\]]*\](?!\()|(?<!\])\([^\)]*\)')

        if malformed_link_pattern.search(content):
            errors.append("Malformed link syntax detected (links should follow [text](url) format)")

        # Check for malformed code blocks
        code_block_markers = content.count('```')
        if code_block_markers % 2 != 0:
            errors.append("Unmatched code block markers (```) detected")

        # Check for malformed lists (should start with -, *, or + followed by space)
        list_pattern = re.compile(r'^[\-\*\+](?!\s)')
        for i, line in enumerate(lines, 1):
            stripped = line.strip()
            if stripped and list_pattern.match(stripped):
                errors.append(f"Malformed list item at line {i} (list markers should be followed by a space)")

        return errors

    def _validate_html(self, content: str) -> list:
        """
        Validate HTML format content.

        Args:
            content: HTML content to validate

        Returns:
            list: List of validation error messages
        """
        from html.parser import HTMLParser

        errors = []

        # Custom HTML parser to detect errors
        class ValidationHTMLParser(HTMLParser):
            def __init__(self):
                super().__init__()
                self.errors = []
                self.tag_stack = []
                self.has_content = False

            def handle_starttag(self, tag, attrs):
                self.tag_stack.append(tag)

            def handle_endtag(self, tag):
                if not self.tag_stack:
                    self.errors.append(f"Closing tag </{tag}> without matching opening tag")
                elif self.tag_stack[-1] != tag:
                    self.errors.append(f"Mismatched tags: expected </{self.tag_stack[-1]}>, got </{tag}>")
                    # Try to find matching opening tag
                    if tag in self.tag_stack:
                        self.tag_stack.remove(tag)
                else:
                    self.tag_stack.pop()

            def handle_data(self, data):
                if data.strip():
                    self.has_content = True

            def error(self, message):
                self.errors.append(f"HTML parsing error: {message}")

        parser = ValidationHTMLParser()

        try:
            parser.feed(content)

            # Check for unclosed tags
            if parser.tag_stack:
                errors.append(f"Unclosed HTML tags: {', '.join(parser.tag_stack)}")

            # Add parser-detected errors
            errors.extend(parser.errors)

            # Check if HTML has actual content
            if not parser.has_content:
                errors.append("HTML content has no text data")

            # Basic HTML structure validation
            content_lower = content.lower()
            if '<html' not in content_lower and '<body' not in content_lower and '<div' not in content_lower and '<p' not in content_lower:
                errors.append("HTML content should contain at least basic structural tags (html, body, div, or p)")

        except Exception as e:
            errors.append(f"HTML parsing failed: {str(e)}")

        return errors

    def _validate_plain_text(self, content: str) -> list:
        """
        Validate plain text format content.

        Args:
            content: Plain text content to validate

        Returns:
            list: List of validation error messages
        """
        import re

        errors = []

        # Check for HTML tags (should not exist in plain text)
        html_tag_pattern = re.compile(r'<[^>]+>')
        if html_tag_pattern.search(content):
            errors.append("Plain text should not contain HTML tags")

        # Check for markdown syntax (should not exist in plain text)
        markdown_patterns = [
            (r'^#{1,6}\s+', "Plain text should not contain markdown headings (#)"),
            (r'\*\*[^\*]+\*\*', "Plain text should not contain markdown bold syntax (**)"),
            (r'\[[^\]]+\]\([^\)]+\)', "Plain text should not contain markdown links"),
            (r'```', "Plain text should not contain markdown code blocks (```)"),
        ]

        for pattern, error_msg in markdown_patterns:
            if re.search(pattern, content, re.MULTILINE):
                errors.append(error_msg)
                break  # Report only first markdown syntax issue

        # Check for excessive special characters that might indicate formatting
        special_char_ratio = len(re.findall(r'[<>{}[\]\\|]', content)) / len(content) if len(content) > 0 else 0
        if special_char_ratio > 0.1:  # More than 10% special formatting characters
            errors.append("Plain text contains excessive formatting characters")

        # Check for valid text content (should have some alphanumeric characters)
        if not re.search(r'[a-zA-Z0-9]', content):
            errors.append("Plain text should contain alphanumeric characters")

        return errors

    @is_tool()
    def lock_document(self, document_id: str, user_id: str):
        # Import datetime for timestamp formatting
        from datetime import datetime

        # Validate input parameters - check for non-empty strings
        if not document_id or not isinstance(document_id, str):
            raise ValueError("document_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")

        # Access the database instance
        db = self.db

        # Get the document_lock table from database
        document_lock_table = getattr(db, "document_lock", None)

        # Initialize the table if it doesn't exist
        if document_lock_table is None:
            document_lock_table = {}
            setattr(db, "document_lock", document_lock_table)

        # Check if the document is already locked (pre-condition validation)
        if document_id in document_lock_table:
            # Document is already locked, raise RuntimeError
            existing_lock = document_lock_table[document_id]
            raise RuntimeError(
                f"Document {document_id} is already locked by user {existing_lock.user_id} "
                f"at {existing_lock.locked_at.strftime('%Y-%m-%d %H:%M:%S')}"
            )

        # Create timestamp for when the document is being locked
        locked_at_datetime = datetime.now()

        # Create a new DocumentLock instance
        new_lock = DocumentLock(
            document_id=document_id,
            user_id=user_id,
            locked_at=locked_at_datetime
        )

        # Add the lock to the document_lock table
        document_lock_table[document_id] = new_lock

        # Update the database with the modified table
        setattr(db, "document_lock", document_lock_table)

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

        # Return success response with formatted timestamp
        return {
            "success": True,
            "locked_at": locked_at_str
        }

    @is_tool()
    def get_linked_documents(self, document_id: str, link_type: Literal["references", "related", "supersedes", "depends_on"] = None):
        """
        Retrieve all documents linked to a specific document.

        This method queries the document_link table to find all documents that are linked
        to the specified document, either as source or target. It can optionally filter
        by link_type.

        Args:
            document_id: The unique identifier of the document to find links for
            link_type: Optional filter to retrieve only specific types of links
                       (references, related, supersedes, or depends_on)

        Returns:
            A dictionary containing:
            - linked_documents: List of linked document information
            - total_count: Total number of linked documents found

        Raises:
            KeyError: If the document_id does not exist in any document links
        """
        # Access the database
        db = self.db

        # Get the document_link table
        document_links = getattr(db, "document_link", None)

        # If document_link table doesn't exist or is empty, raise KeyError
        if document_links is None or len(document_links) == 0:
            raise KeyError(f"Document with id '{document_id}' not found in document links")

        # List to store the linked documents
        linked_documents = []

        # Flag to check if the document exists in any link
        document_exists = False

        # Iterate through all document links
        for link_id, link in document_links.items():
            # Check if the document is either source or target in the link
            is_source = link.source_document_id == document_id
            is_target = link.target_document_id == document_id

            # If document is found in any link, mark it as existing
            if is_source or is_target:
                document_exists = True

                # Apply link_type filter if specified
                if link_type is not None and link.link_type != link_type:
                    continue

                # Determine which document to include in results
                # If current document is source, include target; if target, include source
                if is_source:
                    linked_doc_id = link.target_document_id
                else:
                    linked_doc_id = link.source_document_id

                # Create the linked document entry
                linked_doc_entry = {
                    "document_id": linked_doc_id,
                    "link_type": link.link_type,
                    "link_id": link.link_id,
                    "created_at": link.created_at.strftime("%Y-%m-%d %H:%M:%S")
                }

                # Add to results list
                linked_documents.append(linked_doc_entry)

        # If document was not found in any link, raise KeyError
        if not document_exists:
            raise KeyError(f"Document with id '{document_id}' not found in document links")

        # Return the results with total count
        return {
            "linked_documents": linked_documents,
            "total_count": len(linked_documents)
        }

    @is_tool()
    def get_document_version_history(self, document_id: str):
        """
        Retrieve the version history of a document.

        This method fetches all versions associated with a given document ID,
        sorts them by creation time, and returns a comprehensive version history.

        Args:
            document_id: The unique identifier of the document

        Returns:
            A dictionary containing:
            - versions: List of all versions with their details
            - total_versions: Total count of versions

        Raises:
            KeyError: If the document_id does not exist in the database
        """
        # Import necessary datetime module for time formatting
        from datetime import datetime

        # Access the database instance
        db = self.db

        # Retrieve the document_version table from database
        document_version_table = getattr(db, "document_version", None)

        # Check if the document_version table exists
        if document_version_table is None:
            raise KeyError(f"Document version table not found in database")

        # Filter versions that belong to the specified document_id
        matching_versions = []
        document_found = False

        for version_id, version_obj in document_version_table.items():
            # Check if this version belongs to the requested document
            if version_obj.document_id == document_id:
                document_found = True
                # Build version information dictionary
                version_info = {
                    "version_id": version_obj.version_id,
                    "version_number": version_obj.version_number,
                    "content": version_obj.content,
                    "created_at": version_obj.created_at.strftime("%Y-%m-%d %H:%M:%S")
                }

                # Add optional version_note if it exists
                if version_obj.version_note is not None:
                    version_info["version_note"] = version_obj.version_note

                matching_versions.append(version_info)

        # If no versions found for this document_id, raise KeyError
        if not document_found:
            raise KeyError(f"No versions found for document_id: {document_id}")

        # Sort versions by created_at timestamp (oldest to newest)
        # This provides a chronological history of the document
        matching_versions.sort(key=lambda v: datetime.strptime(v["created_at"], "%Y-%m-%d %H:%M:%S"))

        # Calculate total number of versions
        total_versions = len(matching_versions)

        # Return the complete version history
        return {
            "versions": matching_versions,
            "total_versions": total_versions
        }

    @is_tool()
    def share_document_with_users(self, document_id: str, user_ids: List[str], permission_level: Literal["read", "write"], message: str = None):
        """
        Share a document with multiple users by creating or updating document permissions.

        Args:
            document_id: The unique identifier of the document to share
            user_ids: List of user identifiers to share the document with
            permission_level: The permission level to grant (must be "read" or "write")
            message: Optional message to include with the share (not stored in database, for notification purposes)

        Returns:
            dict: Contains success status and count of users shared with
                - success: Boolean indicating if sharing was successful
                - shared_with_count: Number of users the document was successfully shared with

        Raises:
            KeyError: If document_permission table doesn't exist in database
            ValueError: If parameters are invalid
        """
        from datetime import datetime

        # Validate permission_level is one of the allowed enum values
        if permission_level not in ["read", "write"]:
            raise ValueError(f"Invalid permission_level '{permission_level}'. Must be 'read' or 'write'.")

        # Validate required parameters
        if not document_id:
            raise ValueError("document_id cannot be empty")

        if not user_ids or len(user_ids) == 0:
            raise ValueError("user_ids list cannot be empty")

        # Access the database
        db = self.db

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

        # Get current timestamp for permission updates
        current_time = datetime.now()

        # Counter for successfully shared users
        shared_count = 0

        # Iterate through each user and create/update their permission
        for user_id in user_ids:
            # Skip empty user_ids
            if not user_id:
                continue

            # Check if permission already exists for this document-user pair
            existing_permission_key = None
            for perm_id, perm in document_permission_table.items():
                if perm.document_id == document_id and perm.user_id == user_id:
                    existing_permission_key = perm_id
                    break

            # Create new permission entry
            new_permission = DocumentPermission(
                document_id=document_id,
                user_id=user_id,
                permission_level=permission_level,
                updated_at=current_time
            )

            # Update the permission in the database
            if existing_permission_key:
                # Update existing permission using the found key
                document_permission_table[existing_permission_key] = new_permission
            else:
                # Create new permission entry with unique key
                permission_key = f"{document_id}_{user_id}"
                document_permission_table[permission_key] = new_permission

            # Increment the shared count
            shared_count += 1

        # Save the updated table back to database
        setattr(db, "document_permission", document_permission_table)

        # Return success result with count of users shared with
        return {
            "success": True,
            "shared_with_count": shared_count
        }

    @is_tool()
    def update_document_title(self, document_id: str, title: str):
        # Get the database instance
        db = self.db

        # Retrieve the document table from the database
        document_table = getattr(db, "document", None)

        # Validate that the document table exists
        if document_table is None:
            raise KeyError("Document table does not exist in the database")

        # Check if the document exists in the table
        if document_id not in document_table:
            raise KeyError(f"Document with ID '{document_id}' does not exist")

        # Retrieve the document object
        document = document_table[document_id]

        # Validate that the new title is not empty
        if not title or not title.strip():
            raise ValueError("Title cannot be empty or whitespace only")

        # Update the document title
        document.title = title.strip()

        # Update the timestamp to current time
        current_time = datetime.now()
        document.updated_at = current_time

        # Save the updated document back to the database
        document_table[document_id] = document
        setattr(db, "document", document_table)

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

        # Return success response with updated timestamp
        return {
            "success": True,
            "updated_at": updated_at_str
        }

    @is_tool()
    def remove_document_link(self, link_id: str):
        """
        Remove a link between two documents by link_id.

        This method removes an existing document link from the database.
        It validates that the link exists before attempting removal.

        Args:
            link_id: The unique identifier of the link to be removed

        Returns:
            dict: Contains success status and removal timestamp
                - success: Boolean indicating if removal was successful
                - removed_at: Timestamp string in "yyyy-mm-dd HH:MM:SS" format

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

        # Access the database instance
        db = self.db

        # Get the document_link table from database
        document_links = getattr(db, "document_link", None)

        # Validate that the document_link table exists
        if document_links is None:
            raise KeyError(f"document_link table not found in database")

        # Check if the link exists in the database
        if link_id not in document_links:
            raise KeyError(f"Link with link_id '{link_id}' does not exist")

        # Remove the link from the database
        # Create a new dictionary without the specified link_id
        updated_links = {lid: link for lid, link in document_links.items() if lid != link_id}

        # Update the database with the modified links
        setattr(db, "document_link", updated_links)

        # Get the current timestamp for removal record
        removed_at = datetime.now().strftime("%Y-%m-%d %H:%M:%S")

        # Return success response with removal timestamp
        return {
            "success": True,
            "removed_at": removed_at
        }

    @is_tool()
    def delete_document(self, document_id: str):
        """
        Delete a document from the library

        This method removes a document from the database by its unique identifier.
        It validates that the document exists before deletion and returns the deletion
        timestamp.

        Args:
            document_id: The unique identifier of the document to delete

        Returns:
            dict: Contains success status and deletion timestamp
                - success (bool): True if deletion was successful
                - deleted_at (str): Timestamp in "yyyy-mm-dd HH:MM:SS" format

        Raises:
            KeyError: If the document with the given document_id does not exist
        """
        from datetime import datetime

        # Access the database instance
        db = self.db

        # Get the document table from database
        document_table = getattr(db, "document", None)

        # Check if document table exists
        if document_table is None:
            raise KeyError(f"Document table not found in database")

        # Check if the document exists in the table
        if document_id not in document_table:
            raise KeyError(f"Document with document_id '{document_id}' does not exist")

        # Get current timestamp for deletion record
        deleted_at = datetime.now()

        # Remove the document from the table
        # Create a new dictionary without the deleted document
        updated_table = {key: value for key, value in document_table.items() if key != document_id}

        # Update the database with the new table (without the deleted document)
        setattr(db, "document", updated_table)

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

        # Return success response with deletion timestamp
        return {
            "success": True,
            "deleted_at": deleted_at_str
        }

    @is_tool()
    def revoke_document_access(self, document_id: str, user_id: str):
        """
        Revoke a user's access to a document by removing their permission entry.

        Args:
            document_id: The unique identifier of the document
            user_id: The identifier of the user whose access to revoke

        Returns:
            dict: Contains success status and revoked_at timestamp

        Raises:
            KeyError: If document permission does not exist for the given document_id and user_id
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Get the document_permission table
        document_permission_table = getattr(db, "document_permission", None)

        # Check if the document_permission table exists
        if document_permission_table is None:
            raise KeyError(f"Document permission table does not exist")

        # Find and remove the permission entry for this document_id and user_id
        # The table is a Dict[str, DocumentPermission] where key is document_id
        # We need to find the entry that matches both document_id and user_id
        permission_key_to_remove = None

        for key, permission in document_permission_table.items():
            if permission.document_id == document_id and permission.user_id == user_id:
                permission_key_to_remove = key
                break

        # Check if permission entry exists
        if permission_key_to_remove is None:
            raise KeyError(f"No permission found for document_id: {document_id} and user_id: {user_id}")

        # Remove the permission entry from the table
        # Create a new dictionary without the revoked permission
        updated_permissions = {k: v for k, v in document_permission_table.items() if k != permission_key_to_remove}

        # Update the database with the modified permissions table
        setattr(db, "document_permission", updated_permissions)

        # Get the current timestamp for revoked_at
        revoked_at = datetime.now()

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

        # Return success response with timestamp
        return {
            "success": True,
            "revoked_at": revoked_at_str
        }

    @is_tool()
    def restore_document_version(self, document_id: str, version_id: str):
        """
        Restore a document to a specific version by updating the document's content
        and metadata to match the specified version.

        Args:
            document_id: The unique identifier of the document to restore
            version_id: The unique identifier of the version to restore to

        Returns:
            dict: Contains success status and restoration timestamp

        Raises:
            KeyError: If document_id or version_id does not exist
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Retrieve the document table
        document_table = getattr(db, "document", None)
        if document_table is None:
            raise KeyError(f"Document table does not exist in database")

        # Retrieve the document_version table
        version_table = getattr(db, "document_version", None)
        if version_table is None:
            raise KeyError(f"Document version table does not exist in database")

        # Check if the document exists
        if document_id not in document_table:
            raise KeyError(f"Document with id '{document_id}' does not exist")

        # Check if the version exists
        if version_id not in version_table:
            raise KeyError(f"Version with id '{version_id}' does not exist")

        # Get the document and version objects
        document = document_table[document_id]
        version = version_table[version_id]

        # Verify that the version belongs to the specified document
        if version.document_id != document_id:
            raise KeyError(f"Version '{version_id}' does not belong to document '{document_id}'")

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

        # Restore the document content to the version's content
        document.content = version.content

        # Update the document's updated_at timestamp to reflect the restoration
        document.updated_at = restored_at

        # Save the updated document back to the database
        document_table[document_id] = document
        setattr(db, "document", document_table)

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

        # Return success response
        return {
            "success": True,
            "restored_at": restored_at_str
        }

    @is_tool()
    def get_documents_by_author(self, author: str):
        """
        Retrieve all documents created by a specific author.

        This method searches through all documents in the database and returns those
        whose author field matches the provided author name. Uses fuzzy matching to
        handle variations in author name formatting.

        Args:
            author: The author name to filter by

        Returns:
            dict: Contains 'documents' (list of matching document dicts) and 'total_count' (int)

        Raises:
            ValueError: If author parameter is empty or invalid
            Exception: If database access fails
        """
        from thefuzz import process

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

        author = author.strip()
        if not author:
            raise ValueError("Author parameter cannot be empty or whitespace only")

        try:
            # Access the database
            db = self.db

            # Get the document table from database
            document_table = getattr(db, "document", None)

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

            # Collect all documents with their authors for fuzzy matching
            all_documents = []
            author_names = []

            for doc_id, doc_obj in document_table.items():
                all_documents.append(doc_obj)
                author_names.append(doc_obj.author)

            # Use fuzzy matching to find similar author names
            # Get all matches with similarity score >= 80 (adjustable threshold)
            matches = process.extract(author, author_names, limit=len(author_names))

            # Filter documents whose author names have similarity >= 80
            matching_documents = []
            similarity_threshold = 80

            for i, (matched_author, score) in enumerate(matches):
                if score >= similarity_threshold:
                    # Find all documents with this author name
                    for doc_obj in all_documents:
                        if doc_obj.author == matched_author:
                            # Convert document object to dictionary format
                            doc_dict = {
                                "document_id": doc_obj.document_id,
                                "title": doc_obj.title,
                                "content": doc_obj.content,
                                "author": doc_obj.author,
                                "category": doc_obj.category,
                                "folder_id": doc_obj.folder_id,
                                "is_archived": doc_obj.is_archived,
                                "file_size": doc_obj.file_size,
                                # Convert datetime objects to string format "yyyy-mm-dd HH:MM:SS"
                                "created_at": doc_obj.created_at.strftime("%Y-%m-%d %H:%M:%S"),
                                "updated_at": doc_obj.updated_at.strftime("%Y-%m-%d %H:%M:%S"),
                                "archived_at": doc_obj.archived_at.strftime("%Y-%m-%d %H:%M:%S") if doc_obj.archived_at else None
                            }
                            matching_documents.append(doc_dict)

            # Sort documents by created_at timestamp (newest first)
            matching_documents.sort(key=lambda x: x["created_at"], reverse=True)

            # Return results with document list and total count
            return {
                "documents": matching_documents,
                "total_count": len(matching_documents)
            }

        except AttributeError as e:
            raise Exception(f"Database access error: {str(e)}")
        except Exception as e:
            raise Exception(f"Failed to retrieve documents by author: {str(e)}")

    @is_tool()
    def get_recently_modified_documents(self, limit: int = None, days: int = None):
        """
        Retrieve documents that were recently modified within the specified number of days.

        Args:
            limit: Maximum number of documents to return (optional)
            days: Number of days to look back for modifications (optional)

        Returns:
            dict: Contains 'documents' (list of recently modified documents) and 'total_count' (total number found)

        Raises:
            Exception: If there are errors accessing the database or processing documents
        """
        from datetime import datetime, timedelta

        try:
            # Access the database
            db = self.db

            # Get the document table
            document_table = getattr(db, "document", None)

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

            # Get current time for comparison
            current_time = datetime.now()

            # Calculate the cutoff time if days parameter is provided
            # If days is not provided, include all documents
            cutoff_time = None
            if days is not None:
                if not isinstance(days, int) or days < 0:
                    raise ValueError("Parameter 'days' must be a non-negative integer")
                cutoff_time = current_time - timedelta(days=days)

            # Collect all documents that meet the criteria
            matching_documents = []

            for doc_id, doc in document_table.items():
                # Check if document was updated within the specified time range
                if cutoff_time is None or doc.updated_at >= cutoff_time:
                    # Format the document data for output
                    doc_data = {
                        "document_id": doc.document_id,
                        "title": doc.title,
                        "updated_at": doc.updated_at.strftime("%Y-%m-%d %H:%M:%S")
                    }
                    matching_documents.append(doc_data)

            # Sort documents by updated_at in descending order (most recent first)
            matching_documents.sort(key=lambda x: x["updated_at"], reverse=True)

            # Get total count before applying limit
            total_count = len(matching_documents)

            # Apply limit if specified
            if limit is not None:
                if not isinstance(limit, int) or limit < 0:
                    raise ValueError("Parameter 'limit' must be a non-negative integer")
                matching_documents = matching_documents[:limit]

            # Return the result
            return {
                "documents": matching_documents,
                "total_count": total_count
            }

        except ValueError as e:
            # Re-raise ValueError for parameter validation errors
            raise e
        except Exception as e:
            # Catch any other unexpected errors
            raise Exception(f"Error retrieving recently modified documents: {str(e)}")

    @is_tool()
    def add_document_attachment(self, document_id: str, file_path: str, file_name: str, file_type: Optional[str] = None):
        """
        Add an attachment file to a document.

        This method creates a new attachment record associated with a specific document.
        It generates a unique attachment ID, stores the file metadata, and records the creation timestamp.

        Args:
            document_id: The unique identifier of the document to attach the file to
            file_path: The path or URL to the attachment file
            file_name: The name of the attachment file
            file_type: Optional MIME type of the attachment (e.g., "image/png", "application/pdf")

        Returns:
            A dictionary containing:
                - attachment_id: The unique identifier of the newly created attachment
                - created_at: The timestamp when the attachment was added (yyyy-mm-dd HH:MM:SS format)

        Raises:
            KeyError: If the document_id does not exist in the database
        """
        import secrets
        import hashlib
        from datetime import datetime

        # Access the database
        db = self.db

        # Get the document_attachment table
        document_attachment_table = getattr(db, "document_attachment", None)

        # Initialize the table if it doesn't exist
        if document_attachment_table is None:
            document_attachment_table = {}
            setattr(db, "document_attachment", document_attachment_table)

        # Verify that the document exists by checking if any attachment with this document_id exists
        # or if this is the first attachment for this document
        # Since we don't have access to the document table, we assume the document_id is valid
        # as per the pre-condition "Document exists and user has edit permission"
        # If document doesn't exist, we raise KeyError as specified in the schema

        # Note: Since we cannot verify document existence without access to the document table,
        # we will create the attachment and trust the pre-condition is met by the caller
        # In a production system, this would be validated against the document table

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

        # Ensure the generated ID is unique (collision prevention)
        while attachment_id in document_attachment_table:
            attachment_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp
        created_at = datetime.now()

        # Create the new attachment object
        new_attachment = DocumentAttachment(
            attachment_id=attachment_id,
            document_id=document_id,
            file_path=file_path,
            file_name=file_name,
            file_type=file_type,
            created_at=created_at
        )

        # Add the attachment to the database
        document_attachment_table[attachment_id] = new_attachment
        setattr(db, "document_attachment", document_attachment_table)

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

        # Return the result
        return {
            "attachment_id": attachment_id,
            "created_at": created_at_str
        }

    @is_tool()
    def compare_document_versions(self, content: str):
        """
        Compare two versions of a document and identify differences.

        This method analyzes the provided content and compares it with all existing document versions
        in the database to find the most similar version, then identifies the differences between them.

        Args:
            content: The content of the document version to compare

        Returns:
            dict: A dictionary containing:
                - differences: List of difference objects with type and text
                - similarity_score: Similarity percentage between versions (0-100)

        Raises:
            ValueError: If content is empty or no document versions exist for comparison
        """
        from thefuzz import fuzz
        from difflib import SequenceMatcher, unified_diff

        # Validate input
        if not content or not content.strip():
            raise ValueError("Content cannot be empty")

        # Access the database
        db = self.db
        document_version_table = getattr(db, "document_version", None)

        # Check if there are any document versions to compare with
        if not document_version_table or len(document_version_table) == 0:
            raise ValueError("No document versions available for comparison")

        # Find the most similar document version using fuzzy matching
        best_match_version = None
        best_similarity = 0

        for version_id, version in document_version_table.items():
            # Calculate similarity score using thefuzz
            similarity = fuzz.ratio(content, version.content)

            if similarity > best_similarity:
                best_similarity = similarity
                best_match_version = version

        # If no reasonable match found, compare with the first version
        if best_match_version is None:
            best_match_version = next(iter(document_version_table.values()))
            best_similarity = fuzz.ratio(content, best_match_version.content)

        # Calculate detailed differences using difflib
        original_lines = best_match_version.content.splitlines(keepends=True)
        new_lines = content.splitlines(keepends=True)

        # Use SequenceMatcher to identify differences
        matcher = SequenceMatcher(None, original_lines, new_lines)
        differences = []

        for tag, i1, i2, j1, j2 in matcher.get_opcodes():
            if tag == 'replace':
                # Lines were replaced
                original_text = ''.join(original_lines[i1:i2]).strip()
                new_text = ''.join(new_lines[j1:j2]).strip()
                differences.append({
                    "type": "modification",
                    "text": f"Changed from '{original_text}' to '{new_text}'"
                })
            elif tag == 'delete':
                # Lines were deleted from original
                deleted_text = ''.join(original_lines[i1:i2]).strip()
                differences.append({
                    "type": "deletion",
                    "text": deleted_text
                })
            elif tag == 'insert':
                # Lines were added in new version
                added_text = ''.join(new_lines[j1:j2]).strip()
                differences.append({
                    "type": "addition",
                    "text": added_text
                })
            # 'equal' tag means no changes, so we skip it

        # Calculate more accurate similarity score using SequenceMatcher
        # This provides a ratio between 0 and 1, we convert to percentage
        sequence_similarity = matcher.ratio() * 100

        # Use the average of fuzzy matching and sequence matching for final score
        final_similarity_score = (best_similarity + sequence_similarity) / 2

        # Round to one decimal place
        final_similarity_score = round(final_similarity_score, 1)

        return {
            "differences": differences,
            "similarity_score": final_similarity_score
        }

    @is_tool()
    def unlock_document(self, document_id: str, user_id: str):
        """
        Unlock a document to allow editing by others.

        This method removes the lock from a document, allowing other users to edit it.
        It verifies that the document is currently locked by the specified user before unlocking.

        Args:
            document_id: The unique identifier of the document to unlock
            user_id: The identifier of the user unlocking the document

        Returns:
            dict: A dictionary containing:
                - success (bool): True if unlock was successful
                - unlocked_at (str): Timestamp when document was unlocked in "yyyy-mm-dd HH:MM:SS" format

        Raises:
            RuntimeError: If document is not locked, or if the user is not the one who locked it
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Get the document_lock table
        document_lock_table = getattr(db, "document_lock", None)

        # Check if the document_lock table exists
        if document_lock_table is None:
            raise RuntimeError(f"Document lock table not found in database")

        # Check if the document is currently locked
        if document_id not in document_lock_table:
            raise RuntimeError(f"Document '{document_id}' is not currently locked")

        # Get the lock record
        lock_record = document_lock_table[document_id]

        # Verify that the user attempting to unlock is the one who locked it
        if lock_record.user_id != user_id:
            raise RuntimeError(
                f"Document '{document_id}' is locked by user '{lock_record.user_id}', "
                f"but unlock attempted by user '{user_id}'. Only the user who locked the document can unlock it"
            )

        # Remove the lock by deleting the record from the table
        del document_lock_table[document_id]

        # Update the database with the modified table
        setattr(db, "document_lock", document_lock_table)

        # Get the current timestamp for unlocked_at
        unlocked_at = datetime.now()
        unlocked_at_str = unlocked_at.strftime("%Y-%m-%d %H:%M:%S")

        # Return success response
        return {
            "success": True,
            "unlocked_at": unlocked_at_str
        }

    @is_tool()
    def convert_document_format(self, content: str, format: Literal["markdown", "html", "plain_text"]):
        """
        Convert document content from one format to another.

        This method converts content between markdown, html, and plain_text formats.
        The conversion strategy:
        - markdown -> html: Convert markdown syntax to HTML tags
        - markdown -> plain_text: Strip markdown syntax, keep text only
        - html -> markdown: Convert HTML tags to markdown syntax
        - html -> plain_text: Strip HTML tags, keep text only
        - plain_text -> markdown: Keep as-is (plain text is valid markdown)
        - plain_text -> html: Wrap in paragraph tags

        Args:
            content: The content to convert
            format: The current format of the content (markdown, html, or plain_text)

        Returns:
            dict: Contains 'converted_content' with the converted content in target format

        Raises:
            ValueError: If format is invalid or content cannot be converted
        """
        import re

        # Validate format parameter (enum constraint with safety protection)
        valid_formats = ["markdown", "html", "plain_text"]
        if format not in valid_formats:
            raise ValueError(f"Invalid format '{format}'. Must be one of: {', '.join(valid_formats)}")

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

        try:
            if format == "markdown":
                # Convert markdown to html
                html_content = content

                # Convert headers (# -> <h1>, ## -> <h2>, etc.)
                for i in range(6, 0, -1):
                    pattern = r'^' + '#' * i + r'\s+(.+)$'
                    replacement = f'<h{i}>\\1</h{i}>'
                    html_content = re.sub(pattern, replacement, html_content, flags=re.MULTILINE)

                # Convert bold (**text** or __text__ -> <strong>text</strong>)
                html_content = re.sub(r'\*\*(.+?)\*\*', r'<strong>\1</strong>', html_content)
                html_content = re.sub(r'__(.+?)__', r'<strong>\1</strong>', html_content)

                # Convert italic (*text* or _text_ -> <em>text</em>)
                html_content = re.sub(r'\*(.+?)\*', r'<em>\1</em>', html_content)
                html_content = re.sub(r'_(.+?)_', r'<em>\1</em>', html_content)

                # Convert links ([text](url) -> <a href="url">text</a>)
                html_content = re.sub(r'\[([^\]]+)\]\(([^\)]+)\)', r'<a href="\2">\1</a>', html_content)

                # Convert unordered lists (- item or * item -> <ul><li>item</li></ul>)
                lines = html_content.split('\n')
                in_ul = False
                processed_lines = []
                for line in lines:
                    if re.match(r'^[\*\-]\s+(.+)$', line):
                        if not in_ul:
                            processed_lines.append('<ul>')
                            in_ul = True
                        item = re.sub(r'^[\*\-]\s+(.+)$', r'<li>\1</li>', line)
                        processed_lines.append(item)
                    else:
                        if in_ul:
                            processed_lines.append('</ul>')
                            in_ul = False
                        processed_lines.append(line)
                if in_ul:
                    processed_lines.append('</ul>')
                html_content = '\n'.join(processed_lines)

                # Convert paragraphs (wrap non-empty lines not already in tags)
                lines = html_content.split('\n')
                processed_lines = []
                for line in lines:
                    stripped = line.strip()
                    if stripped and not re.match(r'^<[^>]+>', stripped):
                        processed_lines.append(f'<p>{stripped}</p>')
                    else:
                        processed_lines.append(line)
                html_content = '\n'.join(processed_lines)

                return {"converted_content": html_content}

            elif format == "html":
                # Convert html to markdown
                markdown_content = content

                # Convert headers (<h1> -> #, <h2> -> ##, etc.)
                for i in range(1, 7):
                    markdown_content = re.sub(
                        f'<h{i}>(.+?)</h{i}>', 
                        lambda m: '#' * i + ' ' + m.group(1), 
                        markdown_content, 
                        flags=re.IGNORECASE
                    )

                # Convert bold (<strong> or <b> -> **)
                markdown_content = re.sub(r'<strong>(.+?)</strong>', r'**\1**', markdown_content, flags=re.IGNORECASE)
                markdown_content = re.sub(r'<b>(.+?)</b>', r'**\1**', markdown_content, flags=re.IGNORECASE)

                # Convert italic (<em> or <i> -> *)
                markdown_content = re.sub(r'<em>(.+?)</em>', r'*\1*', markdown_content, flags=re.IGNORECASE)
                markdown_content = re.sub(r'<i>(.+?)</i>', r'*\1*', markdown_content, flags=re.IGNORECASE)

                # Convert links (<a href="url">text</a> -> [text](url))
                markdown_content = re.sub(r'<a\s+href=["\']([^"\']+)["\']>(.+?)</a>', r'[\2](\1)', markdown_content, flags=re.IGNORECASE)

                # Convert lists (<ul><li> -> -)
                markdown_content = re.sub(r'<ul>', '', markdown_content, flags=re.IGNORECASE)
                markdown_content = re.sub(r'</ul>', '', markdown_content, flags=re.IGNORECASE)
                markdown_content = re.sub(r'<li>(.+?)</li>', r'- \1', markdown_content, flags=re.IGNORECASE)

                # Remove paragraphs tags
                markdown_content = re.sub(r'<p>(.+?)</p>', r'\1', markdown_content, flags=re.IGNORECASE)

                # Remove any remaining HTML tags
                markdown_content = re.sub(r'<[^>]+>', '', markdown_content)

                # Clean up extra whitespace
                markdown_content = re.sub(r'\n\s*\n', '\n\n', markdown_content)

                return {"converted_content": markdown_content.strip()}

            elif format == "plain_text":
                # Convert plain_text to html (wrap in paragraph tags)
                lines = content.split('\n')
                html_lines = []
                for line in lines:
                    stripped = line.strip()
                    if stripped:
                        html_lines.append(f'<p>{stripped}</p>')
                    else:
                        html_lines.append('')
                html_content = '\n'.join(html_lines)

                return {"converted_content": html_content}

        except Exception as e:
            raise ValueError(f"Failed to convert content from {format}: {str(e)}")

    @is_tool()
    def search_documents_by_keyword(self, keyword: str, search_field: Literal["title", "content", "both"] = "both") -> Dict[str, Any]:
        """
        Search documents by keyword in title or content using fuzzy matching.

        Args:
            keyword: The keyword to search for
            search_field: The field to search in - "title", "content", or "both"

        Returns:
            Dictionary containing:
            - documents: List of matching documents with their details
            - total_count: Total number of matching documents

        Raises:
            ValueError: If keyword is empty or search_field is invalid
        """
        from thefuzz import fuzz

        # Validate keyword parameter
        if not keyword or not keyword.strip():
            raise ValueError("Keyword cannot be empty")

        # Validate search_field parameter (safety protection for enum)
        valid_fields = ["title", "content", "both"]
        if search_field not in valid_fields:
            raise ValueError(f"Invalid search_field '{search_field}'. Must be one of {valid_fields}")

        # Access the database
        db = self.db
        document_table = getattr(db, "document", None)

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

        # Normalize keyword for better matching
        keyword_lower = keyword.lower().strip()

        # Define similarity threshold for fuzzy matching (adjustable based on requirements)
        similarity_threshold = 60

        matching_documents = []

        # Iterate through all documents in the table
        for doc_id, document in document_table.items():
            is_match = False
            match_score = 0

            # Search in title field
            if search_field in ["title", "both"]:
                title_lower = document.title.lower() if document.title else ""
                # Use partial_ratio for substring matching in title
                title_score = fuzz.partial_ratio(keyword_lower, title_lower)
                if title_score >= similarity_threshold:
                    is_match = True
                    match_score = max(match_score, title_score)

            # Search in content field
            if search_field in ["content", "both"]:
                content_lower = document.content.lower() if document.content else ""
                # Use partial_ratio for substring matching in content
                content_score = fuzz.partial_ratio(keyword_lower, content_lower)
                if content_score >= similarity_threshold:
                    is_match = True
                    match_score = max(match_score, content_score)

            # If document matches, add it to results
            if is_match:
                # Convert document to dictionary format for output
                doc_dict = {
                    "document_id": document.document_id,
                    "title": document.title,
                    "content": document.content,
                    "author": document.author,
                    "category": document.category,
                    "folder_id": document.folder_id,
                    "is_archived": document.is_archived,
                    "file_size": document.file_size,
                    "created_at": document.created_at.strftime("%Y-%m-%d %H:%M:%S"),
                    "updated_at": document.updated_at.strftime("%Y-%m-%d %H:%M:%S"),
                    "archived_at": document.archived_at.strftime("%Y-%m-%d %H:%M:%S") if document.archived_at else None,
                    "match_score": match_score  # Include match score for reference
                }
                matching_documents.append((doc_dict, match_score))

        # Sort documents by match score in descending order (highest match first)
        matching_documents.sort(key=lambda x: x[1], reverse=True)

        # Extract just the document dictionaries (without scores) for final output
        result_documents = [doc for doc, score in matching_documents]

        # Return results with total count
        return {
            "documents": result_documents,
            "total_count": len(result_documents)
        }

    @is_tool()
    def add_comment_to_document(self, document_id: str, user_id: str, comment_text: str):
        """
        Add a comment to a document

        Args:
            document_id: The unique identifier of the document
            user_id: The identifier of the user adding the comment
            comment_text: The text content of the comment

        Returns:
            dict: Contains comment_id and created_at timestamp

        Raises:
            KeyError: If document_id doesn't exist or user doesn't have permission
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Access the database
        db = self.db

        # Validate input parameters - check for empty strings
        if not document_id or not document_id.strip():
            raise ValueError("document_id cannot be empty")
        if not user_id or not user_id.strip():
            raise ValueError("user_id cannot be empty")
        if not comment_text or not comment_text.strip():
            raise ValueError("comment_text cannot be empty")

        # Get the document_comment table from database
        document_comment_table = getattr(db, "document_comment", None)

        # Initialize the table if it doesn't exist
        if document_comment_table is None:
            document_comment_table = {}
            setattr(db, "document_comment", document_comment_table)

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

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

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

        # Create new DocumentComment instance
        new_comment = DocumentComment(
            comment_id=comment_id,
            document_id=document_id,
            user_id=user_id,
            comment_text=comment_text,
            created_at=created_at
        )

        # Add the new comment to the table
        document_comment_table[comment_id] = new_comment

        # Update the database with the modified table
        setattr(db, "document_comment", document_comment_table)

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

        # Return the comment_id and created_at as specified in the schema
        return {
            "comment_id": comment_id,
            "created_at": created_at_str
        }

    @is_tool()
    def create_folder(self, folder_name: str, parent_folder_id: str = None, description: str = None):
        """
        Create a new folder in the document library.

        Args:
            folder_name: The name of the folder to create
            parent_folder_id: The identifier of the parent folder, None for root level
            description: Description of the folder purpose

        Returns:
            dict: Contains folder_id and created_at timestamp

        Raises:
            ValueError: If folder_name is empty or if parent_folder_id is provided but doesn't exist
        """
        from datetime import datetime
        import secrets
        import hashlib

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

        # Get database instance
        db = self.db

        # Get or initialize folder table
        folder_table = getattr(db, 'folder', None)
        if folder_table is None:
            folder_table = {}
            setattr(db, 'folder', folder_table)

        # Validate parent_folder_id exists if provided
        if parent_folder_id is not None:
            if parent_folder_id not in folder_table:
                raise ValueError(f"Parent folder with id '{parent_folder_id}' does not exist")

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

        # Ensure the generated folder_id is unique (collision check)
        while folder_id in folder_table:
            folder_id = "folder_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Create timestamp for folder creation
        created_at = datetime.now()

        # Create new Folder instance
        new_folder = Folder(
            folder_id=folder_id,
            folder_name=folder_name.strip(),
            parent_folder_id=parent_folder_id,
            description=description.strip() if description else None,
            created_at=created_at
        )

        # Add the new folder to the database
        folder_table[folder_id] = new_folder
        setattr(db, 'folder', folder_table)

        # Return the result with formatted timestamp
        return {
            "folder_id": folder_id,
            "created_at": created_at.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def duplicate_document(self, document_id: str, title: str = None):
        """
        Create a duplicate copy of an existing document

        Args:
            document_id: The unique identifier of the document to duplicate
            title: The title for the duplicated document (optional, defaults to "Copy of {original_title}")

        Returns:
            dict: Contains document_id and created_at of the duplicated document

        Raises:
            KeyError: If the document with the given document_id does not exist
        """
        import secrets
        import hashlib
        from datetime import datetime

        # Access the database
        db = self.db

        # Get the document table
        document_table = getattr(db, "document", None)
        if document_table is None:
            raise KeyError(f"Document table not found in database")

        # Check if the original document exists
        if document_id not in document_table:
            raise KeyError(f"Document with id '{document_id}' does not exist")

        # Get the original document
        original_doc = document_table[document_id]

        # Generate a new unique document ID for the duplicate
        # Ensure uniqueness by checking against existing IDs
        while True:
            new_document_id = "doc_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]
            if new_document_id not in document_table:
                break

        # Determine the title for the duplicate
        # If title is not provided, use "Copy of {original_title}"
        if title is None:
            duplicate_title = f"Copy of {original_doc.title}"
        else:
            duplicate_title = title

        # Create timestamp for the duplicate document
        created_timestamp = datetime.now()

        # Create the duplicate document with all properties from the original
        # except for the document_id, title, created_at, and updated_at
        duplicate_doc = Document(
            document_id=new_document_id,
            title=duplicate_title,
            content=original_doc.content,
            author=original_doc.author,
            category=original_doc.category,
            folder_id=original_doc.folder_id,
            is_archived=original_doc.is_archived,
            file_size=original_doc.file_size,
            created_at=created_timestamp,
            updated_at=created_timestamp,
            archived_at=original_doc.archived_at
        )

        # Add the duplicate document to the database
        document_table[new_document_id] = duplicate_doc
        setattr(db, "document", document_table)

        # Handle document tags - duplicate all tags associated with the original document
        document_tag_table = getattr(db, "document_tag", None)
        if document_tag_table is not None:
            # Find all tags associated with the original document
            original_tags = [tag_entry for tag_entry in document_tag_table.values() 
                            if tag_entry.document_id == document_id]

            # Create duplicate tag entries for the new document
            for original_tag in original_tags:
                # Generate a unique key for the tag entry
                # Use composite key to allow multiple tags per document
                tag_key = f"{new_document_id}_{original_tag.tag}"

                # Ensure the tag key is unique
                if tag_key not in document_tag_table:
                    new_tag_entry = DocumentTag(
                        document_id=new_document_id,
                        tag=original_tag.tag
                    )
                    document_tag_table[tag_key] = new_tag_entry

            # Update the document_tag table in the database
            setattr(db, "document_tag", document_tag_table)

        # Return the required information about the duplicate
        return {
            "document_id": new_document_id,
            "created_at": created_timestamp.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def check_document_lock_status(self, document_id: str):
        """
        Check if a document is currently locked and return its lock status.

        This method retrieves the lock status of a document from the document_lock table.
        If the document is locked, it returns information about who locked it and when.
        If the document is not locked, it returns is_locked=False with None values for other fields.

        Args:
            document_id: The unique identifier of the document to check

        Returns:
            dict: A dictionary containing:
                - is_locked (bool): Whether the document is currently locked
                - locked_by (str or None): User ID who locked the document (None if not locked)
                - locked_at (str or None): Timestamp when locked in "yyyy-mm-dd HH:MM:SS" format (None if not locked)

        Raises:
            KeyError: If the document_id does not exist in the system
        """
        from datetime import datetime

        # Access the database instance
        db = self.db

        # Get the document_lock table from the database
        document_lock_table = getattr(db, "document_lock", None)

        # If the document_lock table doesn't exist or is empty, the document is not locked
        if document_lock_table is None or len(document_lock_table) == 0:
            # Since we can't verify if the document exists without a document table,
            # we assume unlocked documents are valid
            return {
                "is_locked": False,
                "locked_by": None,
                "locked_at": None
            }

        # Check if the document has a lock entry
        lock_entry = document_lock_table.get(document_id, None)

        # If no lock entry exists, the document is not locked
        if lock_entry is None:
            return {
                "is_locked": False,
                "locked_by": None,
                "locked_at": None
            }

        # Document is locked, extract lock information
        # Access lock entry attributes directly
        locked_by = lock_entry.user_id
        locked_at_datetime = lock_entry.locked_at

        # Convert datetime object to string in "yyyy-mm-dd HH:MM:SS" format
        locked_at_str = locked_at_datetime.strftime("%Y-%m-%d %H:%M:%S")

        # Return the lock status with all required information
        return {
            "is_locked": True,
            "locked_by": locked_by,
            "locked_at": locked_at_str
        }

    @is_tool()
    def add_tags_to_document(self, document_id: str, tags: List[str]) -> dict:
        # Get database instance
        db = self.db

        # Retrieve the document_tag table from database
        document_tag_table = getattr(db, "document_tag", None)

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

        # Validate input parameters
        if not document_id:
            raise ValueError("document_id cannot be empty")

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

        # Remove duplicate tags and filter empty strings
        tags = [tag.strip() for tag in tags if tag and isinstance(tag, str) and tag.strip()]

        if not tags:
            raise ValueError("tags list contains no valid tag strings")

        # Check if document exists by looking for any existing tags for this document
        existing_tags = []
        document_exists = False

        for tag_id, tag_entry in document_tag_table.items():
            if tag_entry.document_id == document_id:
                document_exists = True
                existing_tags.append(tag_entry.tag)

        # If document doesn't exist, raise KeyError
        if not document_exists:
            raise KeyError(f"Document with id '{document_id}' does not exist")

        # Add new tags to the document
        # Only add tags that don't already exist for this document
        import secrets
        import hashlib

        for tag in tags:
            # Check if this tag already exists for the document
            tag_exists = any(
                tag_entry.document_id == document_id and tag_entry.tag == tag
                for tag_entry in document_tag_table.values()
            )

            # Only add if tag doesn't already exist
            if not tag_exists:
                # Generate unique tag entry id
                tag_entry_id = "tag_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

                # Create new DocumentTag entry
                new_tag_entry = DocumentTag(
                    document_id=document_id,
                    tag=tag
                )

                # Add to document_tag table
                document_tag_table[tag_entry_id] = new_tag_entry
                existing_tags.append(tag)

        # Update the database
        setattr(db, "document_tag", document_tag_table)

        # Return success response with complete list of tags
        return {
            "success": True,
            "tags": existing_tags
        }

    @is_tool()
    def get_document_comments(self, document_id: str):
        # Get the database instance
        db = self.db

        # Retrieve the document_comment table from the database
        # If the table doesn't exist, return empty results
        document_comment_table = getattr(db, 'document_comment', None)

        if document_comment_table is None:
            # If the table doesn't exist, raise KeyError
            raise KeyError(f"document_comment table not found in database")

        # Check if document_id parameter is valid (non-empty string)
        if not document_id or not isinstance(document_id, str):
            raise ValueError("document_id must be a non-empty string")

        # Initialize the list to store matching comments
        matching_comments = []

        # Iterate through all comments in the document_comment table
        # The table is a Dict[str, DocumentComment] where key is comment_id
        for comment_id, comment_obj in document_comment_table.items():
            # Check if the comment belongs to the specified document
            if comment_obj.document_id == document_id:
                # Convert the comment object to a dictionary format
                # Format the created_at datetime to "yyyy-mm-dd HH:MM:SS" string
                comment_dict = {
                    "comment_id": comment_obj.comment_id,
                    "user_id": comment_obj.user_id,
                    "comment_text": comment_obj.comment_text,
                    "created_at": comment_obj.created_at.strftime("%Y-%m-%d %H:%M:%S")
                }
                matching_comments.append(comment_dict)

        # Sort comments by created_at timestamp (oldest first)
        # This provides a consistent ordering for the returned comments
        matching_comments.sort(key=lambda x: x["created_at"])

        # Calculate the total count of comments for this document
        total_count = len(matching_comments)

        # Return the results in the specified format
        return {
            "comments": matching_comments,
            "total_count": total_count
        }

    @is_tool()
    def remove_document_attachment(self, attachment_id: str):
        """
        Remove an attachment from a document

        This method removes an attachment from the document_attachment table by its unique identifier.
        The removal timestamp is recorded in yyyy-mm-dd HH:MM:SS format.

        Args:
            attachment_id: The unique identifier of the attachment to be removed

        Returns:
            dict: A dictionary containing:
                - success (bool): Indicates whether the removal was successful
                - removed_at (str): The timestamp when the attachment was removed in yyyy-mm-dd HH:MM:SS format

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

        # Access the database instance
        db = self.db

        # Get the document_attachment table from the database
        document_attachment_table = getattr(db, "document_attachment", None)

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

        # Check if the attachment exists in the table
        if attachment_id not in document_attachment_table:
            raise KeyError(f"Attachment with id '{attachment_id}' does not exist")

        # Remove the attachment from the table
        # Create a new dictionary without the specified attachment_id
        updated_table = {k: v for k, v in document_attachment_table.items() if k != attachment_id}

        # Update the database with the modified table
        setattr(db, "document_attachment", updated_table)

        # Record the removal timestamp in yyyy-mm-dd HH:MM:SS format
        removed_at = datetime.now().strftime("%Y-%m-%d %H:%M:%S")

        # Return success response with removal timestamp
        return {
            "success": True,
            "removed_at": removed_at
        }

    @is_tool()
    def archive_document(self, document_id: str):
        """
        Archive a document to move it out of active view.

        This method marks a document as archived by setting its is_archived flag to True
        and recording the archive timestamp. The document remains in the database but
        is moved out of active view.

        Args:
            document_id: The unique identifier of the document to archive

        Returns:
            dict: A dictionary containing:
                - success (bool): True if archiving was successful
                - archived_at (str): Timestamp when document was archived in "yyyy-mm-dd HH:MM:SS" format

        Raises:
            KeyError: If the document with the given document_id does not exist
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Get the document table from database
        document_table = getattr(db, "document", None)

        # Check if document table exists
        if document_table is None:
            raise KeyError(f"Document table does not exist in database")

        # Check if the document exists in the table
        if document_id not in document_table:
            raise KeyError(f"Document with document_id '{document_id}' does not exist")

        # Retrieve the document object
        document = document_table[document_id]

        # Get current timestamp for archiving
        archived_timestamp = datetime.now()

        # Update document attributes to mark it as archived
        document.is_archived = True
        document.archived_at = archived_timestamp
        document.updated_at = archived_timestamp

        # Write the updated document back to the database
        document_table[document_id] = document
        setattr(db, "document", document_table)

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

        # Return success response with archived timestamp
        return {
            "success": True,
            "archived_at": archived_at_str
        }

    @is_tool()
    def get_folder_contents(self, folder_id: str):
        # Get the database instance
        db = self.db

        # Retrieve the folder table from the database
        folder_table = getattr(db, "folder", None)
        if folder_table is None:
            raise KeyError(f"Folder table does not exist in the database")

        # Check if the specified folder exists
        if folder_id not in folder_table:
            raise KeyError(f"Folder with folder_id '{folder_id}' does not exist")

        # Retrieve the document table from the database
        document_table = getattr(db, "document", None)
        if document_table is None:
            raise KeyError(f"Document table does not exist in the database")

        # Initialize lists to store documents and subfolders
        documents = []
        subfolders = []

        # Iterate through all documents to find those belonging to the specified folder
        for doc_id, doc in document_table.items():
            # Check if the document belongs to the specified folder
            if doc.folder_id == folder_id:
                # Add document information to the list
                documents.append({
                    "document_id": doc.document_id,
                    "title": doc.title
                })

        # Iterate through all folders to find subfolders of the specified folder
        for fid, folder in folder_table.items():
            # Check if this folder is a subfolder of the specified folder
            if folder.parent_folder_id == folder_id:
                # Add subfolder information to the list
                subfolders.append({
                    "folder_id": folder.folder_id,
                    "folder_name": folder.folder_name
                })

        # Return the folder contents with documents and subfolders
        return {
            "documents": documents,
            "subfolders": subfolders
        }

    @is_tool()
    def get_document_attachments(self, document_id: str):
        """
        Retrieve all attachments for a document

        This method fetches all attachment records associated with a given document ID
        from the document_attachment table and returns them in a structured format.

        Args:
            document_id: The unique identifier of the document

        Returns:
            dict: A dictionary containing:
                - attachments: List of attachment objects with their details
                - total_count: Total number of attachments found

        Raises:
            KeyError: If the document_id does not exist or no attachments are found
        """
        # Access the database instance
        db = self.db

        # Get the document_attachment table from database
        document_attachment_table = getattr(db, "document_attachment", None)

        # Check if the table exists and has data
        if document_attachment_table is None:
            raise KeyError(f"Document attachment table not found in database")

        # Filter attachments by document_id
        # Iterate through all attachments and collect those matching the document_id
        matching_attachments = []
        for attachment_id, attachment in document_attachment_table.items():
            if attachment.document_id == document_id:
                # Build attachment object with required fields
                attachment_obj = {
                    "attachment_id": attachment.attachment_id,
                    "file_name": attachment.file_name,
                    "file_type": attachment.file_type if attachment.file_type else "unknown",
                    "file_path": attachment.file_path,
                    "created_at": attachment.created_at.strftime("%Y-%m-%d %H:%M:%S")
                }
                matching_attachments.append(attachment_obj)

        # If no attachments found for the document_id, raise KeyError
        if len(matching_attachments) == 0:
            raise KeyError(f"No attachments found for document_id: {document_id}")

        # Sort attachments by created_at timestamp (newest first) for consistent ordering
        matching_attachments.sort(key=lambda x: x["created_at"], reverse=True)

        # Prepare the return dictionary with attachments list and total count
        result = {
            "attachments": matching_attachments,
            "total_count": len(matching_attachments)
        }

        return result

    @is_tool()
    def get_document_by_id(self, document_id: str):
        """
        Retrieve a document by its unique identifier.

        This method fetches document details from the database including title, content,
        author, tags, and timestamps. It also retrieves associated tags from the document_tag table.

        Args:
            document_id: The unique identifier of the document to retrieve

        Returns:
            dict: A dictionary containing document details with the following keys:
                - document_id: The unique identifier of the document
                - title: The title of the document
                - content: The main content of the document
                - author: The author name of the document
                - tags: List of tags associated with the document
                - created_at: Creation timestamp in "yyyy-mm-dd HH:MM:SS" format
                - updated_at: Last update timestamp in "yyyy-mm-dd HH:MM:SS" format

        Raises:
            KeyError: If the document with the given document_id does not exist
        """
        from datetime import datetime

        # Access the database instance
        db = self.db

        # Retrieve the document table
        document_table = getattr(db, "document", None)
        if document_table is None:
            raise KeyError(f"Document table not found in database")

        # Check if document exists in the table
        if document_id not in document_table:
            raise KeyError(f"Document with id '{document_id}' not found")

        # Retrieve the document instance
        document = document_table[document_id]

        # Retrieve associated tags from document_tag table
        document_tag_table = getattr(db, "document_tag", None)
        tags = []

        if document_tag_table is not None:
            # Iterate through all document_tag entries to find matching tags
            for tag_entry in document_tag_table.values():
                # Check if this tag entry belongs to the current document
                if tag_entry.document_id == document_id:
                    tags.append(tag_entry.tag)

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

        # Construct and return the result dictionary
        result = {
            "document_id": document.document_id,
            "title": document.title,
            "content": document.content,
            "author": document.author,
            "tags": tags,
            "created_at": created_at_str,
            "updated_at": updated_at_str
        }

        return result

    @is_tool()
    def move_document_to_folder(self, document_id: str, target_folder_id: str):
        from datetime import datetime

        # Access the database
        db = self.db

        # Retrieve the document table from database
        document_table = getattr(db, 'document', None)
        if document_table is None:
            raise KeyError("Document table does not exist in the database")

        # Check if the document exists
        if document_id not in document_table:
            raise KeyError(f"Document with ID '{document_id}' does not exist")

        # Retrieve the document object
        document = document_table[document_id]

        # Update the document's folder_id to the target folder
        # Note: We don't validate target_folder_id existence as per requirements
        # (only validate parameters defined in tool schema, not external references)
        document.folder_id = target_folder_id

        # Update the document's updated_at timestamp
        moved_at = datetime.now()
        document.updated_at = moved_at

        # Save the updated document back to the database
        document_table[document_id] = document
        setattr(db, 'document', document_table)

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

        # Return success result with timestamp
        return {
            "success": True,
            "moved_at": moved_at_str
        }

    @is_tool()
    def delete_comment(self, comment_id: str):
        """
        Delete a comment from a document

        This method removes a comment from the document_comment table based on the provided comment_id.

        Args:
            comment_id: The unique identifier of the comment to be deleted

        Returns:
            dict: A dictionary containing:
                - success (bool): Indicates whether the deletion was successful
                - deleted_at (str): The timestamp when the comment was deleted in yyyy-mm-dd HH:MM:SS format

        Raises:
            KeyError: If the comment with the given comment_id does not exist
        """
        from datetime import datetime

        # Get the database instance
        db = self.db

        # Retrieve the document_comment table from the database
        document_comment_table = getattr(db, "document_comment", None)

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

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

        # Delete the comment from the table
        # Create a new dictionary without the deleted comment
        updated_table = {cid: comment for cid, comment in document_comment_table.items() if cid != comment_id}

        # Update the database with the modified table
        setattr(db, "document_comment", updated_table)

        # Record the deletion timestamp
        deleted_at = datetime.now().strftime("%Y-%m-%d %H:%M:%S")

        # Return success response with deletion timestamp
        return {
            "success": True,
            "deleted_at": deleted_at
        }

    @is_tool()
    def restore_archived_document(self, document_id: str):
        """
        Restore an archived document to active status.

        This method retrieves a document by its ID, verifies it is currently archived,
        then restores it to active status by setting is_archived to False and clearing
        the archived_at timestamp. The updated_at timestamp is also refreshed.

        Args:
            document_id: The unique identifier of the document to restore

        Returns:
            dict: A dictionary containing:
                - success (bool): True if restoration was successful
                - restored_at (str): Timestamp when document was restored in "yyyy-mm-dd HH:MM:SS" format

        Raises:
            KeyError: If the document with the given document_id does not exist
            ValueError: If the document is not currently archived
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Retrieve the document table
        document_table = getattr(db, "document", None)

        # Check if document table exists
        if document_table is None:
            raise KeyError(f"Document table does not exist in database")

        # Check if the document exists
        if document_id not in document_table:
            raise KeyError(f"Document with ID '{document_id}' does not exist")

        # Retrieve the document
        document = document_table[document_id]

        # Verify the document is currently archived
        if not document.is_archived:
            raise ValueError(f"Document with ID '{document_id}' is not archived and cannot be restored")

        # Get the current timestamp for restoration
        restored_timestamp = datetime.now()

        # Update the document status to restore it
        document.is_archived = False
        document.archived_at = None
        document.updated_at = restored_timestamp

        # Write the updated document back to the database
        document_table[document_id] = document
        setattr(db, "document", document_table)

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

        # Return success response
        return {
            "success": True,
            "restored_at": restored_at_str
        }

    @is_tool()
    def calculate_document_word_count(self, content: str):
        """
        Calculate the total word count of a document's content.

        This method counts words in the provided text content by splitting on whitespace.
        Empty strings or strings with only whitespace will return a count of 0.

        Args:
            content: The text content to count words from

        Returns:
            dict: A dictionary containing the word count
                - word_count (int): The total number of words in the content

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

        if not isinstance(content, str):
            raise ValueError(f"Content must be a string, got {type(content).__name__}")

        # Strip leading/trailing whitespace and check if content is empty
        trimmed_content = content.strip()

        # If content is empty or only whitespace, return 0
        if not trimmed_content:
            word_count = 0
        else:
            # Split content by whitespace to get individual words
            # This handles multiple spaces, tabs, and newlines automatically
            words = trimmed_content.split()

            # Count the number of words
            word_count = len(words)

        # Return the result in the specified format
        return {
            "word_count": word_count
        }

    @is_tool()
    def calculate_document_reading_time(self, word_count: int, reading_speed: int = 200):
        """
        Estimate the reading time for a document based on word count.

        This method calculates the estimated reading time by dividing the total word count
        by the average reading speed (in words per minute). The default reading speed is
        set to 200 words per minute, which is a common average for adult readers.

        Args:
            word_count: The total number of words in the document (must be non-negative)
            reading_speed: Average reading speed in words per minute (must be positive, default: 200)

        Returns:
            A dictionary containing:
                - reading_time_minutes: Estimated reading time in minutes (float)

        Raises:
            ValueError: If word_count is negative or reading_speed is not positive
        """
        # Validate word_count parameter
        # Word count must be a non-negative integer
        if word_count < 0:
            raise ValueError("word_count must be a non-negative integer")

        # Validate reading_speed parameter
        # Reading speed must be a positive integer (greater than 0)
        if reading_speed <= 0:
            raise ValueError("reading_speed must be a positive integer greater than 0")

        # Calculate reading time in minutes
        # Formula: reading_time = total_words / words_per_minute
        # This gives us the estimated time in minutes as a floating-point number
        reading_time_minutes = word_count / reading_speed

        # Return the result as a dictionary matching the schema
        return {
            "reading_time_minutes": reading_time_minutes
        }

    @is_tool()
    def export_document(self, document_id: str, format: Literal["pdf", "docx", "html", "markdown", "txt"]) -> dict:
        """
        Export a document to a specific format.

        This method retrieves a document from the database and generates an export URL
        based on the requested format. The export timestamp is recorded in yyyy-mm-dd HH:MM:SS format.

        Args:
            document_id: The unique identifier of the document to export
            format: The format to export the document to (must be one of: pdf, docx, html, markdown, txt)

        Returns:
            A dictionary containing:
                - export_url: The URL or path to the exported file
                - exported_at: The timestamp when the document was exported

        Raises:
            KeyError: If the document with the given document_id does not exist
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Retrieve the document table from database
        document_table = getattr(db, "document", None)

        # Check if document table exists
        if document_table is None:
            raise KeyError(f"Document table does not exist in the database")

        # Check if the document exists in the table
        if document_id not in document_table:
            raise KeyError(f"Document with ID '{document_id}' does not exist")

        # Retrieve the document
        document = document_table[document_id]

        # Validate format parameter (additional safety check beyond Literal type hint)
        valid_formats = ["pdf", "docx", "html", "markdown", "txt"]
        if format not in valid_formats:
            raise ValueError(f"Invalid format '{format}'. Must be one of: {', '.join(valid_formats)}")

        # Generate export URL based on document_id and format
        # Using a storage pattern similar to the example in the schema
        export_url = f"https://storage.example.com/exports/{document_id}.{format}"

        # Get current timestamp for export time in yyyy-mm-dd HH:MM:SS format
        exported_at = datetime.now().strftime("%Y-%m-%d %H:%M:%S")

        # Return the export information
        return {
            "export_url": export_url,
            "exported_at": exported_at
        }

    @is_tool()
    def delete_folder(self, folder_id: str, force: bool = False):
        # Get database instance
        db = self.db

        # Retrieve folder table from database
        folder_table = getattr(db, 'folder', None)
        if folder_table is None:
            raise ValueError("Folder table does not exist in database")

        # Check if folder exists
        if folder_id not in folder_table:
            raise ValueError(f"Folder with id '{folder_id}' does not exist")

        # Get the folder to be deleted
        target_folder = folder_table[folder_id]

        # Check if folder has subfolders (check if any folder has this folder as parent)
        has_subfolders = any(
            folder.parent_folder_id == folder_id 
            for folder in folder_table.values()
        )

        # If folder is not empty and force delete is not enabled, raise error
        if has_subfolders and not force:
            raise ValueError(
                f"Folder '{folder_id}' is not empty. "
                "Set force=True to delete non-empty folder"
            )

        # If force delete is enabled and folder has subfolders, delete all subfolders recursively
        if has_subfolders and force:
            # Find all subfolders
            subfolders_to_delete = [
                fid for fid, folder in folder_table.items()
                if folder.parent_folder_id == folder_id
            ]

            # Recursively delete each subfolder
            for subfolder_id in subfolders_to_delete:
                self.delete_folder(subfolder_id, force=True)

        # Delete the folder from database
        updated_folder_table = {
            fid: folder for fid, folder in folder_table.items()
            if fid != folder_id
        }
        setattr(db, 'folder', updated_folder_table)

        # Import datetime for timestamp generation
        from datetime import datetime

        # Generate deletion timestamp in yyyy-mm-dd HH:MM:SS format
        deleted_at = datetime.now().strftime("%Y-%m-%d %H:%M:%S")

        # Return success response with deletion timestamp
        return {
            "success": True,
            "deleted_at": deleted_at
        }

    @is_tool()
    def create_document_template(self, template_name: str, content_structure: str, category: str = None):
        """
        Create a reusable document template in the document library system.

        This method creates a new document template with the specified name, content structure,
        and optional category. It generates a unique template ID and stores the template in the database.

        Args:
            template_name: The name of the template (e.g., "Project Requirements Template")
            content_structure: The structure/outline of the template content (e.g., "1. Introduction\n2. Requirements\n3. Conclusion")
            category: Optional category of the template (e.g., "technical")

        Returns:
            dict: A dictionary containing:
                - template_id: The unique identifier of the created template
                - created_at: The timestamp when the template was created in "yyyy-mm-dd HH:MM:SS" format

        Raises:
            ValueError: If template_name or content_structure is empty/invalid
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Import the DocumentTemplate class (already imported at file header)
        # from database_class import DocumentTemplate

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

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

        # Validate category if provided
        if category is not None and (not isinstance(category, str) or not category.strip()):
            raise ValueError("category must be a non-empty string if provided")

        # Access the database
        db = self.db

        # Generate a unique template ID
        prefix = "template_"
        template_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp
        created_at = datetime.now()

        # Create the new DocumentTemplate instance
        new_template = DocumentTemplate(
            template_id=template_id,
            template_name=template_name.strip(),
            content_structure=content_structure.strip(),
            category=category.strip() if category else None,
            created_at=created_at
        )

        # Get existing document_template data from database
        document_template_data = getattr(db, "document_template", None)

        # Initialize the document_template dictionary if it doesn't exist
        if document_template_data is None:
            document_template_data = {}

        # Add the new template to the database
        document_template_data[template_id] = new_template

        # Save the updated data back to the database
        setattr(db, "document_template", document_template_data)

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

    @is_tool()
    def get_archived_documents(self):
        """
        Retrieve all archived documents from the database.

        This method filters documents where is_archived is True and returns
        them along with the total count of archived documents.

        Returns:
            dict: A dictionary containing:
                - documents: List of archived document dictionaries
                - total_count: Total number of archived documents

        Raises:
            Exception: If there's an error accessing the database
        """
        try:
            # Access the database instance
            db = self.db

            # Get the document table from the database
            document_table = getattr(db, "document", None)

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

            # Filter archived documents from the document table
            archived_documents = []
            for doc_id, doc in document_table.items():
                # Check if the document is archived
                if doc.is_archived:
                    # Convert document to dictionary format
                    doc_dict = {
                        "document_id": doc.document_id,
                        "title": doc.title,
                        "content": doc.content,
                        "author": doc.author,
                        "category": doc.category,
                        "folder_id": doc.folder_id,
                        "is_archived": doc.is_archived,
                        "file_size": doc.file_size,
                        "created_at": doc.created_at.strftime("%Y-%m-%d %H:%M:%S"),
                        "updated_at": doc.updated_at.strftime("%Y-%m-%d %H:%M:%S"),
                        # Format archived_at timestamp if it exists
                        "archived_at": doc.archived_at.strftime("%Y-%m-%d %H:%M:%S") if doc.archived_at else None
                    }
                    archived_documents.append(doc_dict)

            # Calculate total count of archived documents
            total_count = len(archived_documents)

            # Return the results in the specified format
            return {
                "documents": archived_documents,
                "total_count": total_count
            }

        except Exception as e:
            # Re-raise any exceptions that occur during execution
            raise Exception(f"Error retrieving archived documents: {str(e)}")

    @is_tool()
    def import_document(self, file_path: str, title: str, author: str, format: Literal["pdf", "docx", "html", "markdown", "txt"]):
        """
        Import a document from an external file and create it in the document library.

        This method:
        1. Validates the file format against allowed formats
        2. Simulates file content retrieval from the provided path/URL
        3. Generates a unique document ID
        4. Creates a new Document instance with the imported data
        5. Stores the document in the database
        6. Returns the document ID and import timestamp

        Args:
            file_path: The path or URL to the file to import
            title: The title for the imported document
            author: The author name for the imported document
            format: The format of the file being imported (must be one of: pdf, docx, html, markdown, txt)

        Returns:
            dict: Contains document_id and imported_at timestamp in yyyy-mm-dd HH:MM:SS format

        Raises:
            ValueError: If format is not in allowed formats, or if file_path/title/author is empty
        """
        import secrets
        import hashlib
        from datetime import datetime

        # Validate required parameters are not empty
        if not file_path or not file_path.strip():
            raise ValueError("file_path cannot be empty")
        if not title or not title.strip():
            raise ValueError("title cannot be empty")
        if not author or not author.strip():
            raise ValueError("author cannot be empty")

        # Validate format parameter against enum values
        allowed_formats = ["pdf", "docx", "html", "markdown", "txt"]
        if format not in allowed_formats:
            raise ValueError(f"Invalid format '{format}'. Must be one of: {', '.join(allowed_formats)}")

        # Access the database
        db = self.db

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

        # Generate unique document ID with prefix
        prefix = "doc_"
        document_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

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

        # Get current timestamp for creation and import
        current_time = datetime.now()

        # Simulate content extraction based on file format
        # In a real implementation, this would parse the actual file content
        # For now, we create a placeholder content that indicates the source
        simulated_content = f"Content imported from {file_path} (format: {format})"

        # Estimate file size based on content length (simulated)
        # In real implementation, this would be the actual file size
        estimated_file_size = len(simulated_content.encode('utf-8'))

        # Create new Document instance
        new_document = Document(
            document_id=document_id,
            title=title,
            content=simulated_content,
            author=author,
            category=None,  # Category not specified during import
            folder_id=None,  # Folder not specified during import
            is_archived=False,  # Newly imported documents are not archived
            file_size=estimated_file_size,
            created_at=current_time,
            updated_at=current_time,
            archived_at=None  # Not archived
        )

        # Store the document in the database
        document_table[document_id] = new_document
        setattr(db, "document", document_table)

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

        # Return the result
        return {
            "document_id": document_id,
            "imported_at": imported_at_str
        }

    @is_tool()
    def create_document(self, title: str, content: str, author: str, tags: List[str] = None, category: str = None) -> dict:
        """
        Create a new document with metadata including title, content, author, and tags.

        Args:
            title: The title of the document
            content: The main content of the document
            author: The author name of the document
            tags: Optional list of tags for categorizing the document
            category: Optional category of the document

        Returns:
            dict: Contains document_id and created_at timestamp

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

        # Validate required parameters
        if not title or not isinstance(title, str) or not title.strip():
            raise ValueError("Title 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")

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

        # Validate optional parameters
        if tags is not None:
            if not isinstance(tags, list):
                raise ValueError("Tags must be a list of strings")
            for tag in tags:
                if not isinstance(tag, str) or not tag.strip():
                    raise ValueError("Each tag must be a non-empty string")

        if category is not None and (not isinstance(category, str) or not category.strip()):
            raise ValueError("Category must be a non-empty string if provided")

        # Access database
        db = self.db

        # Generate unique document ID
        document_id = "doc_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Create timestamp
        created_at = datetime.now()

        # Get existing document table or initialize empty dict
        document_table = getattr(db, "document", None)
        if document_table is None:
            document_table = {}

        # Create new document object
        new_document = Document(
            document_id=document_id,
            title=title.strip(),
            content=content.strip(),
            author=author.strip(),
            category=category.strip() if category else None,
            folder_id=None,  # Not provided in parameters
            is_archived=False,  # Default value for new document
            file_size=len(content.encode('utf-8')),  # Calculate file size in bytes
            created_at=created_at,
            updated_at=created_at  # Initially same as created_at
        )

        # Add document to table
        document_table[document_id] = new_document
        setattr(db, "document", document_table)

        # Process tags if provided
        if tags:
            # Get existing document_tag table or initialize empty dict
            document_tag_table = getattr(db, "document_tag", None)
            if document_tag_table is None:
                document_tag_table = {}

            # Create tag entries for each tag
            for tag in tags:
                tag_stripped = tag.strip()
                if tag_stripped:  # Only add non-empty tags
                    # Generate unique key for document_tag entry
                    tag_key = f"{document_id}_{tag_stripped}"

                    # Create document tag object
                    new_tag = DocumentTag(
                        document_id=document_id,
                        tag=tag_stripped
                    )

                    # Add tag to table
                    document_tag_table[tag_key] = new_tag

            # Update document_tag table in database
            setattr(db, "document_tag", document_tag_table)

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