"""
Solvers for egalitarian and utilitarian limits for all three SWFs.
"""
import torch

def egal_swf_water_filling(self, u, rates):
    n = rates.shape[0]
    res = torch.zeros_like(rates)

    # sort filling times in ascending order (larger rate => smaller time)
    indices = torch.argsort(rates, descending=True)
    rates = rates[indices]
    rates_suffix_sum = torch.cumsum(rates.flip(0), dim=0).flip(0)

    i, rem = 0, self.num_alloc
    while (rem > 0) and (i < n):
        # find time when next container fills up
        time = 1 / rates[i]

        if time * rates_suffix_sum[i] > rem:
            # partially fill remaining containers and stop
            time = rem / rates_suffix_sum[i]
            res[i:n] = time * rates[i:n]
            rem = 0
            break
        else:
            # fill this container to 1.0 and continue
            res[i] = 1.0
            rem -= 1
            i += 1

    # rearrange results to original order
    final_res = torch.zeros_like(res)
    final_res[indices] = res

    assert torch.abs(final_res.sum() - self.num_alloc) < 1e-6, (
        f"Total allocation mismatch: {final_res.sum()} vs {self.num_alloc}"
    )
    return final_res

def util_swf_water_filling(self, u, rates):
    return rates