trung99's picture
Update app.py
b9ce003 verified
#<--BẮT ĐẦU TOÀN BỘ CODE CHO FILE load_balancer.py (WORKER ĐẦU MỐI) - PHIÊN BẢN CUỐI CÙNG-->
import os
import asyncio
import aiohttp
import random
from fastapi import FastAPI, HTTPException, Request, Header
from pydantic import BaseModel, Field
from typing import List, Dict, Any
# --- Cấu hình ---
# Đọc key xác thực bí mật từ Secrets của Hugging Face.
# Bạn cần vào Settings -> Secrets của Space này để thêm một Secret mới tên là AUTH_KEY
API_AUTH_KEY = os.getenv("AUTH_KEY", "default-secret-key-change-me")
# !!! THAY THẾ BẰNG DANH SÁCH URL WORKER THỢ THẬT CỦA BẠN !!!
WORKER_URLS = [
# Dán danh sách URL của các worker thợ vào đây. Ví dụ:
"https://whf-manager-api-key-sheet-worker-checker-01.hf.space",
"https://whf-manager-api-key-sheet-elevenlabs-worker-02.hf.space",
"https://whf-manager-api-key-sheet-elevenlabs-worker-03.hf.space",
"https://whf-manager-api-key-sheet-elevenlabs-worker-04.hf.space",
"https://whf-manager-api-key-sheet-elevenlabs-worker-05.hf.space",
"https://whf-manager-api-key-sheet-elevenlabs-worker-06.hf.space",
"https://whf-manager-api-key-sheet-elevenlabs-worker-07.hf.space",
"https://whf-manager-api-key-sheet-elevenlabs-worker-08.hf.space",
"https://whf-manager-api-key-sheet-elevenlabs-worker-09.hf.space",
"https://whf-manager-api-key-sheet-elevenlabs-worker-10.hf.space",
"https://whf-manager-api-key-sheet-elevenlabs-worker-11.hf.space",
"https://whf-manager-api-key-sheet-elevenlabs-worker-12.hf.space",
"https://whf-manager-api-key-sheet-elevenlabs-worker-13.hf.space",
"https://whf-manager-api-key-sheet-elevenlabs-worker-14.hf.space",
"https://whf-manager-api-key-sheet-elevenlabs-worker-15.hf.space",
"https://whf-manager-api-key-sheet-elevenlabs-worker-16.hf.space",
"https://whf-manager-api-key-sheet-elevenlabs-worker-17.hf.space",
"https://whf-manager-api-key-sheet-elevenlabs-worker-18.hf.space",
"https://whf-manager-api-key-sheet-elevenlabs-worker-19.hf.space"
]
# --- Pydantic Models ---
class APIKeyPayload(BaseModel):
api_keys: List[str] = Field(..., min_items=1)
# --- Khởi tạo ứng dụng FastAPI ---
app = FastAPI(
title="Maintenance Load Balancer (Đầu mối)",
description="Xác thực, nhận yêu cầu từ client và phân phối đến các worker thợ.",
version="2.0.0",
)
# --- Logic của Load Balancer ---
worker_load = {url: 0 for url in WORKER_URLS}
load_lock = asyncio.Lock()
async def select_worker() -> str:
if not WORKER_URLS: raise ValueError("Danh sách WORKER_URLS chưa được cấu hình!")
async with load_lock:
min_load = min(worker_load.values())
least_busy_workers = [url for url, load in worker_load.items() if load == min_load]
return random.choice(least_busy_workers)
async def forward_request_to_worker(worker_url: str, payload: Dict[str, Any], auth_key: str):
endpoint = "/check-credits" # Endpoint trên worker thợ
target_url = f"{worker_url.rstrip('/')}{endpoint}"
# Chuyển tiếp cả header xác thực đến worker thợ
headers = {"Content-Type": "application/json", "x-api-key": auth_key}
async with load_lock: worker_load[worker_url] += 1
print(f"Forwarding to {worker_url}. Current loads: {worker_load}")
try:
async with aiohttp.ClientSession() as session:
async with session.post(target_url, json=payload, headers=headers, timeout=125) as response:
response.raise_for_status()
return await response.json()
finally:
async with load_lock: worker_load[worker_url] -= 1
print(f"Request to {worker_url} finished. Current loads: {worker_load}")
# --- API Endpoint của Load Balancer ---
@app.post("/dispatch-maintenance")
async def dispatch_maintenance_endpoint(payload: APIKeyPayload, x_api_key: str = Header(...)):
"""Endpoint DUY NHẤT mà Client Tool sẽ gọi."""
# 1. Xác thực request từ client
if x_api_key != API_AUTH_KEY:
raise HTTPException(status_code=401, detail="Invalid API Key for Load Balancer")
# 2. Chọn worker và chuyển tiếp request
try:
worker_to_use = await select_worker()
result = await forward_request_to_worker(worker_to_use, payload.dict(), API_AUTH_KEY)
return result
except ValueError as e:
raise HTTPException(status_code=503, detail=str(e))
except aiohttp.ClientResponseError as e:
raise HTTPException(status_code=e.status, detail=f"Worker thợ báo lỗi: {e.message}")
except Exception as e:
raise HTTPException(status_code=504, detail=f"Lỗi kết nối đến worker thợ: {str(e)}")
@app.get("/")
def read_root():
return {"message": "Load Balancer is running."}
#<--KẾT THÚC TOÀN BỘ CODE CHO FILE load_balancer.py -->