Files
infinifi/server.py

184 lines
4.4 KiB
Python
Raw Normal View History

import asyncio
2024-07-20 17:09:22 +01:00
import threading
2024-07-22 22:24:39 +01:00
import os
import json
from time import sleep
import requests
2024-07-20 21:18:41 +01:00
from contextlib import asynccontextmanager
from fastapi import (
FastAPI,
Request,
HTTPException,
status,
)
2024-07-20 17:09:22 +01:00
from fastapi.responses import FileResponse
from fastapi.staticfiles import StaticFiles
2024-07-24 17:32:46 +01:00
from logger import log_info, log_warn
2024-07-26 22:34:44 +01:00
from websocket_connection_manager import WebSocketConnectionManager
from sse_starlette.sse import EventSourceResponse
2024-07-20 17:09:22 +01:00
# the index of the current audio track from 0 to 9
2024-07-20 21:18:41 +01:00
current_index = -1
# the timer that periodically advances the current audio track
t = None
inference_url = ""
2024-08-26 12:24:09 +01:00
api_key = ""
2024-07-26 22:34:44 +01:00
ws_connection_manager = WebSocketConnectionManager()
active_listeners = set()
@asynccontextmanager
async def lifespan(app: FastAPI):
2024-08-26 12:24:09 +01:00
global ws, inference_url, api_key
2024-07-22 22:24:39 +01:00
inference_url = os.environ.get("INFERENCE_SERVER_URL")
2024-08-26 12:24:09 +01:00
api_key = os.environ.get("API_KEY")
if not inference_url:
2024-08-22 23:50:58 +01:00
inference_url = "http://localhost:8001"
2024-07-22 22:24:39 +01:00
advance()
2024-07-22 22:24:39 +01:00
yield
2024-07-22 22:24:39 +01:00
if t:
t.cancel()
2024-07-20 17:09:22 +01:00
2024-07-21 17:34:09 +01:00
def generate_new_audio():
if not inference_url:
2024-07-22 22:24:39 +01:00
return
2024-07-21 17:34:09 +01:00
global current_index
offset = 0
if current_index == 0:
offset = 5
elif current_index == 5:
offset = 0
else:
return
log_info("requesting new audio...")
2024-07-21 17:34:09 +01:00
2024-07-24 17:32:46 +01:00
try:
2024-08-26 12:24:09 +01:00
requests.post(
f"{inference_url}/generate",
headers={"Authorization": f"key {api_key}"},
)
except:
log_warn(
"inference server potentially unreachable. recycling cached audio for now."
)
return
is_available = False
while not is_available:
try:
2024-08-26 12:24:09 +01:00
res = requests.post(
f"{inference_url}/clips/0",
stream=True,
headers={"Authorization": f"key {api_key}"},
)
except:
log_warn(
"inference server potentially unreachable. recycling cached audio for now."
)
return
2024-07-22 22:24:39 +01:00
if res.status_code != status.HTTP_200_OK:
2024-08-26 12:24:09 +01:00
print(res.status_code)
print("still generating...")
2024-09-03 23:36:04 +01:00
sleep(30)
continue
2024-07-22 22:24:39 +01:00
print("inference complete! downloading new clips")
2024-07-21 17:34:09 +01:00
is_available = True
with open(f"{offset}.mp3", "wb") as f:
for chunk in res.iter_content(chunk_size=128):
f.write(chunk)
2024-07-21 17:34:09 +01:00
for i in range(4):
2024-08-26 12:24:09 +01:00
res = requests.post(
f"{inference_url}/clips/{i + 1}",
stream=True,
headers={"Authorization": f"key {api_key}"},
)
if res.status_code != status.HTTP_200_OK:
continue
with open(f"{i + 1 + offset}.mp3", "wb") as f:
for chunk in res.iter_content(chunk_size=128):
f.write(chunk)
log_info("audio generated.")
2024-07-21 17:34:09 +01:00
2024-07-20 17:09:22 +01:00
def advance():
global current_index, t
2024-07-20 17:09:22 +01:00
if current_index == 9:
current_index = 0
else:
current_index = current_index + 1
2024-07-24 17:32:46 +01:00
threading.Thread(target=generate_new_audio).start()
2024-07-20 21:18:41 +01:00
2024-07-20 17:09:22 +01:00
t = threading.Timer(60, advance)
t.start()
app = FastAPI(lifespan=lifespan)
2024-07-20 21:18:41 +01:00
@app.get("/current.mp3")
def get_current_audio():
return FileResponse(f"{current_index}.mp3")
@app.get("/status")
def status_stream(request: Request):
async def status_generator():
last_listener_count = len(active_listeners)
yield json.dumps({"listeners": last_listener_count})
while True:
if await request.is_disconnected():
break
listener_count = len(active_listeners)
if listener_count != last_listener_count:
last_listener_count = listener_count
yield json.dumps({"listeners": listener_count})
await asyncio.sleep(1)
return EventSourceResponse(status_generator())
2024-07-26 22:34:44 +01:00
@app.post("/client-status")
async def change_status(request: Request):
body = await request.json()
2024-07-26 22:34:44 +01:00
try:
is_listening = body["isListening"]
client = request.client
if not client:
raise HTTPException(status_code=400, detail="ip address unavailable.")
if is_listening:
active_listeners.add(client.host)
else:
active_listeners.discard(client.host)
return {"isListening": is_listening}
except KeyError:
raise HTTPException(status_code=400, detail="'isListening' must be a boolean")
2024-07-26 22:34:44 +01:00
2024-07-20 21:18:41 +01:00
app.mount("/", StaticFiles(directory="web", html=True), name="web")