"""
Utilities for analyzing polynomials using Newton polytopes and computing polytope statistics.
"""

import json
import numpy as np
import re
from typing import Dict, List, Any, Tuple
from sos_transformer.utils.polynomial import get_newton_polytope_basis
from sos_transformer.utils.polytope_characterizer import analyze_lattice_polytope

def parse_monomial(monomial: str, num_vars: int = 5) -> List[int]:
    """
    Parse a monomial string into its exponent vector.
    Example: "x1^2*x3^4" -> [2, 0, 4, 0, 0] for 5 variables
    
    Args:
        monomial: String representation of monomial (e.g., "x1^2*x3^4")
        num_vars: Number of variables in the polynomial system
        
    Returns:
        List of exponents for each variable
    """
    exponents = [0] * num_vars
    # Split by multiplication
    if not monomial or monomial == "1":
        return exponents
        
    parts = monomial.split('*')
    for part in parts:
        if '^' in part:
            var, exp = part.split('^')
            var_idx = int(var[1]) - 1  # x1 -> index 0
            exponents[var_idx] = int(exp)
        else:
            var_idx = int(part[1]) - 1  # x1 -> index 0
            exponents[var_idx] = 1
            
    return exponents

def parse_polynomial(poly_str: str, num_vars: int = 5) -> List[List[int]]:
    """
    Parse a polynomial string into a list of exponent vectors.
    
    Args:
        poly_str: String representation of polynomial
        num_vars: Number of variables in the polynomial system
        
    Returns:
        List of exponent vectors
    """
    # Split into terms
    terms = re.findall(r'[-+]?\s*\d*\.?\d*\*?(?:x\d(?:\^\d+)?(?:\*x\d(?:\^\d+)?)*)', poly_str)
    exponents = []
    
    for term in terms:
        # Extract the monomial part (everything after the coefficient)
        parts = term.split('*', 1)
        if len(parts) == 1:  # Constant term
            exponents.append([0] * num_vars)
            continue
            
        monomial = parts[1]
        exponents.append(parse_monomial(monomial, num_vars))
        
    return exponents

def analyze_polynomial_polytope(exponent_vectors: np.ndarray) -> Dict[str, Any]:
    """
    Analyze a polynomial's Newton polytope and compute various statistics.
    
    Args:
        exponent_vectors: Array where each row is an exponent vector of a term in the polynomial
        
    Returns:
        Dictionary containing various metrics and properties of the polytope
    """
    # Get the Newton polytope vertices using the existing function
    basis = get_newton_polytope_basis(exponent_vectors)
    if basis is None:
        return {
            'success': False,
            'error': 'Failed to compute Newton polytope basis'
        }
        
    # Convert basis points to numpy array if not already
    basis_points = np.array(basis)
    
    # Analyze the polytope using existing characterizer
    try:
        polytope_stats = analyze_lattice_polytope(basis_points)
        
        # Combine results
        return {
            'success': True,
            'n_basis_elements': len(basis),
            'n_vertices': polytope_stats['n_vertices'],
            'n_interior_points': polytope_stats['n_interior_points'],
            'volume': float(polytope_stats['volume']),  # Convert to float for JSON serialization
            'coordinate_means': polytope_stats['means'].tolist(),
            'coordinate_variances': polytope_stats['variances'].tolist()
        }
    except Exception as e:
        return {
            'success': False,
            'error': str(e)
        }

def analyze_polynomials_from_jsonl(input_file: str, output_file: str):
    """
    Load polynomials from a JSONL file, analyze their polytopes, and save results.
    
    Args:
        input_file: Path to input JSONL file containing polynomials
        output_file: Path to output JSONL file for results
        
    Each input line should be a JSON object with a 'polynomial' field containing
    the polynomial string representation.
    """
    with open(input_file, 'r') as f_in, open(output_file, 'w') as f_out:
        for line_num, line in enumerate(f_in, 1):
            try:
                # Load polynomial data
                poly_data = json.loads(line)
                
                # Extract polynomial string and parse it
                poly_str = poly_data['polynomial']
                exponents = parse_polynomial(poly_str)

                basis_str = poly_data['basis']
                basis_exponents = parse_polynomial(basis_str)


                
                # Analyze the polynomial
                analysis = analyze_polynomial_polytope(np.array(exponents))

                analysis_basis = analyze_polynomial_polytope(np.array(basis_exponents))
                
                # Combine original data with analysis
                result = {
                    'polytope_analysis': analysis,
                    'parsed_exponents': exponents,
                    'basis_analysis': analysis_basis,
                    'basis_exponents': basis_exponents
                }


                
                # Write result
                json.dump(result, f_out)
                f_out.write('\n')
                
            except Exception as e:
                # Log error but continue processing
                error_result = {
                    'line_number': line_num,
                    'error': str(e),
                    'original_data': line.strip()
                }
                json.dump(error_result, f_out)
                f_out.write('\n')
