"""
Complete evaluation system setup migration.

This migration:
1. Creates all necessary tables for the model evaluation system
2. Populates companies, tiers, and models with dynamic ID handling
3. Sets up all indexes and constraints
"""

from django.db import migrations


def create_evaluation_system(apps, schema_editor):
    """Create all tables and populate initial data."""
    db = schema_editor.connection
    cursor = db.cursor()
    
    print("Creating evaluation system tables...")
    
    # 1. Create companies table (or add api_key column if missing)
    cursor.execute("""
        CREATE TABLE IF NOT EXISTS companies (
            id INTEGER PRIMARY KEY AUTOINCREMENT,
            company_name TEXT,
            api_key TEXT
        )
    """)
    
    # Check if api_key column exists (main branch doesn't have it in some versions)
    cursor.execute("PRAGMA table_info(companies)")
    columns = [col[1] for col in cursor.fetchall()]
    if 'api_key' not in columns:
        cursor.execute("ALTER TABLE companies ADD COLUMN api_key TEXT")
    
    # 2. Create model_tiers table
    cursor.execute("""
        CREATE TABLE IF NOT EXISTS model_tiers (
            id INTEGER PRIMARY KEY AUTOINCREMENT,
            tier_number INTEGER NOT NULL CHECK (tier_number >= 1 AND tier_number <= 5),
            tier_name TEXT NOT NULL,
            description TEXT,
            requires_main_grading BOOLEAN DEFAULT 1,
            appears_on_leaderboard BOOLEAN DEFAULT 1,
            is_active BOOLEAN DEFAULT 1
        )
    """)
    
    # 3. Handle models table - it might already exist from main branch
    cursor.execute("SELECT name FROM sqlite_master WHERE type='table' AND name='models'")
    if cursor.fetchone():
        # Table exists - check if it has our new columns
        cursor.execute("PRAGMA table_info(models)")
        columns = [col[1] for col in cursor.fetchall()]
        
        if 'tier_id' not in columns:
            cursor.execute("ALTER TABLE models ADD COLUMN tier_id INTEGER REFERENCES model_tiers(id)")
        if 'framework_type' not in columns:
            cursor.execute("ALTER TABLE models ADD COLUMN framework_type TEXT CHECK (framework_type IN ('inspect', 'cli'))")
        if 'is_active' not in columns:
            cursor.execute("ALTER TABLE models ADD COLUMN is_active BOOLEAN DEFAULT 1")
    else:
        # Create new table
        cursor.execute("""
            CREATE TABLE models (
                id INTEGER PRIMARY KEY AUTOINCREMENT,
                company_id INTEGER REFERENCES companies(id),
                model_name TEXT,
                instructions TEXT,
                tier_id INTEGER REFERENCES model_tiers(id),
                framework_type TEXT CHECK (framework_type IN ('inspect', 'cli')),
                is_active BOOLEAN DEFAULT 1
            )
        """)
    
    # 4. Handle model_attempts table - it might already exist from main branch
    cursor.execute("SELECT name FROM sqlite_master WHERE type='table' AND name='model_attempts'")
    if cursor.fetchone():
        # Table exists - check if it has our new columns
        cursor.execute("PRAGMA table_info(model_attempts)")
        columns = [col[1] for col in cursor.fetchall()]
        
        if 'question_id' not in columns:
            cursor.execute("ALTER TABLE model_attempts ADD COLUMN question_id INTEGER REFERENCES questions(id)")
        if 'attempt_number' not in columns:
            cursor.execute("ALTER TABLE model_attempts ADD COLUMN attempt_number INTEGER DEFAULT 1 CHECK (attempt_number >= 1 AND attempt_number <= 2)")
    else:
        # Create new table
        cursor.execute("""
            CREATE TABLE model_attempts (
                id INTEGER PRIMARY KEY AUTOINCREMENT,
                model_id INTEGER REFERENCES models(id),
                question_id INTEGER REFERENCES questions(id),
                time DATETIME DEFAULT CURRENT_TIMESTAMP,
                attempt_number INTEGER DEFAULT 1 CHECK (attempt_number >= 1 AND attempt_number <= 2)
            )
        """)
    
    # 5. Handle model_answers table - it exists but needs new columns
    cursor.execute("SELECT name FROM sqlite_master WHERE type='table' AND name='model_answers'")
    if cursor.fetchone():
        # Table exists - add missing columns
        cursor.execute("PRAGMA table_info(model_answers)")
        columns = [col[1] for col in cursor.fetchall()]
        
        if 'model_id' not in columns:
            cursor.execute("ALTER TABLE model_answers ADD COLUMN model_id INTEGER REFERENCES models(id)")
        if 'terminal_log_hash' not in columns:
            cursor.execute("ALTER TABLE model_answers ADD COLUMN terminal_log_hash TEXT")
        if 'creation_time' not in columns:
            cursor.execute("ALTER TABLE model_answers ADD COLUMN creation_time DATETIME")
        if 'last_modified' not in columns:
            cursor.execute("ALTER TABLE model_answers ADD COLUMN last_modified DATETIME")
    else:
        # Create new table
        cursor.execute("""
            CREATE TABLE model_answers (
                id INTEGER PRIMARY KEY AUTOINCREMENT,
                attempt_id INTEGER REFERENCES model_attempts(id),
                question_id INTEGER REFERENCES questions(id),
                model_id INTEGER REFERENCES models(id),
                answer TEXT,
                terminal_log_hash TEXT,
                creation_time DATETIME DEFAULT CURRENT_TIMESTAMP,
                last_modified DATETIME DEFAULT CURRENT_TIMESTAMP
            )
        """)
    
    # 6. Handle model_subquestion_answers table - it exists but needs new columns
    cursor.execute("SELECT name FROM sqlite_master WHERE type='table' AND name='model_subquestion_answers'")
    if cursor.fetchone():
        # Table exists - add missing columns
        cursor.execute("PRAGMA table_info(model_subquestion_answers)")
        columns = [col[1] for col in cursor.fetchall()]
        
        if 'model_id' not in columns:
            cursor.execute("ALTER TABLE model_subquestion_answers ADD COLUMN model_id INTEGER REFERENCES models(id)")
        if 'creation_time' not in columns:
            cursor.execute("ALTER TABLE model_subquestion_answers ADD COLUMN creation_time DATETIME")
        if 'last_modified' not in columns:
            cursor.execute("ALTER TABLE model_subquestion_answers ADD COLUMN last_modified DATETIME")
    else:
        # Create new table
        cursor.execute("""
            CREATE TABLE model_subquestion_answers (
                id INTEGER PRIMARY KEY AUTOINCREMENT,
                attempt_id INTEGER REFERENCES model_attempts(id),
                subquestion_id INTEGER REFERENCES subquestions(id),
                model_id INTEGER REFERENCES models(id),
                answer TEXT,
                is_correct INTEGER,
                creation_time DATETIME DEFAULT CURRENT_TIMESTAMP,
                last_modified DATETIME DEFAULT CURRENT_TIMESTAMP
            )
        """)
    
    # 7. Create evaluation_queue table
    cursor.execute("""
        CREATE TABLE IF NOT EXISTS evaluation_queue (
            id INTEGER PRIMARY KEY AUTOINCREMENT,
            attempt_id INTEGER REFERENCES model_attempts(id),
            status TEXT DEFAULT 'pending' CHECK (status IN ('pending', 'running', 'completed', 'failed', 'cancelled')),
            submitted_at DATETIME DEFAULT CURRENT_TIMESTAMP,
            started_at DATETIME,
            completed_at DATETIME,
            error_message TEXT,
            estimated_runtime_seconds INTEGER,
            actual_runtime_seconds INTEGER
        )
    """)
    
    # 8. Create company_execution_locks table
    cursor.execute("""
        CREATE TABLE IF NOT EXISTS company_execution_locks (
            id INTEGER PRIMARY KEY AUTOINCREMENT,
            company_id INTEGER UNIQUE REFERENCES companies(id),
            is_locked BOOLEAN DEFAULT 0,
            locked_at DATETIME,
            locked_by_queue_id INTEGER REFERENCES evaluation_queue(id)
        )
    """)
    
    # 9. Create terminal_logs table
    cursor.execute("""
        CREATE TABLE IF NOT EXISTS terminal_logs (
            id INTEGER PRIMARY KEY AUTOINCREMENT,
            hash TEXT UNIQUE NOT NULL,
            content TEXT NOT NULL,
            created_at DATETIME DEFAULT CURRENT_TIMESTAMP
        )
    """)
    
    # 10. Create model_answer_gradings table with ungradable handling
    cursor.execute("""
        CREATE TABLE IF NOT EXISTS model_answer_gradings (
            id INTEGER PRIMARY KEY AUTOINCREMENT,
            model_answer_id INTEGER REFERENCES model_answers(id),
            grader_id INTEGER REFERENCES participants(id),
            grading_scheme_id INTEGER REFERENCES grading_schemes(id),
            total_score REAL,
            grading_weight REAL DEFAULT 1.0,
            created_at DATETIME DEFAULT CURRENT_TIMESTAMP,
            notes TEXT,
            is_skipped BOOLEAN DEFAULT 0,
            skip_reason TEXT,
            skip_comment TEXT
        )
    """)
    
    print("Tables created successfully!")
    
    # Now populate initial data
    print("Populating companies...")
    
    # Insert companies without depending on specific IDs
    companies_data = [
        'openai',
        'anthropic',
        'google',
        'xai'
    ]
    
    for company_name in companies_data:
        # Check if company already exists
        cursor.execute("SELECT id FROM companies WHERE company_name = %s", [company_name])
        if not cursor.fetchone():
            # Only insert if it doesn't exist
            cursor.execute("INSERT INTO companies (company_name) VALUES (%s)", [company_name])
    
    # Get company IDs dynamically
    company_ids = {}
    for company_name in companies_data:
        cursor.execute("SELECT id FROM companies WHERE company_name = %s", [company_name])
        result = cursor.fetchone()
        if result:
            company_ids[company_name] = result[0]
            print(f"  {company_name}: ID {result[0]}")
    
    print("Populating model tiers...")
    
    # Populate tiers
    tiers_data = [
        (1, 1, 'Priority Models', 'Top priority models requiring thorough evaluation', 1, 1, 1),
        (2, 2, 'Standard Models', 'Standard models with normal evaluation requirements', 1, 1, 1),
        (3, 3, 'Basic Models', 'Basic models requiring minimal evaluation', 0, 1, 1),
        (4, 4, 'Experimental Models', 'Experimental or testing models', 0, 0, 1),
        (5, 5, 'Retired Models', 'Models no longer actively evaluated', 0, 0, 0),
    ]
    
    for tier_data in tiers_data:
        cursor.execute("""
            INSERT OR IGNORE INTO model_tiers 
            (id, tier_number, tier_name, description, requires_main_grading, appears_on_leaderboard, is_active)
            VALUES (%s, %s, %s, %s, %s, %s, %s)
        """, tier_data)
    
    print("Populating models...")
    
    # Clear all dependent tables first to avoid foreign key violations
    tables_to_clear = [
        'step_feedback',  # This depends on answer_reviews
        'answer_reviews',  # This depends on model_answers
        'model_answer_gradings',
        'model_subquestion_answers', 
        'model_answers',
        'evaluation_queue',
        'model_attempts',
        'models'
    ]
    
    for table in tables_to_clear:
        try:
            cursor.execute(f"DELETE FROM {table}")
        except:
            pass  # Table might not exist yet
    
    # Reset the models table completely
    
    # Models data with company names instead of IDs
    models_data = [
        # (model_id, company_name, model_name, instructions, tier_id, framework_type)
        (2, 'anthropic', 'Claude Code', 'Carefully read the instructions in `PATH_TO/INSTRUCTIONS.md` and follow them.', None, None),
        (3, 'google', 'Gemini', 'Google Gemini via gemini-cli', None, None),
        (4, 'openai', 'o3-mini', 'AI model for mathematical reasoning and problem solving', None, None),
        # Removed incorrect entry: (5, 'deepmind', 'Grok 4-Heavy') - Grok models belong to xAI/grok company
        (7, 'openai', 'o4-mini-2025-04-16', None, 3, 'inspect'),  # Changed to Tier 3
        (8, 'google', 'gemini-2.5-pro', None, 1, 'inspect'),
        (9, 'xai', 'grok-4-0709', None, 1, 'inspect'),
        # Removed CLI models - Tier 2 not needed currently
        (12, 'openai', 'o3-2025-04-16', None, 4, 'inspect'),  # Changed to Tier 4
        (13, 'openai', 'gpt-4o-2024-11-20', None, 4, 'inspect'),
        (14, 'anthropic', 'claude-sonnet-4-20250514', None, 4, 'inspect'),  # Updated to Claude Sonnet 4
        (15, 'xai', 'grok-3', None, 4, 'inspect'),
        (16, 'google', 'gemini-2.5-flash', None, 4, 'inspect'),  # Added Gemini 2.5 Flash
        (17, 'anthropic', 'claude-opus-4-1-20250805', None, 1, 'inspect'),  # Claude Opus 4.1
        (18, 'openai', 'gpt-5', None, 1, 'inspect'),  # GPT-5
    ]
    
    for model_data in models_data:
        model_id, company_name, model_name, instructions, tier_id, framework_type = model_data
        
        # Look up company ID dynamically
        company_id = company_ids.get(company_name)
        if not company_id:
            print(f"  Warning: Company '{company_name}' not found, skipping model {model_name}")
            continue
        
        cursor.execute("""
            INSERT OR IGNORE INTO models 
            (id, company_id, model_name, instructions, tier_id, framework_type, is_active)
            VALUES (%s, %s, %s, %s, %s, %s, 1)
        """, [model_id, company_id, model_name, instructions, tier_id, framework_type])
        
        tier_info = f"Tier {tier_id}" if tier_id else "No tier"
        print(f"  Model {model_id}: {model_name} ({company_name}, {tier_info})")
    
    print("Creating indexes...")
    
    # Create indexes for better performance
    indexes = [
        "CREATE INDEX IF NOT EXISTS idx_models_tier ON models(tier_id)",
        "CREATE INDEX IF NOT EXISTS idx_models_company ON models(company_id)",
        "CREATE INDEX IF NOT EXISTS idx_models_active ON models(is_active)",
        "CREATE INDEX IF NOT EXISTS idx_attempts_model_question ON model_attempts(model_id, question_id)",
        "CREATE INDEX IF NOT EXISTS idx_attempts_question ON model_attempts(question_id)",
        "CREATE INDEX IF NOT EXISTS idx_attempts_model ON model_attempts(model_id)",
        "CREATE INDEX IF NOT EXISTS idx_evaluation_queue_status ON evaluation_queue(status)",
        "CREATE INDEX IF NOT EXISTS idx_evaluation_queue_attempt ON evaluation_queue(attempt_id)",
        "CREATE INDEX IF NOT EXISTS idx_model_answers_attempt ON model_answers(attempt_id)",
        "CREATE INDEX IF NOT EXISTS idx_model_subquestion_answers_attempt ON model_subquestion_answers(attempt_id)",
    ]
    
    for index in indexes:
        try:
            cursor.execute(index)
        except Exception as e:
            print(f"  Note: Index creation skipped or already exists: {e}")
    
    print("\n✅ Evaluation system setup complete!")
    
    # Show summary
    cursor.execute("SELECT COUNT(*) FROM companies")
    company_count = cursor.fetchone()[0]
    
    cursor.execute("SELECT COUNT(*) FROM models")
    model_count = cursor.fetchone()[0]
    
    cursor.execute("SELECT COUNT(*) FROM models WHERE tier_id IS NOT NULL")
    tiered_model_count = cursor.fetchone()[0]
    
    print(f"\nSummary:")
    print(f"  Companies: {company_count}")
    print(f"  Models: {model_count}")
    print(f"  Models with tiers: {tiered_model_count}")
    
    # Promote all admin users to Django staff
    print("\nPromoting admin users to Django staff...")
    
    # First promote superusers
    cursor.execute("""
        UPDATE accounts_customuser 
        SET is_staff = 1 
        WHERE is_superuser = 1 AND is_staff = 0
    """)
    superuser_count = cursor.rowcount
    
    # Also promote users with 'admin' role in participants table
    cursor.execute("""
        UPDATE accounts_customuser 
        SET is_staff = 1, is_superuser = 1
        WHERE id IN (
            SELECT user_id FROM participants 
            WHERE role = 'admin' AND user_id IS NOT NULL
        ) AND (is_staff = 0 OR is_superuser = 0)
    """)
    participant_admin_count = cursor.rowcount
    
    total_promoted = superuser_count + participant_admin_count
    if total_promoted > 0:
        print(f"  ✅ Promoted {total_promoted} user(s) to staff/admin")
        if superuser_count > 0:
            print(f"     - {superuser_count} existing superuser(s)")
        if participant_admin_count > 0:
            print(f"     - {participant_admin_count} participant admin(s)")
    else:
        print("  ℹ️  All admin users already have staff privileges")


def reverse_evaluation_system(apps, schema_editor):
    """Reverse the migration by dropping all evaluation tables."""
    db = schema_editor.connection
    cursor = db.cursor()
    
    # Drop tables in reverse order of dependencies
    tables_to_drop = [
        'model_answer_gradings',
        'terminal_logs',
        'company_execution_locks',
        'evaluation_queue',
        'model_subquestion_answers',
        'model_answers',
        'model_attempts',
        'models',
        'model_tiers',
        'companies',
    ]
    
    for table in tables_to_drop:
        cursor.execute(f"DROP TABLE IF EXISTS {table}")
    
    print("Evaluation system tables dropped.")


class Migration(migrations.Migration):
    
    dependencies = [
        ('model_evaluation', '0001_initial'),
    ]
    
    operations = [
        migrations.RunPython(
            create_evaluation_system,
            reverse_evaluation_system,
        ),
    ]