def get_auto_temperature(n_components=10):
    min_temp = 0.035
    max_temp = 0.3
    min_n = 5  # n_components threshold for max_temp
    max_n = 20  # n_components threshold for min_temp

    if n_components <= min_n:
        temperature = max_temp
    elif n_components >= max_n:
        temperature = min_temp
    else:
        # Linearly scale temperature between max_temp and min_temp
        # as n_components goes from min_n to max_n
        scale_factor = (n_components - min_n) / (max_n - min_n)
        temperature = max_temp - scale_factor * (max_temp - min_temp)
    return temperature


class TemperatureAnnealer:
    """
    Smooth temperature annealing scheduler that can be queried for any epoch.

    Provides linear interpolation between start and end temperatures within
    the specified epoch fraction range. Outside this range, returns constant
    start or end temperatures.
    """

    def __init__(
        self,
        start_temp: float,
        end_temp: float,
        start_step: float = 0.2,
        end_step: float = 0.8,
    ):
        """
        Initialize temperature annealer.

        Args:
            start_temp: Initial temperature value
            end_temp: Final temperature value
            start_step: Epoch fraction when annealing begins (0.0-1.0)
            end_step: Epoch fraction when annealing ends (0.0-1.0)
        """
        self.start_temp = start_temp
        self.end_temp = end_temp
        self.start_step = start_step
        self.end_step = end_step

        if start_step >= end_step:
            raise ValueError("start_step must be less than end_step")
        if not (0.0 <= start_step <= 1.0) or not (0.0 <= end_step <= 1.0):
            raise ValueError("step fractions must be between 0.0 and 1.0")

    def get_temperature(self, epoch: int, total_epochs: int) -> float:
        """
        Get temperature for the given epoch.

        Args:
            epoch: Current epoch (0-based)
            total_epochs: Total number of epochs

        Returns:
            Temperature value for this epoch
        """
        epoch_fraction = epoch / total_epochs

        if epoch_fraction <= self.start_step:
            return self.start_temp
        elif epoch_fraction >= self.end_step:
            return self.end_temp
        else:
            # Linear interpolation between start and end
            progress = (epoch_fraction - self.start_step) / (
                self.end_step - self.start_step
            )
            return self.start_temp + progress * (self.end_temp - self.start_temp)

    def __call__(self, epoch: int, total_epochs: int) -> float:
        """Allow callable interface."""
        return self.get_temperature(epoch, total_epochs)

    @classmethod
    def from_dict(cls, config: dict) -> "TemperatureAnnealer":
        """Create annealer from configuration dictionary."""
        return cls(**config)

    def to_dict(self) -> dict:
        """Convert annealer to configuration dictionary."""
        return {
            "start_temp": self.start_temp,
            "end_temp": self.end_temp,
            "start_step": self.start_step,
            "end_step": self.end_step,
        }

    def __str__(self):
        """String representation for debugging."""
        return (
            f"TemperatureAnnealer(start_temp={self.start_temp}, "
            f"end_temp={self.end_temp}, start_step={self.start_step}, "
            f"end_step={self.end_step})"
        )


def create_auto_temperature_annealer(
    start_temp: float,
    end_temp: float = 0.005,
    start_step: float = 0.1,
    end_step: float = 0.9,
) -> TemperatureAnnealer:
    """
    Create a TemperatureAnnealer

    Args:
        start_temp: Initial temperature
        end_temp: Final temperature
        start_step: When to start annealing
        end_step: When to finish annealing
    """
    return TemperatureAnnealer(start_temp, end_temp, start_step, end_step)
