import threading


class MinSize:

    def __init__(self, minimum):
        assert 1 <= minimum, minimum
        self.minimum = minimum
        self.size = 0
        self.lock = threading.Lock()

    def want_load(self):
        with self.lock:
            self.size += 1
        return True, "ok"

    def want_insert(self):
        with self.lock:
            self.size += 1
        return True, "ok"

    def want_remove(self):
        with self.lock:
            if self.size < 1:
                return False, "is empty"
            self.size -= 1
        return True, "ok"

    def want_sample(self):
        if self.size < self.minimum:
            return False, f"too empty: {self.size} < {self.minimum}"
        return True, "ok"


class SamplesPerInsert:

    def __init__(self, samples_per_insert, tolerance, minimum=1):
        assert 1 <= minimum
        self.samples_per_insert = samples_per_insert
        self.minimum = minimum
        self.avail = -minimum
        self.min_avail = -tolerance
        self.max_avail = tolerance * samples_per_insert
        self.size = 0
        self.lock = threading.Lock()

    def want_load(self):
        with self.lock:
            self.size += 1
        return True, "ok"

    def want_insert(self):
        with self.lock:
            if self.avail >= self.max_avail:
                return False, f"rate limited: {self.avail:.3f} >= {self.max_avail:.3f}"
            self.avail += self.samples_per_insert
            self.size += 1
        return True, "ok"

    def want_remove(self):
        with self.lock:
            if self.size < 1:
                return False, "is empty"
            self.size -= 1
        return True, "ok"

    def want_sample(self):
        with self.lock:
            if self.size < self.minimum:
                return False, f"too empty: {self.size} < {self.minimum}"
            if self.avail <= self.min_avail:
                return False, f"rate limited: {self.avail:.3f} <= {self.min_avail:.3f}"
            self.avail -= 1
        return True, "ok"


class Queue:

    def __init__(self, capacity):
        assert 1 <= capacity
        self.capacity = capacity
        self.size = 0
        self.lock = threading.Lock()

    def want_load(self):
        with self.lock:
            self.size += 1
        return True, "ok"

    def want_insert(self):
        with self.lock:
            if self.size >= self.capacity:
                return False, f"is full: {self.size} >= {self.capacity}"
            self.size += 1
        return True, "ok"

    def want_remove(self):
        with self.lock:
            if self.size < 1:
                return False, "is empty"
            self.size -= 1
        return True, "ok"

    def want_sample(self):
        if self.size < 1:
            return False, "is empty"
        else:
            return True, "ok"
