2024-02-13 14:11:53 +00:00
|
|
|
import os
|
2024-03-20 23:11:36 +00:00
|
|
|
import logging
|
2024-02-11 08:17:50 +00:00
|
|
|
from fastapi import (
|
|
|
|
FastAPI,
|
|
|
|
Request,
|
|
|
|
Depends,
|
|
|
|
HTTPException,
|
|
|
|
status,
|
|
|
|
UploadFile,
|
|
|
|
File,
|
|
|
|
Form,
|
|
|
|
)
|
2024-04-20 20:15:59 +00:00
|
|
|
|
|
|
|
from fastapi.responses import StreamingResponse, JSONResponse, FileResponse
|
|
|
|
|
2024-02-11 08:17:50 +00:00
|
|
|
from fastapi.middleware.cors import CORSMiddleware
|
|
|
|
from faster_whisper import WhisperModel
|
2024-04-20 20:21:52 +00:00
|
|
|
from pydantic import BaseModel
|
|
|
|
|
2024-06-07 03:44:42 +00:00
|
|
|
import uuid
|
2024-04-20 20:15:59 +00:00
|
|
|
import requests
|
|
|
|
import hashlib
|
|
|
|
from pathlib import Path
|
|
|
|
import json
|
|
|
|
|
|
|
|
|
2024-02-11 08:17:50 +00:00
|
|
|
from constants import ERROR_MESSAGES
|
|
|
|
from utils.utils import (
|
|
|
|
decode_token,
|
|
|
|
get_current_user,
|
|
|
|
get_verified_user,
|
|
|
|
get_admin_user,
|
|
|
|
)
|
|
|
|
from utils.misc import calculate_sha256
|
|
|
|
|
2024-03-31 08:13:39 +00:00
|
|
|
from config import (
|
|
|
|
SRC_LOG_LEVELS,
|
|
|
|
CACHE_DIR,
|
|
|
|
UPLOAD_DIR,
|
|
|
|
WHISPER_MODEL,
|
|
|
|
WHISPER_MODEL_DIR,
|
2024-04-11 02:30:00 +00:00
|
|
|
WHISPER_MODEL_AUTO_UPDATE,
|
2024-04-02 12:47:52 +00:00
|
|
|
DEVICE_TYPE,
|
2024-06-08 03:18:48 +00:00
|
|
|
AUDIO_STT_OPENAI_API_BASE_URL,
|
|
|
|
AUDIO_STT_OPENAI_API_KEY,
|
|
|
|
AUDIO_TTS_OPENAI_API_BASE_URL,
|
|
|
|
AUDIO_TTS_OPENAI_API_KEY,
|
|
|
|
AUDIO_STT_ENGINE,
|
|
|
|
AUDIO_STT_MODEL,
|
|
|
|
AUDIO_TTS_ENGINE,
|
|
|
|
AUDIO_TTS_MODEL,
|
|
|
|
AUDIO_TTS_VOICE,
|
2024-05-10 07:03:24 +00:00
|
|
|
AppConfig,
|
2024-03-31 08:13:39 +00:00
|
|
|
)
|
2024-03-20 23:11:36 +00:00
|
|
|
|
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
log.setLevel(SRC_LOG_LEVELS["AUDIO"])
|
2024-02-11 08:17:50 +00:00
|
|
|
|
|
|
|
app = FastAPI()
|
|
|
|
app.add_middleware(
|
|
|
|
CORSMiddleware,
|
|
|
|
allow_origins=["*"],
|
|
|
|
allow_credentials=True,
|
|
|
|
allow_methods=["*"],
|
|
|
|
allow_headers=["*"],
|
|
|
|
)
|
|
|
|
|
2024-05-10 07:03:24 +00:00
|
|
|
app.state.config = AppConfig()
|
2024-06-08 03:18:48 +00:00
|
|
|
|
|
|
|
app.state.config.STT_OPENAI_API_BASE_URL = AUDIO_STT_OPENAI_API_BASE_URL
|
|
|
|
app.state.config.STT_OPENAI_API_KEY = AUDIO_STT_OPENAI_API_KEY
|
|
|
|
app.state.config.STT_ENGINE = AUDIO_STT_ENGINE
|
|
|
|
app.state.config.STT_MODEL = AUDIO_STT_MODEL
|
|
|
|
|
|
|
|
app.state.config.TTS_OPENAI_API_BASE_URL = AUDIO_TTS_OPENAI_API_BASE_URL
|
|
|
|
app.state.config.TTS_OPENAI_API_KEY = AUDIO_TTS_OPENAI_API_KEY
|
|
|
|
app.state.config.TTS_ENGINE = AUDIO_TTS_ENGINE
|
|
|
|
app.state.config.TTS_MODEL = AUDIO_TTS_MODEL
|
|
|
|
app.state.config.TTS_VOICE = AUDIO_TTS_VOICE
|
2024-04-20 20:15:59 +00:00
|
|
|
|
2024-04-02 12:47:52 +00:00
|
|
|
# setting device type for whisper model
|
|
|
|
whisper_device_type = DEVICE_TYPE if DEVICE_TYPE and DEVICE_TYPE == "cuda" else "cpu"
|
|
|
|
log.info(f"whisper_device_type: {whisper_device_type}")
|
|
|
|
|
2024-04-20 20:15:59 +00:00
|
|
|
SPEECH_CACHE_DIR = Path(CACHE_DIR).joinpath("./audio/speech/")
|
|
|
|
SPEECH_CACHE_DIR.mkdir(parents=True, exist_ok=True)
|
|
|
|
|
|
|
|
|
2024-06-08 03:18:48 +00:00
|
|
|
class TTSConfigForm(BaseModel):
|
|
|
|
OPENAI_API_BASE_URL: str
|
|
|
|
OPENAI_API_KEY: str
|
|
|
|
ENGINE: str
|
|
|
|
MODEL: str
|
|
|
|
VOICE: str
|
|
|
|
|
|
|
|
|
|
|
|
class STTConfigForm(BaseModel):
|
|
|
|
OPENAI_API_BASE_URL: str
|
|
|
|
OPENAI_API_KEY: str
|
|
|
|
ENGINE: str
|
|
|
|
MODEL: str
|
|
|
|
|
|
|
|
|
|
|
|
class AudioConfigUpdateForm(BaseModel):
|
|
|
|
tts: TTSConfigForm
|
|
|
|
stt: STTConfigForm
|
2024-04-20 20:21:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
@app.get("/config")
|
2024-06-08 03:18:48 +00:00
|
|
|
async def get_audio_config(user=Depends(get_admin_user)):
|
2024-04-20 20:21:52 +00:00
|
|
|
return {
|
2024-06-08 03:18:48 +00:00
|
|
|
"tts": {
|
|
|
|
"OPENAI_API_BASE_URL": app.state.config.TTS_OPENAI_API_BASE_URL,
|
|
|
|
"OPENAI_API_KEY": app.state.config.TTS_OPENAI_API_KEY,
|
|
|
|
"ENGINE": app.state.config.TTS_ENGINE,
|
|
|
|
"MODEL": app.state.config.TTS_MODEL,
|
|
|
|
"VOICE": app.state.config.TTS_VOICE,
|
|
|
|
},
|
|
|
|
"stt": {
|
|
|
|
"OPENAI_API_BASE_URL": app.state.config.STT_OPENAI_API_BASE_URL,
|
|
|
|
"OPENAI_API_KEY": app.state.config.STT_OPENAI_API_KEY,
|
|
|
|
"ENGINE": app.state.config.STT_ENGINE,
|
|
|
|
"MODEL": app.state.config.STT_MODEL,
|
|
|
|
},
|
2024-04-20 20:21:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@app.post("/config/update")
|
2024-06-08 03:18:48 +00:00
|
|
|
async def update_audio_config(
|
|
|
|
form_data: AudioConfigUpdateForm, user=Depends(get_admin_user)
|
2024-04-20 20:21:52 +00:00
|
|
|
):
|
2024-06-08 03:18:48 +00:00
|
|
|
app.state.config.TTS_OPENAI_API_BASE_URL = form_data.tts.OPENAI_API_BASE_URL
|
|
|
|
app.state.config.TTS_OPENAI_API_KEY = form_data.tts.OPENAI_API_KEY
|
|
|
|
app.state.config.TTS_ENGINE = form_data.tts.ENGINE
|
|
|
|
app.state.config.TTS_MODEL = form_data.tts.MODEL
|
|
|
|
app.state.config.TTS_VOICE = form_data.tts.VOICE
|
2024-04-20 20:21:52 +00:00
|
|
|
|
2024-06-08 03:18:48 +00:00
|
|
|
app.state.config.STT_OPENAI_API_BASE_URL = form_data.stt.OPENAI_API_BASE_URL
|
|
|
|
app.state.config.STT_OPENAI_API_KEY = form_data.stt.OPENAI_API_KEY
|
|
|
|
app.state.config.STT_ENGINE = form_data.stt.ENGINE
|
|
|
|
app.state.config.STT_MODEL = form_data.stt.MODEL
|
2024-04-20 20:21:52 +00:00
|
|
|
|
|
|
|
return {
|
2024-06-08 03:18:48 +00:00
|
|
|
"tts": {
|
|
|
|
"OPENAI_API_BASE_URL": app.state.config.TTS_OPENAI_API_BASE_URL,
|
|
|
|
"OPENAI_API_KEY": app.state.config.TTS_OPENAI_API_KEY,
|
|
|
|
"ENGINE": app.state.config.TTS_ENGINE,
|
|
|
|
"MODEL": app.state.config.TTS_MODEL,
|
|
|
|
"VOICE": app.state.config.TTS_VOICE,
|
|
|
|
},
|
|
|
|
"stt": {
|
|
|
|
"OPENAI_API_BASE_URL": app.state.config.STT_OPENAI_API_BASE_URL,
|
|
|
|
"OPENAI_API_KEY": app.state.config.STT_OPENAI_API_KEY,
|
|
|
|
"ENGINE": app.state.config.STT_ENGINE,
|
|
|
|
"MODEL": app.state.config.STT_MODEL,
|
|
|
|
},
|
2024-04-20 20:21:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2024-04-20 20:15:59 +00:00
|
|
|
@app.post("/speech")
|
|
|
|
async def speech(request: Request, user=Depends(get_verified_user)):
|
2024-04-20 21:00:24 +00:00
|
|
|
body = await request.body()
|
|
|
|
name = hashlib.sha256(body).hexdigest()
|
|
|
|
|
|
|
|
file_path = SPEECH_CACHE_DIR.joinpath(f"{name}.mp3")
|
|
|
|
file_body_path = SPEECH_CACHE_DIR.joinpath(f"{name}.json")
|
|
|
|
|
|
|
|
# Check if the file already exists in the cache
|
|
|
|
if file_path.is_file():
|
|
|
|
return FileResponse(file_path)
|
|
|
|
|
|
|
|
headers = {}
|
2024-06-08 03:18:48 +00:00
|
|
|
headers["Authorization"] = f"Bearer {app.state.config.TTS_OPENAI_API_KEY}"
|
2024-04-20 21:00:24 +00:00
|
|
|
headers["Content-Type"] = "application/json"
|
|
|
|
|
|
|
|
r = None
|
2024-04-20 20:15:59 +00:00
|
|
|
try:
|
2024-04-20 21:00:24 +00:00
|
|
|
r = requests.post(
|
2024-06-08 03:18:48 +00:00
|
|
|
url=f"{app.state.config.TTS_OPENAI_API_BASE_URL}/audio/speech",
|
2024-04-20 21:00:24 +00:00
|
|
|
data=body,
|
|
|
|
headers=headers,
|
|
|
|
stream=True,
|
|
|
|
)
|
|
|
|
|
|
|
|
r.raise_for_status()
|
|
|
|
|
|
|
|
# Save the streaming content to a file
|
|
|
|
with open(file_path, "wb") as f:
|
|
|
|
for chunk in r.iter_content(chunk_size=8192):
|
|
|
|
f.write(chunk)
|
|
|
|
|
|
|
|
with open(file_body_path, "w") as f:
|
|
|
|
json.dump(json.loads(body.decode("utf-8")), f)
|
|
|
|
|
|
|
|
# Return the saved file
|
|
|
|
return FileResponse(file_path)
|
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
log.exception(e)
|
|
|
|
error_detail = "Open WebUI: Server Connection Error"
|
|
|
|
if r is not None:
|
|
|
|
try:
|
|
|
|
res = r.json()
|
|
|
|
if "error" in res:
|
|
|
|
error_detail = f"External: {res['error']['message']}"
|
|
|
|
except:
|
|
|
|
error_detail = f"External: {e}"
|
|
|
|
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=r.status_code if r != None else 500,
|
|
|
|
detail=error_detail,
|
|
|
|
)
|
2024-04-20 20:15:59 +00:00
|
|
|
|
2024-02-11 08:17:50 +00:00
|
|
|
|
2024-04-20 20:15:59 +00:00
|
|
|
@app.post("/transcriptions")
|
2024-02-11 08:17:50 +00:00
|
|
|
def transcribe(
|
|
|
|
file: UploadFile = File(...),
|
|
|
|
user=Depends(get_current_user),
|
|
|
|
):
|
2024-03-20 23:11:36 +00:00
|
|
|
log.info(f"file.content_type: {file.content_type}")
|
2024-02-11 08:17:50 +00:00
|
|
|
|
|
|
|
if file.content_type not in ["audio/mpeg", "audio/wav"]:
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=status.HTTP_400_BAD_REQUEST,
|
|
|
|
detail=ERROR_MESSAGES.FILE_NOT_SUPPORTED,
|
|
|
|
)
|
|
|
|
|
|
|
|
try:
|
2024-06-07 03:44:42 +00:00
|
|
|
ext = file.filename.split(".")[-1]
|
|
|
|
|
|
|
|
id = uuid.uuid4()
|
|
|
|
filename = f"{id}.{ext}"
|
|
|
|
|
|
|
|
file_dir = f"{CACHE_DIR}/audio/transcriptions"
|
|
|
|
os.makedirs(file_dir, exist_ok=True)
|
|
|
|
file_path = f"{file_dir}/{filename}"
|
|
|
|
|
2024-02-11 08:17:50 +00:00
|
|
|
contents = file.file.read()
|
|
|
|
with open(file_path, "wb") as f:
|
|
|
|
f.write(contents)
|
|
|
|
f.close()
|
|
|
|
|
2024-06-08 03:31:52 +00:00
|
|
|
if app.state.config.STT_ENGINE == "":
|
|
|
|
whisper_kwargs = {
|
|
|
|
"model_size_or_path": WHISPER_MODEL,
|
|
|
|
"device": whisper_device_type,
|
|
|
|
"compute_type": "int8",
|
|
|
|
"download_root": WHISPER_MODEL_DIR,
|
|
|
|
"local_files_only": not WHISPER_MODEL_AUTO_UPDATE,
|
|
|
|
}
|
|
|
|
|
|
|
|
log.debug(f"whisper_kwargs: {whisper_kwargs}")
|
|
|
|
|
|
|
|
try:
|
|
|
|
model = WhisperModel(**whisper_kwargs)
|
|
|
|
except:
|
|
|
|
log.warning(
|
|
|
|
"WhisperModel initialization failed, attempting download with local_files_only=False"
|
|
|
|
)
|
|
|
|
whisper_kwargs["local_files_only"] = False
|
|
|
|
model = WhisperModel(**whisper_kwargs)
|
|
|
|
|
|
|
|
segments, info = model.transcribe(file_path, beam_size=5)
|
|
|
|
log.info(
|
|
|
|
"Detected language '%s' with probability %f"
|
|
|
|
% (info.language, info.language_probability)
|
2024-04-11 02:44:44 +00:00
|
|
|
)
|
2024-02-11 08:17:50 +00:00
|
|
|
|
2024-06-08 03:31:52 +00:00
|
|
|
transcript = "".join([segment.text for segment in list(segments)])
|
|
|
|
|
|
|
|
# save the transcript to a json file
|
|
|
|
transcript_file = f"{file_dir}/{id}.json"
|
|
|
|
with open(transcript_file, "w") as f:
|
|
|
|
json.dump({"transcript": transcript}, f)
|
2024-02-11 08:17:50 +00:00
|
|
|
|
2024-06-08 03:31:52 +00:00
|
|
|
return {"text": transcript.strip()}
|
2024-02-11 08:17:50 +00:00
|
|
|
|
2024-06-08 03:31:52 +00:00
|
|
|
elif app.state.config.STT_ENGINE == "openai":
|
|
|
|
headers = {"Authorization": f"Bearer {app.state.config.STT_OPENAI_API_KEY}"}
|
2024-06-07 03:44:42 +00:00
|
|
|
|
2024-06-08 03:31:52 +00:00
|
|
|
files = {"file": (filename, open(file_path, "rb"))}
|
|
|
|
data = {"model": "whisper-1"}
|
|
|
|
|
|
|
|
r = None
|
|
|
|
try:
|
|
|
|
r = requests.post(
|
|
|
|
url=f"{app.state.config.STT_OPENAI_API_BASE_URL}/audio/transcriptions",
|
|
|
|
headers=headers,
|
|
|
|
files=files,
|
|
|
|
data=data,
|
|
|
|
)
|
|
|
|
|
|
|
|
r.raise_for_status()
|
|
|
|
return r.json()
|
|
|
|
except Exception as e:
|
|
|
|
log.exception(e)
|
|
|
|
error_detail = "Open WebUI: Server Connection Error"
|
|
|
|
if r is not None:
|
|
|
|
try:
|
|
|
|
res = r.json()
|
|
|
|
if "error" in res:
|
|
|
|
error_detail = f"External: {res['error']['message']}"
|
|
|
|
except:
|
|
|
|
error_detail = f"External: {e}"
|
|
|
|
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=r.status_code if r != None else 500,
|
|
|
|
detail=error_detail,
|
|
|
|
)
|
2024-02-11 08:17:50 +00:00
|
|
|
|
|
|
|
except Exception as e:
|
2024-03-20 23:11:36 +00:00
|
|
|
log.exception(e)
|
2024-02-11 08:17:50 +00:00
|
|
|
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=status.HTTP_400_BAD_REQUEST,
|
|
|
|
detail=ERROR_MESSAGES.DEFAULT(e),
|
|
|
|
)
|