"""
Model Configuration Migration

This migration consolidates all model configuration changes:
1. Adds display_name column to models table
2. Creates model_args and reasoning_args tables for dynamic configuration
3. Populates all configuration data with final values
4. Updates company API key mappings

Replaces migrations: 0010, 0011, 0012
"""

from django.db import migrations, models
import json


def create_model_configuration(apps, schema_editor):
    """Create model configuration system and populate with data."""
    db = schema_editor.connection
    cursor = db.cursor()
    
    print("Creating model configuration system...")
    
    # 1. Add display_name column to models table (from original 0010)
    print("  Adding display_name column...")
    cursor.execute("PRAGMA table_info(models)")
    columns = [col[1] for col in cursor.fetchall()]
    
    if 'display_name' not in columns:
        cursor.execute("ALTER TABLE models ADD COLUMN display_name TEXT")
    
    # 2. Create model_args table (from original 0010)
    print("  Creating model_args table...")
    cursor.execute("""
        CREATE TABLE IF NOT EXISTS model_args (
            id INTEGER PRIMARY KEY AUTOINCREMENT,
            model_id INTEGER NOT NULL REFERENCES models(id) ON DELETE CASCADE,
            arg_name TEXT NOT NULL,
            arg_value TEXT NOT NULL,
            created_at DATETIME DEFAULT CURRENT_TIMESTAMP,
            updated_at DATETIME DEFAULT CURRENT_TIMESTAMP,
            UNIQUE(model_id, arg_name)
        )
    """)
    
    # 3. Create reasoning_args table (from original 0010)
    print("  Creating reasoning_args table...")
    cursor.execute("""
        CREATE TABLE IF NOT EXISTS reasoning_args (
            id INTEGER PRIMARY KEY AUTOINCREMENT,
            model_id INTEGER NOT NULL REFERENCES models(id) ON DELETE CASCADE,
            arg_name TEXT NOT NULL,
            arg_value TEXT NOT NULL,
            created_at DATETIME DEFAULT CURRENT_TIMESTAMP,
            updated_at DATETIME DEFAULT CURRENT_TIMESTAMP,
            UNIQUE(model_id, arg_name)
        )
    """)
    
    # 4. Create indexes (from original 0010)
    indexes = [
        "CREATE INDEX IF NOT EXISTS idx_model_args_model ON model_args(model_id)",
        "CREATE INDEX IF NOT EXISTS idx_model_args_name ON model_args(arg_name)",
        "CREATE INDEX IF NOT EXISTS idx_reasoning_args_model ON reasoning_args(model_id)",
        "CREATE INDEX IF NOT EXISTS idx_reasoning_args_name ON reasoning_args(arg_name)",
    ]
    
    for index in indexes:
        try:
            cursor.execute(index)
        except Exception as e:
            print(f"    Note: Index creation skipped: {e}")
    
    # 5. Create triggers for timestamp updates (from original 0010)
    cursor.execute("""
        CREATE TRIGGER IF NOT EXISTS update_model_args_timestamp 
        AFTER UPDATE ON model_args
        BEGIN
            UPDATE model_args 
            SET updated_at = CURRENT_TIMESTAMP 
            WHERE id = NEW.id;
        END
    """)
    
    cursor.execute("""
        CREATE TRIGGER IF NOT EXISTS update_reasoning_args_timestamp 
        AFTER UPDATE ON reasoning_args
        BEGIN
            UPDATE reasoning_args 
            SET updated_at = CURRENT_TIMESTAMP 
            WHERE id = NEW.id;
        END
    """)
    
    # 6. Update company names and API keys (from original 0011, corrected by 0012)
    print("  Updating company configuration...")
    
    # Update company name: xai -> grok (from original 0011)
    cursor.execute("""
        UPDATE companies 
        SET company_name = 'grok' 
        WHERE company_name = 'xai'
    """)
    
    # Set final API key environment variable names (from original 0012)
    api_key_mappings = {
        'openai': 'OPENAI_API_KEY',
        'anthropic': 'ANTHROPIC_API_KEY',
        'google': 'GOOGLE_API_KEY',
        'grok': 'GROK_API_KEY'
    }
    
    for company_name, api_key_env in api_key_mappings.items():
        cursor.execute("""
            UPDATE companies 
            SET api_key = %s 
            WHERE company_name = %s
        """, [api_key_env, company_name])
        print(f"    Updated {company_name} -> {api_key_env}")
    
    # 7. Update model display names (from original 0011)
    print("  Updating model display names...")
    
    model_display_names = {
        'o4-mini-2025-04-16': 'o4-mini',
        'gemini-2.5-pro': 'Gemini 2.5 Pro',
        'grok-4-0709': 'Grok 4 Heavy',
        'o3-2025-04-16': 'o3',
        'gpt-4o-2024-11-20': 'GPT-4o',
        'claude-sonnet-4-20250514': 'Claude Sonnet 4',
        'grok-3': 'Grok 3',
        'gemini-2.5-flash': 'Gemini 2.5 Flash',
        'claude-opus-4-1-20250805': 'Claude Opus 4.1',
        'claude-3-opus-20240229': 'Claude 3 Opus (Legacy)',
        'gpt-5': 'GPT-5'
    }
    
    for model_name, display_name in model_display_names.items():
        cursor.execute("""
            UPDATE models 
            SET display_name = %s 
            WHERE model_name = %s
        """, [display_name, model_name])
        if cursor.rowcount > 0:
            print(f"    Updated {model_name} -> {display_name}")
    
    # 8. Populate model arguments (from original 0011)
    print("  Populating model arguments...")
    
    # Model configurations from the original hardcoded config
    # Note: reasoning_args are passed to the eval() function
    # model_args are for client initialization (e.g., API configuration)
    # Fixed: o3 parameters moved from model_args to reasoning_args where they belong
    model_configurations = {
        'o4-mini-2025-04-16': {
            'model_args': {},
            'reasoning_args': {
                'reasoning_effort': 'high',
                'reasoning_summary': 'auto'
            }
        },
        'gemini-2.5-pro': {
            'model_args': {},
            'reasoning_args': {
                'reasoning_tokens': 32768
            }
        },
        'grok-4-0709': {
            'model_args': {},
            'reasoning_args': {}
        },
        'o3-2025-04-16': {
            'model_args': {},
            'reasoning_args': {
                'reasoning_effort': 'high',
                'reasoning_tokens': 100000,
                'reasoning_summary': 'auto',
                'reasoning_history': 'auto'
            }
        },
        'gpt-4o-2024-11-20': {
            'model_args': {},
            'reasoning_args': {}
        },
        'claude-sonnet-4-20250514': {
            'model_args': {},
            'reasoning_args': {
                'max_tokens': 64000,
                'cache_prompt': 'auto',
                'reasoning_tokens': 63000
            }
        },
        'grok-3': {
            'model_args': {},
            'reasoning_args': {}
        },
        'gemini-2.5-flash': {
            'model_args': {},
            'reasoning_args': {
                'reasoning_tokens': 24576
            }
        },
        'claude-opus-4-1-20250805': {
            'model_args': {},
            'reasoning_args': {
                'max_tokens': 32000,
                'cache_prompt': 'auto',
                'reasoning_tokens': 31000
            }
        },
        'claude-3-opus-20240229': {
            'model_args': {},
            'reasoning_args': {}
        },
        'gpt-5': {
            'model_args': {},
            'reasoning_args': {
                'reasoning_effort': 'high',
                'reasoning_summary': 'auto'
            }
        }
    }
    
    for model_name, config in model_configurations.items():
        # Get model ID
        cursor.execute("SELECT id FROM models WHERE model_name = %s", [model_name])
        result = cursor.fetchone()
        if not result:
            print(f"    Warning: Model '{model_name}' not found, skipping configuration")
            continue
        
        model_id = result[0]
        
        # Insert model_args
        for arg_name, arg_value in config['model_args'].items():
            cursor.execute("""
                INSERT OR REPLACE INTO model_args 
                (model_id, arg_name, arg_value, created_at, updated_at)
                VALUES (%s, %s, %s, datetime('now'), datetime('now'))
            """, [model_id, arg_name, json.dumps(arg_value)])
        
        # Insert reasoning_args
        for arg_name, arg_value in config['reasoning_args'].items():
            cursor.execute("""
                INSERT OR REPLACE INTO reasoning_args 
                (model_id, arg_name, arg_value, created_at, updated_at)
                VALUES (%s, %s, %s, datetime('now'), datetime('now'))
            """, [model_id, arg_name, json.dumps(arg_value)])
        
        total_args = len(config['model_args']) + len(config['reasoning_args'])
        if total_args > 0:
            print(f"    Configured {model_name}: {total_args} arguments")
    
    print("✅ Model configuration system created successfully!")


def reverse_model_configuration(apps, schema_editor):
    """Reverse the model configuration migration."""
    db = schema_editor.connection
    cursor = db.cursor()
    
    # Drop configuration tables
    cursor.execute("DROP TABLE IF EXISTS reasoning_args")
    cursor.execute("DROP TABLE IF EXISTS model_args")
    
    # Remove display_name column (SQLite doesn't support DROP COLUMN easily)
    # This would require table recreation - left as manual operation if needed
    
    # Revert company name changes
    cursor.execute("UPDATE companies SET company_name = 'xai' WHERE company_name = 'grok'")
    
    print("Model configuration reversed.")


class Migration(migrations.Migration):

    dependencies = [
        ('model_evaluation', '0003_enhanced_features'),
    ]

    operations = [
        migrations.RunPython(
            create_model_configuration,
            reverse_model_configuration,
        ),
        
        # Create Django model definitions for the configuration system
        migrations.CreateModel(
            name='ModelArg',
            fields=[
                ('id', models.AutoField(primary_key=True, serialize=False)),
                ('arg_name', models.TextField()),
                ('arg_value', models.TextField()),
                ('created_at', models.DateTimeField(auto_now_add=True)),
                ('updated_at', models.DateTimeField(auto_now=True)),
            ],
            options={
                'db_table': 'model_args',
                'ordering': ['arg_name'],
                'managed': False,
            },
        ),
        migrations.CreateModel(
            name='ReasoningArg',
            fields=[
                ('id', models.AutoField(primary_key=True, serialize=False)),
                ('arg_name', models.TextField()),
                ('arg_value', models.TextField()),
                ('created_at', models.DateTimeField(auto_now_add=True)),
                ('updated_at', models.DateTimeField(auto_now=True)),
            ],
            options={
                'db_table': 'reasoning_args',
                'ordering': ['arg_name'],
                'managed': False,
            },
        ),
    ]