"""Active Ranking algorithm for top-k selection"""
import numpy as np
import math


def active_ranking(S, P, k, delta):
    n = len(S)
    if n <= k:
        return list(S), 0

    R = []  # confirmed top-k
    T = []  # confirmed NOT top-k
    estimated = np.zeros(n)
    assured = np.zeros(n, dtype=bool)
    num_assured = 0
    t = 0
    comps = 0

    while len(R) < k:
        t += 1
        for i in range(n):
            if assured[i]:
                continue
            j = np.random.randint(n)
            win = np.random.random() < P[S[i], S[j]]
            if win:
                estimated[i] = (1.0 + estimated[i] * (t - 1)) / t
            else:
                estimated[i] = estimated[i] * (t - 1) / t
            comps += 1

        fs = [(estimated[i], i) for i in range(n) if not assured[i]]
        fs.sort()

        alpha_t = math.sqrt(math.log(125.0 * n * math.log(1.12 * t) / delta) / t)
        remaining = n - num_assured
        k_remaining = k - len(R)
        idx_low = max(0, remaining - 1 - k_remaining)
        idx_high = min(len(fs) - 1, remaining - k_remaining)

        anchor_low = fs[idx_low][0]
        anchor_high = fs[idx_high][0]

        for i in range(n):
            if assured[i]:
                continue
            if estimated[i] > anchor_low + 4.0 * alpha_t:
                R.append(S[i])
                assured[i] = True
                num_assured += 1
                if len(R) >= k:
                    return R, comps
            elif estimated[i] < anchor_high - 4.0 * alpha_t:
                T.append(S[i])
                assured[i] = True
                num_assured += 1
                if len(T) >= n - k:
                    for ii in range(n):
                        if not assured[ii]:
                            R.append(S[ii])
                    return R, comps

    return R, comps
