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

class ProjectManagementTools(ToolKitBase):
    """All tools for project_management."""
    
    db: ProjectManagementDB
    
    def __init__(self, db: ProjectManagementDB):
        """Initialize tools with database."""
        super().__init__(db)
    
    @is_tool()
    def calculate_resource_cost(self, resource_rates: dict, allocated_hours: dict):
        """
        Calculate total cost for resources based on hourly rates and allocated hours.

        This method computes the cost for each resource by multiplying their hourly rate
        by their allocated hours, and also calculates the total cost across all resources.

        Args:
            resource_rates: Dictionary mapping resource names to their hourly rates in currency units
            allocated_hours: Dictionary mapping resource names to their allocated hours

        Returns:
            Dictionary containing:
                - resource_costs: Dictionary mapping resource names to their total costs
                - total_cost: Total cost across all resources in currency units

        Raises:
            ValueError: If input dictionaries are empty, contain invalid values, or have mismatched keys
        """
        # Validate input parameters are not None
        if resource_rates is None or allocated_hours is None:
            raise ValueError("Both resource_rates and allocated_hours must be provided and cannot be None")

        # Validate input dictionaries are not empty
        if not resource_rates:
            raise ValueError("resource_rates dictionary cannot be empty")
        if not allocated_hours:
            raise ValueError("allocated_hours dictionary cannot be empty")

        # Validate that both dictionaries have the same set of resource names
        rates_keys = set(resource_rates.keys())
        hours_keys = set(allocated_hours.keys())

        if rates_keys != hours_keys:
            missing_in_rates = hours_keys - rates_keys
            missing_in_hours = rates_keys - hours_keys
            error_msg = []
            if missing_in_rates:
                error_msg.append(f"Resources in allocated_hours but not in resource_rates: {missing_in_rates}")
            if missing_in_hours:
                error_msg.append(f"Resources in resource_rates but not in allocated_hours: {missing_in_hours}")
            raise ValueError("; ".join(error_msg))

        # Initialize result dictionary for storing individual resource costs
        resource_costs = {}

        # Calculate cost for each resource
        for resource_name in resource_rates:
            # Get hourly rate and allocated hours for current resource
            hourly_rate = resource_rates[resource_name]
            hours = allocated_hours[resource_name]

            # Validate that rate and hours are numeric values
            if not isinstance(hourly_rate, (int, float)):
                raise ValueError(f"Hourly rate for resource '{resource_name}' must be a number, got {type(hourly_rate).__name__}")
            if not isinstance(hours, (int, float)):
                raise ValueError(f"Allocated hours for resource '{resource_name}' must be a number, got {type(hours).__name__}")

            # Validate that rate and hours are non-negative
            if hourly_rate < 0:
                raise ValueError(f"Hourly rate for resource '{resource_name}' cannot be negative: {hourly_rate}")
            if hours < 0:
                raise ValueError(f"Allocated hours for resource '{resource_name}' cannot be negative: {hours}")

            # Calculate total cost for this resource (rate * hours)
            resource_cost = hourly_rate * hours
            resource_costs[resource_name] = resource_cost

        # Calculate total cost across all resources by summing individual costs
        total_cost = sum(resource_costs.values())

        # Return results as dictionary with resource_costs and total_cost
        return {
            "resource_costs": resource_costs,
            "total_cost": total_cost
        }

    @is_tool()
    def assess_change_impact(
        self,
        change_description: str,
        estimated_hours: float,
        estimated_cost: float,
        schedule_impact_days: int,
        affected_deliverables: List[str]
    ) -> dict:
        """
        Assess the impact of proposed changes on project scope, schedule, cost, quality, and risks.

        This method performs a comprehensive impact analysis based on the provided change parameters
        and returns an assessment with impact level, detailed summary, and recommendation.

        Args:
            change_description: Description of the proposed change
            estimated_hours: Estimated effort required in hours
            estimated_cost: Estimated cost impact in currency units
            schedule_impact_days: Estimated schedule delay in days
            affected_deliverables: List of deliverables affected by the change

        Returns:
            dict: Contains impact_level, impact_summary, and recommendation

        Raises:
            ValueError: If input parameters are invalid
        """

        # Input validation
        if not change_description or not change_description.strip():
            raise ValueError("change_description cannot be empty")

        if estimated_hours < 0:
            raise ValueError("estimated_hours must be non-negative")

        if estimated_cost < 0:
            raise ValueError("estimated_cost must be non-negative")

        if schedule_impact_days < 0:
            raise ValueError("schedule_impact_days must be non-negative")

        if not isinstance(affected_deliverables, list) or len(affected_deliverables) == 0:
            raise ValueError("affected_deliverables must be a non-empty list")

        # Initialize impact scores (0-100 scale for easier calculation)
        scope_score = 0
        schedule_score = 0
        cost_score = 0
        quality_score = 0

        # Analyze scope impact based on affected deliverables count
        # More affected deliverables = higher scope impact
        deliverable_count = len(affected_deliverables)
        if deliverable_count >= 5:
            scope_impact = "major_addition"
            scope_score = 90
        elif deliverable_count >= 3:
            scope_impact = "moderate_addition"
            scope_score = 60
        elif deliverable_count >= 1:
            scope_impact = "minor_addition"
            scope_score = 30
        else:
            scope_impact = "no_change"
            scope_score = 0

        # Analyze schedule impact based on delay days
        # Thresholds: <5 days = low, 5-15 days = medium, 15-30 days = high, >30 days = critical
        if schedule_impact_days == 0:
            schedule_impact = "no_delay"
            schedule_score = 0
        elif schedule_impact_days < 5:
            schedule_impact = f"{schedule_impact_days}_days_delay"
            schedule_score = 25
        elif schedule_impact_days < 15:
            schedule_impact = f"{schedule_impact_days}_days_delay"
            schedule_score = 50
        elif schedule_impact_days < 30:
            schedule_impact = f"{schedule_impact_days}_days_delay"
            schedule_score = 75
        else:
            schedule_impact = f"{schedule_impact_days}_days_delay"
            schedule_score = 95

        # Analyze cost impact
        # Thresholds: <10k = low, 10k-50k = medium, 50k-100k = high, >100k = critical
        cost_impact = estimated_cost
        if estimated_cost < 10000:
            cost_score = 20
        elif estimated_cost < 50000:
            cost_score = 50
        elif estimated_cost < 100000:
            cost_score = 75
        else:
            cost_score = 95

        # Analyze quality impact based on effort hours
        # Higher effort hours typically require more testing and quality assurance
        # Thresholds: <50 hours = minimal, 50-200 hours = moderate, 200-500 hours = significant, >500 hours = extensive
        if estimated_hours < 50:
            quality_impact = "minimal_testing_required"
            quality_score = 20
        elif estimated_hours < 200:
            quality_impact = "moderate_testing_required"
            quality_score = 45
        elif estimated_hours < 500:
            quality_impact = "additional_testing_required"
            quality_score = 70
        else:
            quality_impact = "extensive_testing_required"
            quality_score = 90

        # Calculate overall impact score (weighted average)
        # Weights: scope=25%, schedule=30%, cost=30%, quality=15%
        overall_score = (
            scope_score * 0.25 +
            schedule_score * 0.30 +
            cost_score * 0.30 +
            quality_score * 0.15
        )

        # Determine overall impact level based on composite score
        # Thresholds: <30 = low, 30-55 = medium, 55-80 = high, >80 = critical
        if overall_score < 30:
            impact_level = "low"
        elif overall_score < 55:
            impact_level = "medium"
        elif overall_score < 80:
            impact_level = "high"
        else:
            impact_level = "critical"

        # Generate recommendation based on impact level and specific factors
        # Decision logic:
        # - Critical impact or very high cost (>150k) -> reject
        # - High impact with significant schedule delay (>25 days) -> defer
        # - High impact but manageable -> modify (request scope reduction)
        # - Medium or low impact -> approve
        if impact_level == "critical" or estimated_cost > 150000:
            recommendation = "reject"
        elif impact_level == "high" and schedule_impact_days > 25:
            recommendation = "defer"
        elif impact_level == "high":
            recommendation = "modify"
        else:
            recommendation = "approve"

        # Construct detailed impact summary
        impact_summary = {
            "scope": scope_impact,
            "schedule": schedule_impact,
            "cost": cost_impact,
            "quality": quality_impact
        }

        # Return comprehensive assessment result
        return {
            "impact_level": impact_level,
            "impact_summary": impact_summary,
            "recommendation": recommendation
        }

    @is_tool()
    def calculate_project_duration(self, task_durations: dict, dependencies: list):
        """
        Calculate the total duration of a project based on task durations and dependencies 
        using critical path method (CPM).

        Args:
            task_durations: Dictionary mapping task names to their durations in days
            dependencies: List of task dependencies where each dependency is [predecessor, successor]

        Returns:
            Dictionary containing:
            - total_duration: Total project duration in days
            - critical_path: List of tasks on the critical path

        Raises:
            ValueError: If task durations or dependencies are invalid
        """
        # Validate input parameters
        if not task_durations:
            raise ValueError("Task durations cannot be empty")

        if not isinstance(task_durations, dict):
            raise ValueError("Task durations must be a dictionary")

        if not isinstance(dependencies, list):
            raise ValueError("Dependencies must be a list")

        # Validate all task durations are positive numbers
        for task, duration in task_durations.items():
            if not isinstance(duration, (int, float)) or duration < 0:
                raise ValueError(f"Task duration for '{task}' must be a non-negative number")

        # Validate dependencies format and that all tasks in dependencies exist in task_durations
        for dep in dependencies:
            if not isinstance(dep, list) or len(dep) != 2:
                raise ValueError("Each dependency must be a list of [predecessor, successor]")
            predecessor, successor = dep
            if predecessor not in task_durations:
                raise ValueError(f"Predecessor task '{predecessor}' not found in task_durations")
            if successor not in task_durations:
                raise ValueError(f"Successor task '{successor}' not found in task_durations")

        # Build adjacency list for the task dependency graph
        # predecessors[task] = list of tasks that must complete before this task
        # successors[task] = list of tasks that depend on this task
        predecessors = {task: [] for task in task_durations}
        successors = {task: [] for task in task_durations}

        for predecessor, successor in dependencies:
            predecessors[successor].append(predecessor)
            successors[predecessor].append(successor)

        # Check for circular dependencies using topological sort
        in_degree = {task: len(predecessors[task]) for task in task_durations}
        queue = [task for task in task_durations if in_degree[task] == 0]
        sorted_tasks = []

        while queue:
            current = queue.pop(0)
            sorted_tasks.append(current)
            for successor in successors[current]:
                in_degree[successor] -= 1
                if in_degree[successor] == 0:
                    queue.append(successor)

        if len(sorted_tasks) != len(task_durations):
            raise ValueError("Circular dependency detected in task dependencies")

        # Calculate earliest start time (ES) and earliest finish time (EF) for each task
        # ES[task] = maximum EF of all predecessors (or 0 if no predecessors)
        # EF[task] = ES[task] + duration[task]
        earliest_start = {task: 0 for task in task_durations}
        earliest_finish = {task: 0 for task in task_durations}

        for task in sorted_tasks:
            if predecessors[task]:
                earliest_start[task] = max(earliest_finish[pred] for pred in predecessors[task])
            earliest_finish[task] = earliest_start[task] + task_durations[task]

        # Total project duration is the maximum earliest finish time
        total_duration = max(earliest_finish.values()) if earliest_finish else 0

        # Calculate latest start time (LS) and latest finish time (LF) for each task
        # Working backwards from the end
        # LF[task] = minimum LS of all successors (or total_duration if no successors)
        # LS[task] = LF[task] - duration[task]
        latest_finish = {task: total_duration for task in task_durations}
        latest_start = {task: total_duration for task in task_durations}

        for task in reversed(sorted_tasks):
            if successors[task]:
                latest_finish[task] = min(latest_start[succ] for succ in successors[task])
            latest_start[task] = latest_finish[task] - task_durations[task]

        # Calculate slack/float for each task
        # Slack = LS - ES (or LF - EF, which gives the same result)
        # Tasks with zero slack are on the critical path
        slack = {task: latest_start[task] - earliest_start[task] for task in task_durations}

        # Identify critical tasks (tasks with zero slack)
        critical_tasks = {task for task in task_durations if abs(slack[task]) < 1e-9}

        # Build the critical path by following critical tasks from start to end
        # Find starting tasks (tasks with no predecessors that are critical)
        critical_path = []

        if critical_tasks:
            # Find the critical task(s) with no critical predecessors
            start_candidates = [
                task for task in critical_tasks 
                if not any(pred in critical_tasks for pred in predecessors[task])
            ]

            if start_candidates:
                # Start with the task that has the earliest start time
                current = min(start_candidates, key=lambda t: earliest_start[t])
                visited = set()

                # Follow the critical path through successors
                while current and current not in visited:
                    critical_path.append(current)
                    visited.add(current)

                    # Find the next critical successor
                    critical_successors = [
                        succ for succ in successors[current] 
                        if succ in critical_tasks and succ not in visited
                    ]

                    if critical_successors:
                        # Choose the successor with the earliest start time
                        current = min(critical_successors, key=lambda t: earliest_start[t])
                    else:
                        current = None

        return {
            "total_duration": int(total_duration),
            "critical_path": critical_path
        }

    @is_tool()
    def calculate_quality_metrics(
        self,
        total_defects_found: int,
        defects_found_before_release: int,
        project_size: int,
        customer_satisfaction_ratings: list
    ) -> dict:
        """
        Calculate quality metrics including defect density, defect removal efficiency, 
        and customer satisfaction scores.

        Args:
            total_defects_found: Total number of defects found during project
            defects_found_before_release: Number of defects found before release
            project_size: Project size in lines of code or function points
            customer_satisfaction_ratings: List of customer satisfaction ratings on scale of 1-5

        Returns:
            Dictionary containing:
            - defect_density: Defects per thousand lines of code or function points
            - defect_removal_efficiency: Percentage of defects found before release
            - average_customer_satisfaction: Average customer satisfaction score

        Raises:
            ValueError: If input parameters are invalid
        """

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

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

        # Validate that defects_found_before_release cannot exceed total_defects_found
        if defects_found_before_release > total_defects_found:
            raise ValueError(
                "defects_found_before_release cannot exceed total_defects_found"
            )

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

        # Validate customer_satisfaction_ratings
        if not isinstance(customer_satisfaction_ratings, list):
            raise ValueError("customer_satisfaction_ratings must be a list")

        if len(customer_satisfaction_ratings) == 0:
            raise ValueError("customer_satisfaction_ratings cannot be empty")

        # Validate each rating is an integer between 1 and 5
        for rating in customer_satisfaction_ratings:
            if not isinstance(rating, int):
                raise ValueError(
                    "All customer satisfaction ratings must be integers"
                )
            if rating < 1 or rating > 5:
                raise ValueError(
                    "All customer satisfaction ratings must be between 1 and 5"
                )

        # Calculate defect density (defects per thousand lines of code or function points)
        # Formula: (total_defects_found / project_size) * 1000
        defect_density = (total_defects_found / project_size) * 1000

        # Calculate defect removal efficiency (percentage of defects found before release)
        # Formula: (defects_found_before_release / total_defects_found) * 100
        # Handle edge case where total_defects_found is 0
        if total_defects_found == 0:
            defect_removal_efficiency = 100.0  # If no defects found, efficiency is 100%
        else:
            defect_removal_efficiency = (
                defects_found_before_release / total_defects_found
            ) * 100

        # Calculate average customer satisfaction score
        # Formula: sum of all ratings / number of ratings
        average_customer_satisfaction = sum(customer_satisfaction_ratings) / len(
            customer_satisfaction_ratings
        )

        # Round results to 2 decimal places for better readability
        defect_density = round(defect_density, 2)
        defect_removal_efficiency = round(defect_removal_efficiency, 2)
        average_customer_satisfaction = round(average_customer_satisfaction, 2)

        # Return the calculated metrics as a dictionary
        return {
            "defect_density": defect_density,
            "defect_removal_efficiency": defect_removal_efficiency,
            "average_customer_satisfaction": average_customer_satisfaction
        }

    @is_tool()
    def list_project_tasks(self, project_id: str, status: Literal["not_started", "in_progress", "blocked", "completed", "cancelled", "all"] = "all", assigned_to: Optional[str] = None):
        """
        Retrieve list of all tasks belonging to a specific project with optional filtering

        Args:
            project_id: Unique identifier of the project
            status: Filter tasks by status (default: "all" to retrieve all tasks)
            assigned_to: Filter tasks by assignee name or ID (optional)

        Returns:
            dict: Contains 'tasks' (list of task summary objects) and 'total_count' (total number of matching tasks)

        Raises:
            KeyError: If the project does not exist in the system
        """
        from thefuzz import fuzz, process

        # Access the database
        db = self.db

        # Get project and task tables from database
        project_table = getattr(db, "project", None)
        task_table = getattr(db, "task", None)

        # Validate that project table exists and is not empty
        if project_table is None or len(project_table) == 0:
            raise KeyError(f"Project table does not exist or is empty")

        # Validate that the specified project exists (pre-condition check)
        if project_id not in project_table:
            raise KeyError(f"Project with ID '{project_id}' does not exist in the system")

        # Initialize result list
        matching_tasks = []

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

        # Iterate through all tasks and filter based on criteria
        for task_id, task in task_table.items():
            # Filter by project_id (exact match required for ID fields)
            if task.project_id != project_id:
                continue

            # Filter by status if not "all"
            # Status parameter uses enum constraint, safe to use direct comparison
            if status != "all" and task.status != status:
                continue

            # Filter by assigned_to if provided (use fuzzy matching for natural language text)
            if assigned_to is not None:
                # Use fuzzy matching to handle variations in assignee names
                # Set threshold to 80 for reasonable matching tolerance
                similarity_score = fuzz.ratio(assigned_to.lower(), task.assigned_to.lower())
                if similarity_score < 80:
                    continue

            # Create task summary object with essential information
            task_summary = {
                "task_id": task.task_id,
                "task_name": task.task_name,
                "status": task.status
            }

            matching_tasks.append(task_summary)

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

        return result

    @is_tool()
    def calculate_return_on_investment(self, total_costs: float, annual_benefits: list, discount_rate: float):
        """
        Calculate project ROI based on project costs and expected benefits over time.

        This method computes three key financial metrics:
        1. ROI Percentage: Overall return on investment as a percentage
        2. Net Present Value (NPV): Present value of future benefits minus initial costs
        3. Payback Period: Time required to recover the initial investment

        Args:
            total_costs: Total project investment cost in currency units
            annual_benefits: List of expected annual benefits for each year
            discount_rate: Annual discount rate as decimal (e.g., 0.1 for 10%)

        Returns:
            dict: Contains roi_percentage, net_present_value, and payback_period_years

        Raises:
            ValueError: If inputs are invalid (negative costs, empty benefits, invalid discount rate)
            TypeError: If inputs are not of expected types
        """
        # Input validation
        if not isinstance(total_costs, (int, float)):
            raise TypeError("total_costs must be a number")

        if not isinstance(annual_benefits, list):
            raise TypeError("annual_benefits must be a list")

        if not isinstance(discount_rate, (int, float)):
            raise TypeError("discount_rate must be a number")

        if total_costs <= 0:
            raise ValueError("total_costs must be positive")

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

        if any(not isinstance(benefit, (int, float)) for benefit in annual_benefits):
            raise TypeError("All annual_benefits must be numbers")

        if any(benefit < 0 for benefit in annual_benefits):
            raise ValueError("annual_benefits cannot contain negative values")

        if discount_rate < 0:
            raise ValueError("discount_rate cannot be negative")

        # Calculate Net Present Value (NPV)
        # NPV = Sum of (Annual Benefit / (1 + discount_rate)^year) - Initial Investment
        net_present_value = 0.0
        for year, benefit in enumerate(annual_benefits, start=1):
            # Discount each year's benefit to present value
            discounted_benefit = benefit / ((1 + discount_rate) ** year)
            net_present_value += discounted_benefit

        # Subtract initial investment to get NPV
        net_present_value -= total_costs

        # Calculate ROI Percentage
        # ROI % = ((Total Benefits - Total Costs) / Total Costs) * 100
        total_benefits = sum(annual_benefits)
        roi_percentage = ((total_benefits - total_costs) / total_costs) * 100

        # Calculate Payback Period (in years)
        # Payback period is when cumulative benefits equal or exceed initial investment
        cumulative_benefit = 0.0
        payback_period_years = 0.0

        for year, benefit in enumerate(annual_benefits, start=1):
            if cumulative_benefit >= total_costs:
                # Already recovered investment in previous years
                break

            if cumulative_benefit + benefit >= total_costs:
                # Investment will be recovered during this year
                # Calculate fractional year needed
                remaining_to_recover = total_costs - cumulative_benefit
                fraction_of_year = remaining_to_recover / benefit if benefit > 0 else 0
                payback_period_years = year - 1 + fraction_of_year
                break
            else:
                # Haven't recovered yet, add this year's benefit
                cumulative_benefit += benefit
        else:
            # If loop completes without break, investment not recovered within given period
            # Set payback period to total years (indicating it extends beyond available data)
            payback_period_years = len(annual_benefits)

            # If still not recovered after all years, we might want to extrapolate
            # but for safety, we'll just indicate it's beyond the projection period
            if cumulative_benefit < total_costs:
                # Investment not recovered within the projection period
                # Calculate approximate years needed assuming last year's benefit continues
                if len(annual_benefits) > 0 and annual_benefits[-1] > 0:
                    remaining = total_costs - cumulative_benefit
                    additional_years = remaining / annual_benefits[-1]
                    payback_period_years = len(annual_benefits) + additional_years
                else:
                    # Cannot determine payback period (no benefits or zero benefits)
                    payback_period_years = float('inf')

        # Round results to reasonable precision
        roi_percentage = round(roi_percentage, 2)
        net_present_value = round(net_present_value, 2)
        payback_period_years = round(payback_period_years, 2) if payback_period_years != float('inf') else payback_period_years

        return {
            "roi_percentage": roi_percentage,
            "net_present_value": net_present_value,
            "payback_period_years": payback_period_years
        }

    @is_tool()
    def calculate_cost_benefit_ratio(
        self,
        total_costs: float,
        total_benefits: float,
        discount_rate: float = None,
        analysis_period_years: int = None
    ) -> dict:
        """
        Calculate cost-benefit ratio and benefit-cost ratio for project investment analysis.

        This method computes the benefit-cost ratio (BCR), net benefit, and provides an
        investment recommendation based on the calculated ratios. If discount rate and
        analysis period are provided, it can perform NPV-based calculations.

        Args:
            total_costs: Total project costs in currency units (must be positive)
            total_benefits: Total expected benefits in currency units (must be positive)
            discount_rate: Optional annual discount rate as decimal for NPV calculation (e.g., 0.1 for 10%)
            analysis_period_years: Optional number of years for benefit realization

        Returns:
            dict containing:
                - benefit_cost_ratio: Benefit-cost ratio (benefits divided by costs)
                - net_benefit: Net benefit amount (total_benefits - total_costs)
                - investment_decision: Investment recommendation ("proceed", "reconsider", or "reject")

        Raises:
            ValueError: If costs or benefits are negative, or if costs are zero
        """

        # Validate input parameters
        if total_costs <= 0:
            raise ValueError("Total costs must be greater than zero")

        if total_benefits < 0:
            raise ValueError("Total benefits cannot be negative")

        if total_costs < 0:
            raise ValueError("Total costs cannot be negative")

        # Validate discount rate if provided
        if discount_rate is not None and (discount_rate < 0 or discount_rate > 1):
            raise ValueError("Discount rate must be between 0 and 1")

        # Validate analysis period if provided
        if analysis_period_years is not None and analysis_period_years <= 0:
            raise ValueError("Analysis period must be greater than zero")

        # Calculate present values if discount rate and period are provided
        if discount_rate is not None and analysis_period_years is not None:
            # Calculate present value of costs (assuming costs occur at the beginning)
            pv_costs = total_costs

            # Calculate present value of benefits (assuming benefits are distributed evenly over the period)
            # Using annuity formula: PV = PMT * [(1 - (1 + r)^-n) / r]
            annual_benefit = total_benefits / analysis_period_years
            if discount_rate == 0:
                # Special case: no discounting
                pv_benefits = total_benefits
            else:
                pv_benefits = annual_benefit * ((1 - (1 + discount_rate) ** -analysis_period_years) / discount_rate)

            # Use present values for calculation
            effective_costs = pv_costs
            effective_benefits = pv_benefits
        else:
            # Use nominal values if no discounting
            effective_costs = total_costs
            effective_benefits = total_benefits

        # Calculate benefit-cost ratio (BCR = Benefits / Costs)
        benefit_cost_ratio = effective_benefits / effective_costs

        # Calculate net benefit (Net Benefit = Benefits - Costs)
        net_benefit = effective_benefits - effective_costs

        # Determine investment decision based on BCR
        # BCR > 1.0: Benefits exceed costs -> Proceed
        # BCR between 0.8 and 1.0: Marginal case -> Reconsider
        # BCR < 0.8: Costs significantly exceed benefits -> Reject
        if benefit_cost_ratio >= 1.0:
            investment_decision = "proceed"
        elif benefit_cost_ratio >= 0.8:
            investment_decision = "reconsider"
        else:
            investment_decision = "reject"

        # Return the analysis results
        return {
            "benefit_cost_ratio": round(benefit_cost_ratio, 2),
            "net_benefit": round(net_benefit, 2),
            "investment_decision": investment_decision
        }

    @is_tool()
    def generate_project_charter(
        self,
        name: str,
        business_case: str,
        objectives: list,
        high_level_scope: str,
        key_stakeholders: list,
        success_criteria: list
    ) -> dict:
        """
        Generate a comprehensive project charter document with all key project information.

        This method creates a structured project charter that serves as the formal authorization
        document for a project, including objectives, scope, stakeholders, and success criteria.

        Args:
            name: Name of the project
            business_case: Business justification for the project
            objectives: List of project objectives
            high_level_scope: High-level description of project scope
            key_stakeholders: List of key stakeholder names and roles
            success_criteria: List of measurable success criteria

        Returns:
            dict: A structured project charter document containing all sections

        Raises:
            ValueError: If any required parameter is empty or invalid
        """
        from datetime import datetime

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

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

        if not objectives or not isinstance(objectives, list) or len(objectives) == 0:
            raise ValueError("Objectives must be a non-empty list")

        # Validate each objective is a non-empty string
        for idx, obj in enumerate(objectives):
            if not isinstance(obj, str) or not obj.strip():
                raise ValueError(f"Objective at index {idx} must be a non-empty string")

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

        if not key_stakeholders or not isinstance(key_stakeholders, list) or len(key_stakeholders) == 0:
            raise ValueError("Key stakeholders must be a non-empty list")

        # Validate each stakeholder is a non-empty string
        for idx, stakeholder in enumerate(key_stakeholders):
            if not isinstance(stakeholder, str) or not stakeholder.strip():
                raise ValueError(f"Stakeholder at index {idx} must be a non-empty string")

        if not success_criteria or not isinstance(success_criteria, list) or len(success_criteria) == 0:
            raise ValueError("Success criteria must be a non-empty list")

        # Validate each success criterion is a non-empty string
        for idx, criterion in enumerate(success_criteria):
            if not isinstance(criterion, str) or not criterion.strip():
                raise ValueError(f"Success criterion at index {idx} must be a non-empty string")

        # Generate charter creation timestamp
        charter_creation_date = datetime.now().strftime("%Y-%m-%d %H:%M:%S")

        # Parse stakeholders into structured format (name and role)
        parsed_stakeholders = []
        for stakeholder in key_stakeholders:
            # Try to split stakeholder string by common delimiters (-, :, |)
            if ' - ' in stakeholder:
                parts = stakeholder.split(' - ', 1)
                parsed_stakeholders.append({
                    "name": parts[0].strip(),
                    "role": parts[1].strip() if len(parts) > 1 else "Not specified"
                })
            elif ': ' in stakeholder:
                parts = stakeholder.split(': ', 1)
                parsed_stakeholders.append({
                    "name": parts[0].strip(),
                    "role": parts[1].strip() if len(parts) > 1 else "Not specified"
                })
            elif ' | ' in stakeholder:
                parts = stakeholder.split(' | ', 1)
                parsed_stakeholders.append({
                    "name": parts[0].strip(),
                    "role": parts[1].strip() if len(parts) > 1 else "Not specified"
                })
            else:
                # If no delimiter found, treat entire string as name
                parsed_stakeholders.append({
                    "name": stakeholder.strip(),
                    "role": "Not specified"
                })

        # Build the comprehensive project charter document
        charter_document = {
            "project_name": name.strip(),
            "charter_creation_date": charter_creation_date,
            "sections": {
                # Executive Summary section
                "executive_summary": {
                    "title": "Executive Summary",
                    "content": f"This project charter formally authorizes the '{name.strip()}' project. {business_case.strip()}"
                },

                # Business Case section
                "business_case": {
                    "title": "Business Case",
                    "content": business_case.strip(),
                    "justification": "This project addresses critical business needs and provides strategic value to the organization."
                },

                # Project Objectives section
                "objectives": {
                    "title": "Project Objectives",
                    "description": "The following objectives define the intended outcomes of this project:",
                    "objective_list": [obj.strip() for obj in objectives]
                },

                # Scope section
                "scope": {
                    "title": "Project Scope",
                    "high_level_description": high_level_scope.strip(),
                    "in_scope": "All items defined in the high-level scope description",
                    "out_of_scope": "Any features, requirements, or deliverables not explicitly mentioned in the scope description"
                },

                # Stakeholders section
                "stakeholders": {
                    "title": "Key Stakeholders",
                    "description": "The following individuals and groups have significant interest or influence in this project:",
                    "stakeholder_list": parsed_stakeholders,
                    "total_count": len(parsed_stakeholders)
                },

                # Success Criteria section
                "success_criteria": {
                    "title": "Success Criteria",
                    "description": "The project will be considered successful when the following measurable criteria are met:",
                    "criteria_list": [criterion.strip() for criterion in success_criteria],
                    "measurement_approach": "Progress against these criteria will be tracked and reported regularly throughout the project lifecycle."
                },

                # High-level Timeline section (placeholder)
                "timeline": {
                    "title": "High-Level Timeline",
                    "description": "Detailed project timeline and milestones will be developed during the planning phase.",
                    "note": "This charter authorizes the project team to proceed with detailed planning activities."
                },

                # Budget section (placeholder)
                "budget": {
                    "title": "Budget Overview",
                    "description": "Detailed budget breakdown will be developed during the planning phase.",
                    "note": "Budget estimates and resource requirements will be refined as project planning progresses."
                },

                # Risks and Assumptions section
                "risks_and_assumptions": {
                    "title": "Initial Risks and Assumptions",
                    "description": "Key risks and assumptions will be identified and managed throughout the project.",
                    "note": "A comprehensive risk management plan will be developed during the planning phase."
                },

                # Authorization section
                "authorization": {
                    "title": "Project Authorization",
                    "authorization_statement": f"This charter formally authorizes the '{name.strip()}' project and empowers the project manager to apply organizational resources to project activities.",
                    "charter_date": charter_creation_date,
                    "approval_status": "Pending formal approval"
                }
            },

            # Metadata
            "metadata": {
                "charter_version": "1.0",
                "generated_date": charter_creation_date,
                "document_type": "Project Charter",
                "total_objectives": len(objectives),
                "total_stakeholders": len(parsed_stakeholders),
                "total_success_criteria": len(success_criteria)
            }
        }

        return {"charter_document": charter_document}

    @is_tool()
    def calculate_resource_utilization_rate(self, allocated_hours: dict, available_hours: dict):
        """
        Calculate the utilization rate of resources based on allocated hours versus available hours.

        This method computes:
        1. Individual utilization rates for each resource (allocated / available * 100)
        2. Average utilization rate across all resources

        Args:
            allocated_hours: Dictionary mapping resource names to their allocated hours
            available_hours: Dictionary mapping resource names to their available hours

        Returns:
            Dictionary containing:
            - utilization_rates: Dictionary mapping resource names to their utilization rates as percentages
            - average_utilization: Average utilization rate across all resources as percentage

        Raises:
            ZeroDivisionError: If any resource has zero available hours
            ValueError: If input dictionaries are empty or have mismatched keys
            TypeError: If allocated_hours or available_hours are not dictionaries
        """
        # Validate input types
        if not isinstance(allocated_hours, dict):
            raise TypeError("allocated_hours must be a dictionary")
        if not isinstance(available_hours, dict):
            raise TypeError("available_hours must be a dictionary")

        # Validate that both dictionaries are not empty
        if not allocated_hours:
            raise ValueError("allocated_hours dictionary cannot be empty")
        if not available_hours:
            raise ValueError("available_hours dictionary cannot be empty")

        # Get all resource names from both dictionaries
        allocated_resources = set(allocated_hours.keys())
        available_resources = set(available_hours.keys())

        # Find resources that exist in both dictionaries
        common_resources = allocated_resources.intersection(available_resources)

        # Validate that there are common resources to calculate
        if not common_resources:
            raise ValueError("No common resources found between allocated_hours and available_hours")

        # Initialize result dictionary
        utilization_rates = {}
        total_utilization = 0.0
        resource_count = 0

        # Calculate utilization rate for each common resource
        for resource_name in common_resources:
            allocated = allocated_hours[resource_name]
            available = available_hours[resource_name]

            # Validate that allocated and available hours are numeric
            if not isinstance(allocated, (int, float)):
                raise TypeError(f"allocated_hours for '{resource_name}' must be numeric")
            if not isinstance(available, (int, float)):
                raise TypeError(f"available_hours for '{resource_name}' must be numeric")

            # Check for zero available hours to prevent division by zero
            if available == 0:
                raise ZeroDivisionError(f"Resource '{resource_name}' has zero available hours, cannot calculate utilization rate")

            # Validate that hours are non-negative
            if allocated < 0:
                raise ValueError(f"allocated_hours for '{resource_name}' cannot be negative")
            if available < 0:
                raise ValueError(f"available_hours for '{resource_name}' cannot be negative")

            # Calculate utilization rate as percentage
            # Formula: (allocated_hours / available_hours) * 100
            utilization_rate = (allocated / available) * 100.0

            # Store the utilization rate for this resource
            utilization_rates[resource_name] = round(utilization_rate, 2)

            # Accumulate for average calculation
            total_utilization += utilization_rate
            resource_count += 1

        # Calculate average utilization rate across all resources
        average_utilization = round(total_utilization / resource_count, 2) if resource_count > 0 else 0.0

        # Return the results
        return {
            "utilization_rates": utilization_rates,
            "average_utilization": average_utilization
        }

    @is_tool()
    def calculate_project_risk_score(self, risks: list):
        """
        Calculate overall project risk score based on individual risk assessments.

        This method computes an aggregate risk score by:
        1. Validating each risk's probability (0-1) and impact (1-5) values
        2. Calculating individual risk scores (probability × impact)
        3. Computing overall risk score using weighted average approach
        4. Classifying the risk level based on the final score

        Args:
            risks: List of risk dictionaries, each containing:
                   - name (optional): Risk identifier/name
                   - probability: Float between 0 and 1 (0% to 100% chance)
                   - impact: Integer between 1 and 5 (severity scale)

        Returns:
            dict: Contains:
                - overall_risk_score: Float between 0 and 5 representing aggregate risk
                - risk_level: String classification ("low", "medium", "high", "critical")

        Raises:
            ValueError: If risks list is empty, or if any risk has invalid probability/impact values
        """
        # Validate input: risks list must not be empty
        if not risks or len(risks) == 0:
            raise ValueError("Risks list cannot be empty")

        # Validate each risk and collect individual risk scores
        individual_scores = []

        for idx, risk in enumerate(risks):
            # Validate risk is a dictionary
            if not isinstance(risk, dict):
                raise ValueError(f"Risk at index {idx} must be a dictionary")

            # Validate required fields exist
            if "probability" not in risk:
                raise ValueError(f"Risk at index {idx} missing required field 'probability'")
            if "impact" not in risk:
                raise ValueError(f"Risk at index {idx} missing required field 'impact'")

            probability = risk["probability"]
            impact = risk["impact"]

            # Validate probability: must be numeric and in range [0, 1]
            try:
                probability = float(probability)
            except (TypeError, ValueError):
                raise ValueError(f"Risk at index {idx} has invalid probability value: must be numeric")

            if not (0 <= probability <= 1):
                raise ValueError(f"Risk at index {idx} has probability {probability} outside valid range [0, 1]")

            # Validate impact: must be numeric and in range [1, 5]
            try:
                impact = float(impact)
            except (TypeError, ValueError):
                raise ValueError(f"Risk at index {idx} has invalid impact value: must be numeric")

            if not (1 <= impact <= 5):
                raise ValueError(f"Risk at index {idx} has impact {impact} outside valid range [1, 5]")

            # Calculate individual risk score: probability × impact
            # This gives a value between 0 (no risk) and 5 (maximum risk)
            risk_score = probability * impact
            individual_scores.append(risk_score)

        # Calculate overall risk score using weighted average
        # We use the maximum score as a conservative approach, but weight it with the average
        # to account for multiple risks. This balances between:
        # - Single high-severity risk scenarios
        # - Multiple moderate-risk scenarios
        max_score = max(individual_scores)
        avg_score = sum(individual_scores) / len(individual_scores)

        # Overall score is weighted combination: 60% max + 40% average
        # This ensures high-impact risks are prominent while considering cumulative effect
        overall_risk_score = 0.6 * max_score + 0.4 * avg_score

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

        # Classify risk level based on overall score
        # Thresholds:
        # - [0, 1.5): low risk
        # - [1.5, 2.5): medium risk  
        # - [2.5, 3.5): high risk
        # - [3.5, 5.0]: critical risk
        if overall_risk_score < 1.5:
            risk_level = "low"
        elif overall_risk_score < 2.5:
            risk_level = "medium"
        elif overall_risk_score < 3.5:
            risk_level = "high"
        else:
            risk_level = "critical"

        return {
            "overall_risk_score": overall_risk_score,
            "risk_level": risk_level
        }

    @is_tool()
    def calculate_float_time_distribution(self, task_float_times: dict):
        """
        Calculate the distribution of float time across all project tasks to identify scheduling flexibility.

        This method analyzes task float times to:
        1. Identify critical path tasks (zero float)
        2. Identify near-critical tasks (float < 5 days)
        3. Calculate average float time
        4. Assess overall scheduling flexibility

        Args:
            task_float_times: Dictionary mapping task names to their total float time in days
                             Example: {"task1": 0, "task2": 5, "task3": 10, "task4": 3}

        Returns:
            Dictionary containing:
            - critical_path: List of tasks with zero float (critical path)
            - near_critical_tasks: List of tasks with float less than 5 days (excluding zero float)
            - average_float: Average float time across all tasks in days
            - scheduling_flexibility: Overall assessment ("low", "medium", or "high")

        Raises:
            ValueError: If task_float_times is empty, None, or contains invalid values
        """
        # Validate input parameter
        if not task_float_times:
            raise ValueError("task_float_times cannot be empty or None")

        if not isinstance(task_float_times, dict):
            raise ValueError("task_float_times must be a dictionary")

        # Validate all float time values are numeric and non-negative
        for task_name, float_time in task_float_times.items():
            if not isinstance(float_time, (int, float)):
                raise ValueError(f"Float time for task '{task_name}' must be a number, got {type(float_time).__name__}")
            if float_time < 0:
                raise ValueError(f"Float time for task '{task_name}' cannot be negative, got {float_time}")

        # Initialize result containers
        critical_path = []
        near_critical_tasks = []

        # Identify critical path tasks (zero float) and near-critical tasks (0 < float < 5)
        for task_name, float_time in task_float_times.items():
            if float_time == 0:
                # Tasks with zero float are on the critical path
                critical_path.append(task_name)
            elif 0 < float_time < 5:
                # Tasks with float less than 5 days (but not zero) are near-critical
                near_critical_tasks.append(task_name)

        # Calculate average float time across all tasks
        total_float = sum(task_float_times.values())
        task_count = len(task_float_times)
        average_float = total_float / task_count

        # Assess overall scheduling flexibility based on multiple factors
        # Factors considered:
        # 1. Percentage of critical path tasks (zero float)
        # 2. Percentage of near-critical tasks (float < 5)
        # 3. Average float time

        critical_percentage = len(critical_path) / task_count
        near_critical_percentage = len(near_critical_tasks) / task_count
        constrained_percentage = critical_percentage + near_critical_percentage

        # Determine scheduling flexibility level
        # Low flexibility: High percentage of constrained tasks or low average float
        # Medium flexibility: Moderate constraints
        # High flexibility: Low constraints and high average float

        if constrained_percentage >= 0.6 or average_float < 3:
            # More than 60% of tasks are constrained (critical or near-critical)
            # OR average float is less than 3 days
            scheduling_flexibility = "low"
        elif constrained_percentage >= 0.3 or average_float < 7:
            # 30-60% of tasks are constrained
            # OR average float is between 3-7 days
            scheduling_flexibility = "medium"
        else:
            # Less than 30% of tasks are constrained
            # AND average float is 7 days or more
            scheduling_flexibility = "high"

        # Return the analysis results
        return {
            "critical_path": critical_path,
            "near_critical_tasks": near_critical_tasks,
            "average_float": average_float,
            "scheduling_flexibility": scheduling_flexibility
        }

    @is_tool()
    def analyze_project_dependencies(self, dependencies: list):
        """
        Analyze project dependencies to identify mandatory, discretionary, external, and internal dependencies.

        This method processes a list of dependency dictionaries and provides:
        1. A summary count of each dependency type
        2. A list of critical dependencies (mandatory and internal)
        3. A list of external dependencies that pose scheduling risks

        Args:
            dependencies: List of dependency dictionaries, each containing:
                - predecessor: str, the predecessor task name
                - successor: str, the successor task name
                - type: str, one of "mandatory", "discretionary", "external", "internal"
                - lead_lag_days: int, number of days lead (negative) or lag (positive)

        Returns:
            dict containing:
                - dependency_summary: dict with counts of each dependency type
                - critical_dependencies: list of dependencies on the critical path (mandatory + internal)
                - external_risks: list of external dependencies that pose scheduling risks

        Raises:
            ValueError: If dependencies parameter is invalid or contains invalid data
        """
        # Validate input parameter
        if not isinstance(dependencies, list):
            raise ValueError("dependencies parameter must be a list")

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

        # Initialize counters for dependency summary
        dependency_summary = {
            "mandatory": 0,
            "discretionary": 0,
            "external": 0,
            "internal": 0
        }

        # Initialize lists for analysis results
        critical_dependencies = []
        external_risks = []

        # Valid dependency types
        valid_types = {"mandatory", "discretionary", "external", "internal"}

        # Process each dependency
        for idx, dep in enumerate(dependencies):
            # Validate dependency structure
            if not isinstance(dep, dict):
                raise ValueError(f"Dependency at index {idx} must be a dictionary")

            # Check required fields
            required_fields = ["predecessor", "successor", "type", "lead_lag_days"]
            for field in required_fields:
                if field not in dep:
                    raise ValueError(f"Dependency at index {idx} is missing required field: {field}")

            # Extract and validate dependency fields
            predecessor = dep["predecessor"]
            successor = dep["successor"]
            dep_type = dep["type"]
            lead_lag_days = dep["lead_lag_days"]

            # Validate field types and values
            if not isinstance(predecessor, str) or not predecessor.strip():
                raise ValueError(f"Dependency at index {idx} has invalid predecessor: must be a non-empty string")

            if not isinstance(successor, str) or not successor.strip():
                raise ValueError(f"Dependency at index {idx} has invalid successor: must be a non-empty string")

            if not isinstance(dep_type, str) or dep_type.lower() not in valid_types:
                raise ValueError(f"Dependency at index {idx} has invalid type: must be one of {valid_types}")

            if not isinstance(lead_lag_days, (int, float)):
                raise ValueError(f"Dependency at index {idx} has invalid lead_lag_days: must be a number")

            # Normalize dependency type to lowercase for consistent processing
            dep_type_normalized = dep_type.lower()

            # Count dependency types
            dependency_summary[dep_type_normalized] += 1

            # Identify critical dependencies
            # Critical dependencies are those that are mandatory or internal
            # as they directly impact the project timeline and are within project control
            if dep_type_normalized in ["mandatory", "internal"]:
                critical_dependencies.append({
                    "predecessor": predecessor,
                    "successor": successor
                })

            # Identify external risks
            # External dependencies pose scheduling risks because they depend on
            # factors outside the project team's direct control
            if dep_type_normalized == "external":
                external_risks.append({
                    "predecessor": predecessor,
                    "successor": successor
                })

        # Prepare and return the analysis results
        return {
            "dependency_summary": dependency_summary,
            "critical_dependencies": critical_dependencies,
            "external_risks": external_risks
        }

    @is_tool()
    def create_milestone(self, project_id: str, milestone_name: str, target_date: str, deliverables: List[str] = None):
        # Import necessary modules for ID generation and datetime handling
        import secrets
        import hashlib
        from datetime import datetime

        # Access the database instance
        db = self.db

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

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

        # Validate and parse target_date format (yyyy-mm-dd)
        try:
            parsed_target_date = datetime.strptime(target_date, "%Y-%m-%d").date()
        except ValueError:
            raise ValueError("target_date must be in yyyy-mm-dd format")

        # Retrieve the project table from database
        project_table = getattr(db, "project", None)
        if project_table is None:
            raise KeyError("Project table does not exist in database")

        # Check if the project exists in the system (pre-condition)
        if project_id not in project_table:
            raise KeyError(f"Project with project_id '{project_id}' does not exist")

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

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

        # Create the Milestone object
        new_milestone = Milestone(
            milestone_id=milestone_id,
            project_id=project_id,
            milestone_name=milestone_name,
            target_date=parsed_target_date,
            is_completed=False,
            actual_completion_date=None,
            completion_notes=None,
            creation_timestamp=creation_timestamp,
            updated_timestamp=creation_timestamp
        )

        # Retrieve the milestone table from database
        milestone_table = getattr(db, "milestone", None)
        if milestone_table is None:
            milestone_table = {}

        # Add the new milestone to the milestone table
        milestone_table[milestone_id] = new_milestone
        setattr(db, "milestone", milestone_table)

        # Process deliverables if provided
        if deliverables:
            # Retrieve the milestone_deliverable table from database
            deliverable_table = getattr(db, "milestone_deliverable", None)
            if deliverable_table is None:
                deliverable_table = {}

            # Create MilestoneDeliverable objects for each deliverable
            for deliverable_desc in deliverables:
                # Validate deliverable description is not empty
                if deliverable_desc and deliverable_desc.strip():
                    # Generate unique deliverable_id
                    deliverable_id = "DLV-" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

                    # Create MilestoneDeliverable object
                    new_deliverable = MilestoneDeliverable(
                        deliverable_id=deliverable_id,
                        milestone_id=milestone_id,
                        deliverable_description=deliverable_desc
                    )

                    # Add deliverable to the table
                    deliverable_table[deliverable_id] = new_deliverable

            # Update the deliverable table in database
            setattr(db, "milestone_deliverable", deliverable_table)

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

        # Return the milestone_id and creation_timestamp
        return {
            "milestone_id": milestone_id,
            "creation_timestamp": creation_timestamp_str
        }

    @is_tool()
    def validate_budget_allocation(self, budget: float, allocations: dict) -> dict:
        """
        Validate that budget allocations do not exceed total budget and all categories are properly allocated.

        Args:
            budget: Total project budget in currency units
            allocations: Dictionary mapping budget categories to allocated amounts in currency units

        Returns:
            Dictionary containing validation results with is_valid, total_allocated, 
            remaining_budget, and validation_errors fields

        Raises:
            ValueError: If budget is negative or allocations contain invalid values
        """
        # Initialize validation errors list
        validation_errors = []

        # Validate budget parameter
        if not isinstance(budget, (int, float)):
            raise ValueError("Budget must be a numeric value")

        if budget < 0:
            raise ValueError("Budget cannot be negative")

        # Validate allocations parameter
        if not isinstance(allocations, dict):
            raise ValueError("Allocations must be a dictionary")

        if not allocations:
            validation_errors.append("Allocations dictionary is empty")

        # Calculate total allocated amount and validate individual allocations
        total_allocated = 0.0

        for category, amount in allocations.items():
            # Validate category name
            if not isinstance(category, str) or not category.strip():
                validation_errors.append(f"Invalid category name: '{category}'")
                continue

            # Validate allocation amount
            if not isinstance(amount, (int, float)):
                validation_errors.append(f"Invalid allocation amount for category '{category}': must be numeric")
                continue

            if amount < 0:
                validation_errors.append(f"Negative allocation amount for category '{category}': {amount}")
                continue

            # Add to total allocated
            total_allocated += amount

        # Calculate remaining budget
        remaining_budget = budget - total_allocated

        # Check if total allocation exceeds budget
        if total_allocated > budget:
            validation_errors.append(
                f"Total allocated amount ({total_allocated}) exceeds total budget ({budget})"
            )

        # Determine if allocation is valid (no errors)
        is_valid = len(validation_errors) == 0

        # Return validation results
        return {
            "is_valid": is_valid,
            "total_allocated": total_allocated,
            "remaining_budget": remaining_budget,
            "validation_errors": validation_errors
        }

    @is_tool()
    def update_milestone_status(
        self,
        milestone_id: str,
        is_completed: bool,
        actual_completion_date: Optional[str] = None,
        completion_notes: Optional[str] = None
    ):
        """
        Update the completion status of a milestone

        This method updates the completion status of an existing milestone in the database.
        It modifies the is_completed flag, and optionally updates the actual_completion_date
        and completion_notes fields. The updated_timestamp is automatically set to current time.

        Args:
            milestone_id: Unique identifier of the milestone to update
            is_completed: Whether the milestone is completed
            actual_completion_date: Actual completion date in yyyy-mm-dd format (optional)
            completion_notes: Notes about milestone completion (optional)

        Returns:
            dict: Contains success status and updated_timestamp
                - success (bool): Whether the status update was successful
                - updated_timestamp (str): Timestamp when status was updated in yyyy-mm-dd HH:MM:SS format

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

        # Get database instance
        db = self.db

        # Retrieve milestone table from database
        milestone_table = getattr(db, "milestone", None)

        # Check if milestone table exists
        if milestone_table is None:
            raise KeyError(f"Milestone table does not exist in database")

        # Check if milestone exists in the table
        if milestone_id not in milestone_table:
            raise KeyError(f"Milestone with id '{milestone_id}' does not exist")

        # Get the milestone object
        milestone = milestone_table[milestone_id]

        # Update is_completed status
        milestone.is_completed = is_completed

        # Update actual_completion_date if provided
        if actual_completion_date is not None:
            # Parse the date string in yyyy-mm-dd format
            try:
                completion_date = datetime.strptime(actual_completion_date, "%Y-%m-%d").date()
                milestone.actual_completion_date = completion_date
            except ValueError as e:
                raise ValueError(f"Invalid date format for actual_completion_date. Expected yyyy-mm-dd format: {e}")

        # Update completion_notes if provided
        if completion_notes is not None:
            milestone.completion_notes = completion_notes

        # Update the updated_timestamp to current time
        current_timestamp = datetime.now()
        milestone.updated_timestamp = current_timestamp

        # Save the updated milestone back to database
        milestone_table[milestone_id] = milestone
        setattr(db, "milestone", milestone_table)

        # Format the timestamp for return value
        updated_timestamp_str = current_timestamp.strftime("%Y-%m-%d %H:%M:%S")

        # Return success response
        return {
            "success": True,
            "updated_timestamp": updated_timestamp_str
        }

    @is_tool()
    def calculate_task_slack_time(self, task_schedules: dict) -> dict:
        """
        Calculate the slack time (float) for each task based on early start, early finish, 
        late start, and late finish times.

        Slack time (also known as float) is the amount of time a task can be delayed without 
        affecting the project completion date. It is calculated as:
        Slack Time = Late Start - Early Start = Late Finish - Early Finish

        Args:
            task_schedules: Dictionary mapping task names to their scheduling information.
                           Each task should contain:
                           - early_start: The earliest time a task can start
                           - early_finish: The earliest time a task can finish
                           - late_start: The latest time a task can start without delaying project
                           - late_finish: The latest time a task can finish without delaying project

        Returns:
            Dictionary mapping task names to their calculated slack time in days

        Raises:
            ValueError: If task_schedules is empty, None, or contains invalid scheduling data
        """
        # Validate input parameter
        if not task_schedules:
            raise ValueError("task_schedules cannot be empty or None")

        if not isinstance(task_schedules, dict):
            raise ValueError("task_schedules must be a dictionary")

        # Initialize result dictionary to store slack times
        slack_times = {}

        # Iterate through each task and calculate its slack time
        for task_name, schedule_info in task_schedules.items():
            # Validate that schedule_info is a dictionary
            if not isinstance(schedule_info, dict):
                raise ValueError(f"Schedule information for task '{task_name}' must be a dictionary")

            # Extract required scheduling parameters
            required_fields = ["early_start", "early_finish", "late_start", "late_finish"]
            for field in required_fields:
                if field not in schedule_info:
                    raise ValueError(f"Task '{task_name}' is missing required field '{field}'")

            early_start = schedule_info["early_start"]
            early_finish = schedule_info["early_finish"]
            late_start = schedule_info["late_start"]
            late_finish = schedule_info["late_finish"]

            # Validate that all time values are numeric
            try:
                early_start = float(early_start)
                early_finish = float(early_finish)
                late_start = float(late_start)
                late_finish = float(late_finish)
            except (ValueError, TypeError):
                raise ValueError(f"All time values for task '{task_name}' must be numeric")

            # Validate logical consistency of time values
            if early_finish < early_start:
                raise ValueError(f"Task '{task_name}': early_finish cannot be less than early_start")

            if late_finish < late_start:
                raise ValueError(f"Task '{task_name}': late_finish cannot be less than late_start")

            if late_start < early_start:
                raise ValueError(f"Task '{task_name}': late_start cannot be less than early_start")

            if late_finish < early_finish:
                raise ValueError(f"Task '{task_name}': late_finish cannot be less than early_finish")

            # Calculate slack time using the formula: Slack = Late Start - Early Start
            # (which is equivalent to Late Finish - Early Finish)
            slack_time = late_start - early_start

            # Verify consistency: both formulas should yield the same result
            slack_time_alt = late_finish - early_finish
            if abs(slack_time - slack_time_alt) > 0.001:  # Allow small floating point differences
                raise ValueError(
                    f"Task '{task_name}': Inconsistent scheduling data. "
                    f"Slack calculated from start times ({slack_time}) does not match "
                    f"slack calculated from finish times ({slack_time_alt})"
                )

            # Store the calculated slack time for this task
            slack_times[task_name] = slack_time

        # Return the dictionary containing slack times for all tasks
        return {"slack_times": slack_times}

    @is_tool()
    def generate_risk_register(self, risks: list):
        """
        Generate comprehensive risk register documenting identified risks with assessment and response plans.

        This method processes a list of risk dictionaries and generates a structured risk register
        with calculated risk scores, priority levels, and response strategies.

        Args:
            risks: List of risk dictionaries, each containing:
                - name: Risk name/identifier
                - description: Detailed description of the risk
                - category: Risk category (technical, resource, schedule, budget, external, quality)
                - probability: Probability of occurrence (0-1)
                - impact: Impact level if occurs (1-5)
                - owner: Risk owner name/ID

        Returns:
            dict containing:
                - risk_register: List of risk entries with calculated fields
                - risk_summary: Summary statistics by category and priority

        Raises:
            ValueError: If input validation fails
        """
        import secrets
        import hashlib

        # Input validation
        if not risks:
            raise ValueError("risks list cannot be empty")

        if not isinstance(risks, list):
            raise ValueError("risks must be a list")

        # Validate each risk dictionary
        required_fields = ["name", "description", "category", "probability", "impact", "owner"]
        valid_categories = ["technical", "resource", "schedule", "budget", "external", "quality"]

        for idx, risk in enumerate(risks):
            if not isinstance(risk, dict):
                raise ValueError(f"Risk at index {idx} must be a dictionary")

            # Check required fields
            missing_fields = [field for field in required_fields if field not in risk]
            if missing_fields:
                raise ValueError(f"Risk at index {idx} missing required fields: {missing_fields}")

            # Validate category
            if risk["category"] not in valid_categories:
                raise ValueError(f"Risk at index {idx} has invalid category '{risk['category']}'. Must be one of: {valid_categories}")

            # Validate probability (0-1)
            try:
                prob = float(risk["probability"])
                if not (0 <= prob <= 1):
                    raise ValueError(f"Risk at index {idx} probability must be between 0 and 1, got {prob}")
            except (TypeError, ValueError) as e:
                raise ValueError(f"Risk at index {idx} has invalid probability value: {e}")

            # Validate impact (1-5)
            try:
                impact = int(risk["impact"])
                if not (1 <= impact <= 5):
                    raise ValueError(f"Risk at index {idx} impact must be between 1 and 5, got {impact}")
            except (TypeError, ValueError) as e:
                raise ValueError(f"Risk at index {idx} has invalid impact value: {e}")

        # Initialize counters for summary
        risk_summary = {
            "total_risks": len(risks),
            "high_priority": 0,
            "medium_priority": 0,
            "low_priority": 0,
            "by_category": {}
        }

        # Process each risk and build the register
        risk_register = []

        for risk in risks:
            # Generate unique risk ID
            risk_id_prefix = "R"
            risk_id = risk_id_prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

            # Calculate risk score (probability × impact)
            probability = float(risk["probability"])
            impact = int(risk["impact"])
            risk_score = probability * impact

            # Determine priority based on risk score
            # Risk score ranges from 0 to 5 (probability 0-1 × impact 1-5)
            # Critical: >= 4.0, High: >= 2.5, Medium: >= 1.0, Low: < 1.0
            if risk_score >= 4.0:
                priority = "critical"
                risk_summary["high_priority"] += 1  # Count critical as high for summary
            elif risk_score >= 2.5:
                priority = "high"
                risk_summary["high_priority"] += 1
            elif risk_score >= 1.0:
                priority = "medium"
                risk_summary["medium_priority"] += 1
            else:
                priority = "low"
                risk_summary["low_priority"] += 1

            # Determine response strategy based on priority and probability
            # High probability + high impact = mitigate
            # Low probability + high impact = transfer or contingency plan
            # High probability + low impact = accept with monitoring
            # Low probability + low impact = accept
            if priority in ["critical", "high"]:
                if probability >= 0.5:
                    response_strategy = "mitigate"
                else:
                    response_strategy = "transfer"
            elif priority == "medium":
                if probability >= 0.5:
                    response_strategy = "mitigate"
                else:
                    response_strategy = "monitor"
            else:  # low priority
                response_strategy = "accept"

            # Build risk register entry
            risk_entry = {
                "risk_id": risk_id,
                "name": risk["name"],
                "description": risk["description"],
                "category": risk["category"],
                "probability": probability,
                "impact": impact,
                "risk_score": round(risk_score, 2),
                "priority": priority,
                "owner": risk["owner"],
                "response_strategy": response_strategy,
                "status": "identified"  # Initial status for newly registered risks
            }

            risk_register.append(risk_entry)

            # Update category statistics
            category = risk["category"]
            if category not in risk_summary["by_category"]:
                risk_summary["by_category"][category] = 0
            risk_summary["by_category"][category] += 1

        # Sort risk register by priority (critical > high > medium > low) and then by risk_score
        priority_order = {"critical": 0, "high": 1, "medium": 2, "low": 3}
        risk_register.sort(key=lambda x: (priority_order[x["priority"]], -x["risk_score"]))

        return {
            "risk_register": risk_register,
            "risk_summary": risk_summary
        }

    @is_tool()
    def validate_resource_availability(self, resource_capacity: dict, assignments: dict) -> dict:
        """
        Validate that resource assignments do not exceed available capacity and identify over-allocations.

        This method compares the assigned hours for each resource against their available capacity
        and identifies any over-allocations. It returns a comprehensive validation report including
        overall validity status, list of over-allocated resources, and detailed allocation information.

        Args:
            resource_capacity: Dictionary mapping resource names to their available hours per time period
                              Example: {"john_doe": 160, "jane_smith": 160}
            assignments: Dictionary mapping resource names to their assigned hours per time period
                        Example: {"john_doe": 180, "jane_smith": 140}

        Returns:
            Dictionary containing:
            - is_valid: Boolean indicating if all resources are within capacity
            - over_allocated_resources: List of resource names that exceed capacity
            - allocation_details: Dictionary with detailed allocation status for each resource

        Raises:
            ValueError: If inputs are invalid (not dictionaries, negative values, etc.)
        """
        # Validate input parameters
        if not isinstance(resource_capacity, dict):
            raise ValueError("resource_capacity must be a dictionary")

        if not isinstance(assignments, dict):
            raise ValueError("assignments must be a dictionary")

        # Validate that all capacity values are non-negative numbers
        for resource, capacity in resource_capacity.items():
            if not isinstance(capacity, (int, float)):
                raise ValueError(f"Capacity for resource '{resource}' must be a number, got {type(capacity).__name__}")
            if capacity < 0:
                raise ValueError(f"Capacity for resource '{resource}' cannot be negative: {capacity}")

        # Validate that all assignment values are non-negative numbers
        for resource, assigned in assignments.items():
            if not isinstance(assigned, (int, float)):
                raise ValueError(f"Assigned hours for resource '{resource}' must be a number, got {type(assigned).__name__}")
            if assigned < 0:
                raise ValueError(f"Assigned hours for resource '{resource}' cannot be negative: {assigned}")

        # Initialize result structures
        over_allocated_resources = []
        allocation_details = {}

        # Get all unique resource names from both capacity and assignments
        all_resources = set(resource_capacity.keys()) | set(assignments.keys())

        # Process each resource
        for resource in all_resources:
            # Get capacity (default to 0 if not specified)
            capacity = resource_capacity.get(resource, 0)

            # Get assigned hours (default to 0 if not specified)
            assigned = assignments.get(resource, 0)

            # Calculate over-allocation (positive if over-allocated, negative/zero if within capacity)
            over_allocation = assigned - capacity

            # Build allocation details for this resource
            allocation_details[resource] = {
                "capacity": capacity,
                "assigned": assigned,
                "over_allocation": over_allocation
            }

            # Check if resource is over-allocated
            if over_allocation > 0:
                over_allocated_resources.append(resource)

        # Determine overall validity (valid if no resources are over-allocated)
        is_valid = len(over_allocated_resources) == 0

        # Return comprehensive validation results
        return {
            "is_valid": is_valid,
            "over_allocated_resources": over_allocated_resources,
            "allocation_details": allocation_details
        }

    @is_tool()
    def create_project(
        self,
        name: str,
        description: str,
        start_date: str,
        end_date: str,
        budget: float = None,
        project_manager: str = None
    ) -> dict:
        # Import required modules at the beginning
        import secrets
        import hashlib
        from datetime import datetime

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

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

        if not start_date or not isinstance(start_date, str):
            raise ValueError("Project start_date is required and must be a string in yyyy-mm-dd format")

        if not end_date or not isinstance(end_date, str):
            raise ValueError("Project end_date is required and must be a string in yyyy-mm-dd format")

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

        try:
            parsed_end_date = datetime.strptime(end_date, "%Y-%m-%d").date()
        except ValueError:
            raise ValueError(f"Invalid end_date format: {end_date}. Expected format: yyyy-mm-dd")

        # Validate that end_date is not before start_date
        if parsed_end_date < parsed_start_date:
            raise ValueError(f"Project end_date ({end_date}) cannot be before start_date ({start_date})")

        # Validate optional parameters
        if budget is not None:
            if not isinstance(budget, (int, float)):
                raise ValueError("Budget must be a numeric value")
            if budget < 0:
                raise ValueError("Budget cannot be negative")

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

        # Generate unique project ID using hash of timestamp and random bytes
        prefix = "PRJ-"
        project_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

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

        # Create Project instance with all required and optional fields
        new_project = Project(
            project_id=project_id,
            name=name.strip(),
            description=description.strip(),
            start_date=parsed_start_date,
            end_date=parsed_end_date,
            budget=budget,
            project_manager=project_manager.strip() if project_manager else None,
            status="planning",  # Default status for new projects
            completion_percentage=0.0,
            creation_timestamp=creation_time,
            updated_timestamp=creation_time
        )

        # Access database through self.db
        db = self.db

        # Get existing project data or initialize empty dict
        project_data = getattr(db, "project", None)
        if project_data is None:
            project_data = {}

        # Store new project in database using project_id as key
        project_data[project_id] = new_project
        setattr(db, "project", project_data)

        # Return project_id and creation_timestamp in required format (yyyy-mm-dd HH:MM:SS)
        return {
            "project_id": project_id,
            "creation_timestamp": creation_time.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def calculate_sprint_burndown(
        self,
        total_story_points: int,
        daily_completed_points: list,
        sprint_length_days: int
    ) -> dict:
        """
        Calculate sprint burndown data showing remaining work over time for agile sprint tracking.

        This method computes:
        1. Actual burndown: remaining story points after each day based on completed work
        2. Ideal burndown: theoretical linear decrease of remaining work
        3. On-track status: whether the sprint is likely to complete all work on time

        Args:
            total_story_points: Total story points committed for the sprint
            daily_completed_points: List of story points completed each day
            sprint_length_days: Total length of the sprint in days

        Returns:
            dict containing:
                - burndown_data: List of remaining story points for each day (including day 0)
                - ideal_burndown: List of ideal remaining story points for each day (including day 0)
                - is_on_track: Boolean indicating if sprint is on track
        """
        # Input validation
        if total_story_points < 0:
            raise ValueError("total_story_points must be non-negative")

        if sprint_length_days <= 0:
            raise ValueError("sprint_length_days must be positive")

        if not isinstance(daily_completed_points, list):
            raise ValueError("daily_completed_points must be a list")

        # Validate that daily_completed_points contains valid integers
        for i, points in enumerate(daily_completed_points):
            if not isinstance(points, int) or points < 0:
                raise ValueError(f"daily_completed_points[{i}] must be a non-negative integer")

        # Calculate actual burndown data
        # Start with day 0 having all story points remaining
        burndown_data = [total_story_points]
        remaining_points = total_story_points

        # For each day, subtract the completed points from remaining
        for completed in daily_completed_points:
            remaining_points -= completed
            # Ensure remaining points don't go negative (handle over-completion)
            remaining_points = max(0, remaining_points)
            burndown_data.append(remaining_points)

        # Calculate ideal burndown (linear decrease)
        # Ideal burndown assumes equal work completion each day
        ideal_burndown = []
        points_per_day = total_story_points / sprint_length_days

        for day in range(sprint_length_days + 1):  # Include day 0
            ideal_remaining = total_story_points - (points_per_day * day)
            # Round to reasonable precision and ensure non-negative
            ideal_remaining = max(0, round(ideal_remaining, 2))
            ideal_burndown.append(ideal_remaining)

        # Determine if sprint is on track
        # Compare actual progress with ideal progress
        # If we have data for the current day, check if actual <= ideal (more work done is better)
        # Also check if we can complete remaining work in remaining days
        is_on_track = True

        if len(burndown_data) > 1:
            # Get the latest actual remaining points (last element in burndown_data)
            current_remaining = burndown_data[-1]
            current_day = len(burndown_data) - 1  # Current day index

            # Check if all work is already completed
            if current_remaining == 0:
                is_on_track = True
            else:
                # Calculate if remaining work can be completed in remaining days
                remaining_days = sprint_length_days - current_day

                if remaining_days <= 0:
                    # Sprint should be finished but work remains
                    is_on_track = (current_remaining == 0)
                else:
                    # Check if current pace can complete remaining work
                    # Calculate average velocity from completed days
                    total_completed = total_story_points - current_remaining
                    if current_day > 0:
                        avg_velocity = total_completed / current_day
                        projected_completion = current_remaining / avg_velocity if avg_velocity > 0 else float('inf')
                        is_on_track = projected_completion <= remaining_days
                    else:
                        # No days completed yet, assume on track
                        is_on_track = True

                    # Additional check: compare with ideal burndown at current day
                    if current_day < len(ideal_burndown):
                        ideal_at_current_day = ideal_burndown[current_day]
                        # If actual remaining is significantly higher than ideal, may not be on track
                        if current_remaining > ideal_at_current_day * 1.2:  # 20% tolerance
                            is_on_track = False

        return {
            "burndown_data": burndown_data,
            "ideal_burndown": ideal_burndown,
            "is_on_track": is_on_track
        }

    @is_tool()
    def prioritize_tasks_by_dependencies(self, tasks: list, dependencies: list):
        """
        Prioritize tasks based on their dependencies using topological sorting (Kahn's algorithm).

        This method implements a topological sort to determine the execution order of tasks
        based on their dependencies. It also detects circular dependencies.

        Args:
            tasks: List of task names to be prioritized
            dependencies: List of [predecessor, successor] pairs representing task dependencies

        Returns:
            dict with:
                - prioritized_tasks: List of tasks in dependency order
                - has_circular_dependency: Boolean indicating if circular dependencies exist

        Raises:
            ValueError: If tasks list is empty, or if dependencies reference non-existent tasks
        """
        # Validate input parameters
        if not tasks:
            raise ValueError("Tasks list cannot be empty")

        if not isinstance(tasks, list):
            raise ValueError("Tasks must be a list")

        if not isinstance(dependencies, list):
            raise ValueError("Dependencies must be a list")

        # Convert tasks to set for quick lookup
        task_set = set(tasks)

        # Validate that all dependencies reference existing tasks
        for dependency in dependencies:
            if not isinstance(dependency, list) or len(dependency) != 2:
                raise ValueError(f"Each dependency must be a list of two elements [predecessor, successor], got: {dependency}")

            predecessor, successor = dependency
            if predecessor not in task_set:
                raise ValueError(f"Dependency references non-existent task: {predecessor}")
            if successor not in task_set:
                raise ValueError(f"Dependency references non-existent task: {successor}")

        # Build adjacency list (graph) and in-degree map
        # graph[task] = list of tasks that depend on this task
        graph = {task: [] for task in tasks}
        # in_degree[task] = number of tasks this task depends on
        in_degree = {task: 0 for task in tasks}

        # Populate graph and in-degree based on dependencies
        for predecessor, successor in dependencies:
            graph[predecessor].append(successor)
            in_degree[successor] += 1

        # Initialize queue with tasks that have no dependencies (in-degree = 0)
        from collections import deque
        queue = deque([task for task in tasks if in_degree[task] == 0])

        # Perform topological sort using Kahn's algorithm
        prioritized_tasks = []

        while queue:
            # Remove a task with no dependencies
            current_task = queue.popleft()
            prioritized_tasks.append(current_task)

            # For each task that depends on current_task
            for dependent_task in graph[current_task]:
                # Decrease in-degree (remove the dependency)
                in_degree[dependent_task] -= 1

                # If dependent_task now has no dependencies, add to queue
                if in_degree[dependent_task] == 0:
                    queue.append(dependent_task)

        # Check for circular dependencies
        # If we couldn't process all tasks, there's a cycle
        has_circular_dependency = len(prioritized_tasks) != len(tasks)

        # If there's a circular dependency, include remaining tasks at the end
        # (they form the cycle or are blocked by the cycle)
        if has_circular_dependency:
            remaining_tasks = [task for task in tasks if task not in prioritized_tasks]
            prioritized_tasks.extend(remaining_tasks)

        return {
            "prioritized_tasks": prioritized_tasks,
            "has_circular_dependency": has_circular_dependency
        }

    @is_tool()
    def calculate_team_capacity(
        self,
        team_members: list,
        planning_period_weeks: int,
        capacity_buffer_percentage: float = 0
    ) -> dict:
        """
        Calculate total team capacity based on team member availability, working hours, and planned time off.

        This method computes:
        1. Individual capacity for each team member (weekly_hours * weeks - time_off)
        2. Total team capacity across all members
        3. Available capacity after applying buffer percentage

        Args:
            team_members: List of dictionaries with 'name', 'weekly_hours', and 'planned_time_off_days'
            planning_period_weeks: Number of weeks in the planning period
            capacity_buffer_percentage: Percentage to reduce capacity for buffer (0-100), defaults to 0

        Returns:
            Dictionary containing:
            - total_capacity_hours: Total team capacity in hours
            - available_capacity_hours: Capacity after applying buffer
            - individual_capacities: Dict mapping member names to their capacity hours

        Raises:
            ValueError: If input validation fails
        """
        # Validate input parameters
        if not isinstance(team_members, list) or len(team_members) == 0:
            raise ValueError("team_members must be a non-empty list")

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

        if not isinstance(capacity_buffer_percentage, (int, float)) or not (0 <= capacity_buffer_percentage <= 100):
            raise ValueError("capacity_buffer_percentage must be a number between 0 and 100")

        # Initialize result containers
        individual_capacities = {}
        total_capacity_hours = 0.0

        # Calculate capacity for each team member
        for member in team_members:
            # Validate member structure
            if not isinstance(member, dict):
                raise ValueError("Each team member must be a dictionary")

            if "name" not in member:
                raise ValueError("Each team member must have a 'name' field")

            if "weekly_hours" not in member:
                raise ValueError(f"Team member '{member.get('name', 'unknown')}' must have 'weekly_hours' field")

            # Extract member data
            name = member["name"]
            weekly_hours = member["weekly_hours"]
            planned_time_off_days = member.get("planned_time_off_days", 0)

            # Validate member data types and values
            if not isinstance(weekly_hours, (int, float)) or weekly_hours < 0:
                raise ValueError(f"weekly_hours for '{name}' must be a non-negative number")

            if not isinstance(planned_time_off_days, (int, float)) or planned_time_off_days < 0:
                raise ValueError(f"planned_time_off_days for '{name}' must be a non-negative number")

            # Calculate total working hours for the planning period
            total_working_hours = weekly_hours * planning_period_weeks

            # Calculate time off in hours (assuming 8-hour work days)
            # This is a common industry standard for converting days to hours
            hours_per_day = 8
            time_off_hours = planned_time_off_days * hours_per_day

            # Calculate individual capacity (cannot be negative)
            individual_capacity = max(0, total_working_hours - time_off_hours)

            # Store individual capacity
            individual_capacities[name] = individual_capacity

            # Add to total capacity
            total_capacity_hours += individual_capacity

        # Calculate available capacity after applying buffer
        # Buffer reduces available capacity (e.g., 20% buffer means 80% available)
        buffer_multiplier = 1 - (capacity_buffer_percentage / 100)
        available_capacity_hours = total_capacity_hours * buffer_multiplier

        # Return results
        return {
            "total_capacity_hours": total_capacity_hours,
            "available_capacity_hours": available_capacity_hours,
            "individual_capacities": individual_capacities
        }

    @is_tool()
    def get_project_details(self, project_id: str):
        """
        Retrieve complete details of a specific project including all metadata, status, and key metrics.

        Args:
            project_id: Unique identifier of the project

        Returns:
            Dictionary containing complete project details

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

        # Retrieve the project table from the database
        project_table = getattr(db, "project", None)

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

        # Check if the project_id exists in the project table
        if project_id not in project_table:
            raise KeyError(f"Project with project_id '{project_id}' does not exist")

        # Retrieve the project instance
        project = project_table[project_id]

        # Convert date objects to string format (yyyy-mm-dd)
        # The database stores dates as date objects, need to convert to string
        start_date_str = project.start_date.strftime("%Y-%m-%d")
        end_date_str = project.end_date.strftime("%Y-%m-%d")

        # Construct the return dictionary with all project details
        # Include all metadata, status, and key metrics as specified in the schema
        project_details = {
            "project_id": project.project_id,
            "name": project.name,
            "description": project.description,
            "status": project.status,  # Enum value: planning, active, on_hold, completed, cancelled
            "start_date": start_date_str,
            "end_date": end_date_str,
            "budget": project.budget,  # May be None if not set
            "completion_percentage": project.completion_percentage
        }

        return project_details

    @is_tool()
    def calculate_earned_value_metrics(
        self,
        planned_value: float,
        earned_value: float,
        actual_cost: float,
        budget: float
    ) -> dict:
        """
        Calculate earned value management (EVM) metrics for project performance analysis.

        This method computes key EVM metrics including:
        - Cost Variance (CV): Indicates whether the project is under or over budget
        - Schedule Variance (SV): Indicates whether the project is ahead or behind schedule
        - Cost Performance Index (CPI): Efficiency of cost utilization
        - Schedule Performance Index (SPI): Efficiency of time utilization
        - Estimate at Completion (EAC): Projected total cost at project completion
        - Estimate to Complete (ETC): Estimated cost to finish remaining work

        Args:
            planned_value: Planned value (PV) or budgeted cost of work scheduled in currency units
            earned_value: Earned value (EV) or budgeted cost of work performed in currency units
            actual_cost: Actual cost (AC) or actual cost of work performed in currency units
            budget: Total planned budget (BAC - Budget at Completion) for the project in currency units

        Returns:
            dict: Dictionary containing all calculated EVM metrics with the following keys:
                - cost_variance: CV = EV - AC (negative means over budget)
                - schedule_variance: SV = EV - PV (negative means behind schedule)
                - cost_performance_index: CPI = EV / AC (< 1 means over budget, > 1 means under budget)
                - schedule_performance_index: SPI = EV / PV (< 1 means behind schedule, > 1 means ahead)
                - estimate_at_completion: EAC = BAC / CPI (projected total cost)
                - estimate_to_complete: ETC = EAC - AC (remaining cost to complete)

        Raises:
            ValueError: If any input parameter is negative or if budget is zero
            ZeroDivisionError: If actual_cost is zero (cannot calculate CPI) or if CPI is zero (cannot calculate EAC)
        """

        # Validate input parameters - all values should be non-negative
        if planned_value < 0:
            raise ValueError("Planned value cannot be negative")
        if earned_value < 0:
            raise ValueError("Earned value cannot be negative")
        if actual_cost < 0:
            raise ValueError("Actual cost cannot be negative")
        if budget < 0:
            raise ValueError("Budget cannot be negative")
        if budget == 0:
            raise ValueError("Budget cannot be zero")

        # Check for division by zero conditions
        if actual_cost == 0:
            raise ZeroDivisionError("Actual cost cannot be zero - cannot calculate Cost Performance Index (CPI)")
        if planned_value == 0:
            raise ZeroDivisionError("Planned value cannot be zero - cannot calculate Schedule Performance Index (SPI)")

        # Calculate Cost Variance (CV)
        # CV = EV - AC
        # Positive CV indicates under budget, negative CV indicates over budget
        cost_variance = earned_value - actual_cost

        # Calculate Schedule Variance (SV)
        # SV = EV - PV
        # Positive SV indicates ahead of schedule, negative SV indicates behind schedule
        schedule_variance = earned_value - planned_value

        # Calculate Cost Performance Index (CPI)
        # CPI = EV / AC
        # CPI > 1 means project is under budget (good)
        # CPI < 1 means project is over budget (bad)
        # CPI = 1 means project is on budget
        cost_performance_index = earned_value / actual_cost

        # Calculate Schedule Performance Index (SPI)
        # SPI = EV / PV
        # SPI > 1 means project is ahead of schedule (good)
        # SPI < 1 means project is behind schedule (bad)
        # SPI = 1 means project is on schedule
        schedule_performance_index = earned_value / planned_value

        # Check if CPI is zero before calculating EAC
        if cost_performance_index == 0:
            raise ZeroDivisionError("Cost Performance Index (CPI) is zero - cannot calculate Estimate at Completion (EAC)")

        # Calculate Estimate at Completion (EAC)
        # EAC = BAC / CPI
        # This formula assumes current cost performance will continue for the remainder of the project
        # EAC represents the expected total cost of the project at completion
        estimate_at_completion = budget / cost_performance_index

        # Calculate Estimate to Complete (ETC)
        # ETC = EAC - AC
        # This represents the estimated cost needed to complete the remaining work
        estimate_to_complete = estimate_at_completion - actual_cost

        # Return all calculated metrics as a dictionary
        return {
            "cost_variance": cost_variance,
            "schedule_variance": schedule_variance,
            "cost_performance_index": cost_performance_index,
            "schedule_performance_index": schedule_performance_index,
            "estimate_at_completion": estimate_at_completion,
            "estimate_to_complete": estimate_to_complete
        }

    @is_tool()
    def calculate_contingency_reserve(
        self,
        budget: float,
        risk_exposure: float,
        project_complexity: Literal["low", "medium", "high"],
        historical_variance_percentage: float = None
    ) -> dict:
        """
        Calculate appropriate contingency reserve for project budget based on risk assessment and uncertainty.

        The calculation considers:
        1. Base risk exposure amount
        2. Project complexity multiplier
        3. Historical variance from similar projects (if available)

        Args:
            budget: Base project budget in currency units
            risk_exposure: Total risk exposure amount in currency units
            project_complexity: Project complexity level (low/medium/high)
            historical_variance_percentage: Optional historical budget variance percentage from similar projects

        Returns:
            dict: Contains recommended_contingency, contingency_percentage, and total_budget_with_contingency

        Raises:
            ValueError: If budget or risk_exposure is negative, or if inputs are invalid
        """

        # Validate input parameters
        if budget <= 0:
            raise ValueError("Budget must be a positive number")

        if risk_exposure < 0:
            raise ValueError("Risk exposure cannot be negative")

        if historical_variance_percentage is not None and historical_variance_percentage < 0:
            raise ValueError("Historical variance percentage cannot be negative")

        # Define complexity multipliers based on project complexity level
        # Higher complexity requires higher contingency reserves
        complexity_multipliers = {
            "low": 1.0,      # 100% - baseline
            "medium": 1.3,   # 130% - moderate increase
            "high": 1.6      # 160% - significant increase
        }

        # Get the appropriate complexity multiplier
        complexity_multiplier = complexity_multipliers.get(project_complexity)
        if complexity_multiplier is None:
            raise ValueError(f"Invalid project_complexity value: {project_complexity}. Must be 'low', 'medium', or 'high'")

        # Calculate base contingency from risk exposure
        # Apply complexity multiplier to account for project uncertainty
        base_contingency = risk_exposure * complexity_multiplier

        # If historical variance data is available, incorporate it into the calculation
        # This helps adjust contingency based on past project performance
        if historical_variance_percentage is not None:
            # Convert percentage to decimal (e.g., 12.5% -> 0.125)
            historical_factor = historical_variance_percentage / 100.0

            # Calculate historical-based contingency amount
            historical_contingency = budget * historical_factor

            # Take weighted average: 60% from risk-based, 40% from historical
            # This balances current risk assessment with past experience
            recommended_contingency = (base_contingency * 0.6) + (historical_contingency * 0.4)
        else:
            # If no historical data, use risk-based calculation only
            recommended_contingency = base_contingency

        # Ensure contingency is not less than the original risk exposure
        # This provides a minimum safety threshold
        recommended_contingency = max(recommended_contingency, risk_exposure)

        # Calculate contingency as percentage of base budget
        contingency_percentage = (recommended_contingency / budget) * 100.0

        # Calculate total budget including contingency reserve
        total_budget_with_contingency = budget + recommended_contingency

        # Round results to 2 decimal places for currency precision
        recommended_contingency = round(recommended_contingency, 2)
        contingency_percentage = round(contingency_percentage, 2)
        total_budget_with_contingency = round(total_budget_with_contingency, 2)

        # Return comprehensive contingency analysis
        return {
            "recommended_contingency": recommended_contingency,
            "contingency_percentage": contingency_percentage,
            "total_budget_with_contingency": total_budget_with_contingency
        }

    @is_tool()
    def update_project_status(self, project_id: str, status: Literal["planning", "active", "on_hold", "completed", "cancelled"], status_reason: str = None):
        # Import required modules
        from datetime import datetime

        # Access the database instance
        db = self.db

        # Retrieve the project table from database
        project_table = getattr(db, "project", None)

        # Validate that project table exists
        if project_table is None:
            raise KeyError(f"Project table does not exist in database")

        # Check if the project exists in the database
        if project_id not in project_table:
            raise KeyError(f"Project with ID '{project_id}' does not exist in the system")

        # Retrieve the project object
        project = project_table[project_id]

        # Update the project status
        project.status = status

        # Update the timestamp to current time
        current_timestamp = datetime.now()
        project.updated_timestamp = current_timestamp

        # Note: status_reason parameter is accepted but not stored in the Project schema
        # In a production system, this could be logged to a separate audit/history table

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

        # Return success response with updated timestamp
        return {
            "success": True,
            "updated_timestamp": updated_timestamp_str
        }

    @is_tool()
    def generate_communication_plan(self, stakeholders: list, project_phase: str):
        """
        Generate a comprehensive communication plan for project stakeholders.

        This method creates a structured communication plan that defines how and when
        to communicate with different stakeholders based on their roles, information needs,
        and the current project phase.

        Args:
            stakeholders: List of stakeholder dictionaries, each containing:
                - name: Stakeholder name or title
                - role: Stakeholder role (e.g., 'executive', 'team_member', 'client')
                - information_needs: List of information types they need (e.g., ['status', 'budget', 'risks'])
            project_phase: Current project phase, must be one of:
                'initiation', 'planning', 'execution', 'monitoring', 'closing'

        Returns:
            dict: Communication plan containing:
                - communication_plan: List of communication items, each with:
                    - stakeholder: Name of the stakeholder
                    - frequency: How often to communicate (e.g., 'daily', 'weekly', 'monthly')
                    - channel: Communication channel (e.g., 'email', 'meeting', 'dashboard')
                    - content: What content to communicate (e.g., 'status report', 'risk updates')

        Raises:
            ValueError: If stakeholders list is empty, invalid, or project_phase is not valid
            TypeError: If parameters are not of expected types
        """
        # Validate input parameters
        if not isinstance(stakeholders, list):
            raise TypeError("stakeholders must be a list")

        if not stakeholders:
            raise ValueError("stakeholders list cannot be empty")

        # Validate project_phase enum
        valid_phases = ["initiation", "planning", "execution", "monitoring", "closing"]
        if project_phase not in valid_phases:
            raise ValueError(f"project_phase must be one of {valid_phases}, got '{project_phase}'")

        # Validate stakeholder structure
        for idx, stakeholder in enumerate(stakeholders):
            if not isinstance(stakeholder, dict):
                raise TypeError(f"Stakeholder at index {idx} must be a dictionary")

            if "name" not in stakeholder:
                raise ValueError(f"Stakeholder at index {idx} missing required field 'name'")

            if "role" not in stakeholder:
                raise ValueError(f"Stakeholder at index {idx} missing required field 'role'")

            if "information_needs" not in stakeholder:
                raise ValueError(f"Stakeholder at index {idx} missing required field 'information_needs'")

            if not isinstance(stakeholder["information_needs"], list):
                raise TypeError(f"Stakeholder at index {idx} 'information_needs' must be a list")

        # Define communication frequency mapping based on role and project phase
        # Different phases require different communication intensities
        frequency_map = {
            "initiation": {
                "executive": "weekly",
                "sponsor": "weekly",
                "manager": "weekly",
                "team_member": "bi-weekly",
                "client": "bi-weekly",
                "stakeholder": "monthly",
                "default": "bi-weekly"
            },
            "planning": {
                "executive": "weekly",
                "sponsor": "weekly",
                "manager": "bi-weekly",
                "team_member": "weekly",
                "client": "weekly",
                "stakeholder": "bi-weekly",
                "default": "weekly"
            },
            "execution": {
                "executive": "weekly",
                "sponsor": "weekly",
                "manager": "daily",
                "team_member": "daily",
                "client": "weekly",
                "stakeholder": "weekly",
                "default": "weekly"
            },
            "monitoring": {
                "executive": "weekly",
                "sponsor": "weekly",
                "manager": "daily",
                "team_member": "daily",
                "client": "weekly",
                "stakeholder": "bi-weekly",
                "default": "weekly"
            },
            "closing": {
                "executive": "bi-weekly",
                "sponsor": "bi-weekly",
                "manager": "weekly",
                "team_member": "weekly",
                "client": "weekly",
                "stakeholder": "monthly",
                "default": "weekly"
            }
        }

        # Define communication channels based on role
        channel_map = {
            "executive": "email",
            "sponsor": "meeting",
            "manager": "meeting",
            "team_member": "dashboard",
            "client": "email",
            "stakeholder": "email",
            "default": "email"
        }

        # Define content mapping based on information needs
        content_map = {
            "status": "status report",
            "budget": "budget report",
            "risks": "risk updates",
            "schedule": "schedule updates",
            "quality": "quality metrics",
            "resources": "resource allocation",
            "issues": "issue log",
            "changes": "change requests",
            "deliverables": "deliverable status",
            "performance": "performance metrics",
            "milestones": "milestone progress",
            "dependencies": "dependency tracking"
        }

        # Generate communication plan
        communication_plan = []

        for stakeholder in stakeholders:
            stakeholder_name = stakeholder["name"]
            role = stakeholder["role"].lower()
            information_needs = stakeholder["information_needs"]

            # Determine frequency based on role and project phase
            phase_frequencies = frequency_map.get(project_phase, frequency_map["execution"])
            frequency = phase_frequencies.get(role, phase_frequencies["default"])

            # Determine channel based on role
            channel = channel_map.get(role, channel_map["default"])

            # Generate content based on information needs
            # Consolidate multiple information needs into appropriate content descriptions
            content_items = []
            for need in information_needs:
                need_lower = need.lower()
                # Map the information need to appropriate content
                if need_lower in content_map:
                    content_items.append(content_map[need_lower])
                else:
                    # For unmapped needs, create a generic content description
                    content_items.append(f"{need_lower} updates")

            # Create consolidated content description
            if len(content_items) == 1:
                content = content_items[0]
            elif len(content_items) == 2:
                content = f"{content_items[0]} and {content_items[1]}"
            else:
                # For multiple items, create a comprehensive description
                content = ", ".join(content_items[:-1]) + f", and {content_items[-1]}"

            # Add communication item to plan
            communication_item = {
                "stakeholder": stakeholder_name,
                "frequency": frequency,
                "channel": channel,
                "content": content
            }

            communication_plan.append(communication_item)

        # Return the complete communication plan
        return {
            "communication_plan": communication_plan
        }

    @is_tool()
    def calculate_budget_variance(self, planned_budget: dict, actual_spending: dict) -> dict:
        """
        Calculate budget variance by comparing planned budget against actual spending.

        This method computes variances for each budget category and calculates overall
        variance statistics. Negative variance indicates over budget (actual > planned),
        positive variance indicates under budget (actual < planned).

        Args:
            planned_budget: Dictionary mapping budget categories to planned amounts
            actual_spending: Dictionary mapping budget categories to actual spending amounts

        Returns:
            Dictionary containing:
            - variances: Category-wise variance amounts (negative = over budget)
            - total_variance: Total variance across all categories
            - variance_percentage: Overall variance as percentage of total planned budget

        Raises:
            ValueError: If inputs are invalid (empty, non-numeric values, or zero total budget)
        """
        # Validate input parameters are dictionaries
        if not isinstance(planned_budget, dict):
            raise ValueError("planned_budget must be a dictionary")
        if not isinstance(actual_spending, dict):
            raise ValueError("actual_spending must be a dictionary")

        # Validate that both dictionaries are not empty
        if not planned_budget:
            raise ValueError("planned_budget cannot be empty")
        if not actual_spending:
            raise ValueError("actual_spending cannot be empty")

        # Initialize result dictionary to store variances for each category
        variances = {}

        # Get all unique categories from both planned and actual budgets
        # This ensures we handle cases where categories exist in one but not the other
        all_categories = set(planned_budget.keys()) | set(actual_spending.keys())

        # Calculate variance for each category
        # Variance = Planned - Actual (negative means over budget, positive means under budget)
        for category in all_categories:
            # Get planned amount for this category (default to 0 if not present)
            planned = planned_budget.get(category, 0)

            # Get actual spending for this category (default to 0 if not present)
            actual = actual_spending.get(category, 0)

            # Validate that amounts are numeric
            if not isinstance(planned, (int, float)):
                raise ValueError(f"planned_budget value for category '{category}' must be numeric")
            if not isinstance(actual, (int, float)):
                raise ValueError(f"actual_spending value for category '{category}' must be numeric")

            # Validate that amounts are non-negative
            if planned < 0:
                raise ValueError(f"planned_budget value for category '{category}' cannot be negative")
            if actual < 0:
                raise ValueError(f"actual_spending value for category '{category}' cannot be negative")

            # Calculate variance: positive means under budget, negative means over budget
            variance = planned - actual
            variances[category] = variance

        # Calculate total variance across all categories
        total_variance = sum(variances.values())

        # Calculate total planned budget for percentage calculation
        total_planned = sum(planned_budget.get(cat, 0) for cat in all_categories)

        # Validate that total planned budget is not zero to avoid division by zero
        if total_planned == 0:
            raise ValueError("Total planned budget cannot be zero")

        # Calculate variance percentage relative to total planned budget
        # Formula: (Total Variance / Total Planned Budget) × 100
        variance_percentage = (total_variance / total_planned) * 100

        # Return comprehensive variance analysis
        return {
            "variances": variances,
            "total_variance": total_variance,
            "variance_percentage": variance_percentage
        }

    @is_tool()
    def create_risk(
        self,
        project_id: str,
        risk_name: str,
        probability: float,
        impact: int,
        category: Literal["technical", "resource", "schedule", "budget", "external", "quality"],
        owner: str,
        description: Optional[str] = None
    ) -> Dict[str, Any]:
        """
        Create and register a new risk for a project with assessment details.

        This method:
        1. Validates that the project exists in the system
        2. Validates input parameters (probability range, impact range, category enum)
        3. Generates a unique risk_id
        4. Calculates risk_score (probability × impact)
        5. Determines priority level based on risk_score
        6. Creates and stores the new Risk instance in the database
        7. Returns the risk details including risk_id, risk_score, and priority

        Args:
            project_id: Unique identifier of the project
            risk_name: Name of the risk
            probability: Probability of risk occurrence (must be between 0 and 1)
            impact: Impact level if risk occurs (must be between 1 and 5)
            category: Risk category (must be one of the enum values)
            owner: Name or ID of risk owner
            description: Optional detailed description of the risk

        Returns:
            Dictionary containing:
            - risk_id: Unique identifier for the created risk
            - risk_score: Calculated risk score (probability × impact)
            - priority: Risk priority level based on risk_score

        Raises:
            KeyError: If the specified project_id does not exist in the system
            ValueError: If probability or impact values are out of valid range
        """
        import secrets
        import hashlib
        from datetime import datetime

        # Access the database
        db = self.db

        # Validate that the project exists
        project_table = getattr(db, "project", None)
        if project_table is None or project_id not in project_table:
            raise KeyError(f"Project with project_id '{project_id}' does not exist in the system")

        # Validate probability range (0-1)
        if not (0 <= probability <= 1):
            raise ValueError(f"Probability must be between 0 and 1, got {probability}")

        # Validate impact range (1-5)
        if not (1 <= impact <= 5):
            raise ValueError(f"Impact must be between 1 and 5, got {impact}")

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

        # Calculate risk_score (probability × impact)
        risk_score = probability * impact

        # Determine priority level based on risk_score
        # Priority thresholds: critical (>3.5), high (2.5-3.5), medium (1.5-2.5), low (<1.5)
        if risk_score > 3.5:
            priority = "critical"
        elif risk_score > 2.5:
            priority = "high"
        elif risk_score > 1.5:
            priority = "medium"
        else:
            priority = "low"

        # Get current timestamp
        current_time = datetime.now()

        # Create new Risk instance
        new_risk = Risk(
            risk_id=risk_id,
            project_id=project_id,
            risk_name=risk_name,
            description=description,
            probability=probability,
            impact=impact,
            category=category,
            owner=owner,
            status="identified",  # Default status for new risks
            risk_score=risk_score,
            priority=priority,
            creation_timestamp=current_time,
            updated_timestamp=current_time
        )

        # Get the risk table from database
        risk_table = getattr(db, "risk", None)

        # Initialize risk table if it doesn't exist
        if risk_table is None:
            risk_table = {}

        # Add the new risk to the table
        risk_table[risk_id] = new_risk

        # Update the database with the new risk table
        setattr(db, "risk", risk_table)

        # Return the required information
        return {
            "risk_id": risk_id,
            "risk_score": risk_score,
            "priority": priority
        }

    @is_tool()
    def update_risk_status(
        self,
        risk_id: str,
        status: Literal["identified", "assessed", "mitigated", "accepted", "occurred", "closed"],
        mitigation_actions: Optional[List[str]] = None,
        probability: Optional[float] = None
    ) -> Dict[str, Any]:
        """
        Update the status and details of an existing risk.

        This method updates a risk's status and optionally its probability and mitigation actions.
        It recalculates the risk score based on the updated probability if provided.

        Args:
            risk_id: Unique identifier of the risk to update
            status: New status for the risk (must be one of the predefined enum values)
            mitigation_actions: Optional list of mitigation action descriptions to add
            probability: Optional updated probability value (0-1) after mitigation

        Returns:
            Dictionary containing:
                - success: Boolean indicating if the update was successful
                - updated_risk_score: The recalculated risk score after updates

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

        # Access the database
        db = self.db

        # Get the risk table from database
        risk_table = getattr(db, "risk", None)
        if risk_table is None:
            raise KeyError(f"Risk table not found in database")

        # Check if the risk exists
        if risk_id not in risk_table:
            raise KeyError(f"Risk with ID '{risk_id}' does not exist")

        # Get the existing risk object
        risk = risk_table[risk_id]

        # Update the risk status
        risk.status = status

        # Update probability if provided
        if probability is not None:
            # Validate probability is in valid range [0, 1]
            if not (0 <= probability <= 1):
                raise ValueError(f"Probability must be between 0 and 1, got {probability}")
            risk.probability = probability

        # Recalculate risk score based on updated probability and existing impact
        # Risk score = probability × impact
        updated_risk_score = risk.probability * risk.impact
        risk.risk_score = updated_risk_score

        # Update the timestamp to reflect the modification
        risk.updated_timestamp = datetime.now()

        # Update the risk in the database
        risk_table[risk_id] = risk
        setattr(db, "risk", risk_table)

        # Add mitigation actions if provided
        if mitigation_actions is not None and len(mitigation_actions) > 0:
            # Get the risk_mitigation_action table
            mitigation_table = getattr(db, "risk_mitigation_action", None)
            if mitigation_table is None:
                # Initialize the table if it doesn't exist
                mitigation_table = {}

            # Add each mitigation action to the database
            for action_description in mitigation_actions:
                # Generate unique action_id using hash
                action_id = "ACT-" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

                # Create new mitigation action object
                new_action = RiskMitigationAction(
                    action_id=action_id,
                    risk_id=risk_id,
                    action_description=action_description,
                    action_date=datetime.now()
                )

                # Add to the mitigation actions table
                mitigation_table[action_id] = new_action

            # Update the mitigation actions table in database
            setattr(db, "risk_mitigation_action", mitigation_table)

        # Return success response with updated risk score
        return {
            "success": True,
            "updated_risk_score": updated_risk_score
        }

    @is_tool()
    def generate_work_breakdown_structure(self, name: str, deliverables: List[str], decomposition_levels: int) -> dict:
        """
        Generate a hierarchical work breakdown structure from project scope by decomposing 
        deliverables into manageable work packages.

        Args:
            name: Name of the project
            deliverables: List of major project deliverables
            decomposition_levels: Number of hierarchical levels to decompose (1-5)

        Returns:
            dict: Contains wbs_structure (hierarchical dictionary) and total_work_packages (integer)

        Raises:
            ValueError: If parameters are invalid
        """
        # Validate input parameters
        if not name or not isinstance(name, str):
            raise ValueError("Project name must be a non-empty string")

        if not deliverables or not isinstance(deliverables, list) or len(deliverables) == 0:
            raise ValueError("Deliverables must be a non-empty list")

        if not all(isinstance(d, str) and d.strip() for d in deliverables):
            raise ValueError("All deliverables must be non-empty strings")

        if not isinstance(decomposition_levels, int) or decomposition_levels < 1 or decomposition_levels > 5:
            raise ValueError("Decomposition levels must be an integer between 1 and 5")

        # Initialize WBS structure with project root
        wbs_structure = {
            "1": {
                "name": name
            }
        }

        # Counter for total work packages (starting with 1 for the root project)
        total_work_packages = 1

        # Common work package categories for decomposition
        # These represent typical breakdown patterns in project management
        level_2_categories = [
            "Planning", "Design", "Development", "Testing", "Deployment"
        ]

        level_3_categories = [
            "Requirements", "Architecture", "Implementation", "Review"
        ]

        level_4_categories = [
            "Analysis", "Documentation", "Execution", "Validation"
        ]

        level_5_categories = [
            "Research", "Specification", "Build", "Verification"
        ]

        # Map decomposition levels to category lists
        category_map = {
            2: level_2_categories,
            3: level_3_categories,
            4: level_4_categories,
            5: level_5_categories
        }

        # Process each deliverable at level 1
        for deliverable_idx, deliverable in enumerate(deliverables, start=1):
            deliverable_key = f"1.{deliverable_idx}"
            wbs_structure["1"][deliverable_key] = {
                "name": deliverable
            }
            total_work_packages += 1

            # Decompose further if levels > 1
            if decomposition_levels > 1:
                # Recursively decompose the deliverable
                current_parent = wbs_structure["1"][deliverable_key]
                current_prefix = deliverable_key

                # For each additional level beyond 1
                for level in range(2, decomposition_levels + 1):
                    # Get appropriate categories for this level
                    categories = category_map.get(level, level_2_categories)

                    # Determine how many sub-packages to create at this level
                    # Use min to avoid creating too many packages
                    num_packages = min(len(categories), 3 + (level - 2))

                    # Create work packages at current level
                    for pkg_idx in range(1, num_packages + 1):
                        # Generate work package key
                        package_key = f"{current_prefix}.{pkg_idx}"

                        # Select category name (cycle through if needed)
                        category_name = categories[(pkg_idx - 1) % len(categories)]

                        # Create package name combining deliverable context and category
                        package_name = f"{deliverable} - {category_name}"

                        # Add to parent's structure
                        if level == 2:
                            # For level 2, add directly under deliverable
                            wbs_structure["1"][deliverable_key][package_key] = {
                                "name": package_name
                            }
                        else:
                            # For deeper levels, need to navigate to correct parent
                            # Build path to parent
                            path_parts = current_prefix.split('.')
                            parent_ref = wbs_structure["1"]

                            # Navigate to the parent node
                            for part_idx in range(1, len(path_parts)):
                                parent_key = '.'.join(path_parts[:part_idx + 1])
                                parent_ref = parent_ref[parent_key]

                            # Add work package to parent
                            parent_ref[package_key] = {
                                "name": package_name
                            }

                        total_work_packages += 1

                    # For next level, update prefix to first package at current level
                    # This creates a tree structure rather than all packages at same level
                    if level < decomposition_levels:
                        current_prefix = f"{current_prefix}.1"

        return {
            "wbs_structure": wbs_structure,
            "total_work_packages": total_work_packages
        }

    @is_tool()
    def calculate_project_buffer(self, tasks: list, buffer_percentage: float):
        """
        Calculate project buffer size using critical chain methodology based on task uncertainties.

        This method implements the Critical Chain Project Management (CCPM) buffer calculation:
        1. Calculate aggressive duration for each task (typically using optimistic or most_likely estimates)
        2. Calculate safety time for each task (difference between pessimistic and aggressive estimates)
        3. Sum up all safety times
        4. Apply buffer percentage to total safety time to get project buffer
        5. Return aggressive duration, buffer, and total buffered duration

        Args:
            tasks: List of task dictionaries, each containing:
                - name: Task name (string)
                - optimistic_duration: Best case duration in days (number)
                - most_likely_duration: Most likely duration in days (number)
                - pessimistic_duration: Worst case duration in days (number)
            buffer_percentage: Percentage of total safety time to allocate as buffer (0-100)

        Returns:
            Dictionary containing:
                - project_buffer_days: Calculated buffer size in days
                - total_aggressive_duration: Sum of aggressive task estimates
                - buffered_project_duration: Total duration including buffer

        Raises:
            ValueError: If input validation fails (empty tasks, invalid durations, invalid percentage)
        """

        # Validate input parameters
        if not tasks or not isinstance(tasks, list) or len(tasks) == 0:
            raise ValueError("Tasks list cannot be empty")

        if not isinstance(buffer_percentage, (int, float)) or buffer_percentage < 0 or buffer_percentage > 100:
            raise ValueError("Buffer percentage must be a number between 0 and 100")

        # Initialize accumulators
        total_aggressive_duration = 0.0
        total_safety_time = 0.0

        # Process each task
        for idx, task in enumerate(tasks):
            # Validate task structure
            if not isinstance(task, dict):
                raise ValueError(f"Task at index {idx} must be a dictionary")

            # Extract required fields
            task_name = task.get("name", f"Task_{idx}")
            optimistic = task.get("optimistic_duration")
            most_likely = task.get("most_likely_duration")
            pessimistic = task.get("pessimistic_duration")

            # Validate that all duration fields are present and valid
            if optimistic is None or most_likely is None or pessimistic is None:
                raise ValueError(
                    f"Task '{task_name}' must have optimistic_duration, most_likely_duration, "
                    f"and pessimistic_duration fields"
                )

            # Convert to float and validate
            try:
                optimistic = float(optimistic)
                most_likely = float(most_likely)
                pessimistic = float(pessimistic)
            except (TypeError, ValueError):
                raise ValueError(
                    f"Task '{task_name}' durations must be numeric values"
                )

            # Validate duration logic: optimistic <= most_likely <= pessimistic
            if not (optimistic <= most_likely <= pessimistic):
                raise ValueError(
                    f"Task '{task_name}' must satisfy: "
                    f"optimistic_duration <= most_likely_duration <= pessimistic_duration"
                )

            # Validate non-negative durations
            if optimistic < 0 or most_likely < 0 or pessimistic < 0:
                raise ValueError(f"Task '{task_name}' durations must be non-negative")

            # Use most_likely as aggressive estimate (common CCPM practice)
            # Alternative: could use optimistic, but most_likely is more realistic
            aggressive_duration = most_likely

            # Calculate safety time (buffer removed from individual task)
            # Safety time = pessimistic - aggressive
            safety_time = pessimistic - aggressive_duration

            # Accumulate totals
            total_aggressive_duration += aggressive_duration
            total_safety_time += safety_time

        # Calculate project buffer using the specified percentage
        # Buffer = (sum of all safety times) * (buffer_percentage / 100)
        project_buffer_days = total_safety_time * (buffer_percentage / 100.0)

        # Calculate total buffered project duration
        # Total duration = aggressive duration + project buffer
        buffered_project_duration = total_aggressive_duration + project_buffer_days

        # Return results
        return {
            "project_buffer_days": project_buffer_days,
            "total_aggressive_duration": total_aggressive_duration,
            "buffered_project_duration": buffered_project_duration
        }

    @is_tool()
    def allocate_resource_to_task(self, task_id: str, resource_id: str, allocated_hours: float, start_date: str, end_date: str):
        """
        Allocate a resource to a specific task with defined hours and time period.

        This method creates a new resource allocation entry in the database after validating
        that the task exists and the date format is correct. It calculates the utilization
        impact based on the allocated hours and time period.

        Args:
            task_id: Unique identifier of the task
            resource_id: Unique identifier of the resource
            allocated_hours: Number of hours to allocate
            start_date: Allocation start date in yyyy-mm-dd format
            end_date: Allocation end date in yyyy-mm-dd format

        Returns:
            dict: Contains allocation_id and utilization_impact

        Raises:
            KeyError: If task does not exist in the system
            ValueError: If date format is invalid or end_date is before start_date
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Access the database
        db = self.db

        # Validate that task exists in the system
        task_table = getattr(db, "task", None)
        if task_table is None or task_id not in task_table:
            raise KeyError(f"Task with id '{task_id}' does not exist in the system")

        # Parse and validate date strings
        try:
            start_datetime = datetime.strptime(start_date, "%Y-%m-%d")
            end_datetime = datetime.strptime(end_date, "%Y-%m-%d")
        except ValueError as e:
            raise ValueError(f"Invalid date format. Expected yyyy-mm-dd format. Error: {str(e)}")

        # Validate that end_date is not before start_date
        if end_datetime < start_datetime:
            raise ValueError(f"End date '{end_date}' cannot be before start date '{start_date}'")

        # Validate allocated_hours is positive
        if allocated_hours <= 0:
            raise ValueError(f"Allocated hours must be positive, got {allocated_hours}")

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

        # Calculate utilization impact
        # Calculate the number of days in the allocation period
        allocation_days = (end_datetime - start_datetime).days + 1  # +1 to include both start and end dates

        # Assume standard work hours per day (8 hours) and 5 working days per week
        # Calculate available working hours in the period
        # Simplified calculation: assume all days are working days for basic calculation
        # For more accurate calculation, would need to account for weekends and holidays
        working_days = allocation_days  # Simplified assumption
        available_hours = working_days * 8  # 8 hours per working day

        # Calculate utilization impact as percentage
        # This represents what percentage of available time is being allocated
        if available_hours > 0:
            utilization_impact = round((allocated_hours / available_hours) * 100, 2)
        else:
            utilization_impact = 0.0

        # Create ResourceAllocation instance
        new_allocation = ResourceAllocation(
            allocation_id=allocation_id,
            task_id=task_id,
            resource_id=resource_id,
            allocated_hours=allocated_hours,
            start_date=start_datetime,
            end_date=end_datetime,
            creation_timestamp=datetime.now()
        )

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

        # Add new allocation to the table
        resource_allocation_table[allocation_id] = new_allocation

        # Update the database
        setattr(db, "resource_allocation", resource_allocation_table)

        # Return the result
        return {
            "allocation_id": allocation_id,
            "utilization_impact": utilization_impact
        }

    @is_tool()
    def calculate_team_velocity(self, sprint_story_points: list[int]):
        """
        Calculate team velocity based on completed story points over multiple sprints.

        This method computes the average velocity and determines the velocity trend
        by analyzing the story points completed across multiple sprints.

        Args:
            sprint_story_points: List of story points completed in each sprint

        Returns:
            dict: Contains average_velocity (float) and velocity_trend (str)

        Raises:
            ValueError: If sprint_story_points is empty or contains invalid values
        """
        # Validate input: sprint_story_points must not be empty
        if not sprint_story_points:
            raise ValueError("sprint_story_points list cannot be empty")

        # Validate input: all elements must be non-negative integers
        if not all(isinstance(points, int) and points >= 0 for points in sprint_story_points):
            raise ValueError("All story points must be non-negative integers")

        # Calculate average velocity
        # Average is the sum of all story points divided by the number of sprints
        total_points = sum(sprint_story_points)
        num_sprints = len(sprint_story_points)
        average_velocity = total_points / num_sprints

        # Determine velocity trend
        # Strategy: Compare the first half and second half of sprints to determine trend
        # If there's only 1 sprint, trend is "stable"
        # If there are 2+ sprints, we analyze the trend

        if num_sprints == 1:
            # Single sprint - no trend to analyze
            velocity_trend = "stable"
        else:
            # Split sprints into two halves for trend analysis
            mid_point = num_sprints // 2

            # Calculate average of first half and second half
            first_half = sprint_story_points[:mid_point] if mid_point > 0 else []
            second_half = sprint_story_points[mid_point:]

            # Handle edge case where first_half might be empty (when num_sprints == 2)
            if not first_half:
                first_half_avg = sprint_story_points[0]
            else:
                first_half_avg = sum(first_half) / len(first_half)

            second_half_avg = sum(second_half) / len(second_half)

            # Calculate the percentage change between first and second half
            # Using a threshold of 5% to determine if the change is significant
            change_threshold = 0.05  # 5% threshold

            if first_half_avg == 0:
                # Special case: if first half average is 0
                if second_half_avg > 0:
                    velocity_trend = "increasing"
                else:
                    velocity_trend = "stable"
            else:
                change_ratio = (second_half_avg - first_half_avg) / first_half_avg

                if change_ratio > change_threshold:
                    velocity_trend = "increasing"
                elif change_ratio < -change_threshold:
                    velocity_trend = "decreasing"
                else:
                    velocity_trend = "stable"

        # Return the calculated metrics
        return {
            "average_velocity": round(average_velocity, 1),  # Round to 1 decimal place
            "velocity_trend": velocity_trend
        }

    @is_tool()
    def get_task_details(self, task_id: str):
        """
        Retrieve complete details of a specific task including status, progress, and metadata.

        Args:
            task_id: Unique identifier of the task to retrieve

        Returns:
            dict: Dictionary containing complete task details including:
                - task_id: Unique identifier of the task
                - task_name: Name of the task
                - project_id: Unique identifier of the parent project
                - assigned_to: Name or ID of the person assigned to the task
                - status: Current task status (not_started, in_progress, blocked, completed, cancelled)
                - completion_percentage: Task completion percentage
                - due_date: Task due date in yyyy-mm-dd format

        Raises:
            KeyError: If the task with the given task_id does not exist in the system
        """
        # Access the database instance
        db = self.db

        # Retrieve the task table from the database
        task_table = getattr(db, "task", None)

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

        # Check if the task_id exists in the task table
        if task_id not in task_table:
            raise KeyError(f"Task with task_id '{task_id}' does not exist in the system")

        # Retrieve the task object
        task = task_table[task_id]

        # Format the due_date to yyyy-mm-dd string format
        # The due_date in the database is a date object, convert it to string
        due_date_str = task.due_date.strftime("%Y-%m-%d")

        # Construct the return dictionary with all required fields
        task_details = {
            "task_id": task.task_id,
            "task_name": task.task_name,
            "project_id": task.project_id,
            "assigned_to": task.assigned_to,
            "status": task.status,
            "completion_percentage": task.completion_percentage,
            "due_date": due_date_str
        }

        return task_details

    @is_tool()
    def generate_status_report(
        self,
        name: str,
        reporting_period: str,
        status: Literal["on_track", "at_risk", "off_track"],
        completed_milestones: List[str],
        active_issues: List[str],
        top_risks: List[str],
        upcoming_activities: List[str]
    ) -> dict:
        """
        Generate comprehensive project status report including progress, issues, risks, and upcoming activities.

        This method creates a structured status report document with executive summary and detailed sections
        covering completed milestones, active issues, identified risks, and planned activities.
        """
        from datetime import datetime

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

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

        # Validate status is one of the allowed enum values
        valid_statuses = ["on_track", "at_risk", "off_track"]
        if status not in valid_statuses:
            raise ValueError(f"Status must be one of {valid_statuses}, got '{status}'")

        # Validate list parameters
        if not isinstance(completed_milestones, list):
            raise ValueError("Completed milestones must be a list")

        if not isinstance(active_issues, list):
            raise ValueError("Active issues must be a list")

        if not isinstance(top_risks, list):
            raise ValueError("Top risks must be a list")

        if not isinstance(upcoming_activities, list):
            raise ValueError("Upcoming activities must be a list")

        # Generate report timestamp
        report_timestamp = datetime.now().strftime("%Y-%m-%d %H:%M:%S")

        # Map status to human-readable format
        status_display = {
            "on_track": "On Track",
            "at_risk": "At Risk",
            "off_track": "Off Track"
        }

        # Create header section with project and report metadata
        header = {
            "project_name": name,
            "reporting_period": reporting_period,
            "report_date": report_timestamp,
            "overall_status": status_display.get(status, status)
        }

        # Generate executive summary based on status and key metrics
        # Calculate summary metrics
        total_completed = len(completed_milestones)
        total_issues = len(active_issues)
        total_risks = len(top_risks)
        total_upcoming = len(upcoming_activities)

        # Build executive summary narrative
        executive_summary_parts = []

        # Status overview
        if status == "on_track":
            executive_summary_parts.append(
                f"Project '{name}' is currently on track for the {reporting_period}."
            )
        elif status == "at_risk":
            executive_summary_parts.append(
                f"Project '{name}' is at risk during the {reporting_period}. Immediate attention required."
            )
        else:  # off_track
            executive_summary_parts.append(
                f"Project '{name}' is off track for the {reporting_period}. Critical intervention needed."
            )

        # Milestones summary
        if total_completed > 0:
            executive_summary_parts.append(
                f"During this period, {total_completed} milestone(s) were successfully completed."
            )
        else:
            executive_summary_parts.append(
                "No milestones were completed during this period."
            )

        # Issues summary
        if total_issues > 0:
            executive_summary_parts.append(
                f"There are currently {total_issues} active issue(s) requiring resolution."
            )
        else:
            executive_summary_parts.append(
                "No active issues are currently reported."
            )

        # Risks summary
        if total_risks > 0:
            executive_summary_parts.append(
                f"{total_risks} top risk(s) have been identified and are being monitored."
            )
        else:
            executive_summary_parts.append(
                "No significant risks are currently identified."
            )

        # Upcoming activities summary
        if total_upcoming > 0:
            executive_summary_parts.append(
                f"The team has {total_upcoming} planned activit(ies) for the next period."
            )

        executive_summary = " ".join(executive_summary_parts)

        # Create detailed sections with structured information
        detailed_sections = {}

        # Section 1: Completed Milestones
        detailed_sections["completed_milestones"] = {
            "title": "Completed Milestones",
            "count": total_completed,
            "items": [
                {
                    "index": idx + 1,
                    "description": milestone
                }
                for idx, milestone in enumerate(completed_milestones)
            ],
            "summary": f"{total_completed} milestone(s) completed" if total_completed > 0 else "No milestones completed"
        }

        # Section 2: Active Issues
        detailed_sections["active_issues"] = {
            "title": "Active Issues",
            "count": total_issues,
            "items": [
                {
                    "index": idx + 1,
                    "description": issue,
                    "severity": "high" if status == "off_track" else "medium" if status == "at_risk" else "low"
                }
                for idx, issue in enumerate(active_issues)
            ],
            "summary": f"{total_issues} active issue(s) require attention" if total_issues > 0 else "No active issues"
        }

        # Section 3: Top Risks
        detailed_sections["top_risks"] = {
            "title": "Top Risks",
            "count": total_risks,
            "items": [
                {
                    "index": idx + 1,
                    "description": risk,
                    "priority": "critical" if idx == 0 and total_risks > 0 else "high" if idx < 2 else "medium"
                }
                for idx, risk in enumerate(top_risks)
            ],
            "summary": f"{total_risks} risk(s) identified" if total_risks > 0 else "No significant risks identified"
        }

        # Section 4: Upcoming Activities
        detailed_sections["upcoming_activities"] = {
            "title": "Upcoming Activities",
            "count": total_upcoming,
            "items": [
                {
                    "index": idx + 1,
                    "description": activity
                }
                for idx, activity in enumerate(upcoming_activities)
            ],
            "summary": f"{total_upcoming} planned activit(ies) for next period" if total_upcoming > 0 else "No upcoming activities planned"
        }

        # Section 5: Recommendations (generated based on status and metrics)
        recommendations = []

        if status == "off_track":
            recommendations.append("Conduct immediate project review meeting with stakeholders")
            recommendations.append("Reassess project timeline and resource allocation")
            if total_issues > 0:
                recommendations.append("Prioritize resolution of active issues")
        elif status == "at_risk":
            recommendations.append("Monitor critical path activities closely")
            recommendations.append("Implement risk mitigation strategies for identified risks")
            if total_issues > 2:
                recommendations.append("Increase issue resolution cadence")
        else:  # on_track
            recommendations.append("Continue current project execution approach")
            recommendations.append("Maintain regular stakeholder communication")

        if total_risks > 3:
            recommendations.append("Conduct comprehensive risk assessment and update mitigation plans")

        detailed_sections["recommendations"] = {
            "title": "Recommendations",
            "items": [
                {
                    "index": idx + 1,
                    "recommendation": rec
                }
                for idx, rec in enumerate(recommendations)
            ]
        }

        # Assemble complete status report structure
        status_report = {
            "header": header,
            "executive_summary": executive_summary,
            "detailed_sections": detailed_sections,
            "metadata": {
                "generated_at": report_timestamp,
                "report_type": "project_status_report",
                "version": "1.0"
            }
        }

        return {"status_report": status_report}

    @is_tool()
    def calculate_resource_leveling(self, tasks: list, resource_capacity: dict) -> dict:
        """
        Calculate resource leveling adjustments to resolve resource over-allocations 
        while minimizing schedule impact.

        This method implements a heuristic resource leveling algorithm that:
        1. Identifies resource over-allocations by day
        2. Delays non-critical tasks to resolve conflicts
        3. Minimizes total schedule delay

        Args:
            tasks: List of task dictionaries with keys:
                - name: Task name
                - start_date: Start date in "yyyy-mm-dd" format
                - duration_days: Duration in days
                - assigned_resources: List of resource names
                - is_on_critical_path: Boolean indicating if task is on critical path
            resource_capacity: Dictionary mapping resource names to daily capacity hours

        Returns:
            Dictionary containing:
                - leveled_schedule: List of tasks with adjusted start dates
                - schedule_impact_days: Total schedule delay in days

        Raises:
            ValueError: If input data is invalid or missing required fields
        """
        from datetime import datetime, timedelta
        from collections import defaultdict

        # Validate inputs
        if not tasks:
            raise ValueError("Tasks list cannot be empty")
        if not resource_capacity:
            raise ValueError("Resource capacity dictionary cannot be empty")

        # Validate each task has required fields
        required_fields = ["name", "start_date", "duration_days", "assigned_resources", "is_on_critical_path"]
        for task in tasks:
            for field in required_fields:
                if field not in task:
                    raise ValueError(f"Task {task.get('name', 'unknown')} missing required field: {field}")

            # Validate data types
            if not isinstance(task["duration_days"], (int, float)) or task["duration_days"] <= 0:
                raise ValueError(f"Task {task['name']} has invalid duration_days")
            if not isinstance(task["assigned_resources"], list) or not task["assigned_resources"]:
                raise ValueError(f"Task {task['name']} has invalid assigned_resources")
            if not isinstance(task["is_on_critical_path"], bool):
                raise ValueError(f"Task {task['name']} has invalid is_on_critical_path")

        # Parse dates and create working copies of tasks
        working_tasks = []
        for task in tasks:
            try:
                start_date = datetime.strptime(task["start_date"], "%Y-%m-%d").date()
            except ValueError:
                raise ValueError(f"Task {task['name']} has invalid start_date format. Expected yyyy-mm-dd")

            working_tasks.append({
                "name": task["name"],
                "original_start_date": start_date,
                "current_start_date": start_date,
                "duration_days": int(task["duration_days"]),
                "assigned_resources": task["assigned_resources"],
                "is_on_critical_path": task["is_on_critical_path"],
                "delay_days": 0
            })

        # Calculate resource utilization by day for initial schedule
        def calculate_daily_utilization(tasks_list):
            """Calculate resource hours used per day"""
            utilization = defaultdict(lambda: defaultdict(float))

            for task in tasks_list:
                current_date = task["current_start_date"]
                # Assume 8 hours per day per resource for each task
                hours_per_day = 8.0

                for day in range(task["duration_days"]):
                    date_key = current_date + timedelta(days=day)
                    for resource in task["assigned_resources"]:
                        utilization[date_key][resource] += hours_per_day

            return utilization

        # Identify over-allocations
        def find_overallocations(utilization, capacity):
            """Find dates and resources that are over-allocated"""
            overallocations = []

            for date_key, resources in utilization.items():
                for resource, hours_used in resources.items():
                    capacity_hours = capacity.get(resource, 8.0)
                    if hours_used > capacity_hours:
                        overallocations.append({
                            "date": date_key,
                            "resource": resource,
                            "hours_used": hours_used,
                            "capacity": capacity_hours,
                            "overallocation": hours_used - capacity_hours
                        })

            return overallocations

        # Iteratively resolve over-allocations
        max_iterations = 100  # Prevent infinite loops
        iteration = 0

        while iteration < max_iterations:
            utilization = calculate_daily_utilization(working_tasks)
            overallocations = find_overallocations(utilization, resource_capacity)

            if not overallocations:
                # No more over-allocations, leveling complete
                break

            # Sort over-allocations by severity (highest overallocation first)
            overallocations.sort(key=lambda x: x["overallocation"], reverse=True)

            # Try to resolve the most severe over-allocation
            resolved = False
            for overalloc in overallocations:
                conflict_date = overalloc["date"]
                conflict_resource = overalloc["resource"]

                # Find tasks that use this resource on this date and are not on critical path
                candidate_tasks = []
                for task in working_tasks:
                    if task["is_on_critical_path"]:
                        continue

                    task_start = task["current_start_date"]
                    task_end = task_start + timedelta(days=task["duration_days"] - 1)

                    if (conflict_resource in task["assigned_resources"] and
                        task_start <= conflict_date <= task_end):
                        candidate_tasks.append(task)

                if not candidate_tasks:
                    continue

                # Sort candidate tasks by current delay (prioritize tasks with less delay)
                candidate_tasks.sort(key=lambda x: x["delay_days"])

                # Delay the first candidate task by 1 day
                task_to_delay = candidate_tasks[0]
                task_to_delay["current_start_date"] += timedelta(days=1)
                task_to_delay["delay_days"] += 1
                resolved = True
                break

            if not resolved:
                # Cannot resolve further without delaying critical path tasks
                break

            iteration += 1

        # Build leveled schedule result
        leveled_schedule = []
        total_delay = 0

        for task in working_tasks:
            if task["delay_days"] > 0:
                leveled_schedule.append({
                    "name": task["name"],
                    "new_start_date": task["current_start_date"].strftime("%Y-%m-%d"),
                    "delay_days": task["delay_days"]
                })
                total_delay = max(total_delay, task["delay_days"])

        # Calculate overall schedule impact
        # Schedule impact is the maximum delay among all tasks
        schedule_impact_days = total_delay

        return {
            "leveled_schedule": leveled_schedule,
            "schedule_impact_days": schedule_impact_days
        }

    @is_tool()
    def calculate_milestone_completion_percentage(self, total_milestones: int, completed_milestones: int):
        """
        Calculate the completion percentage of project milestones based on completed versus total milestones.

        This method computes the percentage of milestones that have been completed and determines
        how many milestones remain to be completed.

        Args:
            total_milestones: Total number of milestones in the project (must be non-negative)
            completed_milestones: Number of completed milestones (must be non-negative and <= total_milestones)

        Returns:
            dict: A dictionary containing:
                - completion_percentage (float): The percentage of completed milestones (0-100)
                - remaining_milestones (int): The number of milestones still to be completed

        Raises:
            ValueError: If input parameters are invalid (negative values, completed > total, etc.)
        """

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

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

        # Validate that completed_milestones does not exceed total_milestones
        if completed_milestones > total_milestones:
            raise ValueError(
                f"completed_milestones ({completed_milestones}) cannot exceed total_milestones ({total_milestones})"
            )

        # Handle edge case: if there are no milestones, completion is 0%
        if total_milestones == 0:
            completion_percentage = 0.0
            remaining_milestones = 0
        else:
            # Calculate completion percentage: (completed / total) * 100
            completion_percentage = (completed_milestones / total_milestones) * 100.0

            # Calculate remaining milestones: total - completed
            remaining_milestones = total_milestones - completed_milestones

        # Return the results as a dictionary
        return {
            "completion_percentage": completion_percentage,
            "remaining_milestones": remaining_milestones
        }

    @is_tool()
    def generate_gantt_chart_data(self, tasks: list, dependencies: list, milestones: list = None):
        """
        Generate data structure for Gantt chart visualization including task bars, dependencies, and milestones.

        This method processes task schedules, dependencies, and milestones to create a structured data format
        suitable for Gantt chart rendering. It calculates timeline boundaries, validates date formats,
        and organizes all chart elements.

        Args:
            tasks: List of task dictionaries containing name, start_date (yyyy-mm-dd), end_date (yyyy-mm-dd), 
                   and progress percentage
            dependencies: List of task dependencies where each dependency is [predecessor, successor]
            milestones: Optional list of milestone dictionaries containing name and date (yyyy-mm-dd)

        Returns:
            dict: Gantt chart data structure with tasks, dependencies, milestones, and timeline information

        Raises:
            ValueError: If input data is invalid or date formats are incorrect
        """
        from datetime import datetime, timedelta

        # Validate required parameters
        if not tasks or not isinstance(tasks, list):
            raise ValueError("Tasks parameter must be a non-empty list")

        if not isinstance(dependencies, list):
            raise ValueError("Dependencies parameter must be a list")

        # Initialize milestones if not provided
        if milestones is None:
            milestones = []

        if not isinstance(milestones, list):
            raise ValueError("Milestones parameter must be a list")

        # Process and validate tasks
        processed_tasks = []
        task_names = set()
        min_date = None
        max_date = None

        for task in tasks:
            # Validate task structure
            if not isinstance(task, dict):
                raise ValueError("Each task must be a dictionary")

            if "name" not in task or "start_date" not in task or "end_date" not in task:
                raise ValueError("Each task must contain 'name', 'start_date', and 'end_date' fields")

            task_name = task["name"]
            if not task_name or not isinstance(task_name, str):
                raise ValueError("Task name must be a non-empty string")

            task_names.add(task_name)

            # Parse and validate dates
            try:
                start_date = datetime.strptime(task["start_date"], "%Y-%m-%d")
                end_date = datetime.strptime(task["end_date"], "%Y-%m-%d")
            except (ValueError, TypeError) as e:
                raise ValueError(f"Invalid date format for task '{task_name}'. Expected 'yyyy-mm-dd' format: {str(e)}")

            # Validate date logic
            if start_date > end_date:
                raise ValueError(f"Task '{task_name}' has start_date after end_date")

            # Calculate duration in days
            duration = (end_date - start_date).days + 1

            # Validate and process progress
            progress = task.get("progress", 0)
            if not isinstance(progress, (int, float)):
                raise ValueError(f"Progress for task '{task_name}' must be a number")

            # Ensure progress is within valid range
            progress = max(0, min(100, progress))

            # Update timeline boundaries
            if min_date is None or start_date < min_date:
                min_date = start_date
            if max_date is None or end_date > max_date:
                max_date = end_date

            # Create processed task entry
            processed_task = {
                "name": task_name,
                "start_date": task["start_date"],
                "end_date": task["end_date"],
                "start_datetime": start_date,
                "end_datetime": end_date,
                "duration_days": duration,
                "progress": progress,
                "progress_decimal": progress / 100.0
            }

            processed_tasks.append(processed_task)

        # Process and validate dependencies
        processed_dependencies = []
        for dependency in dependencies:
            # Validate dependency structure
            if not isinstance(dependency, (list, tuple)) or len(dependency) != 2:
                raise ValueError("Each dependency must be a list/tuple with exactly 2 elements [predecessor, successor]")

            predecessor, successor = dependency

            # Validate that both tasks exist
            if predecessor not in task_names:
                raise ValueError(f"Predecessor task '{predecessor}' not found in tasks list")

            if successor not in task_names:
                raise ValueError(f"Successor task '{successor}' not found in tasks list")

            # Avoid self-dependencies
            if predecessor == successor:
                raise ValueError(f"Task '{predecessor}' cannot depend on itself")

            processed_dependencies.append({
                "predecessor": predecessor,
                "successor": successor,
                "type": "finish_to_start"  # Default dependency type for Gantt charts
            })

        # Process and validate milestones
        processed_milestones = []
        for milestone in milestones:
            # Validate milestone structure
            if not isinstance(milestone, dict):
                raise ValueError("Each milestone must be a dictionary")

            if "name" not in milestone or "date" not in milestone:
                raise ValueError("Each milestone must contain 'name' and 'date' fields")

            milestone_name = milestone["name"]
            if not milestone_name or not isinstance(milestone_name, str):
                raise ValueError("Milestone name must be a non-empty string")

            # Parse and validate milestone date
            try:
                milestone_date = datetime.strptime(milestone["date"], "%Y-%m-%d")
            except (ValueError, TypeError) as e:
                raise ValueError(f"Invalid date format for milestone '{milestone_name}'. Expected 'yyyy-mm-dd' format: {str(e)}")

            # Update timeline boundaries
            if min_date is None or milestone_date < min_date:
                min_date = milestone_date
            if max_date is None or milestone_date > max_date:
                max_date = milestone_date

            processed_milestones.append({
                "name": milestone_name,
                "date": milestone["date"],
                "datetime": milestone_date
            })

        # Calculate timeline information
        timeline = {}
        if min_date and max_date:
            total_days = (max_date - min_date).days + 1
            timeline = {
                "start_date": min_date.strftime("%Y-%m-%d"),
                "end_date": max_date.strftime("%Y-%m-%d"),
                "total_days": total_days,
                "start_datetime": min_date.strftime("%Y-%m-%d %H:%M:%S"),
                "end_datetime": max_date.strftime("%Y-%m-%d %H:%M:%S")
            }

        # Build final Gantt chart data structure
        gantt_data = {
            "tasks": [
                {
                    "name": task["name"],
                    "start_date": task["start_date"],
                    "end_date": task["end_date"],
                    "duration_days": task["duration_days"],
                    "progress": task["progress"]
                }
                for task in processed_tasks
            ],
            "dependencies": processed_dependencies,
            "milestones": [
                {
                    "name": milestone["name"],
                    "date": milestone["date"]
                }
                for milestone in processed_milestones
            ],
            "timeline": timeline
        }

        return {"gantt_data": gantt_data}

    @is_tool()
    def validate_project_closure_readiness(
        self,
        deliverables_accepted: bool,
        documentation_complete: bool,
        lessons_learned_captured: bool,
        resources_released: bool,
        financials_closed: bool,
        stakeholder_signoff: bool
    ) -> dict:
        """
        Validate that all project closure criteria are met including deliverable acceptance,
        documentation, and lessons learned.

        This method evaluates each closure criterion and determines if the project is ready
        for formal closure. It returns a comprehensive assessment including:
        - Overall readiness status
        - List of pending items that need completion
        - Complete checklist showing status of all criteria

        Args:
            deliverables_accepted: Whether all deliverables have been formally accepted
            documentation_complete: Whether all project documentation is complete
            lessons_learned_captured: Whether lessons learned have been documented
            resources_released: Whether all project resources have been released
            financials_closed: Whether all financial accounts have been closed
            stakeholder_signoff: Whether final stakeholder sign-off has been obtained

        Returns:
            dict: Dictionary containing:
                - ready_for_closure (bool): True if all criteria are met
                - pending_items (list): List of items still needing completion
                - closure_checklist (dict): Complete status of all closure criteria

        Raises:
            ValueError: If any of the required parameters are not boolean values
        """
        # Validate that all inputs are boolean values
        params = {
            "deliverables_accepted": deliverables_accepted,
            "documentation_complete": documentation_complete,
            "lessons_learned_captured": lessons_learned_captured,
            "resources_released": resources_released,
            "financials_closed": financials_closed,
            "stakeholder_signoff": stakeholder_signoff
        }

        for param_name, param_value in params.items():
            if not isinstance(param_value, bool):
                raise ValueError(f"Parameter '{param_name}' must be a boolean value, got {type(param_value).__name__}")

        # Build the closure checklist with all criteria and their status
        closure_checklist = {
            "deliverables_accepted": deliverables_accepted,
            "documentation_complete": documentation_complete,
            "lessons_learned_captured": lessons_learned_captured,
            "resources_released": resources_released,
            "financials_closed": financials_closed,
            "stakeholder_signoff": stakeholder_signoff
        }

        # Define human-readable names for each criterion for pending items list
        criterion_names = {
            "deliverables_accepted": "Complete deliverable acceptance",
            "documentation_complete": "Complete project documentation",
            "lessons_learned_captured": "Complete lessons learned documentation",
            "resources_released": "Release all project resources",
            "financials_closed": "Close all financial accounts",
            "stakeholder_signoff": "Obtain final stakeholder sign-off"
        }

        # Identify pending items (criteria that are False)
        pending_items = []
        for criterion, status in closure_checklist.items():
            if not status:
                pending_items.append(criterion_names[criterion])

        # Determine if project is ready for closure
        # All criteria must be True for the project to be ready
        ready_for_closure = all(closure_checklist.values())

        # Return the comprehensive closure readiness assessment
        return {
            "ready_for_closure": ready_for_closure,
            "pending_items": pending_items,
            "closure_checklist": closure_checklist
        }

    @is_tool()
    def calculate_productivity_metrics(self, completed_items: list, time_period_days: int):
        """
        Calculate team productivity metrics including throughput, cycle time, and lead time.

        This method computes three key productivity metrics:
        1. Throughput: Number of items completed per day over the specified time period
        2. Average Cycle Time: Average time from commit date to completion date
        3. Average Lead Time: Average time from start date to completion date

        Args:
            completed_items: List of completed item dictionaries containing:
                - name: Item name
                - start_date: When work started (yyyy-mm-dd HH:MM:SS format)
                - commit_date: When item was committed (yyyy-mm-dd HH:MM:SS format)
                - completion_date: When item was completed (yyyy-mm-dd HH:MM:SS format)
            time_period_days: Time period for throughput calculation in days

        Returns:
            Dictionary containing:
                - throughput: Number of items completed per day
                - average_cycle_time_days: Average time from commit to completion in days
                - average_lead_time_days: Average time from start to completion in days

        Raises:
            ValueError: If input parameters are invalid or if date formats are incorrect
        """
        from datetime import datetime

        # Validate input parameters
        if not isinstance(completed_items, list):
            raise ValueError("completed_items must be a list")

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

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

        # Initialize accumulators for cycle time and lead time
        total_cycle_time_days = 0.0
        total_lead_time_days = 0.0
        valid_items_count = 0

        # Process each completed item
        for item in completed_items:
            # Validate item structure
            if not isinstance(item, dict):
                raise ValueError("Each item in completed_items must be a dictionary")

            required_fields = ["name", "start_date", "commit_date", "completion_date"]
            for field in required_fields:
                if field not in item:
                    raise ValueError(f"Item missing required field: {field}")

            try:
                # Parse date strings to datetime objects
                # Expected format: "yyyy-mm-dd HH:MM:SS"
                start_date = datetime.strptime(item["start_date"], "%Y-%m-%d %H:%M:%S")
                commit_date = datetime.strptime(item["commit_date"], "%Y-%m-%d %H:%M:%S")
                completion_date = datetime.strptime(item["completion_date"], "%Y-%m-%d %H:%M:%S")

            except ValueError as e:
                raise ValueError(f"Invalid date format for item '{item.get('name', 'unknown')}'. Expected format: yyyy-mm-dd HH:MM:SS. Error: {str(e)}")

            # Validate date logical consistency
            if start_date > commit_date:
                raise ValueError(f"Item '{item['name']}': start_date cannot be after commit_date")

            if commit_date > completion_date:
                raise ValueError(f"Item '{item['name']}': commit_date cannot be after completion_date")

            if start_date > completion_date:
                raise ValueError(f"Item '{item['name']}': start_date cannot be after completion_date")

            # Calculate cycle time (commit_date to completion_date)
            cycle_time = completion_date - commit_date
            cycle_time_days = cycle_time.total_seconds() / (24 * 3600)  # Convert to days

            # Calculate lead time (start_date to completion_date)
            lead_time = completion_date - start_date
            lead_time_days = lead_time.total_seconds() / (24 * 3600)  # Convert to days

            # Accumulate times
            total_cycle_time_days += cycle_time_days
            total_lead_time_days += lead_time_days
            valid_items_count += 1

        # Calculate throughput: number of items completed per day
        # Throughput = total completed items / time period
        throughput = valid_items_count / time_period_days

        # Calculate average cycle time
        average_cycle_time_days = total_cycle_time_days / valid_items_count

        # Calculate average lead time
        average_lead_time_days = total_lead_time_days / valid_items_count

        # Return metrics with appropriate precision
        return {
            "throughput": round(throughput, 2),
            "average_cycle_time_days": round(average_cycle_time_days, 2),
            "average_lead_time_days": round(average_lead_time_days, 2)
        }

    @is_tool()
    def create_task(
        self,
        project_id: str,
        task_name: str,
        assigned_to: str,
        due_date: str,
        description: str = None,
        estimated_hours: float = None,
        priority: Literal["low", "medium", "high", "critical"] = "medium"
    ):
        """
        Create a new task within a project with details including name, assignee, and due date.

        Args:
            project_id: Unique identifier of the parent project
            task_name: Name of the task
            assigned_to: Name or ID of the person assigned to the task
            due_date: Task due date in yyyy-mm-dd format
            description: Detailed description of the task (optional)
            estimated_hours: Estimated hours to complete the task (optional)
            priority: Task priority level, must be one of: low, medium, high, critical (default: medium)

        Returns:
            dict: Contains task_id and creation_timestamp

        Raises:
            KeyError: If the project does not exist in the system
            ValueError: If input parameters are invalid
        """
        from datetime import datetime
        import secrets
        import hashlib

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

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

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

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

        # Validate and parse due_date format (yyyy-mm-dd)
        try:
            parsed_due_date = datetime.strptime(due_date, "%Y-%m-%d").date()
        except ValueError as e:
            raise ValueError(f"due_date must be in yyyy-mm-dd format: {str(e)}")

        # Validate optional parameters
        if estimated_hours is not None:
            if not isinstance(estimated_hours, (int, float)) or estimated_hours < 0:
                raise ValueError("estimated_hours must be a non-negative number")

        # Validate priority enum (additional safety check beyond Literal type hint)
        valid_priorities = ["low", "medium", "high", "critical"]
        if priority not in valid_priorities:
            raise ValueError(f"priority must be one of {valid_priorities}, got: {priority}")

        # Access database
        db = self.db

        # Check if project table exists
        project_table = getattr(db, "project", None)
        if project_table is None:
            raise KeyError(f"Project table does not exist in the database")

        # Verify that the project exists (pre-condition)
        if project_id not in project_table:
            raise KeyError(f"Project with ID '{project_id}' does not exist in the system")

        # Generate unique task_id using secure hash
        task_id_prefix = "TSK-"
        task_id = task_id_prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

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

        # Task class is already imported at file header via: from database import Task
        # Create new task object with all provided parameters
        new_task = Task(
            task_id=task_id,
            project_id=project_id,
            task_name=task_name,
            description=description,
            assigned_to=assigned_to,
            due_date=parsed_due_date,
            estimated_hours=estimated_hours,
            priority=priority,
            status="not_started",  # Default status for new tasks
            completion_percentage=0.0,  # New tasks start at 0% completion
            creation_timestamp=creation_timestamp,
            updated_timestamp=creation_timestamp
        )

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

        # Add new task to the task table
        task_table[task_id] = new_task

        # Update database with new task
        setattr(db, "task", task_table)

        # Format creation timestamp for return (yyyy-mm-dd HH:MM:SS format)
        formatted_timestamp = creation_timestamp.strftime("%Y-%m-%d %H:%M:%S")

        # Return task_id and creation_timestamp as specified in schema
        return {
            "task_id": task_id,
            "creation_timestamp": formatted_timestamp
        }

    @is_tool()
    def assess_project_health(
        self,
        schedule_performance_index: float,
        cost_performance_index: float,
        scope_change_count: int,
        quality_defect_rate: float,
        high_risk_count: int
    ) -> dict:
        """
        Assess overall project health by analyzing schedule, budget, scope, quality, and risk indicators.

        This method calculates a comprehensive health score based on multiple project performance metrics
        and provides actionable recommendations for improvement.

        Args:
            schedule_performance_index: SPI value (1.0 = on schedule, <1.0 = behind, >1.0 = ahead)
            cost_performance_index: CPI value (1.0 = on budget, <1.0 = over budget, >1.0 = under budget)
            scope_change_count: Number of scope changes since baseline
            quality_defect_rate: Percentage of deliverables with quality defects (0-100)
            high_risk_count: Number of high-priority active risks

        Returns:
            dict: Contains health_status, health_score, key_concerns, and recommendations

        Raises:
            ValueError: If input parameters are invalid or out of expected ranges
        """

        # Input validation
        if schedule_performance_index <= 0:
            raise ValueError("schedule_performance_index must be greater than 0")

        if cost_performance_index <= 0:
            raise ValueError("cost_performance_index must be greater than 0")

        if scope_change_count < 0:
            raise ValueError("scope_change_count must be non-negative")

        if quality_defect_rate < 0 or quality_defect_rate > 100:
            raise ValueError("quality_defect_rate must be between 0 and 100")

        if high_risk_count < 0:
            raise ValueError("high_risk_count must be non-negative")

        # Initialize lists for concerns and recommendations
        key_concerns = []
        recommendations = []

        # Calculate individual dimension scores (0-100 scale)
        # Each dimension contributes equally to the overall health score

        # 1. Schedule Health (20% weight)
        # SPI >= 1.0 is excellent (100 points), SPI < 0.8 is critical (0 points)
        if schedule_performance_index >= 1.0:
            schedule_score = 100.0
        elif schedule_performance_index >= 0.9:
            schedule_score = 80.0 + (schedule_performance_index - 0.9) * 200  # 80-100
        elif schedule_performance_index >= 0.8:
            schedule_score = 50.0 + (schedule_performance_index - 0.8) * 300  # 50-80
        else:
            schedule_score = max(0.0, schedule_performance_index * 62.5)  # 0-50

        if schedule_performance_index < 0.9:
            key_concerns.append("Schedule delay")
            if schedule_performance_index < 0.8:
                recommendations.append("Accelerate critical path tasks")
                recommendations.append("Consider adding resources to critical activities")
            else:
                recommendations.append("Monitor schedule closely and adjust resource allocation")

        # 2. Cost Health (20% weight)
        # CPI >= 1.0 is excellent (100 points), CPI < 0.8 is critical (0 points)
        if cost_performance_index >= 1.0:
            cost_score = 100.0
        elif cost_performance_index >= 0.9:
            cost_score = 80.0 + (cost_performance_index - 0.9) * 200  # 80-100
        elif cost_performance_index >= 0.8:
            cost_score = 50.0 + (cost_performance_index - 0.8) * 300  # 50-80
        else:
            cost_score = max(0.0, cost_performance_index * 62.5)  # 0-50

        if cost_performance_index < 0.9:
            key_concerns.append("Budget overrun")
            if cost_performance_index < 0.8:
                recommendations.append("Implement strict cost control measures")
                recommendations.append("Review and optimize resource utilization")
            else:
                recommendations.append("Monitor budget variance and control discretionary spending")

        # 3. Scope Health (20% weight)
        # 0-2 changes = healthy (100-80), 3-5 changes = at risk (80-50), >5 = critical (<50)
        if scope_change_count <= 2:
            scope_score = max(80.0, 100.0 - scope_change_count * 10)
        elif scope_change_count <= 5:
            scope_score = max(50.0, 80.0 - (scope_change_count - 2) * 10)
        else:
            scope_score = max(0.0, 50.0 - (scope_change_count - 5) * 8)

        if scope_change_count >= 3:
            key_concerns.append("Multiple scope changes")
            if scope_change_count >= 5:
                recommendations.append("Implement change control process")
                recommendations.append("Freeze scope and focus on current commitments")
            else:
                recommendations.append("Strengthen change management procedures")

        # 4. Quality Health (20% weight)
        # 0-5% defects = healthy (100-80), 5-15% = at risk (80-50), >15% = critical (<50)
        if quality_defect_rate <= 5.0:
            quality_score = max(80.0, 100.0 - quality_defect_rate * 4)
        elif quality_defect_rate <= 15.0:
            quality_score = max(50.0, 80.0 - (quality_defect_rate - 5.0) * 3)
        else:
            quality_score = max(0.0, 50.0 - (quality_defect_rate - 15.0) * 2)

        if quality_defect_rate > 5.0:
            key_concerns.append("Quality issues")
            if quality_defect_rate > 15.0:
                recommendations.append("Implement comprehensive quality assurance program")
                recommendations.append("Conduct root cause analysis of defects")
            else:
                recommendations.append("Enhance quality control processes")
                recommendations.append("Increase testing and review activities")

        # 5. Risk Health (20% weight)
        # 0-1 high risks = healthy (100-85), 2-3 = at risk (85-60), >3 = critical (<60)
        if high_risk_count <= 1:
            risk_score = max(85.0, 100.0 - high_risk_count * 15)
        elif high_risk_count <= 3:
            risk_score = max(60.0, 85.0 - (high_risk_count - 1) * 12.5)
        else:
            risk_score = max(0.0, 60.0 - (high_risk_count - 3) * 15)

        if high_risk_count >= 2:
            key_concerns.append("Multiple high-priority risks")
            if high_risk_count >= 3:
                recommendations.append("Develop and execute risk mitigation plans immediately")
                recommendations.append("Escalate critical risks to stakeholders")
            else:
                recommendations.append("Actively manage and monitor high-priority risks")

        # Calculate overall health score (weighted average of all dimensions)
        health_score = (
            schedule_score * 0.20 +
            cost_score * 0.20 +
            scope_score * 0.20 +
            quality_score * 0.20 +
            risk_score * 0.20
        )

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

        # Determine overall health status based on score
        if health_score >= 80.0:
            health_status = "healthy"
        elif health_score >= 60.0:
            health_status = "at_risk"
        else:
            health_status = "critical"

        # Add general recommendations based on overall health status
        if health_status == "critical":
            if "Conduct project health review with stakeholders" not in recommendations:
                recommendations.append("Conduct project health review with stakeholders")
            if "Consider project recovery plan or re-baselining" not in recommendations:
                recommendations.append("Consider project recovery plan or re-baselining")
        elif health_status == "at_risk":
            if "Increase monitoring frequency and stakeholder communication" not in recommendations:
                recommendations.append("Increase monitoring frequency and stakeholder communication")

        # If no concerns identified (healthy project), add positive note
        if not key_concerns:
            key_concerns.append("No major concerns identified")
            recommendations.append("Continue current project management practices")
            recommendations.append("Maintain proactive monitoring of all metrics")

        return {
            "health_status": health_status,
            "health_score": health_score,
            "key_concerns": key_concerns,
            "recommendations": recommendations
        }

    @is_tool()
    def update_task_progress(self, task_id: str, progress_percentage: float, status: Literal["not_started", "in_progress", "blocked", "completed", "cancelled"] = None, notes: str = None):
        """
        Update the progress percentage and status of a task.

        This method updates a task's completion percentage and optionally its status.
        The updated timestamp is automatically set to the current time.

        Args:
            task_id: Unique identifier of the task to update
            progress_percentage: New progress percentage (must be between 0-100)
            status: Optional new task status (must be one of the enum values)
            notes: Optional progress update notes (currently logged for reference)

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

        Raises:
            KeyError: If task_id does not exist in the database
            ValueError: If progress_percentage is not between 0-100
        """
        from datetime import datetime

        # Validate progress_percentage range
        if not (0 <= progress_percentage <= 100):
            raise ValueError(f"progress_percentage must be between 0 and 100, got {progress_percentage}")

        # Access the database
        db = self.db

        # Retrieve the task table
        task_table = getattr(db, "task", None)
        if task_table is None:
            raise KeyError(f"Task table not found in database")

        # Check if task exists
        if task_id not in task_table:
            raise KeyError(f"Task with task_id '{task_id}' does not exist")

        # Get the task object
        task = task_table[task_id]

        # Update the completion percentage
        task.completion_percentage = progress_percentage

        # Update status if provided
        if status is not None:
            # Safety check: ensure status is one of the allowed enum values
            allowed_statuses = ["not_started", "in_progress", "blocked", "completed", "cancelled"]
            if status not in allowed_statuses:
                raise ValueError(f"Invalid status '{status}'. Must be one of {allowed_statuses}")
            task.status = status

        # Update the timestamp to current time
        current_time = datetime.now()
        task.updated_timestamp = current_time

        # Save the updated task back to the database
        task_table[task_id] = task
        setattr(db, "task", task_table)

        # Log notes if provided (for audit trail purposes)
        if notes:
            # Notes are logged here but not persisted to the task object
            # as the schema doesn't include a notes field
            pass

        # Return success response with formatted timestamp
        return {
            "success": True,
            "updated_timestamp": current_time.strftime("%Y-%m-%d %H:%M:%S")
        }
