blob: 3a3790ff64d05305d7a0604672761d667693d260 [file] [log] [blame] [edit]
from math import modf
import random as r
def _enumerate_with_random(xs):
""" List[(key * value)] -> List[(key * value * rand)]
like 'enumerate' but with a third argument that can be used as a random
tiebreaker for fair sorting
@param xs : an iterator of things
@return : an iterator of triples consisting of
1) the index into the original iterator
2) the element drawn from the iterator
3) a number in the range [0,1) chosen uniformly at random
"""
for (k, v) in enumerate(xs):
yield (k, v, r.random())
def _normalize_entitlement(entitlement):
"""normalize a list of entitlements so it has unit sum
@param entitlement : a list of constants proportional to the share of
the total that the nth item is entitled to.
@result : same as entitlement, but normalized to sum to 1.
"""
s = sum(entitlement)
return tuple(x / float(s) for x in entitlement)
def descending_fair_sort_indices(xs):
"""fairly sort an iterator of values in descending order.
each item in the iterator is a pair consisting of an index into xs
and the original value
(4, 5, 6) --> iter([(2, 6), (1, 5), (0, 4)])
@param xs : an iterator of things
@return : the indices of xs, but with ties resolved fairly
"""
for idx, _, _ in sorted(
_enumerate_with_random(xs),
key=(lambda __v_tiebreaker: (__v_tiebreaker[1], __v_tiebreaker[2])),
reverse=True):
yield idx
def _full_partial_remaining(quota, seats):
"""number of full seats, partial seats, and remaining seats to be filled.
given a list of numbers with unit sum (e.g. [0.5, 0.2, 0.3])
perform the first step of fairly allocating a non-negative integer
number of seats between them.
@param quota : a list of numbers with unit sum
@param seats : the number of items to be distributed.
@return : a triple containing three things
1) the number of full seats each index is entitled to
2) the number of partial seats each index is entitled to
3) the number of remaining seats that need to be filled from (2)
"""
full = []
partial = []
must_fill = seats
for x in quota:
partial_seat, full_seats = modf(x * seats)
full.append(full_seats)
partial.append(partial_seat)
must_fill -= full_seats
return full, partial, must_fill
def _largest_remainder(entitlement, seats):
"""distribute stuff according to the largest remainder method.
@param entitlement : a not-necessarily-normalized list of numbers
representing
how many seats/things each index is entitled to.
@param seats : the number of seats to distribute
@return : a list of integers of the same length as entitlement summing to
seats. The allocation of seats is intended to be as close as
possible
To the original entitlement.
"""
quota = _normalize_entitlement(entitlement)
out, rems, remaining = _full_partial_remaining(quota, seats)
indices = descending_fair_sort_indices(rems)
for idx in indices:
if remaining <= 0:
break
out[idx] += 1
remaining -= 1
assert sum(out) == seats
return out
def _indices(xs):
"""get an iterator of indices over an iterator.
Do not materialize the entire iterator first.
@param xs : an iterator of things
@return : an iterator of indices of the same length as (xs)
"""
for k, _ in enumerate(xs):
yield k
def partition(xs, ratio):
"""take a list of items and a ratio and return two lists.
The ratio determines which fraction of the items are transferred.
@param xs : a list of things to split between the transfer and retain group.
@param ratio : the ratio of things to transfer.
@return : a list of two things
1) the elements of xs that are going to be transferred
2) the elements of xs that are going to be retained
"""
ratios = [ratio, 1.0 - ratio]
transfer_count, _ = _largest_remainder(ratios, len(xs))
transfer_count = int(round(transfer_count))
to_transfer_indices = r.sample(
population=list(_indices(xs)), k=transfer_count)
to_transfer = []
to_retain = []
for k, v in enumerate(xs):
if k in to_transfer_indices:
to_transfer.append(v)
else:
to_retain.append(v)
return to_transfer, to_retain