"""
Core set selection algorithms.
"""
import random
import numpy as np
from typing import List, Callable


def k_center(items: List, k: int, dist_fn: Callable) -> List[int]:
    """
    K-Center greedy selection.

    Selects k items maximizing minimum distance to nearest selected item.

    Args:
        items: Input items
        k: Number to select
        dist_fn: Distance function(item1, item2) -> float

    Returns:
        Selected indices
    """
    n = len(items)
    if k >= n:
        return list(range(n))

    # Precompute distances
    dist = np.zeros((n, n))
    for i in range(n):
        for j in range(i + 1, n):
            d = dist_fn(items[i], items[j])
            dist[i, j] = dist[j, i] = d

    # Greedy selection
    selected = [random.randint(0, n - 1)]
    min_dist = dist[selected[0]].copy()

    for _ in range(k - 1):
        # Mask selected
        candidates = min_dist.copy()
        for idx in selected:
            candidates[idx] = -np.inf

        # Select farthest
        next_idx = np.argmax(candidates)
        selected.append(next_idx)
        min_dist = np.minimum(min_dist, dist[next_idx])

    return [int(i) for i in selected]


def facility_location(items: List, k: int, sim_fn: Callable) -> List[int]:
    """
    Facility location greedy selection.

    Maximizes coverage based on similarity scores.

    Args:
        items: Input items
        k: Number to select
        sim_fn: Similarity function(item1, item2) -> float

    Returns:
        Selected indices
    """
    n = len(items)
    if k >= n:
        return list(range(n))
    if k <= 0:
        return []

    # Precompute similarities
    sim = np.zeros((n, n))
    for i in range(n):
        sim[i, i] = sim_fn(items[i], items[i])
        for j in range(i + 1, n):
            s = sim_fn(items[i], items[j])
            sim[i, j] = sim[j, i] = s

    # First selection: medoid
    selected = [int(np.argmax(sim.sum(axis=1)))]
    is_selected = np.zeros(n, dtype=bool)
    is_selected[selected[0]] = True
    max_sim = sim[selected[0]].copy()

    # Greedy selection
    for _ in range(k - 1):
        # Marginal gain
        gain = np.maximum(sim - max_sim, 0).sum(axis=1)
        gain[is_selected] = -np.inf

        best = np.argmax(gain)
        selected.append(int(best))
        is_selected[best] = True
        max_sim = np.maximum(max_sim, sim[best])

    return selected
