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

class OnlineLearningTools(ToolKitBase):
    """All tools for online_learning."""
    
    db: OnlineLearningDB
    
    def __init__(self, db: OnlineLearningDB):
        """Initialize tools with database."""
        super().__init__(db)
    
    @is_tool()
    def calculate_assessment_statistics(self, scores: list, passing_score: int):
        """
        Calculate statistical metrics for an assessment including average score, pass rate, and score distribution.

        This method performs statistical analysis on a list of assessment scores to provide insights into
        overall performance, including average score, pass rate, and the range of scores achieved.

        Args:
            scores: List of all scores from graded attempts (list of integers)
            passing_score: Minimum score required to pass (integer)

        Returns:
            dict: Dictionary containing:
                - average_score: Average score across all attempts (float)
                - pass_rate: Percentage of attempts that passed (float)
                - highest_score: Highest score achieved (int)
                - lowest_score: Lowest score achieved (int)

        Raises:
            ValueError: If scores list is empty, contains invalid values, or passing_score is negative
        """
        # Validate input parameters
        if not scores:
            raise ValueError("Scores list cannot be empty - at least one graded attempt is required")

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

        if passing_score < 0:
            raise ValueError("Passing score cannot be negative")

        # Validate all scores are integers and non-negative
        for score in scores:
            if not isinstance(score, int):
                raise ValueError(f"All scores must be integers, found: {type(score).__name__}")
            if score < 0:
                raise ValueError(f"Scores cannot be negative, found: {score}")

        # Calculate average score
        # Sum all scores and divide by the number of scores
        total_score = sum(scores)
        average_score = total_score / len(scores)

        # Calculate pass rate
        # Count how many scores meet or exceed the passing score
        passed_count = sum(1 for score in scores if score >= passing_score)
        pass_rate = (passed_count / len(scores)) * 100.0

        # Find highest and lowest scores
        highest_score = max(scores)
        lowest_score = min(scores)

        # Return statistical metrics
        return {
            "average_score": average_score,
            "pass_rate": pass_rate,
            "highest_score": highest_score,
            "lowest_score": lowest_score
        }

    @is_tool()
    def calculate_learner_points(
        self,
        completed_lessons: int,
        points_per_lesson: int,
        assessment_scores: list,
        discussion_posts: int,
        points_per_post: int
    ) -> dict:
        """
        Calculate total points earned by a learner from various activities.

        This method computes points from three sources:
        1. Lesson completion points
        2. Assessment score points
        3. Discussion participation points

        Args:
            completed_lessons: Number of lessons completed by the learner
            points_per_lesson: Points awarded for each completed lesson
            assessment_scores: List of integer scores from assessments
            discussion_posts: Number of discussion posts made by the learner
            points_per_post: Points awarded for each discussion post

        Returns:
            dict: Dictionary containing 'total_points' key with the calculated total

        Raises:
            ValueError: If any input parameter has invalid value (negative numbers, invalid types)
        """
        # Validate completed_lessons parameter
        if not isinstance(completed_lessons, int):
            raise ValueError("completed_lessons must be an integer")
        if completed_lessons < 0:
            raise ValueError("completed_lessons cannot be negative")

        # Validate points_per_lesson parameter
        if not isinstance(points_per_lesson, int):
            raise ValueError("points_per_lesson must be an integer")
        if points_per_lesson < 0:
            raise ValueError("points_per_lesson cannot be negative")

        # Validate assessment_scores parameter
        if not isinstance(assessment_scores, list):
            raise ValueError("assessment_scores must be a list")
        for score in assessment_scores:
            if not isinstance(score, int):
                raise ValueError("All assessment scores must be integers")
            if score < 0:
                raise ValueError("Assessment scores cannot be negative")

        # Validate discussion_posts parameter
        if not isinstance(discussion_posts, int):
            raise ValueError("discussion_posts must be an integer")
        if discussion_posts < 0:
            raise ValueError("discussion_posts cannot be negative")

        # Validate points_per_post parameter
        if not isinstance(points_per_post, int):
            raise ValueError("points_per_post must be an integer")
        if points_per_post < 0:
            raise ValueError("points_per_post cannot be negative")

        # Calculate points from completed lessons
        # Each completed lesson earns points_per_lesson points
        lesson_points = completed_lessons * points_per_lesson

        # Calculate points from assessment scores
        # Sum all assessment scores to get total assessment points
        assessment_points = sum(assessment_scores)

        # Calculate points from discussion participation
        # Each discussion post earns points_per_post points
        discussion_points = discussion_posts * points_per_post

        # Calculate total points by summing all three components
        total_points = lesson_points + assessment_points + discussion_points

        # Return result as dictionary with total_points key
        return {
            "total_points": total_points
        }

    @is_tool()
    def get_learner_enrollments(self, learner_id: str, status_filter: Literal["active", "completed", "dropped", "all"] = "all"):
        """
        Retrieve all course enrollments for a specific learner including enrollment status and progress.

        Args:
            learner_id: Unique identifier of the learner
            status_filter: Filter enrollments by status (active, completed, dropped, or all)

        Returns:
            Dictionary containing list of enrollments with their details

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

        # Get the enrollment table from database
        enrollment_table = getattr(db, "enrollment", None)

        # If enrollment table doesn't exist or is empty, return empty list
        if enrollment_table is None or len(enrollment_table) == 0:
            raise KeyError(f"Learner with ID '{learner_id}' not found in the system")

        # Collect all enrollments for the specified learner
        learner_enrollments = []
        learner_found = False

        # Iterate through all enrollments in the table
        for enrollment_id, enrollment in enrollment_table.items():
            # Check if this enrollment belongs to the specified learner
            if enrollment.learner_id == learner_id:
                learner_found = True

                # Apply status filter if not "all"
                # Note: status_filter uses "dropped" while schema uses "dropped" or "suspended"
                # We'll match "dropped" status and also consider "suspended" as a valid status
                if status_filter == "all" or enrollment.status == status_filter:
                    # Build enrollment information dictionary
                    enrollment_info = {
                        "enrollment_id": enrollment.enrollment_id,
                        "course_id": enrollment.course_id,
                        "enrollment_date": enrollment.enrollment_date.strftime("%Y-%m-%d %H:%M:%S"),
                        "status": enrollment.status,
                        "progress_percentage": enrollment.progress_percentage if enrollment.progress_percentage is not None else 0.0
                    }

                    # Add optional fields if they exist
                    if enrollment.final_grade is not None:
                        enrollment_info["final_grade"] = enrollment.final_grade

                    if enrollment.last_updated is not None:
                        enrollment_info["last_updated"] = enrollment.last_updated.strftime("%Y-%m-%d %H:%M:%S")

                    learner_enrollments.append(enrollment_info)

        # If no enrollments found for this learner, raise KeyError
        if not learner_found:
            raise KeyError(f"Learner with ID '{learner_id}' not found in the system")

        # Return the list of enrollments
        # Note: Even if status_filter results in empty list, we return empty list
        # as long as the learner exists (learner_found is True)
        return {
            "enrollments": learner_enrollments
        }

    @is_tool()
    def grade_assessment_attempt(self, attempt_id: str, score: int, graded_by: str, grading_timestamp: str, feedback: str = None):
        """
        Grade a submitted assessment attempt and record the score and feedback.

        This method updates an existing assessment attempt with grading information including
        score, feedback, grader identifier, and grading timestamp. The attempt must exist
        and be in pending status (not already graded).

        Args:
            attempt_id: Unique identifier of the assessment attempt to grade
            score: Score achieved by the learner (integer value)
            graded_by: Identifier of the instructor or system that performed the grading
            grading_timestamp: Timestamp when grading was completed in yyyy-mm-dd HH:MM:SS format
            feedback: Optional detailed feedback on the assessment performance

        Returns:
            dict: {"success": True} if grading was successful

        Raises:
            KeyError: If attempt_id does not exist in the database
            ValueError: If the attempt has already been graded or if timestamp format is invalid
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Retrieve the assessment_attempt table
        assessment_attempt_table = getattr(db, "assessment_attempt", None)

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

        # Check if the attempt exists
        if attempt_id not in assessment_attempt_table:
            raise KeyError(f"Assessment attempt with ID '{attempt_id}' does not exist")

        # Retrieve the specific attempt
        attempt = assessment_attempt_table[attempt_id]

        # Pre-condition: Check if attempt is in pending status (not already graded)
        # An attempt is considered pending if it hasn't been graded yet (score is None)
        if attempt.score is not None:
            raise ValueError(f"Assessment attempt '{attempt_id}' has already been graded")

        # Validate and parse the grading timestamp
        try:
            # Parse the timestamp string to datetime object
            grading_dt = datetime.strptime(grading_timestamp, "%Y-%m-%d %H:%M:%S")
        except ValueError as e:
            raise ValueError(f"Invalid timestamp format. Expected 'yyyy-mm-dd HH:MM:SS', got '{grading_timestamp}'") from e

        # Validate score is a non-negative integer
        if not isinstance(score, int):
            raise TypeError(f"Score must be an integer, got {type(score).__name__}")

        # Update the attempt with grading information
        attempt.score = score
        attempt.feedback = feedback
        attempt.graded_by = graded_by
        attempt.grading_timestamp = grading_dt

        # Write the updated attempt back to the database
        assessment_attempt_table[attempt_id] = attempt
        setattr(db, "assessment_attempt", assessment_attempt_table)

        # Return success indicator
        return {"success": True}

    @is_tool()
    def search_courses(
        self,
        keywords: str,
        category: Literal["programming", "data_science", "business", "design", "language", "mathematics", "science", "humanities", "health", "personal_development"] = None,
        difficulty_level: Literal["beginner", "intermediate", "advanced", "all_levels"] = None,
        min_rating: float = None,
        max_duration_hours: int = None
    ) -> Dict[str, Any]:
        """
        Search for courses based on keywords, categories, difficulty levels, and other filters.
        Uses fuzzy matching for course titles and descriptions to improve search robustness.

        Args:
            keywords: Search keywords to match course titles and descriptions
            category: Optional course category filter (enum constraint)
            difficulty_level: Optional difficulty level filter (enum constraint)
            min_rating: Optional minimum average rating (0.0 to 5.0)
            max_duration_hours: Optional maximum course duration in hours

        Returns:
            Dictionary containing list of courses matching the search criteria

        Raises:
            ValueError: If parameters are invalid
        """
        # Import required modules
        from thefuzz import fuzz, process

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

        # Validate min_rating if provided
        if min_rating is not None:
            if not isinstance(min_rating, (int, float)):
                raise ValueError("min_rating must be a number")
            if min_rating < 0.0 or min_rating > 5.0:
                raise ValueError("min_rating must be between 0.0 and 5.0")

        # Validate max_duration_hours if provided
        if max_duration_hours is not None:
            if not isinstance(max_duration_hours, int):
                raise ValueError("max_duration_hours must be an integer")
            if max_duration_hours <= 0:
                raise ValueError("max_duration_hours must be greater than 0")

        # Safety check for enum parameters (recommended for robustness)
        valid_categories = ["programming", "data_science", "business", "design", "language", 
                           "mathematics", "science", "humanities", "health", "personal_development"]
        valid_difficulty_levels = ["beginner", "intermediate", "advanced", "all_levels"]

        if category is not None and category not in valid_categories:
            raise ValueError(f"Invalid category. Must be one of: {', '.join(valid_categories)}")

        if difficulty_level is not None and difficulty_level not in valid_difficulty_levels:
            raise ValueError(f"Invalid difficulty_level. Must be one of: {', '.join(valid_difficulty_levels)}")

        # Access database
        db = self.db
        course_table = getattr(db, "course", None)

        # Handle empty database case
        if course_table is None or not course_table:
            return {"courses": []}

        # Normalize keywords for searching
        keywords_lower = keywords.lower().strip()

        # List to store matching courses with their match scores
        matching_courses = []

        # Iterate through all courses and apply filters
        for course_id, course in course_table.items():
            # Apply category filter (exact match for enum field)
            if category is not None and course.category != category:
                continue

            # Apply difficulty level filter (exact match for enum field)
            if difficulty_level is not None and course.difficulty_level != difficulty_level:
                continue

            # Apply min_rating filter
            if min_rating is not None:
                # Handle None average_rating by treating it as 0.0
                course_rating = course.average_rating if course.average_rating is not None else 0.0
                if course_rating < min_rating:
                    continue

            # Apply max_duration_hours filter
            if max_duration_hours is not None and course.duration_hours > max_duration_hours:
                continue

            # Apply keyword matching using fuzzy matching for title and description
            # Calculate match scores for title and description
            title_score = fuzz.partial_ratio(keywords_lower, course.title.lower())
            description_score = fuzz.partial_ratio(keywords_lower, course.description.lower())

            # Use the higher score between title and description
            max_score = max(title_score, description_score)

            # Set a reasonable threshold for matching (e.g., 60)
            # This can be adjusted based on requirements
            match_threshold = 60

            if max_score >= match_threshold:
                # Store course with its match score for sorting
                matching_courses.append({
                    "course": course,
                    "score": max_score
                })

        # Sort courses by match score (descending order)
        matching_courses.sort(key=lambda x: x["score"], reverse=True)

        # Build result list with course details
        result_courses = []
        for item in matching_courses:
            course = item["course"]
            course_dict = {
                "course_id": course.course_id,
                "title": course.title,
                "description": course.description,
                "category": course.category,
                "difficulty_level": course.difficulty_level,
                "instructor_name": course.instructor_name,
                "duration_hours": course.duration_hours,
                "average_rating": course.average_rating if course.average_rating is not None else 0.0
            }

            # Include optional fields if they exist
            if course.prerequisites is not None:
                course_dict["prerequisites"] = course.prerequisites

            if course.learning_outcomes is not None:
                course_dict["learning_outcomes"] = course.learning_outcomes

            result_courses.append(course_dict)

        return {"courses": result_courses}

    @is_tool()
    def record_interactive_exercise_attempt(
        self,
        enrollment_id: str,
        exercise_id: str,
        submitted_code: str,
        is_correct: bool,
        attempt_timestamp: str,
        execution_time_ms: Optional[int] = None
    ) -> Dict[str, str]:
        """
        Record a learner's attempt at an interactive coding exercise or practice problem.

        This method creates a new exercise attempt record in the lesson_completion table,
        storing the submitted code, correctness status, execution time, and timestamp.

        Args:
            enrollment_id: Unique identifier of the enrollment
            exercise_id: Unique identifier of the exercise
            submitted_code: Code submitted by the learner
            is_correct: Whether the solution was correct
            attempt_timestamp: Timestamp of the attempt in yyyy-mm-dd HH:MM:SS format
            execution_time_ms: Optional time taken to execute the code in milliseconds

        Returns:
            Dictionary containing the generated attempt_id

        Raises:
            ValueError: If enrollment doesn't exist, timestamp format is invalid,
                       or required parameters are invalid
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Access the database
        db = self.db

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

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

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

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

        try:
            # Parse the timestamp to validate format (yyyy-mm-dd HH:MM:SS)
            parsed_timestamp = datetime.strptime(attempt_timestamp, "%Y-%m-%d %H:%M:%S")
        except ValueError as e:
            raise ValueError(f"Invalid attempt_timestamp format. Expected 'yyyy-mm-dd HH:MM:SS', got '{attempt_timestamp}': {str(e)}")

        # Validate execution_time_ms if provided
        if execution_time_ms is not None and execution_time_ms < 0:
            raise ValueError("execution_time_ms must be non-negative")

        # Check if enrollment exists
        enrollment_table = getattr(db, "enrollment", None)
        if enrollment_table is None:
            raise ValueError("Enrollment table does not exist in database")

        enrollment = enrollment_table.get(enrollment_id)
        if enrollment is None:
            raise ValueError(f"Enrollment with ID '{enrollment_id}' does not exist")

        # Generate unique attempt_id using the specified pattern
        prefix = "EA"
        attempt_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get or initialize lesson_completion table
        lesson_completion_table = getattr(db, "lesson_completion", None)
        if lesson_completion_table is None:
            lesson_completion_table = {}
            setattr(db, "lesson_completion", lesson_completion_table)

        # Calculate time_spent_minutes from execution_time_ms
        # If execution_time_ms is provided, convert to minutes (round up to at least 1)
        # If not provided, default to 0
        if execution_time_ms is not None:
            time_spent_minutes = max(1, int(execution_time_ms / 60000))  # Convert ms to minutes, minimum 1
        else:
            time_spent_minutes = 0

        # Create the exercise attempt record using LessonCompletion schema
        # Note: We're repurposing the lesson_completion table to store exercise attempts
        # completion_id will be the attempt_id
        # lesson_id will store the exercise_id
        new_attempt = LessonCompletion(
            completion_id=attempt_id,
            enrollment_id=enrollment_id,
            lesson_id=exercise_id,  # Store exercise_id in lesson_id field
            completion_timestamp=parsed_timestamp,
            time_spent_minutes=time_spent_minutes
        )

        # Add the new attempt to the lesson_completion table
        lesson_completion_table[attempt_id] = new_attempt
        setattr(db, "lesson_completion", lesson_completion_table)

        # Return the generated attempt_id
        return {
            "attempt_id": attempt_id
        }

    @is_tool()
    def add_bookmark(
        self,
        enrollment_id: str,
        content_id: str,
        content_type: Literal["video", "reading", "exercise", "quiz", "discussion"],
        bookmark_timestamp: str,
        position_reference: str = None,
        note: str = None
    ) -> dict:
        """
        Add a bookmark to a specific point in course content for later reference.

        This method creates a new bookmark entry associated with an enrollment and content item.
        It validates that the enrollment exists and the content_type is valid, then generates
        a unique bookmark ID and stores the bookmark in the database.
        """
        import secrets
        import hashlib
        from datetime import datetime

        # Access the database
        db = self.db

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

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

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

        try:
            # Parse bookmark_timestamp to validate format (yyyy-mm-dd HH:MM:SS)
            parsed_timestamp = datetime.strptime(bookmark_timestamp, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise ValueError("bookmark_timestamp must be in 'yyyy-mm-dd HH:MM:SS' format")

        # Validate content_type is one of the allowed enum values
        valid_content_types = ["video", "reading", "exercise", "quiz", "discussion"]
        if content_type not in valid_content_types:
            raise ValueError(f"content_type must be one of {valid_content_types}, got '{content_type}'")

        # Retrieve enrollment table from database
        enrollment_table = getattr(db, "enrollment", None)
        if enrollment_table is None:
            raise ValueError("Enrollment table does not exist in database")

        # Check if the enrollment exists (pre-condition)
        if enrollment_id not in enrollment_table:
            raise ValueError(f"Enrollment with ID '{enrollment_id}' does not exist")

        # Retrieve bookmark table from database
        bookmark_table = getattr(db, "bookmark", None)
        if bookmark_table is None:
            # Initialize bookmark table if it doesn't exist
            bookmark_table = {}
            setattr(db, "bookmark", bookmark_table)

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

        # Ensure the generated bookmark_id is unique
        while bookmark_id in bookmark_table:
            bookmark_id = "BM" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Import Bookmark class using absolute import from database module
        # This follows the requirement to use "from database import {ClassName}" format

        # Create new bookmark instance with provided parameters
        new_bookmark = Bookmark(
            bookmark_id=bookmark_id,
            enrollment_id=enrollment_id,
            content_id=content_id,
            content_type=content_type,
            position_reference=position_reference,  # Optional, can be None
            note=note,  # Optional, can be None
            bookmark_timestamp=parsed_timestamp  # Store as datetime object
        )

        # Add the new bookmark to the bookmark table
        bookmark_table[bookmark_id] = new_bookmark

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

        # Return the bookmark_id of the created bookmark
        return {
            "bookmark_id": bookmark_id
        }

    @is_tool()
    def get_leaderboard(self, course_id: str, ranking_metric: Literal["total_points", "completion_percentage", "assessment_average", "participation_score"], top_n: int = None):
        # Access database directly via self.db (no import needed)
        db = self.db

        # Retrieve course, enrollment, and learner tables
        course_table = getattr(db, "course", None)
        enrollment_table = getattr(db, "enrollment", None)
        learner_table = getattr(db, "learner", None)

        # Validate that required tables exist
        if course_table is None or enrollment_table is None or learner_table is None:
            raise KeyError("Required database tables (course, enrollment, learner) not found")

        # Validate course exists
        if course_id not in course_table:
            raise KeyError(f"Course with course_id '{course_id}' does not exist")

        # Get all enrollments for this course
        course_enrollments = [
            enrollment for enrollment in enrollment_table.values()
            if enrollment.course_id == course_id
        ]

        # Check if there are any enrollments for this course
        if not course_enrollments:
            raise KeyError(f"No enrollments found for course '{course_id}'")

        # Build leaderboard data based on ranking metric
        leaderboard_data = []

        for enrollment in course_enrollments:
            learner_id = enrollment.learner_id

            # Validate learner exists
            if learner_id not in learner_table:
                continue  # Skip if learner not found

            learner = learner_table[learner_id]

            # Calculate score based on ranking metric
            score = None

            if ranking_metric == "total_points":
                # Use learner's total points
                score = learner.total_points if learner.total_points is not None else 0

            elif ranking_metric == "completion_percentage":
                # Use enrollment's progress percentage
                score = enrollment.progress_percentage if enrollment.progress_percentage is not None else 0.0

            elif ranking_metric == "assessment_average":
                # Use enrollment's final grade as assessment average
                score = enrollment.final_grade if enrollment.final_grade is not None else 0.0

            elif ranking_metric == "participation_score":
                # For participation score, use progress percentage as proxy
                # (no explicit participation field in schema)
                score = enrollment.progress_percentage if enrollment.progress_percentage is not None else 0.0

            # Add learner to leaderboard data
            leaderboard_data.append({
                "learner_id": learner_id,
                "score": score
            })

        # Sort leaderboard by score in descending order (highest score first)
        leaderboard_data.sort(key=lambda x: x["score"], reverse=True)

        # Limit to top_n if specified
        if top_n is not None and top_n > 0:
            leaderboard_data = leaderboard_data[:top_n]

        # Add rank to each entry
        leaderboard = []
        for rank, entry in enumerate(leaderboard_data, start=1):
            leaderboard.append({
                "rank": rank,
                "learner_id": entry["learner_id"],
                "score": entry["score"]
            })

        # Return leaderboard
        return {
            "leaderboard": leaderboard
        }

    @is_tool()
    def update_enrollment_status(self, enrollment_id: str, new_status: Literal["active", "completed", "dropped", "suspended"], update_timestamp: str):
        """
        Update the status of a course enrollment

        Args:
            enrollment_id: Unique identifier of the enrollment to update
            new_status: New status for the enrollment (must be one of: active, completed, dropped, suspended)
            update_timestamp: Timestamp of status update in yyyy-mm-dd HH:MM:SS format

        Returns:
            dict: {"success": bool} indicating whether the update was successful

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

        # Access the database
        db = self.db

        # Get the enrollment table from database
        enrollment_table = getattr(db, "enrollment", None)

        # Check if enrollment table exists
        if enrollment_table is None:
            raise KeyError(f"Enrollment table not found in database")

        # Check if the enrollment exists
        if enrollment_id not in enrollment_table:
            raise KeyError(f"Enrollment with ID '{enrollment_id}' does not exist")

        # Validate and parse the update timestamp
        try:
            parsed_timestamp = datetime.strptime(update_timestamp, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise ValueError(f"Invalid timestamp format. Expected 'yyyy-mm-dd HH:MM:SS', got '{update_timestamp}'")

        # Validate new_status is one of the allowed enum values
        # This is enforced by Literal type hint, but we add a safety check
        allowed_statuses = ["active", "completed", "dropped", "suspended"]
        if new_status not in allowed_statuses:
            raise ValueError(f"Invalid status '{new_status}'. Must be one of: {', '.join(allowed_statuses)}")

        # Get the existing enrollment record
        enrollment = enrollment_table[enrollment_id]

        # Update the enrollment status and last_updated timestamp
        enrollment.status = new_status
        enrollment.last_updated = parsed_timestamp

        # If status is completed and final_grade is not set, ensure progress is 100%
        if new_status == "completed" and enrollment.progress_percentage < 100.0:
            enrollment.progress_percentage = 100.0

        # Save the updated enrollment back to the database
        enrollment_table[enrollment_id] = enrollment
        setattr(db, "enrollment", enrollment_table)

        # Return success indicator
        return {"success": True}

    @is_tool()
    def issue_certificate(
        self,
        enrollment_id: str,
        issue_date: str,
        final_grade: float,
        certificate_type: Literal["completion", "achievement", "professional", "verified"]
    ) -> dict:
        """
        Issue a certificate of completion to a learner who has successfully completed a course.

        Args:
            enrollment_id: Unique identifier of the enrollment
            issue_date: Date of certificate issuance in yyyy-mm-dd HH:MM:SS format
            final_grade: Final grade or score for the course
            certificate_type: Type of certificate being issued (must be one of: completion, achievement, professional, verified)

        Returns:
            dict: Contains certificate_id and verification_code

        Raises:
            ValueError: If enrollment doesn't exist, enrollment not completed, invalid date format, invalid grade, or invalid certificate type
        """
        import secrets
        import hashlib
        from datetime import datetime

        # Access database
        db = self.db

        # Validate certificate_type against enum values (safety protection)
        valid_certificate_types = ["completion", "achievement", "professional", "verified"]
        if certificate_type not in valid_certificate_types:
            raise ValueError(
                f"Invalid certificate_type '{certificate_type}'. Must be one of: {', '.join(valid_certificate_types)}"
            )

        # Validate and parse issue_date format
        try:
            parsed_issue_date = datetime.strptime(issue_date, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise ValueError(
                f"Invalid issue_date format '{issue_date}'. Expected format: yyyy-mm-dd HH:MM:SS (e.g., '2024-03-20 12:00:00')"
            )

        # Validate final_grade is a valid number
        if not isinstance(final_grade, (int, float)):
            raise ValueError(f"Invalid final_grade '{final_grade}'. Must be a number")

        # Typically grades should be between 0 and 100, but allow flexibility
        if final_grade < 0:
            raise ValueError(f"Invalid final_grade '{final_grade}'. Grade cannot be negative")

        # Retrieve enrollment table
        enrollment_table = getattr(db, "enrollment", None)
        if enrollment_table is None:
            raise ValueError("Enrollment table does not exist in database")

        # Check if enrollment exists
        enrollment = enrollment_table.get(enrollment_id)
        if enrollment is None:
            raise ValueError(f"Enrollment with ID '{enrollment_id}' does not exist")

        # Pre-condition: Verify that the learner has completed all course requirements
        # Check enrollment status is 'completed'
        if enrollment.status != "completed":
            raise ValueError(
                f"Enrollment '{enrollment_id}' has not completed the course. "
                f"Current status: '{enrollment.status}'. Certificate can only be issued for completed enrollments"
            )

        # Verify progress is 100% (if progress_percentage is tracked)
        if enrollment.progress_percentage is not None and enrollment.progress_percentage < 100.0:
            raise ValueError(
                f"Enrollment '{enrollment_id}' has not completed all course requirements. "
                f"Progress: {enrollment.progress_percentage}%. Certificate requires 100% completion"
            )

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

        # Generate verification_code for certificate authenticity
        verification_code = hashlib.sha256(
            f"{certificate_id}{enrollment_id}{issue_date}".encode()
        ).hexdigest()[:12].upper()

        # Create new certificate record
        new_certificate = Certificate(
            certificate_id=certificate_id,
            enrollment_id=enrollment_id,
            issue_date=parsed_issue_date,
            final_grade=final_grade,
            certificate_type=certificate_type,
            verification_code=verification_code
        )

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

        # Add new certificate to the table
        certificate_table[certificate_id] = new_certificate

        # Save certificate table back to database
        setattr(db, "certificate", certificate_table)

        # Update enrollment's final_grade if not already set
        if enrollment.final_grade is None or enrollment.final_grade != final_grade:
            enrollment.final_grade = final_grade
            enrollment.last_updated = parsed_issue_date
            enrollment_table[enrollment_id] = enrollment
            setattr(db, "enrollment", enrollment_table)

        # Return certificate information
        return {
            "certificate_id": certificate_id,
            "verification_code": verification_code
        }

    @is_tool()
    def get_learner_bookmarks(self, learner_id: str, course_id: Optional[str] = None):
        """
        Retrieve all bookmarks created by a learner across their enrolled courses.

        Args:
            learner_id: Unique identifier of the learner
            course_id: Optional filter to get bookmarks for a specific course

        Returns:
            Dictionary containing list of bookmarks with their details

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

        # Validate that learner exists in the system
        learner_table = getattr(db, "learner", None)
        if learner_table is None or learner_id not in learner_table:
            raise KeyError(f"Learner with ID '{learner_id}' does not exist in the system")

        # Get all enrollments for this learner
        enrollment_table = getattr(db, "enrollment", None)
        if enrollment_table is None:
            # No enrollments exist, return empty bookmarks list
            return {"bookmarks": []}

        # Find all enrollment IDs for this learner
        # If course_id is provided, filter enrollments by course_id as well
        learner_enrollment_ids = []
        for enrollment_id, enrollment in enrollment_table.items():
            if enrollment.learner_id == learner_id:
                # If course_id filter is provided, check if enrollment matches the course
                if course_id is None or enrollment.course_id == course_id:
                    learner_enrollment_ids.append(enrollment_id)

        # If no enrollments found, return empty bookmarks list
        if not learner_enrollment_ids:
            return {"bookmarks": []}

        # Get all bookmarks for the learner's enrollments
        bookmark_table = getattr(db, "bookmark", None)
        if bookmark_table is None:
            # No bookmarks exist, return empty list
            return {"bookmarks": []}

        # Collect all bookmarks that belong to the learner's enrollments
        bookmarks_list = []
        for bookmark_id, bookmark in bookmark_table.items():
            if bookmark.enrollment_id in learner_enrollment_ids:
                # Convert bookmark to dictionary format for return
                bookmark_dict = {
                    "bookmark_id": bookmark.bookmark_id,
                    "enrollment_id": bookmark.enrollment_id,
                    "content_id": bookmark.content_id,
                    "content_type": bookmark.content_type,
                    "bookmark_timestamp": bookmark.bookmark_timestamp.strftime("%Y-%m-%d %H:%M:%S")
                }

                # Add optional fields if they exist
                if bookmark.position_reference is not None:
                    bookmark_dict["position_reference"] = bookmark.position_reference

                if bookmark.note is not None:
                    bookmark_dict["note"] = bookmark.note

                bookmarks_list.append(bookmark_dict)

        # Sort bookmarks by timestamp (most recent first) for better user experience
        bookmarks_list.sort(key=lambda x: x["bookmark_timestamp"], reverse=True)

        return {"bookmarks": bookmarks_list}

    @is_tool()
    def validate_prerequisites(self, required_prerequisites: List[str], learner_completed_courses: List[str]) -> dict:
        """
        Validate whether a learner meets the prerequisites required to enroll in a course.

        This method compares the required prerequisites against the learner's completed courses
        to determine if all prerequisites are satisfied.

        Args:
            required_prerequisites: List of required prerequisite course IDs or skills
            learner_completed_courses: List of course IDs completed by the learner

        Returns:
            dict: Contains:
                - prerequisites_met (bool): Whether all prerequisites are satisfied
                - missing_prerequisites (List[str]): List of missing prerequisites if any

        Raises:
            ValueError: If input parameters are invalid
        """
        # Input validation: check if parameters are provided
        if required_prerequisites is None:
            raise ValueError("required_prerequisites cannot be None")
        if learner_completed_courses is None:
            raise ValueError("learner_completed_courses cannot be None")

        # Input validation: check if parameters are lists
        if not isinstance(required_prerequisites, list):
            raise ValueError("required_prerequisites must be a list")
        if not isinstance(learner_completed_courses, list):
            raise ValueError("learner_completed_courses must be a list")

        # Input validation: check if all elements in lists are strings
        if not all(isinstance(item, str) for item in required_prerequisites):
            raise ValueError("All elements in required_prerequisites must be strings")
        if not all(isinstance(item, str) for item in learner_completed_courses):
            raise ValueError("All elements in learner_completed_courses must be strings")

        # Convert learner's completed courses to a set for efficient lookup
        # This allows O(1) lookup time instead of O(n) for list
        completed_set = set(learner_completed_courses)

        # Initialize list to track missing prerequisites
        missing_prerequisites = []

        # Check each required prerequisite against completed courses
        for prerequisite in required_prerequisites:
            # If prerequisite is not in the completed set, add it to missing list
            if prerequisite not in completed_set:
                missing_prerequisites.append(prerequisite)

        # Determine if all prerequisites are met
        # Prerequisites are met if there are no missing prerequisites
        prerequisites_met = len(missing_prerequisites) == 0

        # Return validation result with all required information
        return {
            "prerequisites_met": prerequisites_met,
            "missing_prerequisites": missing_prerequisites
        }

    @is_tool()
    def get_course_details(self, course_id: str):
        """
        Retrieve comprehensive information about a specific course including syllabus, 
        instructor details, prerequisites, and learning outcomes.

        Args:
            course_id: Unique identifier of the course

        Returns:
            dict: Complete course information including:
                - course_id: Unique identifier of the course
                - title: Course title
                - description: Detailed course description
                - instructor_name: Name of the course instructor
                - duration_hours: Total course duration in hours
                - prerequisites: List of prerequisite courses or skills
                - learning_outcomes: Expected learning outcomes after course completion

        Raises:
            KeyError: If the course does not exist in the system
        """
        import json

        # Access the database
        db = self.db

        # Retrieve the course table from the database
        course_table = getattr(db, "course", None)

        # Check if the course table exists
        if course_table is None:
            raise KeyError(f"Course table not found in database")

        # Check if the course exists in the table
        if course_id not in course_table:
            raise KeyError(f"Course with ID '{course_id}' not found in the system")

        # Retrieve the course object
        course = course_table[course_id]

        # Parse prerequisites from JSON string to list
        # If prerequisites is None or empty, default to empty list
        prerequisites = []
        if course.prerequisites:
            try:
                prerequisites = json.loads(course.prerequisites)
                # Ensure it's a list
                if not isinstance(prerequisites, list):
                    prerequisites = []
            except (json.JSONDecodeError, TypeError):
                # If parsing fails, default to empty list
                prerequisites = []

        # Parse learning_outcomes from JSON string to list
        # If learning_outcomes is None or empty, default to empty list
        learning_outcomes = []
        if course.learning_outcomes:
            try:
                learning_outcomes = json.loads(course.learning_outcomes)
                # Ensure it's a list
                if not isinstance(learning_outcomes, list):
                    learning_outcomes = []
            except (json.JSONDecodeError, TypeError):
                # If parsing fails, default to empty list
                learning_outcomes = []

        # Construct the return dictionary with all course details
        course_details = {
            "course_id": course.course_id,
            "title": course.title,
            "description": course.description,
            "instructor_name": course.instructor_name,
            "duration_hours": course.duration_hours,
            "prerequisites": prerequisites,
            "learning_outcomes": learning_outcomes
        }

        return course_details

    @is_tool()
    def calculate_study_time_statistics(self, session_durations_minutes: List[int], date_range_days: int) -> dict:
        """
        Calculate study time statistics for a learner including total time, average daily time, 
        and time distribution across courses.

        Args:
            session_durations_minutes: List of study session durations in minutes
            date_range_days: Number of days in the analysis period

        Returns:
            Dictionary containing:
                - total_time_minutes: Total study time in minutes
                - average_daily_minutes: Average study time per day
                - longest_session_minutes: Duration of longest study session

        Raises:
            ValueError: If input parameters are invalid
        """
        # Validate input parameters
        if not isinstance(session_durations_minutes, list):
            raise ValueError("session_durations_minutes must be a list")

        if not isinstance(date_range_days, int):
            raise ValueError("date_range_days must be an integer")

        if date_range_days <= 0:
            raise ValueError("date_range_days must be a positive integer")

        # Validate all session durations are non-negative integers
        for duration in session_durations_minutes:
            if not isinstance(duration, int):
                raise ValueError("All session durations must be integers")
            if duration < 0:
                raise ValueError("Session durations cannot be negative")

        # Handle empty session list case
        if len(session_durations_minutes) == 0:
            return {
                "total_time_minutes": 0,
                "average_daily_minutes": 0.0,
                "longest_session_minutes": 0
            }

        # Calculate total study time
        # Sum all session durations to get total time spent studying
        total_time_minutes = sum(session_durations_minutes)

        # Calculate average daily study time
        # Divide total time by the number of days in the analysis period
        # This gives us the average time spent per day over the entire period
        average_daily_minutes = total_time_minutes / date_range_days

        # Find the longest study session
        # This helps identify peak study periods and maximum focus duration
        longest_session_minutes = max(session_durations_minutes)

        # Return comprehensive statistics
        return {
            "total_time_minutes": total_time_minutes,
            "average_daily_minutes": round(average_daily_minutes, 2),  # Round to 2 decimal places for readability
            "longest_session_minutes": longest_session_minutes
        }

    @is_tool()
    def verify_certificate(self, certificate_id: str, verification_code: str):
        """
        Verify the authenticity of a certificate using its verification code.

        This method checks if a certificate exists and validates its verification code,
        then returns the certificate details including learner information and course title.

        Args:
            certificate_id: Unique identifier of the certificate to verify
            verification_code: Verification code to validate against the certificate

        Returns:
            dict: Contains verification status and certificate details:
                - is_valid (bool): Whether the certificate is valid
                - learner_name (str): Name of the certificate holder
                - title (str): Title of the completed course
                - issue_date (str): Date the certificate was issued in yyyy-mm-dd format

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

        # Get certificate table data
        certificate_table = getattr(db, "certificate", None)
        if certificate_table is None or certificate_id not in certificate_table:
            raise KeyError(f"Certificate with ID '{certificate_id}' does not exist in the system")

        # Retrieve the certificate record
        certificate = certificate_table[certificate_id]

        # Verify the verification code matches
        is_valid = certificate.verification_code == verification_code

        # If verification fails, return invalid status with empty details
        if not is_valid:
            raise KeyError(f"Invalid verification code for certificate '{certificate_id}'")

        # Get enrollment table to find learner and course information
        enrollment_table = getattr(db, "enrollment", None)
        if enrollment_table is None or certificate.enrollment_id not in enrollment_table:
            raise KeyError(f"Enrollment with ID '{certificate.enrollment_id}' does not exist in the system")

        enrollment = enrollment_table[certificate.enrollment_id]

        # Get learner information
        learner_table = getattr(db, "learner", None)
        if learner_table is None or enrollment.learner_id not in learner_table:
            raise KeyError(f"Learner with ID '{enrollment.learner_id}' does not exist in the system")

        learner = learner_table[enrollment.learner_id]

        # Get course information
        course_table = getattr(db, "course", None)
        if course_table is None or enrollment.course_id not in course_table:
            raise KeyError(f"Course with ID '{enrollment.course_id}' does not exist in the system")

        course = course_table[enrollment.course_id]

        # Format the issue date to yyyy-mm-dd format
        issue_date_str = certificate.issue_date.strftime("%Y-%m-%d")

        # Return verification result with certificate details
        return {
            "is_valid": is_valid,
            "learner_name": learner.name,
            "title": course.title,
            "issue_date": issue_date_str
        }

    @is_tool()
    def record_lesson_completion(self, enrollment_id: str, lesson_id: str, completion_timestamp: str, time_spent_minutes: int):
        """
        Record that a learner has completed a specific lesson within a course.

        This method creates a new lesson completion record after validating that:
        1. The enrollment exists
        2. The lesson hasn't already been completed for this enrollment
        3. The completion timestamp is in valid format
        4. The time spent is a positive integer
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Access the database
        db = self.db

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

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

        # Validate time_spent_minutes is positive
        if time_spent_minutes <= 0:
            raise ValueError("time_spent_minutes must be a positive integer")

        # Parse and validate completion_timestamp format
        try:
            parsed_timestamp = datetime.strptime(completion_timestamp, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise ValueError("completion_timestamp must be in 'yyyy-mm-dd HH:MM:SS' format")

        # Get enrollment table from database
        enrollment_table = getattr(db, "enrollment", None)
        if enrollment_table is None:
            raise ValueError("Enrollment table does not exist in database")

        # Check if enrollment exists
        enrollment = enrollment_table.get(enrollment_id)
        if enrollment is None:
            raise ValueError(f"Enrollment with id '{enrollment_id}' does not exist")

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

        # Check if this lesson has already been completed for this enrollment
        # Iterate through all completion records to check for duplicates
        for completion_id, completion in lesson_completion_table.items():
            if completion.enrollment_id == enrollment_id and completion.lesson_id == lesson_id:
                raise ValueError(f"Lesson '{lesson_id}' has already been completed for enrollment '{enrollment_id}'")

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

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

        # Create new LessonCompletion record
        new_completion = LessonCompletion(
            completion_id=completion_id,
            enrollment_id=enrollment_id,
            lesson_id=lesson_id,
            completion_timestamp=parsed_timestamp,
            time_spent_minutes=time_spent_minutes
        )

        # Add the new completion record to the table
        lesson_completion_table[completion_id] = new_completion

        # Update the database with the modified table
        setattr(db, "lesson_completion", lesson_completion_table)

        # Return the completion_id
        return {
            "completion_id": completion_id
        }

    @is_tool()
    def set_learning_goal(
        self,
        learner_id: str,
        goal_type: Literal["complete_course", "achieve_certification", "master_skill", "study_hours", "complete_projects"],
        goal_description: str,
        target_date: str,
        creation_timestamp: str,
        target_value: str = None
    ):
        # Import necessary modules for datetime handling and ID generation
        from datetime import datetime
        import secrets
        import hashlib

        # Access the database instance
        db = self.db

        # Validate that learner exists in the system (pre-condition)
        learner_table = getattr(db, "learner", None)
        if learner_table is None:
            raise ValueError("Learner table does not exist in the database")

        if learner_id not in learner_table:
            raise ValueError(f"Learner with ID '{learner_id}' does not exist in the system")

        # Validate goal_type against enum values (safety protection for enum parameter)
        valid_goal_types = ["complete_course", "achieve_certification", "master_skill", "study_hours", "complete_projects"]
        if goal_type not in valid_goal_types:
            raise ValueError(f"Invalid goal_type '{goal_type}'. Must be one of: {valid_goal_types}")

        # Parse and validate target_date format (yyyy-mm-dd)
        try:
            target_date_obj = datetime.strptime(target_date, "%Y-%m-%d").date()
        except ValueError:
            raise ValueError(f"Invalid target_date format '{target_date}'. Expected format: yyyy-mm-dd")

        # Parse and validate creation_timestamp format (yyyy-mm-dd HH:MM:SS)
        try:
            creation_timestamp_obj = datetime.strptime(creation_timestamp, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise ValueError(f"Invalid creation_timestamp format '{creation_timestamp}'. Expected format: yyyy-mm-dd HH:MM:SS")

        # Validate that target_date is in the future relative to creation_timestamp
        if target_date_obj < creation_timestamp_obj.date():
            raise ValueError("Target date must be on or after the creation date")

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

        # Ensure the generated goal_id is unique in the learning_goal table
        learning_goal_table = getattr(db, "learning_goal", None)
        if learning_goal_table is None:
            learning_goal_table = {}

        # If by rare chance the ID already exists, regenerate
        while goal_id in learning_goal_table:
            goal_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Create new LearningGoal instance with all required fields
        new_goal = LearningGoal(
            goal_id=goal_id,
            learner_id=learner_id,
            goal_type=goal_type,
            goal_description=goal_description,
            target_value=target_value,  # Optional field, can be None
            current_value=None,  # Initialize as None (no progress yet)
            target_date=target_date_obj,
            creation_timestamp=creation_timestamp_obj,
            is_completed=False  # New goal starts as not completed
        )

        # Add the new goal to the learning_goal table
        learning_goal_table[goal_id] = new_goal
        setattr(db, "learning_goal", learning_goal_table)

        # Return the unique identifier of the created goal
        return {
            "goal_id": goal_id
        }

    @is_tool()
    def calculate_course_progress(
        self,
        total_lessons: int,
        completed_lessons: int,
        total_assessments: int,
        completed_assessments: int
    ) -> dict:
        """
        Calculate the overall progress percentage for a learner in a specific course
        based on completed lessons and assessments.

        The progress is calculated using a weighted average:
        - Lessons contribute 60% to the overall progress
        - Assessments contribute 40% to the overall progress

        Args:
            total_lessons: Total number of lessons in the course
            completed_lessons: Number of lessons completed by the learner
            total_assessments: Total number of assessments in the course
            completed_assessments: Number of assessments completed by the learner

        Returns:
            dict: Dictionary containing progress_percentage (float, 0.0 to 100.0)

        Raises:
            ValueError: If any input parameter is negative, or if completed count exceeds total count
        """
        # Validate input parameters - all must be non-negative integers
        if total_lessons < 0:
            raise ValueError("total_lessons must be a non-negative integer")
        if completed_lessons < 0:
            raise ValueError("completed_lessons must be a non-negative integer")
        if total_assessments < 0:
            raise ValueError("total_assessments must be a non-negative integer")
        if completed_assessments < 0:
            raise ValueError("completed_assessments must be a non-negative integer")

        # Validate that completed counts do not exceed total counts
        if completed_lessons > total_lessons:
            raise ValueError(
                f"completed_lessons ({completed_lessons}) cannot exceed total_lessons ({total_lessons})"
            )
        if completed_assessments > total_assessments:
            raise ValueError(
                f"completed_assessments ({completed_assessments}) cannot exceed total_assessments ({total_assessments})"
            )

        # Define weights for lessons and assessments
        # Lessons contribute 60% and assessments contribute 40% to overall progress
        lesson_weight = 0.6
        assessment_weight = 0.4

        # Calculate lesson progress percentage
        if total_lessons > 0:
            lesson_progress = (completed_lessons / total_lessons) * 100.0
        else:
            # If there are no lessons, lesson progress is 0%
            lesson_progress = 0.0

        # Calculate assessment progress percentage
        if total_assessments > 0:
            assessment_progress = (completed_assessments / total_assessments) * 100.0
        else:
            # If there are no assessments, assessment progress is 0%
            assessment_progress = 0.0

        # Handle edge case: if both totals are 0, progress is 0%
        if total_lessons == 0 and total_assessments == 0:
            overall_progress = 0.0
        # If only lessons exist (no assessments), progress is based solely on lessons
        elif total_assessments == 0:
            overall_progress = lesson_progress
        # If only assessments exist (no lessons), progress is based solely on assessments
        elif total_lessons == 0:
            overall_progress = assessment_progress
        # Normal case: calculate weighted average of both components
        else:
            overall_progress = (lesson_progress * lesson_weight) + (assessment_progress * assessment_weight)

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

        # Ensure progress is within valid range [0.0, 100.0]
        overall_progress = max(0.0, min(100.0, overall_progress))

        return {
            "progress_percentage": overall_progress
        }

    @is_tool()
    def post_discussion_message(self, enrollment_id: str, discussion_topic: str, message_content: str, post_timestamp: str):
        from datetime import datetime
        import secrets
        import hashlib

        # Get database instance
        db = self.db

        # Validate and parse timestamp
        try:
            parsed_timestamp = datetime.strptime(post_timestamp, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise ValueError(f"Invalid timestamp format: {post_timestamp}. Expected format: yyyy-mm-dd HH:MM:SS")

        # Retrieve enrollment table from database
        enrollment_table = getattr(db, "enrollment", None)
        if enrollment_table is None:
            raise ValueError("Enrollment table not found in database")

        # Check if enrollment exists
        enrollment = enrollment_table.get(enrollment_id)
        if enrollment is None:
            raise ValueError(f"Enrollment with ID '{enrollment_id}' does not exist")

        # Verify enrollment status is active (pre-condition: learner must be enrolled)
        if enrollment.status != "active":
            raise ValueError(f"Enrollment '{enrollment_id}' is not active. Current status: {enrollment.status}")

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

        # Create new discussion message instance
        # This is a new top-level message, not a reply, so parent_message_id is None
        new_message = DiscussionMessage(
            message_id=message_id,
            enrollment_id=enrollment_id,
            parent_message_id=None,
            discussion_topic=discussion_topic,
            message_content=message_content,
            post_timestamp=parsed_timestamp
        )

        # Retrieve discussion_message table from database
        discussion_message_table = getattr(db, "discussion_message", None)
        if discussion_message_table is None:
            # Initialize empty dictionary if table doesn't exist
            discussion_message_table = {}

        # Add new message to the discussion_message table
        discussion_message_table[message_id] = new_message

        # Update database with the new message
        setattr(db, "discussion_message", discussion_message_table)

        # Return the generated message_id
        return {
            "message_id": message_id
        }

    @is_tool()
    def submit_peer_review(self, review_request_id: str, reviewer_enrollment_id: str, rating: int, feedback: str, submission_timestamp: str):
        """
        Submit a peer review with ratings and feedback for another learner's work.

        This method creates a new peer review record by:
        1. Validating the review request exists
        2. Validating the reviewer enrollment exists and is active
        3. Validating the rating is within acceptable range (1-5)
        4. Parsing and validating the submission timestamp format
        5. Generating a unique review ID
        6. Creating and storing the peer review record

        Args:
            review_request_id: Unique identifier of the peer review request
            reviewer_enrollment_id: Enrollment ID of the reviewer
            rating: Numerical rating on 1-5 scale
            feedback: Written feedback for the reviewed work
            submission_timestamp: Timestamp when review was submitted (yyyy-mm-dd HH:MM:SS format)

        Returns:
            Dictionary containing the generated review_id

        Raises:
            ValueError: If validation fails for any input parameter
        """
        import secrets
        import hashlib
        from datetime import datetime
        import json

        # Access the database
        db = self.db

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

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

        # Validate that the reviewer enrollment exists in the database
        enrollment_table = getattr(db, "enrollment", None)
        if enrollment_table is None:
            raise ValueError("Enrollment table not found in database")

        reviewer_enrollment = enrollment_table.get(reviewer_enrollment_id)
        if reviewer_enrollment is None:
            raise ValueError(f"Reviewer enrollment with ID '{reviewer_enrollment_id}' does not exist")

        # Validate that the reviewer enrollment is active
        if reviewer_enrollment.status not in ["active", "completed"]:
            raise ValueError(f"Reviewer enrollment status is '{reviewer_enrollment.status}', must be 'active' or 'completed' to submit reviews")

        # Validate rating is within the acceptable range (1-5)
        if not isinstance(rating, int):
            raise ValueError("rating must be an integer")
        if rating < 1 or rating > 5:
            raise ValueError(f"rating must be between 1 and 5, got {rating}")

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

        # Validate and parse submission_timestamp
        try:
            # Try parsing with full datetime format (yyyy-mm-dd HH:MM:SS)
            parsed_timestamp = datetime.strptime(submission_timestamp, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            try:
                # Try parsing with date only format (yyyy-mm-dd)
                parsed_timestamp = datetime.strptime(submission_timestamp, "%Y-%m-%d")
            except ValueError:
                raise ValueError(f"submission_timestamp must be in 'yyyy-mm-dd HH:MM:SS' or 'yyyy-mm-dd' format, got '{submission_timestamp}'")

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

        # Get assessment_attempt table to store the peer review
        # Note: Since there's no dedicated peer_review table, we store it as an assessment attempt
        # with special attributes indicating it's a peer review
        assessment_attempt_table = getattr(db, "assessment_attempt", None)
        if assessment_attempt_table is None:
            raise ValueError("assessment_attempt table not found in database")

        # Create a new assessment attempt record to represent the peer review
        # We use the assessment_attempt structure to store peer review data
        # The answers field will store a JSON with review details
        review_data = {
            "review_request_id": review_request_id,
            "reviewer_enrollment_id": reviewer_enrollment_id,
            "rating": rating,
            "feedback": feedback,
            "review_id": review_id
        }

        # Create new peer review record as an AssessmentAttempt
        new_review = AssessmentAttempt(
            attempt_id=review_id,
            enrollment_id=reviewer_enrollment_id,
            assessment_id=review_request_id,  # Using review_request_id as assessment_id
            submission_timestamp=parsed_timestamp,
            time_spent_minutes=None,
            answers=json.dumps(review_data),  # Store review details in JSON format
            score=rating,  # Store rating as score
            feedback=feedback,
            graded_by=reviewer_enrollment_id,  # Reviewer is the one submitting
            grading_timestamp=parsed_timestamp
        )

        # Store the new review in the database
        assessment_attempt_table[review_id] = new_review

        # Return the generated review_id
        return {
            "review_id": review_id
        }

    @is_tool()
    def export_learner_transcript(self, learner_id: str, format: Literal["json", "pdf", "csv"], include_in_progress: bool = False):
        # Access the database
        db = self.db

        # Verify learner exists
        learner_table = getattr(db, "learner", None)
        if learner_table is None or learner_id not in learner_table:
            raise KeyError(f"Learner with ID '{learner_id}' does not exist")

        learner = learner_table[learner_id]

        # Get all enrollments for this learner
        enrollment_table = getattr(db, "enrollment", None)
        course_table = getattr(db, "course", None)
        certificate_table = getattr(db, "certificate", None)

        if enrollment_table is None:
            raise KeyError("Enrollment table not found in database")
        if course_table is None:
            raise KeyError("Course table not found in database")

        # Filter enrollments by learner_id and status
        learner_enrollments = []
        for enrollment_id, enrollment in enrollment_table.items():
            if enrollment.learner_id == learner_id:
                # Include completed courses always, and in-progress courses if requested
                if enrollment.status == "completed" or (include_in_progress and enrollment.status == "active"):
                    learner_enrollments.append(enrollment)

        # Build transcript data by gathering course details, grades, and certificates
        transcript_courses = []
        for enrollment in learner_enrollments:
            course = course_table.get(enrollment.course_id)
            if course is None:
                # Skip if course data is missing
                continue

            # Find certificate for this enrollment (if exists)
            certificate_info = None
            if certificate_table is not None:
                for cert_id, cert in certificate_table.items():
                    if cert.enrollment_id == enrollment.enrollment_id:
                        certificate_info = cert
                        break

            # Build course entry for transcript
            course_entry = {
                "course_id": course.course_id,
                "title": course.title,
                "category": course.category,
                "instructor": course.instructor_name,
                "enrollment_date": enrollment.enrollment_date.strftime("%Y-%m-%d %H:%M:%S"),
                "status": enrollment.status,
                "progress_percentage": enrollment.progress_percentage if enrollment.progress_percentage is not None else 0.0,
                "final_grade": enrollment.final_grade if enrollment.final_grade is not None else None,
                "completion_date": enrollment.last_updated.strftime("%Y-%m-%d %H:%M:%S") if enrollment.status == "completed" and enrollment.last_updated else None,
                "certificate_id": certificate_info.certificate_id if certificate_info else None,
                "certificate_type": certificate_info.certificate_type if certificate_info else None,
                "verification_code": certificate_info.verification_code if certificate_info else None
            }

            transcript_courses.append(course_entry)

        # Sort courses by enrollment date (most recent first)
        transcript_courses.sort(key=lambda x: x["enrollment_date"], reverse=True)

        # Build complete transcript data structure
        transcript_data_dict = {
            "learner_id": learner.learner_id,
            "learner_name": learner.name,
            "learner_email": learner.email,
            "registration_date": learner.registration_date.strftime("%Y-%m-%d %H:%M:%S"),
            "total_points": learner.total_points if learner.total_points is not None else 0,
            "total_courses": len(transcript_courses),
            "completed_courses": sum(1 for c in transcript_courses if c["status"] == "completed"),
            "in_progress_courses": sum(1 for c in transcript_courses if c["status"] == "active"),
            "courses": transcript_courses
        }

        # Format the transcript based on requested format
        if format == "json":
            import json
            transcript_data = json.dumps(transcript_data_dict, indent=2, ensure_ascii=False)

        elif format == "csv":
            # Build CSV format with header and data rows
            import csv
            from io import StringIO

            output = StringIO()
            writer = csv.writer(output)

            # Write learner summary header
            writer.writerow(["Learner Transcript"])
            writer.writerow(["Learner ID", learner.learner_id])
            writer.writerow(["Name", learner.name])
            writer.writerow(["Email", learner.email])
            writer.writerow(["Registration Date", learner.registration_date.strftime("%Y-%m-%d %H:%M:%S")])
            writer.writerow(["Total Points", learner.total_points if learner.total_points is not None else 0])
            writer.writerow([])

            # Write course data header and rows
            writer.writerow(["Course ID", "Title", "Category", "Instructor", "Enrollment Date", "Status", 
                            "Progress %", "Final Grade", "Completion Date", "Certificate ID", 
                            "Certificate Type", "Verification Code"])

            for course in transcript_courses:
                writer.writerow([
                    course["course_id"],
                    course["title"],
                    course["category"],
                    course["instructor"],
                    course["enrollment_date"],
                    course["status"],
                    course["progress_percentage"],
                    course["final_grade"] if course["final_grade"] is not None else "",
                    course["completion_date"] if course["completion_date"] else "",
                    course["certificate_id"] if course["certificate_id"] else "",
                    course["certificate_type"] if course["certificate_type"] else "",
                    course["verification_code"] if course["verification_code"] else ""
                ])

            transcript_data = output.getvalue()

        elif format == "pdf":
            # Generate a simple text-based PDF representation
            # Note: In a real implementation, you would use a PDF library like reportlab
            # Here we create a formatted text representation that simulates PDF content

            pdf_content = []
            pdf_content.append("=" * 80)
            pdf_content.append("LEARNER TRANSCRIPT")
            pdf_content.append("=" * 80)
            pdf_content.append("")
            pdf_content.append(f"Learner ID: {learner.learner_id}")
            pdf_content.append(f"Name: {learner.name}")
            pdf_content.append(f"Email: {learner.email}")
            pdf_content.append(f"Registration Date: {learner.registration_date.strftime('%Y-%m-%d %H:%M:%S')}")
            pdf_content.append(f"Total Points: {learner.total_points if learner.total_points is not None else 0}")
            pdf_content.append("")
            pdf_content.append(f"Total Courses: {len(transcript_courses)}")
            pdf_content.append(f"Completed: {sum(1 for c in transcript_courses if c['status'] == 'completed')}")
            pdf_content.append(f"In Progress: {sum(1 for c in transcript_courses if c['status'] == 'active')}")
            pdf_content.append("")
            pdf_content.append("-" * 80)
            pdf_content.append("COURSE HISTORY")
            pdf_content.append("-" * 80)
            pdf_content.append("")

            for idx, course in enumerate(transcript_courses, 1):
                pdf_content.append(f"{idx}. {course['title']}")
                pdf_content.append(f"   Course ID: {course['course_id']}")
                pdf_content.append(f"   Category: {course['category']}")
                pdf_content.append(f"   Instructor: {course['instructor']}")
                pdf_content.append(f"   Enrollment Date: {course['enrollment_date']}")
                pdf_content.append(f"   Status: {course['status']}")
                pdf_content.append(f"   Progress: {course['progress_percentage']}%")

                if course['final_grade'] is not None:
                    pdf_content.append(f"   Final Grade: {course['final_grade']}")

                if course['completion_date']:
                    pdf_content.append(f"   Completion Date: {course['completion_date']}")

                if course['certificate_id']:
                    pdf_content.append(f"   Certificate ID: {course['certificate_id']}")
                    pdf_content.append(f"   Certificate Type: {course['certificate_type']}")
                    pdf_content.append(f"   Verification Code: {course['verification_code']}")

                pdf_content.append("")

            pdf_content.append("=" * 80)
            pdf_content.append("END OF TRANSCRIPT")
            pdf_content.append("=" * 80)

            transcript_data = "\n".join(pdf_content)

        else:
            # This should not happen due to Literal type constraint, but add safety check
            raise ValueError(f"Unsupported format: {format}. Must be one of: json, pdf, csv")

        return {"transcript_data": transcript_data}

    @is_tool()
    def reply_to_discussion(self, parent_message_id: str, enrollment_id: str, reply_content: str, reply_timestamp: str) -> dict:
        """
        Post a reply to an existing discussion message in a course forum.

        Args:
            parent_message_id: Unique identifier of the message being replied to
            enrollment_id: Unique identifier of the enrollment of the person replying
            reply_content: Content of the reply
            reply_timestamp: Timestamp when reply was posted in yyyy-mm-dd HH:MM:SS format

        Returns:
            dict: Contains reply_id of the posted reply

        Raises:
            KeyError: If parent message or enrollment does not exist
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Access database
        db = self.db

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

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

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

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

        # Parse timestamp string to datetime object
        try:
            post_datetime = datetime.strptime(reply_timestamp, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise ValueError("reply_timestamp must be in yyyy-mm-dd HH:MM:SS format")

        # Get discussion_message table
        discussion_message_table = getattr(db, "discussion_message", None)
        if discussion_message_table is None:
            raise KeyError("discussion_message table does not exist in database")

        # Verify parent message exists
        parent_message = discussion_message_table.get(parent_message_id)
        if parent_message is None:
            raise KeyError(f"Parent message with ID '{parent_message_id}' does not exist")

        # Get enrollment table
        enrollment_table = getattr(db, "enrollment", None)
        if enrollment_table is None:
            raise KeyError("enrollment table does not exist in database")

        # Verify enrollment exists and has access
        enrollment = enrollment_table.get(enrollment_id)
        if enrollment is None:
            raise KeyError(f"Enrollment with ID '{enrollment_id}' does not exist")

        # Check if enrollment is active (replier must have access)
        if enrollment.status not in ["active", "completed"]:
            raise ValueError(f"Enrollment '{enrollment_id}' does not have active access (status: {enrollment.status})")

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

        # Get discussion topic from parent message
        # If parent message itself is a reply, trace back to find the original topic
        discussion_topic = parent_message.discussion_topic
        if discussion_topic is None and parent_message.parent_message_id is not None:
            # Trace back to find the root message with a topic
            current_msg = parent_message
            while current_msg.parent_message_id is not None and current_msg.discussion_topic is None:
                current_msg = discussion_message_table.get(current_msg.parent_message_id)
                if current_msg is None:
                    break
            if current_msg and current_msg.discussion_topic:
                discussion_topic = current_msg.discussion_topic

        # Create new reply message
        new_reply = DiscussionMessage(
            message_id=reply_id,
            enrollment_id=enrollment_id,
            parent_message_id=parent_message_id,
            discussion_topic=discussion_topic,
            message_content=reply_content,
            post_timestamp=post_datetime
        )

        # Add reply to discussion_message table
        discussion_message_table[reply_id] = new_reply
        setattr(db, "discussion_message", discussion_message_table)

        # Return reply_id
        return {
            "reply_id": reply_id
        }

    @is_tool()
    def create_assessment(
        self,
        course_id: str,
        assessment_title: str,
        assessment_type: Literal["quiz", "exam", "assignment", "project", "peer_review"],
        total_points: int,
        time_limit_minutes: Optional[int] = None,
        passing_score: Optional[int] = None
    ) -> dict:
        """
        Create a new assessment (quiz, exam, assignment) for a course with questions and grading criteria.

        This method:
        1. Validates that the course exists in the system
        2. Validates input parameters (assessment_type enum, total_points, passing_score)
        3. Generates a unique assessment_id
        4. Creates a new Assessment object with the provided information
        5. Stores the assessment in the database
        6. Returns the generated assessment_id
        """
        # Import necessary modules for id generation
        import secrets
        import hashlib

        # Get database instance
        db = self.db

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

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

        # Pre-condition: Verify that the course exists in the system
        course_table = getattr(db, "course", None)
        if course_table is None:
            raise ValueError("Course table does not exist in the database")

        # Check if the specific course exists
        if course_id not in course_table:
            raise ValueError(f"Course with id '{course_id}' does not exist in the system")

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

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

        # Validate passing_score if provided
        if passing_score is not None:
            if not isinstance(passing_score, int):
                raise ValueError("passing_score must be an integer if provided")
            if passing_score < 0:
                raise ValueError("passing_score cannot be negative")
            if passing_score > total_points:
                raise ValueError("passing_score cannot exceed total_points")

        # Validate assessment_type against enum values
        # Note: Literal type annotation already provides compile-time checking,
        # but we add runtime validation for robustness
        valid_types = ["quiz", "exam", "assignment", "project", "peer_review"]
        if assessment_type not in valid_types:
            raise ValueError(
                f"Invalid assessment_type '{assessment_type}'. "
                f"Must be one of: {', '.join(valid_types)}"
            )

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

        # Get or initialize assessment table
        assessment_table = getattr(db, "assessment", None)
        if assessment_table is None:
            assessment_table = {}

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

        # Create new Assessment object with provided parameters
        new_assessment = Assessment(
            assessment_id=assessment_id,
            course_id=course_id,
            assessment_title=assessment_title,
            assessment_type=assessment_type,
            total_points=total_points,
            time_limit_minutes=time_limit_minutes,
            passing_score=passing_score,
            deadline=None  # Optional field, set to None by default
        )

        # Add the new assessment to the assessment table
        assessment_table[assessment_id] = new_assessment

        # Update the database with the modified assessment table
        setattr(db, "assessment", assessment_table)

        # Return the generated assessment_id
        return {
            "assessment_id": assessment_id
        }

    @is_tool()
    def get_discussion_thread(self, message_id: str):
        """
        Retrieve all messages in a discussion thread including the original post and all replies.

        Args:
            message_id: Unique identifier of the root message in the thread

        Returns:
            dict: Complete discussion thread with all messages in chronological order

        Raises:
            KeyError: If message_id not found or discussion_message table doesn't exist
        """
        # Get database instance
        db = self.db

        # Retrieve discussion_message table from database
        discussion_message_table = getattr(db, "discussion_message", None)

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

        # Check if the root message exists in the table
        if message_id not in discussion_message_table:
            raise KeyError(f"Message with message_id '{message_id}' not found in discussion_message table")

        # Get the root message
        root_message = discussion_message_table[message_id]

        # Build a mapping of all messages by message_id for efficient lookup
        all_messages = {msg_id: msg for msg_id, msg in discussion_message_table.items()}

        # Helper function to collect all messages in the thread
        def collect_thread_messages(msg_id: str, collected: set) -> list:
            """
            Collects all messages in a thread starting from the given message_id.
            Returns a list of message dictionaries in chronological order.
            """
            messages = []

            # Get the current message
            message = all_messages.get(msg_id)
            if message is None or msg_id in collected:
                return messages

            # Mark as collected to avoid infinite loops
            collected.add(msg_id)

            # Add current message
            message_dict = {
                "message_id": message.message_id,
                "enrollment_id": message.enrollment_id,
                "parent_message_id": message.parent_message_id,
                "discussion_topic": message.discussion_topic,
                "message_content": message.message_content,
                "post_timestamp": message.post_timestamp.strftime("%Y-%m-%d %H:%M:%S"),
                "replies": []
            }
            messages.append(message_dict)

            # Find all direct replies to this message
            replies = []
            for other_msg_id, other_msg in all_messages.items():
                if other_msg.parent_message_id == msg_id and other_msg_id not in collected:
                    # Recursively collect replies
                    reply_messages = collect_thread_messages(other_msg_id, collected)
                    replies.extend(reply_messages)

            # Sort replies by post_timestamp and add to messages list
            replies.sort(key=lambda x: x["post_timestamp"])
            messages.extend(replies)

            return messages

        # Collect all messages in the thread starting from root
        collected_ids = set()
        thread_list = collect_thread_messages(message_id, collected_ids)

        # Return the complete thread in the required format
        return {
            "thread": thread_list
        }

    @is_tool()
    def enroll_in_course(self, learner_id: str, course_id: str, enrollment_date: str):
        """
        Enroll a learner in a specific course, creating a new enrollment record with initial status.

        Args:
            learner_id: Unique identifier of the learner
            course_id: Unique identifier of the course
            enrollment_date: Date of enrollment in yyyy-mm-dd HH:MM:SS format

        Returns:
            dict: Contains enrollment_id and status

        Raises:
            ValueError: If learner doesn't exist, course doesn't exist, or learner is already enrolled
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Access the database
        db = self.db

        # Validate and parse enrollment_date format
        try:
            parsed_enrollment_date = datetime.strptime(enrollment_date, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise ValueError(f"Invalid enrollment_date format. Expected 'yyyy-mm-dd HH:MM:SS', got '{enrollment_date}'")

        # Get learner table - verify learner exists
        learner_table = getattr(db, "learner", None)
        if learner_table is None or learner_id not in learner_table:
            raise ValueError(f"Learner with ID '{learner_id}' does not exist")

        # Get course table - verify course exists
        course_table = getattr(db, "course", None)
        if course_table is None or course_id not in course_table:
            raise ValueError(f"Course with ID '{course_id}' does not exist")

        # Get enrollment table
        enrollment_table = getattr(db, "enrollment", None)
        if enrollment_table is None:
            enrollment_table = {}

        # Check if learner is already enrolled in this course
        # Search through all enrollments to find any existing enrollment for this learner-course pair
        for enrollment_id, enrollment in enrollment_table.items():
            if enrollment.learner_id == learner_id and enrollment.course_id == course_id:
                raise ValueError(f"Learner '{learner_id}' is already enrolled in course '{course_id}'")

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

        # Ensure the generated ID is unique (collision check)
        while enrollment_id in enrollment_table:
            enrollment_id = "E" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Create new enrollment record with initial active status
        new_enrollment = Enrollment(
            enrollment_id=enrollment_id,
            learner_id=learner_id,
            course_id=course_id,
            enrollment_date=parsed_enrollment_date,
            status="active",  # Initial status is active
            progress_percentage=0.00,  # Start with 0% progress
            final_grade=None,  # No grade initially
            last_updated=parsed_enrollment_date  # Set last_updated to enrollment date
        )

        # Add the new enrollment to the enrollment table
        enrollment_table[enrollment_id] = new_enrollment

        # Save the updated enrollment table back to database
        setattr(db, "enrollment", enrollment_table)

        # Return enrollment_id and status
        return {
            "enrollment_id": enrollment_id,
            "status": "active"
        }

    @is_tool()
    def submit_assessment_attempt(
        self,
        enrollment_id: str,
        assessment_id: str,
        submission_timestamp: str,
        answers: str,
        time_spent_minutes: Optional[int] = None
    ) -> dict:
        """
        Submit a learner's attempt at an assessment with their answers for grading.

        This method creates a new assessment attempt record in the database after validating
        that the enrollment and assessment exist and that the learner has access to the assessment.

        Args:
            enrollment_id: Unique identifier of the enrollment
            assessment_id: Unique identifier of the assessment
            submission_timestamp: Timestamp of submission in yyyy-mm-dd HH:MM:SS format
            answers: JSON string containing the learner's answers
            time_spent_minutes: Optional time spent on the assessment in minutes

        Returns:
            dict: Dictionary containing the unique attempt_id

        Raises:
            ValueError: If enrollment or assessment doesn't exist, or if learner doesn't have access
        """
        from datetime import datetime
        import secrets
        import hashlib
        import json

        # Access the database
        db = self.db

        # Validate that enrollment exists
        enrollment_table = getattr(db, "enrollment", None)
        if enrollment_table is None or enrollment_id not in enrollment_table:
            raise ValueError(f"Enrollment with ID '{enrollment_id}' does not exist")

        enrollment = enrollment_table[enrollment_id]

        # Validate that assessment exists
        assessment_table = getattr(db, "assessment", None)
        if assessment_table is None or assessment_id not in assessment_table:
            raise ValueError(f"Assessment with ID '{assessment_id}' does not exist")

        assessment = assessment_table[assessment_id]

        # Verify learner has access to the assessment by checking if enrollment's course matches assessment's course
        if enrollment.course_id != assessment.course_id:
            raise ValueError(
                f"Learner does not have access to assessment '{assessment_id}'. "
                f"Enrollment course '{enrollment.course_id}' does not match assessment course '{assessment.course_id}'"
            )

        # Verify enrollment status allows submission (must be active)
        if enrollment.status != "active":
            raise ValueError(
                f"Cannot submit assessment attempt. Enrollment status is '{enrollment.status}', must be 'active'"
            )

        # Parse and validate submission timestamp
        try:
            submission_dt = datetime.strptime(submission_timestamp, "%Y-%m-%d %H:%M:%S")
        except ValueError as e:
            raise ValueError(
                f"Invalid submission_timestamp format. Expected 'yyyy-mm-dd HH:MM:SS', got '{submission_timestamp}'"
            ) from e

        # Validate answers is valid JSON string
        try:
            json.loads(answers)
        except json.JSONDecodeError as e:
            raise ValueError(f"Invalid JSON format for answers: {str(e)}") from e

        # Validate time_spent_minutes if provided
        if time_spent_minutes is not None and time_spent_minutes < 0:
            raise ValueError(f"time_spent_minutes must be non-negative, got {time_spent_minutes}")

        # Check if assessment has a time limit and validate time spent doesn't exceed it
        if (time_spent_minutes is not None and 
            assessment.time_limit_minutes is not None and 
            time_spent_minutes > assessment.time_limit_minutes):
            raise ValueError(
                f"Time spent ({time_spent_minutes} minutes) exceeds assessment time limit "
                f"({assessment.time_limit_minutes} minutes)"
            )

        # Check if submission is before deadline (if deadline exists)
        if assessment.deadline is not None and submission_dt > assessment.deadline:
            raise ValueError(
                f"Submission timestamp ({submission_timestamp}) is after assessment deadline "
                f"({assessment.deadline.strftime('%Y-%m-%d %H:%M:%S')})"
            )

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

        # Create new AssessmentAttempt instance
        new_attempt = AssessmentAttempt(
            attempt_id=attempt_id,
            enrollment_id=enrollment_id,
            assessment_id=assessment_id,
            submission_timestamp=submission_dt,
            time_spent_minutes=time_spent_minutes,
            answers=answers,
            score=None,  # Not graded yet
            feedback=None,  # No feedback yet
            graded_by=None,  # Not graded yet
            grading_timestamp=None  # Not graded yet
        )

        # Get assessment_attempt table (create if doesn't exist)
        assessment_attempt_table = getattr(db, "assessment_attempt", None)
        if assessment_attempt_table is None:
            assessment_attempt_table = {}

        # Add new attempt to the table
        assessment_attempt_table[attempt_id] = new_attempt

        # Update the database
        setattr(db, "assessment_attempt", assessment_attempt_table)

        # Return the attempt_id
        return {"attempt_id": attempt_id}

    @is_tool()
    def track_video_progress(
        self,
        enrollment_id: str,
        video_id: str,
        watched_seconds: int,
        video_duration_seconds: int,
        tracking_timestamp: str,
        last_position_seconds: Optional[int] = None
    ):
        """
        Track and record a learner's progress through a video lesson.

        This method creates a lesson completion record for video progress tracking,
        calculates completion percentage, and updates the enrollment's overall progress.
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Access the database
        db = self.db

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

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

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

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

        if watched_seconds > video_duration_seconds:
            raise ValueError("watched_seconds cannot exceed video_duration_seconds")

        # Validate and parse tracking timestamp
        try:
            tracking_dt = datetime.strptime(tracking_timestamp, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise ValueError("tracking_timestamp must be in 'yyyy-mm-dd HH:MM:SS' format")

        # Validate last_position_seconds if provided
        if last_position_seconds is not None:
            if not isinstance(last_position_seconds, int) or last_position_seconds < 0:
                raise ValueError("last_position_seconds must be a non-negative integer")
            if last_position_seconds > video_duration_seconds:
                raise ValueError("last_position_seconds cannot exceed video_duration_seconds")

        # Check if enrollment exists
        enrollment_table = getattr(db, "enrollment", None)
        if enrollment_table is None:
            raise ValueError("Enrollment table not found in database")

        enrollment = enrollment_table.get(enrollment_id)
        if enrollment is None:
            raise ValueError(f"Enrollment with id '{enrollment_id}' does not exist")

        # Check if enrollment is active (learners can only track progress for active enrollments)
        if enrollment.status not in ["active", "completed"]:
            raise ValueError(f"Cannot track progress for enrollment with status '{enrollment.status}'")

        # Calculate completion percentage
        # Completion percentage = (watched_seconds / video_duration_seconds) * 100
        completion_percentage = round((watched_seconds / video_duration_seconds) * 100, 2)

        # Calculate time spent in minutes (rounded up)
        time_spent_minutes = (watched_seconds + 59) // 60  # Round up to nearest minute

        # Generate unique progress_id (which serves as completion_id in lesson_completion table)
        prefix = "VP"
        progress_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Create lesson completion record
        # Note: In this system, video progress is tracked as lesson completion
        # The video_id serves as the lesson_id in the lesson_completion table
        new_completion = LessonCompletion(
            completion_id=progress_id,
            enrollment_id=enrollment_id,
            lesson_id=video_id,  # Video ID is used as lesson ID
            completion_timestamp=tracking_dt,
            time_spent_minutes=time_spent_minutes
        )

        # Get lesson_completion table and add the new record
        lesson_completion_table = getattr(db, "lesson_completion", None)
        if lesson_completion_table is None:
            # Initialize the table if it doesn't exist
            lesson_completion_table = {}

        # Add the new completion record to the table
        lesson_completion_table[progress_id] = new_completion
        setattr(db, "lesson_completion", lesson_completion_table)

        # Update enrollment's last_updated timestamp
        enrollment.last_updated = tracking_dt

        # Update enrollment table with modified enrollment
        enrollment_table[enrollment_id] = enrollment
        setattr(db, "enrollment", enrollment_table)

        # Return progress information
        return {
            "progress_id": progress_id,
            "completion_percentage": completion_percentage
        }
