import time
import json
import os
import collections
import random


class Request:
    def __init__(self, request_id, arrival_time, task_type, batch_size, seq_length):
        self.request_id = request_id
        self.arrival_time = arrival_time
        self.task_type = task_type
        self.batch_size = batch_size
        self.seq_length = seq_length

    def to_dict(self):
        return self.__dict__


class ExperimentalTestbed:
    def __init__(self):
        self.edge_servers = {f"edge_server_{i}": {} for i in range(1, 5)}
        self.cloud_server = {"cloud_server": {}}
        self.all_devices = {**self.edge_servers, **self.cloud_server}
        self.server_status = {name: 'idle' for name in self.all_devices.keys()}

    def get_idle_servers(self):
        return [name for name, status in self.server_status.items() if status == 'idle']

    def set_server_status(self, server_name, status):
        if server_name in self.server_status:
            self.server_status[server_name] = status


class CentralOffloadingScheduler:
    def __init__(self, testbed, kq=10):
        self.global_queue = collections.deque()
        self.testbed = testbed
        self.kq = kq

    def add_request(self, request):
        self.global_queue.append(request)

    def schedule(self):
        assignments = []
        idle_servers = self.testbed.get_idle_servers()
        if not idle_servers or not self.global_queue:
            return assignments

        num_potential_tasks = min(len(self.global_queue), self.kq)
        decision_space = list(collections.deque(itertools.islice(self.global_queue, num_potential_tasks)))

        num_assignments = min(len(idle_servers), len(decision_space))
        if num_assignments == 0:
            return assignments

        for i in range(num_assignments):
            request_to_assign = self.global_queue.popleft()
            server_to_assign = idle_servers[i]

            self.testbed.set_server_status(server_to_assign, 'busy')

            assignment = (request_to_assign, server_to_assign)
            assignments.append(assignment)

        return assignments


def safe_read_json(filepath, default_value):
    try:
        with open(filepath, 'r') as f:
            return json.load(f)
    except (FileNotFoundError, json.JSONDecodeError):
        return default_value


def safe_write_json(filepath, data):
    with open(filepath, 'w') as f:
        json.dump(data, f, indent=4)


def main(total_requests=50, kq_value=5):
    TASK_QUEUE_FILE = "task_queue.json"
    STATUS_UPDATES_FILE = "status_updates.json"

    if os.path.exists(TASK_QUEUE_FILE): os.remove(TASK_QUEUE_FILE)
    if os.path.exists(STATUS_UPDATES_FILE): os.remove(STATUS_UPDATES_FILE)
    safe_write_json(TASK_QUEUE_FILE, [])
    safe_write_json(STATUS_UPDATES_FILE, [])

    testbed = ExperimentalTestbed()
    scheduler = CentralOffloadingScheduler(testbed, kq=kq_value)

    print("Central Cloud Scheduler is running...")
    print(f"Simulating {total_requests} requests with Kq={kq_value}.")

    for i in range(total_requests):
        req = Request(
            request_id=i,
            arrival_time=i * 0.1,
            task_type="inference",
            batch_size=random.choice([4, 8, 16]),
            seq_length=random.choice([64, 128, 256])
        )
        scheduler.add_request(req)

    current_time = 0.0
    completed_tasks_count = 0

    while completed_tasks_count < total_requests:
        status_updates = safe_read_json(STATUS_UPDATES_FILE, [])
        if status_updates:
            for update in status_updates:
                server_name = update.get("completed_by")
                if server_name:
                    testbed.set_server_status(server_name, 'idle')
                    completed_tasks_count += 1
            safe_write_json(STATUS_UPDATES_FILE, [])
            print(
                f"Time: {current_time:.2f}s | Completed: {completed_tasks_count}/{total_requests} | Idle Servers: {len(testbed.get_idle_servers())}")

        new_assignments = scheduler.schedule()
        if new_assignments:
            task_queue = safe_read_json(TASK_QUEUE_FILE, [])
            for request, server in new_assignments:
                task_data = request.to_dict()
                task_data['assign_to'] = server
                task_queue.append(task_data)
                print(f"Time: {current_time:.2f}s | Assigned task {request.request_id} to {server}")
            safe_write_json(TASK_QUEUE_FILE, task_queue)

        time.sleep(0.1)
        current_time += 0.1

    print("Simulation finished. Sending termination signal.")
    task_queue = safe_read_json(TASK_QUEUE_FILE, [])
    for device in testbed.all_devices:
        task_queue.append({"assign_to": device, "task_type": "TERMINATE"})
    safe_write_json(TASK_QUEUE_FILE, task_queue)


if __name__ == "__main__":
    import itertools

    main()