"""
Database schemas for project_management

This module contains all database schema classes for the project_management domain.
All classes are Pydantic BaseModel subclasses.

Generated by ClassDBAgent.
"""

import json
from datetime import datetime
from datetime import datetime, date
from pathlib import Path
from pydantic import Field
from pydantic import PrivateAttr, Field
from pydantic import PrivateAttr, Field, validator
from scale_env.data_model.thread_safe_base import ThreadSafeBase, with_instance_key
from scale_env.environment.db import DB, DictAccessMixin as BaseModel
from typing import Dict, List, Any, Optional
from typing import Dict, List, Any, Optional, Literal

@with_instance_key("project_id")
class Project(BaseModel, ThreadSafeBase["Project"]):      
    project_id: str = Field(..., description="Unique identifier for the project")
    name: str = Field(..., description="Name of the project")
    description: str = Field(..., description="Detailed description of the project")
    start_date: date = Field(..., description="Project start date")
    end_date: date = Field(..., description="Project planned end date")
    budget: Optional[float] = Field(default=None, description="Total project budget in currency units")
    project_manager: Optional[str] = Field(default=None, description="Name of the project manager")
    status: Literal["planning", "active", "on_hold", "completed", "cancelled"] = Field(default="planning", description="Current project status")
    completion_percentage: float = Field(default=0.00, description="Current completion percentage")
    creation_timestamp: datetime = Field(default_factory=datetime.now, description="Timestamp when project was created")
    updated_timestamp: datetime = Field(default_factory=datetime.now, description="Timestamp when project was last updated")

@with_instance_key("task_id")
class Task(BaseModel, ThreadSafeBase["Task"]):
    task_id: str = Field(..., description="Unique identifier for the task")
    project_id: str = Field(..., description="Reference to parent project")
    task_name: str = Field(..., description="Name of the task")
    description: Optional[str] = Field(default=None, description="Detailed description of the task")
    assigned_to: str = Field(..., description="Name or ID of the person assigned to the task")
    due_date: date = Field(..., description="Task due date")
    estimated_hours: Optional[float] = Field(default=None, description="Estimated hours to complete the task")
    priority: Literal["low", "medium", "high", "critical"] = Field(default="medium", description="Task priority level")
    status: Literal["not_started", "in_progress", "blocked", "completed", "cancelled"] = Field(default="not_started", description="Current task status")
    completion_percentage: float = Field(default=0.00, description="Task completion percentage")
    creation_timestamp: datetime = Field(default_factory=datetime.now, description="Timestamp when task was created")
    updated_timestamp: datetime = Field(default_factory=datetime.now, description="Timestamp when task was last updated")

@with_instance_key("milestone_id")
class Milestone(BaseModel, ThreadSafeBase["Milestone"]):
    milestone_id: str = Field(..., description="Unique identifier for the milestone")
    project_id: str = Field(..., description="Reference to parent project")
    milestone_name: str = Field(..., description="Name of the milestone")
    target_date: date = Field(..., description="Target completion date")
    is_completed: bool = Field(default=False, description="Whether the milestone is completed")
    actual_completion_date: Optional[date] = Field(default=None, description="Actual completion date")
    completion_notes: Optional[str] = Field(default=None, description="Notes about milestone completion")
    creation_timestamp: datetime = Field(default_factory=datetime.now, description="Timestamp when milestone was created")
    updated_timestamp: datetime = Field(default_factory=datetime.now, description="Timestamp when milestone was last updated")

@with_instance_key("deliverable_id")
class MilestoneDeliverable(BaseModel, ThreadSafeBase["MilestoneDeliverable"]):      
    deliverable_id: str = Field(..., description="Unique identifier for the deliverable")
    milestone_id: str = Field(..., description="Reference to parent milestone")
    deliverable_description: str = Field(..., description="Description of the deliverable")

@with_instance_key("risk_id")
class Risk(BaseModel, ThreadSafeBase["Risk"]):
    risk_id: str = Field(..., description="Unique identifier for the risk")
    project_id: str = Field(..., description="Reference to parent project")
    risk_name: str = Field(..., description="Name of the risk")
    description: Optional[str] = Field(default=None, description="Detailed description of the risk")
    probability: float = Field(..., description="Probability of risk occurrence (0-1)")
    impact: int = Field(..., description="Impact level if risk occurs (1-5)")
    category: Literal["technical", "resource", "schedule", "budget", "external", "quality"] = Field(
        ..., description="Risk category"
    )
    owner: str = Field(..., description="Name or ID of risk owner")
    status: Literal["identified", "assessed", "mitigated", "accepted", "occurred", "closed"] = Field(
        "identified", description="Current risk status"
    )
    risk_score: float = Field(..., description="Calculated risk score (probability × impact)")
    priority: Literal["low", "medium", "high", "critical"] = Field(
        ..., description="Risk priority level"
    )
    creation_timestamp: datetime = Field(
        default_factory=datetime.now, description="Timestamp when risk was created"
    )
    updated_timestamp: datetime = Field(
        default_factory=datetime.now, description="Timestamp when risk was last updated"
    )

@with_instance_key("action_id")
class RiskMitigationAction(BaseModel, ThreadSafeBase["RiskMitigationAction"]):
    action_id: str = Field(default=..., description="Unique identifier for the mitigation action")
    risk_id: str = Field(default=..., description="Reference to parent risk")
    action_description: str = Field(default=..., description="Description of the mitigation action")
    action_date: datetime = Field(default=..., description="When the action was taken")  # Default to CURRENT_TIMESTAMP handled by DB

@with_instance_key("allocation_id")
class ResourceAllocation(BaseModel, ThreadSafeBase["ResourceAllocation"]):
    allocation_id: str = Field(..., description="Unique identifier for the resource allocation")
    task_id: str = Field(..., description="Reference to task")
    resource_id: str = Field(..., description="Unique identifier of the resource")
    allocated_hours: float = Field(..., description="Number of hours allocated")
    start_date: datetime = Field(..., description="Allocation start datetime")
    end_date: datetime = Field(..., description="Allocation end datetime")
    creation_timestamp: datetime = Field(default_factory=datetime.now, description="Timestamp when allocation was created")

class ProjectManagementDB(DB):
    """Database containing all project_management-related data"""
    project: Optional[Dict[str, Project]] = Field(
        default=None,
        description="Schema Project"
    )
    task: Optional[Dict[str, Task]] = Field(
        default=None,
        description="Schema Task"
    )
    milestone: Optional[Dict[str, Milestone]] = Field(
        default=None,
        description="Schema Milestone"
    )
    milestone_deliverable: Optional[Dict[str, MilestoneDeliverable]] = Field(
        default=None,
        description="Schema MilestoneDeliverable"
    )
    risk: Optional[Dict[str, Risk]] = Field(
        default=None,
        description="Schema Risk"
    )
    risk_mitigation_action: Optional[Dict[str, RiskMitigationAction]] = Field(
        default=None,
        description="Schema RiskMitigationAction"
    )
    resource_allocation: Optional[Dict[str, ResourceAllocation]] = Field(
        default=None,
        description="Schema ResourceAllocation"
    )