open-webui/backend/apps/images/main.py

528 lines
15 KiB
Python
Raw Normal View History

2024-02-23 03:32:36 +00:00
import re
2024-02-22 02:12:01 +00:00
import requests
from fastapi import (
FastAPI,
Request,
Depends,
HTTPException,
status,
UploadFile,
File,
Form,
)
from fastapi.middleware.cors import CORSMiddleware
from faster_whisper import WhisperModel
from constants import ERROR_MESSAGES
from utils.utils import (
get_current_user,
get_admin_user,
)
2024-03-24 00:01:13 +00:00
from apps.images.utils.comfyui import ImageGenerationPayload, comfyui_generate_image
2024-02-22 02:12:01 +00:00
from utils.misc import calculate_sha256
from typing import Optional
from pydantic import BaseModel
2024-03-09 01:38:10 +00:00
from pathlib import Path
import mimetypes
2024-03-09 01:38:10 +00:00
import uuid
import base64
import json
import logging
2024-03-09 01:38:10 +00:00
2024-04-11 05:39:11 +00:00
from config import (
SRC_LOG_LEVELS,
CACHE_DIR,
IMAGE_GENERATION_ENGINE,
ENABLE_IMAGE_GENERATION,
2024-04-11 05:39:11 +00:00
AUTOMATIC1111_BASE_URL,
COMFYUI_BASE_URL,
2024-04-23 11:14:31 +00:00
IMAGES_OPENAI_API_BASE_URL,
IMAGES_OPENAI_API_KEY,
IMAGE_GENERATION_MODEL,
IMAGE_SIZE,
IMAGE_STEPS,
AppConfig,
2024-04-11 05:39:11 +00:00
)
2024-03-09 01:38:10 +00:00
log = logging.getLogger(__name__)
log.setLevel(SRC_LOG_LEVELS["IMAGES"])
2024-03-09 01:38:10 +00:00
IMAGE_CACHE_DIR = Path(CACHE_DIR).joinpath("./image/generations/")
IMAGE_CACHE_DIR.mkdir(parents=True, exist_ok=True)
2024-02-22 02:12:01 +00:00
app = FastAPI()
app.add_middleware(
CORSMiddleware,
allow_origins=["*"],
allow_credentials=True,
allow_methods=["*"],
allow_headers=["*"],
)
app.state.config = AppConfig()
2024-03-09 01:38:10 +00:00
app.state.config.ENGINE = IMAGE_GENERATION_ENGINE
app.state.config.ENABLED = ENABLE_IMAGE_GENERATION
2024-04-20 20:15:59 +00:00
app.state.config.OPENAI_API_BASE_URL = IMAGES_OPENAI_API_BASE_URL
app.state.config.OPENAI_API_KEY = IMAGES_OPENAI_API_KEY
2024-03-09 01:38:10 +00:00
app.state.config.MODEL = IMAGE_GENERATION_MODEL
2024-03-09 01:38:10 +00:00
2024-03-23 22:38:59 +00:00
app.state.config.AUTOMATIC1111_BASE_URL = AUTOMATIC1111_BASE_URL
app.state.config.COMFYUI_BASE_URL = COMFYUI_BASE_URL
2024-03-09 01:38:10 +00:00
app.state.config.IMAGE_SIZE = IMAGE_SIZE
app.state.config.IMAGE_STEPS = IMAGE_STEPS
2024-02-22 02:12:01 +00:00
2024-03-09 01:38:10 +00:00
@app.get("/config")
async def get_config(request: Request, user=Depends(get_admin_user)):
return {
"engine": app.state.config.ENGINE,
"enabled": app.state.config.ENABLED,
}
2024-02-22 02:12:01 +00:00
2024-03-09 01:38:10 +00:00
class ConfigUpdateForm(BaseModel):
engine: str
enabled: bool
@app.post("/config/update")
async def update_config(form_data: ConfigUpdateForm, user=Depends(get_admin_user)):
app.state.config.ENGINE = form_data.engine
app.state.config.ENABLED = form_data.enabled
return {
"engine": app.state.config.ENGINE,
"enabled": app.state.config.ENABLED,
}
2024-02-22 02:12:01 +00:00
2024-03-23 22:38:59 +00:00
class EngineUrlUpdateForm(BaseModel):
AUTOMATIC1111_BASE_URL: Optional[str] = None
COMFYUI_BASE_URL: Optional[str] = None
2024-02-22 02:12:01 +00:00
@app.get("/url")
2024-03-23 22:38:59 +00:00
async def get_engine_url(user=Depends(get_admin_user)):
return {
"AUTOMATIC1111_BASE_URL": app.state.config.AUTOMATIC1111_BASE_URL,
"COMFYUI_BASE_URL": app.state.config.COMFYUI_BASE_URL,
2024-03-23 22:38:59 +00:00
}
2024-02-22 02:12:01 +00:00
@app.post("/url/update")
2024-03-23 22:38:59 +00:00
async def update_engine_url(
form_data: EngineUrlUpdateForm, user=Depends(get_admin_user)
2024-03-09 01:38:10 +00:00
):
2024-02-22 03:06:27 +00:00
2024-03-23 22:38:59 +00:00
if form_data.AUTOMATIC1111_BASE_URL == None:
app.state.config.AUTOMATIC1111_BASE_URL = AUTOMATIC1111_BASE_URL
2024-02-22 03:06:27 +00:00
else:
2024-03-23 22:38:59 +00:00
url = form_data.AUTOMATIC1111_BASE_URL.strip("/")
2024-03-09 01:38:10 +00:00
try:
r = requests.head(url)
app.state.config.AUTOMATIC1111_BASE_URL = url
2024-03-09 01:38:10 +00:00
except Exception as e:
raise HTTPException(status_code=400, detail=ERROR_MESSAGES.DEFAULT(e))
2024-02-22 03:06:27 +00:00
2024-03-23 22:38:59 +00:00
if form_data.COMFYUI_BASE_URL == None:
app.state.config.COMFYUI_BASE_URL = COMFYUI_BASE_URL
2024-03-23 22:38:59 +00:00
else:
url = form_data.COMFYUI_BASE_URL.strip("/")
2024-03-24 00:01:13 +00:00
try:
r = requests.head(url)
app.state.config.COMFYUI_BASE_URL = url
2024-03-24 00:01:13 +00:00
except Exception as e:
raise HTTPException(status_code=400, detail=ERROR_MESSAGES.DEFAULT(e))
2024-03-23 22:38:59 +00:00
2024-02-22 03:06:27 +00:00
return {
"AUTOMATIC1111_BASE_URL": app.state.config.AUTOMATIC1111_BASE_URL,
"COMFYUI_BASE_URL": app.state.config.COMFYUI_BASE_URL,
2024-02-22 03:06:27 +00:00
"status": True,
}
2024-02-22 02:12:01 +00:00
2024-04-23 11:14:31 +00:00
class OpenAIConfigUpdateForm(BaseModel):
url: str
2024-03-09 01:38:10 +00:00
key: str
2024-04-23 11:14:31 +00:00
@app.get("/openai/config")
async def get_openai_config(user=Depends(get_admin_user)):
return {
"OPENAI_API_BASE_URL": app.state.config.OPENAI_API_BASE_URL,
"OPENAI_API_KEY": app.state.config.OPENAI_API_KEY,
2024-04-23 11:14:31 +00:00
}
2024-03-09 01:38:10 +00:00
2024-04-23 11:14:31 +00:00
@app.post("/openai/config/update")
async def update_openai_config(
form_data: OpenAIConfigUpdateForm, user=Depends(get_admin_user)
2024-03-09 01:38:10 +00:00
):
if form_data.key == "":
raise HTTPException(status_code=400, detail=ERROR_MESSAGES.API_KEY_NOT_FOUND)
app.state.config.OPENAI_API_BASE_URL = form_data.url
app.state.config.OPENAI_API_KEY = form_data.key
2024-04-23 11:14:31 +00:00
2024-03-09 01:38:10 +00:00
return {
"status": True,
"OPENAI_API_BASE_URL": app.state.config.OPENAI_API_BASE_URL,
"OPENAI_API_KEY": app.state.config.OPENAI_API_KEY,
2024-03-09 01:38:10 +00:00
}
2024-02-23 03:32:36 +00:00
class ImageSizeUpdateForm(BaseModel):
size: str
@app.get("/size")
async def get_image_size(user=Depends(get_admin_user)):
return {"IMAGE_SIZE": app.state.config.IMAGE_SIZE}
2024-02-23 03:32:36 +00:00
@app.post("/size/update")
async def update_image_size(
form_data: ImageSizeUpdateForm, user=Depends(get_admin_user)
):
pattern = r"^\d+x\d+$" # Regular expression pattern
if re.match(pattern, form_data.size):
app.state.config.IMAGE_SIZE = form_data.size
2024-02-23 03:32:36 +00:00
return {
"IMAGE_SIZE": app.state.config.IMAGE_SIZE,
2024-02-23 03:32:36 +00:00
"status": True,
}
else:
raise HTTPException(
status_code=400,
detail=ERROR_MESSAGES.INCORRECT_FORMAT(" (e.g., 512x512)."),
)
2024-02-25 02:08:35 +00:00
class ImageStepsUpdateForm(BaseModel):
steps: int
@app.get("/steps")
async def get_image_size(user=Depends(get_admin_user)):
return {"IMAGE_STEPS": app.state.config.IMAGE_STEPS}
@app.post("/steps/update")
async def update_image_size(
form_data: ImageStepsUpdateForm, user=Depends(get_admin_user)
):
if form_data.steps >= 0:
app.state.config.IMAGE_STEPS = form_data.steps
return {
"IMAGE_STEPS": app.state.config.IMAGE_STEPS,
"status": True,
}
else:
raise HTTPException(
status_code=400,
detail=ERROR_MESSAGES.INCORRECT_FORMAT(" (e.g., 50)."),
)
2024-02-23 03:32:36 +00:00
2024-02-22 02:12:01 +00:00
@app.get("/models")
def get_models(user=Depends(get_current_user)):
try:
if app.state.config.ENGINE == "openai":
2024-03-09 01:38:10 +00:00
return [
{"id": "dall-e-2", "name": "DALL·E 2"},
{"id": "dall-e-3", "name": "DALL·E 3"},
]
elif app.state.config.ENGINE == "comfyui":
2024-03-23 22:38:59 +00:00
r = requests.get(url=f"{app.state.config.COMFYUI_BASE_URL}/object_info")
2024-03-23 22:38:59 +00:00
info = r.json()
return list(
map(
lambda model: {"id": model, "name": model},
info["CheckpointLoaderSimple"]["input"]["required"]["ckpt_name"][0],
)
)
2024-03-09 01:38:10 +00:00
else:
r = requests.get(
url=f"{app.state.config.AUTOMATIC1111_BASE_URL}/sdapi/v1/sd-models"
2024-03-09 01:38:10 +00:00
)
models = r.json()
return list(
map(
lambda model: {"id": model["title"], "name": model["model_name"]},
models,
)
)
2024-02-22 02:12:01 +00:00
except Exception as e:
app.state.config.ENABLED = False
2024-02-25 02:08:35 +00:00
raise HTTPException(status_code=400, detail=ERROR_MESSAGES.DEFAULT(e))
2024-02-22 02:12:01 +00:00
@app.get("/models/default")
async def get_default_model(user=Depends(get_admin_user)):
try:
if app.state.config.ENGINE == "openai":
return {
"model": (
app.state.config.MODEL if app.state.config.MODEL else "dall-e-2"
)
}
elif app.state.config.ENGINE == "comfyui":
return {"model": (app.state.config.MODEL if app.state.config.MODEL else "")}
2024-03-09 01:38:10 +00:00
else:
r = requests.get(
url=f"{app.state.config.AUTOMATIC1111_BASE_URL}/sdapi/v1/options"
)
2024-03-09 01:38:10 +00:00
options = r.json()
return {"model": options["sd_model_checkpoint"]}
2024-02-22 02:12:01 +00:00
except Exception as e:
app.state.config.ENABLED = False
2024-02-25 02:08:35 +00:00
raise HTTPException(status_code=400, detail=ERROR_MESSAGES.DEFAULT(e))
2024-02-22 02:12:01 +00:00
class UpdateModelForm(BaseModel):
model: str
def set_model_handler(model: str):
if app.state.config.ENGINE in ["openai", "comfyui"]:
app.state.config.MODEL = model
return app.state.config.MODEL
2024-03-09 01:38:10 +00:00
else:
r = requests.get(
url=f"{app.state.config.AUTOMATIC1111_BASE_URL}/sdapi/v1/options"
)
2024-03-09 01:38:10 +00:00
options = r.json()
if model != options["sd_model_checkpoint"]:
options["sd_model_checkpoint"] = model
r = requests.post(
url=f"{app.state.config.AUTOMATIC1111_BASE_URL}/sdapi/v1/options",
json=options,
2024-03-09 01:38:10 +00:00
)
2024-02-22 02:12:01 +00:00
2024-03-09 01:38:10 +00:00
return options
2024-02-22 02:12:01 +00:00
@app.post("/models/default/update")
def update_default_model(
form_data: UpdateModelForm,
user=Depends(get_current_user),
):
return set_model_handler(form_data.model)
class GenerateImageForm(BaseModel):
model: Optional[str] = None
prompt: str
n: int = 1
2024-03-09 03:54:47 +00:00
size: Optional[str] = None
2024-02-22 02:12:01 +00:00
negative_prompt: Optional[str] = None
2024-03-09 01:38:10 +00:00
def save_b64_image(b64_str):
try:
2024-05-02 22:59:45 +00:00
image_id = str(uuid.uuid4())
2024-03-09 01:38:10 +00:00
2024-05-02 22:59:45 +00:00
if "," in b64_str:
header, encoded = b64_str.split(",", 1)
mime_type = header.split(";")[0]
2024-04-30 18:52:08 +00:00
2024-05-02 22:59:45 +00:00
img_data = base64.b64decode(encoded)
image_format = mimetypes.guess_extension(mime_type)
image_filename = f"{image_id}{image_format}"
file_path = IMAGE_CACHE_DIR / f"{image_filename}"
with open(file_path, "wb") as f:
f.write(img_data)
return image_filename
else:
image_filename = f"{image_id}.png"
file_path = IMAGE_CACHE_DIR.joinpath(image_filename)
img_data = base64.b64decode(b64_str)
# Write the image data to a file
with open(file_path, "wb") as f:
f.write(img_data)
return image_filename
2024-04-30 18:52:08 +00:00
2024-03-09 01:38:10 +00:00
except Exception as e:
log.exception(f"Error saving image: {e}")
2024-04-30 18:52:08 +00:00
return None
2024-03-09 01:38:10 +00:00
2024-03-24 00:01:13 +00:00
def save_url_image(url):
image_id = str(uuid.uuid4())
try:
r = requests.get(url)
r.raise_for_status()
if r.headers["content-type"].split("/")[0] == "image":
mime_type = r.headers["content-type"]
image_format = mimetypes.guess_extension(mime_type)
2024-03-24 00:01:13 +00:00
if not image_format:
raise ValueError("Could not determine image type from MIME type")
2024-05-02 22:54:31 +00:00
image_filename = f"{image_id}{image_format}"
file_path = IMAGE_CACHE_DIR.joinpath(f"{image_filename}")
with open(file_path, "wb") as image_file:
for chunk in r.iter_content(chunk_size=8192):
image_file.write(chunk)
2024-05-02 22:54:31 +00:00
return image_filename
else:
log.error(f"Url does not point to an image.")
2024-05-02 22:54:31 +00:00
return None
2024-03-24 00:01:13 +00:00
except Exception as e:
log.exception(f"Error saving image: {e}")
2024-05-02 22:54:31 +00:00
return None
2024-03-24 00:01:13 +00:00
2024-02-22 02:12:01 +00:00
@app.post("/generations")
def generate_image(
form_data: GenerateImageForm,
user=Depends(get_current_user),
):
width, height = tuple(map(int, app.state.config.IMAGE_SIZE).split("x"))
2024-03-24 00:01:13 +00:00
2024-03-09 03:54:47 +00:00
r = None
2024-02-22 02:36:40 +00:00
try:
if app.state.config.ENGINE == "openai":
2024-02-22 02:36:40 +00:00
2024-03-09 01:38:10 +00:00
headers = {}
headers["Authorization"] = f"Bearer {app.state.config.OPENAI_API_KEY}"
2024-03-09 01:38:10 +00:00
headers["Content-Type"] = "application/json"
2024-02-22 02:36:40 +00:00
2024-03-09 01:38:10 +00:00
data = {
"model": (
app.state.config.MODEL
if app.state.config.MODEL != ""
else "dall-e-2"
),
2024-03-09 01:38:10 +00:00
"prompt": form_data.prompt,
"n": form_data.n,
"size": (
form_data.size if form_data.size else app.state.config.IMAGE_SIZE
),
2024-03-09 01:38:10 +00:00
"response_format": "b64_json",
}
2024-03-12 20:35:30 +00:00
2024-03-09 01:38:10 +00:00
r = requests.post(
url=f"{app.state.config.OPENAI_API_BASE_URL}/images/generations",
2024-03-09 01:38:10 +00:00
json=data,
headers=headers,
)
2024-03-09 01:38:10 +00:00
r.raise_for_status()
res = r.json()
2024-02-22 02:12:01 +00:00
2024-03-09 01:38:10 +00:00
images = []
for image in res["data"]:
2024-04-30 18:52:08 +00:00
image_filename = save_b64_image(image["b64_json"])
images.append({"url": f"/cache/image/generations/{image_filename}"})
2024-05-02 22:55:42 +00:00
file_body_path = IMAGE_CACHE_DIR.joinpath(f"{image_filename}.json")
2024-03-09 01:38:10 +00:00
with open(file_body_path, "w") as f:
json.dump(data, f)
return images
elif app.state.config.ENGINE == "comfyui":
2024-03-24 00:01:13 +00:00
data = {
"prompt": form_data.prompt,
"width": width,
"height": height,
"n": form_data.n,
}
if app.state.config.IMAGE_STEPS is not None:
data["steps"] = app.state.config.IMAGE_STEPS
2024-03-24 00:01:13 +00:00
if form_data.negative_prompt is not None:
2024-03-24 00:01:13 +00:00
data["negative_prompt"] = form_data.negative_prompt
data = ImageGenerationPayload(**data)
res = comfyui_generate_image(
app.state.config.MODEL,
2024-03-24 00:01:13 +00:00
data,
user.id,
app.state.config.COMFYUI_BASE_URL,
2024-03-24 00:01:13 +00:00
)
log.debug(f"res: {res}")
2024-03-24 00:01:13 +00:00
images = []
for image in res["data"]:
2024-05-02 22:54:31 +00:00
image_filename = save_url_image(image["url"])
images.append({"url": f"/cache/image/generations/{image_filename}"})
file_body_path = IMAGE_CACHE_DIR.joinpath(f"{image_filename}.json")
2024-03-24 00:01:13 +00:00
with open(file_body_path, "w") as f:
json.dump(data.model_dump(exclude_none=True), f)
log.debug(f"images: {images}")
2024-03-24 00:01:13 +00:00
return images
2024-03-09 01:38:10 +00:00
else:
if form_data.model:
set_model_handler(form_data.model)
data = {
"prompt": form_data.prompt,
"batch_size": form_data.n,
"width": width,
"height": height,
}
if app.state.config.IMAGE_STEPS is not None:
data["steps"] = app.state.config.IMAGE_STEPS
2024-03-09 01:38:10 +00:00
if form_data.negative_prompt is not None:
2024-03-09 01:38:10 +00:00
data["negative_prompt"] = form_data.negative_prompt
r = requests.post(
url=f"{app.state.config.AUTOMATIC1111_BASE_URL}/sdapi/v1/txt2img",
2024-03-09 01:38:10 +00:00
json=data,
)
res = r.json()
log.debug(f"res: {res}")
2024-03-09 01:38:10 +00:00
images = []
for image in res["images"]:
2024-04-30 18:52:08 +00:00
image_filename = save_b64_image(image)
images.append({"url": f"/cache/image/generations/{image_filename}"})
2024-05-02 22:55:42 +00:00
file_body_path = IMAGE_CACHE_DIR.joinpath(f"{image_filename}.json")
2024-03-09 01:38:10 +00:00
with open(file_body_path, "w") as f:
json.dump({**data, "info": res["info"]}, f)
return images
2024-02-22 02:36:40 +00:00
except Exception as e:
2024-03-12 20:35:30 +00:00
error = e
if r != None:
data = r.json()
if "error" in data:
error = data["error"]["message"]
raise HTTPException(status_code=400, detail=ERROR_MESSAGES.DEFAULT(error))