"""
Algorithms Package for SP-UCB-OLP Framework

This package contains implementations of:

Main Algorithm:
- SPUCBOLP: Saddle-Point UCB-OLP (main algorithm from paper)

Baselines:
- SPGreedyOLP: Greedy without exploration (alpha=0)
- OneHotSPUCBOLP: Per-config UCB (wrong abstraction)
- OraclePolicy: Perfect information upper bound
- RandomPolicy: Uniform random lower bound
- FixedConfigPolicy: Single fixed config

Ablation Variants:
- EnvelopeGreedyOLP: No mixture sampling (uses argmax instead)
- MixtureLocalPriceOLP: Per-config prices instead of global saddle-point
- NoSlackSPUCBOLP: No slack (epsilon=0)
- AcceptedOnlySPUCBOLP: Selection bias (estimates from accepted samples only)
"""

from .base import BaseAlgorithm
from .sp_ucb_olp import SPUCBOLP
from .baselines import (
    SPGreedyOLP,
    OneHotSPUCBOLP,
    OraclePolicy,
    RandomPolicy,
    FixedConfigPolicy,
    EnvelopeGreedyOLP,
    MixtureLocalPriceOLP,
    NoSlackSPUCBOLP,
    AcceptedOnlySPUCBOLP,
)


__all__ = [
    'BaseAlgorithm',
    'SPUCBOLP',
    'SPGreedyOLP',
    'OneHotSPUCBOLP',
    'OraclePolicy',
    'RandomPolicy',
    'FixedConfigPolicy',
    # Ablation variants
    'EnvelopeGreedyOLP',
    'MixtureLocalPriceOLP',
    'NoSlackSPUCBOLP',
    'AcceptedOnlySPUCBOLP',
]


def get_algorithm(name: str, K: int, d: int, T: int, B, config=None):
    """
    Factory function to create algorithm by name.

    Parameters
    ----------
    name : str
        Algorithm name: 'SP-UCB-OLP', 'Greedy', 'OneHot', 'Oracle', 'Random', 'Fixed'
    K : int
        Number of configurations
    d : int
        Number of resource dimensions
    T : int
        Time horizon
    B : np.ndarray
        Total budget vector
    config : dict, optional
        Algorithm configuration

    Returns
    -------
    BaseAlgorithm
        Instantiated algorithm
    """
    import numpy as np
    B = np.asarray(B)

    algorithms = {
        'SP-UCB-OLP': SPUCBOLP,
        'SPUCBOLP': SPUCBOLP,
        'Greedy': SPGreedyOLP,
        'SPGreedyOLP': SPGreedyOLP,
        'OneHot': OneHotSPUCBOLP,
        'OneHotSPUCBOLP': OneHotSPUCBOLP,
        'Oracle': OraclePolicy,
        'OraclePolicy': OraclePolicy,
        'Random': RandomPolicy,
        'RandomPolicy': RandomPolicy,
        'Fixed': FixedConfigPolicy,
        'FixedConfigPolicy': FixedConfigPolicy,
        # Ablation variants
        'EnvelopeGreedy': EnvelopeGreedyOLP,
        'EnvelopeGreedyOLP': EnvelopeGreedyOLP,
        'MixtureLocalPrice': MixtureLocalPriceOLP,
        'MixtureLocalPriceOLP': MixtureLocalPriceOLP,
        'NoSlack': NoSlackSPUCBOLP,
        'NoSlackSPUCBOLP': NoSlackSPUCBOLP,
        'AcceptedOnly': AcceptedOnlySPUCBOLP,
        'AcceptedOnlySPUCBOLP': AcceptedOnlySPUCBOLP,
    }

    if name not in algorithms:
        raise ValueError(f"Unknown algorithm: {name}. Available: {list(algorithms.keys())}")

    return algorithms[name](K, d, T, B, config)
