"""ρ manager - computes and manages low-fidelity scaling factor"""
import numpy as np


class RhoManager:
    """ρ manager - manages scaling factor for low-fidelity model."""
    
    def __init__(self):
        """Initialize ρ manager."""
        self.rho = 1.0
    
    def compute_rho(self, y_H: np.ndarray, mu_LF: np.ndarray, iteration: int = 0) -> float:
        """Compute ρ based on least squares (scaling matching).
        
        ρ* = argmin_ρ Σ (y_H(i) - ρ * y_L(i))^2 = Σ y_L(i) y_H(i) / Σ (y_L(i))^2
        
        Args:
            y_H: High-fidelity true values, shape (n,)
            mu_LF: Low-fidelity prediction mean, shape (n,)
            iteration: Iteration number (for debugging)
        
        Returns:
            ρ value
        """
        n_samples = len(y_H)
        
        if n_samples == 0 or len(mu_LF) == 0:
            raise ValueError("Input data cannot be empty")
        
        if n_samples != len(mu_LF):
            raise ValueError("y_H and mu_LF must have the same length")
        
        numerator = float(np.dot(mu_LF, y_H))
        denominator = float(np.dot(mu_LF, mu_LF))
        
        if denominator < 1e-12 or not np.isfinite(denominator):
            self.rho = 1.0
            return self.rho
        
        rho_ls = numerator / denominator
        
        if not np.isfinite(rho_ls):
            self.rho = 1.0
            return self.rho
        
        rho_clamped = float(np.clip(rho_ls, 0.7, 1.5))
        self.rho = rho_clamped
        
        return self.rho
    
    def get_rho(self) -> float:
        """Get current ρ value.
        
        Returns:
            ρ value
        """
        return self.rho
    
    def set_rho(self, rho: float):
        """Manually set ρ value.
        
        Args:
            rho: ρ value
        """
        self.rho = rho

