2024-08-27 22:10:27 +00:00
|
|
|
import asyncio
|
2024-03-09 01:38:10 +00:00
|
|
|
import base64
|
|
|
|
import json
|
2024-03-20 23:11:36 +00:00
|
|
|
import logging
|
2024-08-27 22:10:27 +00:00
|
|
|
import mimetypes
|
2024-08-20 22:35:42 +00:00
|
|
|
import re
|
2024-08-27 22:10:27 +00:00
|
|
|
import uuid
|
|
|
|
from pathlib import Path
|
|
|
|
from typing import Optional
|
2024-08-20 22:35:42 +00:00
|
|
|
|
2024-08-27 22:10:27 +00:00
|
|
|
import requests
|
2024-12-12 02:53:38 +00:00
|
|
|
|
|
|
|
|
|
|
|
from fastapi import Depends, FastAPI, HTTPException, Request, APIRouter
|
|
|
|
from fastapi.middleware.cors import CORSMiddleware
|
|
|
|
from pydantic import BaseModel
|
2024-12-10 08:54:13 +00:00
|
|
|
|
|
|
|
|
|
|
|
from open_webui.config import CACHE_DIR
|
2024-09-04 14:54:48 +00:00
|
|
|
from open_webui.constants import ERROR_MESSAGES
|
2024-11-11 21:45:13 +00:00
|
|
|
from open_webui.env import ENV, SRC_LOG_LEVELS, ENABLE_FORWARD_USER_INFO_HEADERS
|
|
|
|
|
2024-12-09 00:01:56 +00:00
|
|
|
from open_webui.utils.auth import get_admin_user, get_verified_user
|
2024-12-12 02:53:38 +00:00
|
|
|
from open_webui.utils.images.comfyui import (
|
|
|
|
ComfyUIGenerateImageForm,
|
|
|
|
ComfyUIWorkflow,
|
|
|
|
comfyui_generate_image,
|
|
|
|
)
|
|
|
|
|
2024-03-09 01:38:10 +00:00
|
|
|
|
2024-03-20 23:11:36 +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
|
|
|
|
2024-11-10 02:01:23 +00:00
|
|
|
|
2024-12-12 02:53:38 +00:00
|
|
|
router = APIRouter()
|
2024-08-02 20:35:02 +00:00
|
|
|
|
2024-12-12 02:53:38 +00:00
|
|
|
|
|
|
|
@router.get("/config")
|
2024-03-09 01:38:10 +00:00
|
|
|
async def get_config(request: Request, user=Depends(get_admin_user)):
|
2024-05-10 05:36:10 +00:00
|
|
|
return {
|
2024-12-13 04:22:17 +00:00
|
|
|
"enabled": request.app.state.config.ENABLE_IMAGE_GENERATION,
|
2024-12-13 04:24:36 +00:00
|
|
|
"engine": request.app.state.config.IMAGE_GENERATION_ENGINE,
|
2024-08-20 22:35:42 +00:00
|
|
|
"openai": {
|
2024-12-13 04:24:36 +00:00
|
|
|
"OPENAI_API_BASE_URL": request.app.state.config.IMAGES_OPENAI_API_BASE_URL,
|
|
|
|
"OPENAI_API_KEY": request.app.state.config.IMAGES_OPENAI_API_KEY,
|
2024-08-20 22:35:42 +00:00
|
|
|
},
|
|
|
|
"automatic1111": {
|
2024-12-12 02:53:38 +00:00
|
|
|
"AUTOMATIC1111_BASE_URL": request.app.state.config.AUTOMATIC1111_BASE_URL,
|
|
|
|
"AUTOMATIC1111_API_AUTH": request.app.state.config.AUTOMATIC1111_API_AUTH,
|
|
|
|
"AUTOMATIC1111_CFG_SCALE": request.app.state.config.AUTOMATIC1111_CFG_SCALE,
|
|
|
|
"AUTOMATIC1111_SAMPLER": request.app.state.config.AUTOMATIC1111_SAMPLER,
|
|
|
|
"AUTOMATIC1111_SCHEDULER": request.app.state.config.AUTOMATIC1111_SCHEDULER,
|
2024-08-20 22:35:42 +00:00
|
|
|
},
|
|
|
|
"comfyui": {
|
2024-12-12 02:53:38 +00:00
|
|
|
"COMFYUI_BASE_URL": request.app.state.config.COMFYUI_BASE_URL,
|
2024-12-17 07:29:00 +00:00
|
|
|
"COMFYUI_API_KEY": request.app.state.config.COMFYUI_API_KEY,
|
2024-12-12 02:53:38 +00:00
|
|
|
"COMFYUI_WORKFLOW": request.app.state.config.COMFYUI_WORKFLOW,
|
|
|
|
"COMFYUI_WORKFLOW_NODES": request.app.state.config.COMFYUI_WORKFLOW_NODES,
|
2024-08-20 22:35:42 +00:00
|
|
|
},
|
2024-05-10 05:36:10 +00:00
|
|
|
}
|
2024-02-22 02:12:01 +00:00
|
|
|
|
|
|
|
|
2024-08-20 22:35:42 +00:00
|
|
|
class OpenAIConfigForm(BaseModel):
|
|
|
|
OPENAI_API_BASE_URL: str
|
|
|
|
OPENAI_API_KEY: str
|
|
|
|
|
|
|
|
|
|
|
|
class Automatic1111ConfigForm(BaseModel):
|
|
|
|
AUTOMATIC1111_BASE_URL: str
|
|
|
|
AUTOMATIC1111_API_AUTH: str
|
2024-12-10 07:22:47 +00:00
|
|
|
AUTOMATIC1111_CFG_SCALE: Optional[str | float | int]
|
2024-09-12 12:00:24 +00:00
|
|
|
AUTOMATIC1111_SAMPLER: Optional[str]
|
|
|
|
AUTOMATIC1111_SCHEDULER: Optional[str]
|
2024-08-20 22:35:42 +00:00
|
|
|
|
|
|
|
|
|
|
|
class ComfyUIConfigForm(BaseModel):
|
|
|
|
COMFYUI_BASE_URL: str
|
2024-12-17 07:29:00 +00:00
|
|
|
COMFYUI_API_KEY: str
|
2024-08-20 22:35:42 +00:00
|
|
|
COMFYUI_WORKFLOW: str
|
|
|
|
COMFYUI_WORKFLOW_NODES: list[dict]
|
|
|
|
|
|
|
|
|
|
|
|
class ConfigForm(BaseModel):
|
2024-03-09 01:38:10 +00:00
|
|
|
enabled: bool
|
2024-08-20 22:35:42 +00:00
|
|
|
engine: str
|
|
|
|
openai: OpenAIConfigForm
|
|
|
|
automatic1111: Automatic1111ConfigForm
|
|
|
|
comfyui: ComfyUIConfigForm
|
2024-03-09 01:38:10 +00:00
|
|
|
|
|
|
|
|
2024-12-12 02:53:38 +00:00
|
|
|
@router.post("/config/update")
|
|
|
|
async def update_config(
|
|
|
|
request: Request, form_data: ConfigForm, user=Depends(get_admin_user)
|
|
|
|
):
|
2024-12-13 04:24:36 +00:00
|
|
|
request.app.state.config.IMAGE_GENERATION_ENGINE = form_data.engine
|
2024-12-13 04:22:17 +00:00
|
|
|
request.app.state.config.ENABLE_IMAGE_GENERATION = form_data.enabled
|
2024-02-22 02:12:01 +00:00
|
|
|
|
2024-12-13 04:24:36 +00:00
|
|
|
request.app.state.config.IMAGES_OPENAI_API_BASE_URL = (
|
|
|
|
form_data.openai.OPENAI_API_BASE_URL
|
|
|
|
)
|
|
|
|
request.app.state.config.IMAGES_OPENAI_API_KEY = form_data.openai.OPENAI_API_KEY
|
2024-02-22 02:12:01 +00:00
|
|
|
|
2024-12-12 02:53:38 +00:00
|
|
|
request.app.state.config.AUTOMATIC1111_BASE_URL = (
|
2024-08-20 22:35:42 +00:00
|
|
|
form_data.automatic1111.AUTOMATIC1111_BASE_URL
|
|
|
|
)
|
2024-12-12 02:53:38 +00:00
|
|
|
request.app.state.config.AUTOMATIC1111_API_AUTH = (
|
2024-08-20 22:35:42 +00:00
|
|
|
form_data.automatic1111.AUTOMATIC1111_API_AUTH
|
|
|
|
)
|
2024-09-13 04:49:23 +00:00
|
|
|
|
2024-12-12 02:53:38 +00:00
|
|
|
request.app.state.config.AUTOMATIC1111_CFG_SCALE = (
|
2024-09-13 04:49:23 +00:00
|
|
|
float(form_data.automatic1111.AUTOMATIC1111_CFG_SCALE)
|
2024-09-20 01:18:14 +00:00
|
|
|
if form_data.automatic1111.AUTOMATIC1111_CFG_SCALE
|
2024-09-13 04:49:23 +00:00
|
|
|
else None
|
|
|
|
)
|
2024-12-12 02:53:38 +00:00
|
|
|
request.app.state.config.AUTOMATIC1111_SAMPLER = (
|
2024-09-13 04:49:23 +00:00
|
|
|
form_data.automatic1111.AUTOMATIC1111_SAMPLER
|
2024-09-20 01:18:14 +00:00
|
|
|
if form_data.automatic1111.AUTOMATIC1111_SAMPLER
|
2024-09-13 04:49:23 +00:00
|
|
|
else None
|
|
|
|
)
|
2024-12-12 02:53:38 +00:00
|
|
|
request.app.state.config.AUTOMATIC1111_SCHEDULER = (
|
2024-09-13 04:49:23 +00:00
|
|
|
form_data.automatic1111.AUTOMATIC1111_SCHEDULER
|
2024-09-20 01:18:14 +00:00
|
|
|
if form_data.automatic1111.AUTOMATIC1111_SCHEDULER
|
2024-09-13 04:49:23 +00:00
|
|
|
else None
|
|
|
|
)
|
2024-02-22 02:12:01 +00:00
|
|
|
|
2024-12-12 02:53:38 +00:00
|
|
|
request.app.state.config.COMFYUI_BASE_URL = (
|
|
|
|
form_data.comfyui.COMFYUI_BASE_URL.strip("/")
|
|
|
|
)
|
|
|
|
request.app.state.config.COMFYUI_WORKFLOW = form_data.comfyui.COMFYUI_WORKFLOW
|
|
|
|
request.app.state.config.COMFYUI_WORKFLOW_NODES = (
|
|
|
|
form_data.comfyui.COMFYUI_WORKFLOW_NODES
|
|
|
|
)
|
2024-02-22 02:12:01 +00:00
|
|
|
|
2024-03-23 22:38:59 +00:00
|
|
|
return {
|
2024-12-13 04:22:17 +00:00
|
|
|
"enabled": request.app.state.config.ENABLE_IMAGE_GENERATION,
|
2024-12-13 04:24:36 +00:00
|
|
|
"engine": request.app.state.config.IMAGE_GENERATION_ENGINE,
|
2024-08-20 22:35:42 +00:00
|
|
|
"openai": {
|
2024-12-13 04:24:36 +00:00
|
|
|
"OPENAI_API_BASE_URL": request.app.state.config.IMAGES_OPENAI_API_BASE_URL,
|
|
|
|
"OPENAI_API_KEY": request.app.state.config.IMAGES_OPENAI_API_KEY,
|
2024-08-20 22:35:42 +00:00
|
|
|
},
|
|
|
|
"automatic1111": {
|
2024-12-12 02:53:38 +00:00
|
|
|
"AUTOMATIC1111_BASE_URL": request.app.state.config.AUTOMATIC1111_BASE_URL,
|
|
|
|
"AUTOMATIC1111_API_AUTH": request.app.state.config.AUTOMATIC1111_API_AUTH,
|
|
|
|
"AUTOMATIC1111_CFG_SCALE": request.app.state.config.AUTOMATIC1111_CFG_SCALE,
|
|
|
|
"AUTOMATIC1111_SAMPLER": request.app.state.config.AUTOMATIC1111_SAMPLER,
|
|
|
|
"AUTOMATIC1111_SCHEDULER": request.app.state.config.AUTOMATIC1111_SCHEDULER,
|
2024-08-20 22:35:42 +00:00
|
|
|
},
|
|
|
|
"comfyui": {
|
2024-12-12 02:53:38 +00:00
|
|
|
"COMFYUI_BASE_URL": request.app.state.config.COMFYUI_BASE_URL,
|
2024-12-17 07:29:00 +00:00
|
|
|
"COMFYUI_API_KEY": request.app.state.config.COMFYUI_API_KEY,
|
2024-12-12 02:53:38 +00:00
|
|
|
"COMFYUI_WORKFLOW": request.app.state.config.COMFYUI_WORKFLOW,
|
|
|
|
"COMFYUI_WORKFLOW_NODES": request.app.state.config.COMFYUI_WORKFLOW_NODES,
|
2024-08-20 22:35:42 +00:00
|
|
|
},
|
2024-03-23 22:38:59 +00:00
|
|
|
}
|
2024-02-22 02:12:01 +00:00
|
|
|
|
|
|
|
|
2024-12-12 02:53:38 +00:00
|
|
|
def get_automatic1111_api_auth(request: Request):
|
|
|
|
if request.app.state.config.AUTOMATIC1111_API_AUTH is None:
|
2024-08-20 22:35:42 +00:00
|
|
|
return ""
|
2024-06-20 06:15:49 +00:00
|
|
|
else:
|
2024-12-12 02:53:38 +00:00
|
|
|
auth1111_byte_string = request.app.state.config.AUTOMATIC1111_API_AUTH.encode(
|
|
|
|
"utf-8"
|
|
|
|
)
|
2024-08-20 22:35:42 +00:00
|
|
|
auth1111_base64_encoded_bytes = base64.b64encode(auth1111_byte_string)
|
|
|
|
auth1111_base64_encoded_string = auth1111_base64_encoded_bytes.decode("utf-8")
|
|
|
|
return f"Basic {auth1111_base64_encoded_string}"
|
2024-02-22 02:12:01 +00:00
|
|
|
|
|
|
|
|
2024-12-12 02:53:38 +00:00
|
|
|
@router.get("/config/url/verify")
|
|
|
|
async def verify_url(request: Request, user=Depends(get_admin_user)):
|
2024-12-13 04:24:36 +00:00
|
|
|
if request.app.state.config.IMAGE_GENERATION_ENGINE == "automatic1111":
|
2024-08-21 15:27:39 +00:00
|
|
|
try:
|
|
|
|
r = requests.get(
|
2024-12-12 02:53:38 +00:00
|
|
|
url=f"{request.app.state.config.AUTOMATIC1111_BASE_URL}/sdapi/v1/options",
|
|
|
|
headers={"authorization": get_automatic1111_api_auth(request)},
|
2024-08-21 15:27:39 +00:00
|
|
|
)
|
|
|
|
r.raise_for_status()
|
|
|
|
return True
|
2024-08-27 22:10:27 +00:00
|
|
|
except Exception:
|
2024-12-13 04:22:17 +00:00
|
|
|
request.app.state.config.ENABLE_IMAGE_GENERATION = False
|
2024-08-21 15:27:39 +00:00
|
|
|
raise HTTPException(status_code=400, detail=ERROR_MESSAGES.INVALID_URL)
|
2024-12-13 04:24:36 +00:00
|
|
|
elif request.app.state.config.IMAGE_GENERATION_ENGINE == "comfyui":
|
2024-08-21 15:27:39 +00:00
|
|
|
try:
|
2024-12-12 02:53:38 +00:00
|
|
|
r = requests.get(
|
|
|
|
url=f"{request.app.state.config.COMFYUI_BASE_URL}/object_info"
|
|
|
|
)
|
2024-08-21 15:27:39 +00:00
|
|
|
r.raise_for_status()
|
|
|
|
return True
|
2024-08-27 22:10:27 +00:00
|
|
|
except Exception:
|
2024-12-13 04:22:17 +00:00
|
|
|
request.app.state.config.ENABLE_IMAGE_GENERATION = False
|
2024-08-21 15:27:39 +00:00
|
|
|
raise HTTPException(status_code=400, detail=ERROR_MESSAGES.INVALID_URL)
|
|
|
|
else:
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
2024-12-12 02:53:38 +00:00
|
|
|
def set_image_model(request: Request, model: str):
|
2024-09-04 13:25:31 +00:00
|
|
|
log.info(f"Setting image model to {model}")
|
2024-12-13 04:26:28 +00:00
|
|
|
request.app.state.config.IMAGE_GENERATION_MODEL = model
|
2024-12-13 04:24:36 +00:00
|
|
|
if request.app.state.config.IMAGE_GENERATION_ENGINE in ["", "automatic1111"]:
|
2024-08-20 22:35:42 +00:00
|
|
|
api_auth = get_automatic1111_api_auth()
|
|
|
|
r = requests.get(
|
2024-12-12 02:53:38 +00:00
|
|
|
url=f"{request.app.state.config.AUTOMATIC1111_BASE_URL}/sdapi/v1/options",
|
2024-08-20 22:35:42 +00:00
|
|
|
headers={"authorization": api_auth},
|
|
|
|
)
|
|
|
|
options = r.json()
|
|
|
|
if model != options["sd_model_checkpoint"]:
|
|
|
|
options["sd_model_checkpoint"] = model
|
|
|
|
r = requests.post(
|
2024-12-12 02:53:38 +00:00
|
|
|
url=f"{request.app.state.config.AUTOMATIC1111_BASE_URL}/sdapi/v1/options",
|
2024-08-20 22:35:42 +00:00
|
|
|
json=options,
|
|
|
|
headers={"authorization": api_auth},
|
|
|
|
)
|
2024-12-13 04:26:28 +00:00
|
|
|
return request.app.state.config.IMAGE_GENERATION_MODEL
|
2024-03-09 01:38:10 +00:00
|
|
|
|
|
|
|
|
2024-12-13 04:26:28 +00:00
|
|
|
def get_image_model(request):
|
2024-12-13 04:24:36 +00:00
|
|
|
if request.app.state.config.IMAGE_GENERATION_ENGINE == "openai":
|
2024-12-12 02:53:38 +00:00
|
|
|
return (
|
2024-12-13 04:26:28 +00:00
|
|
|
request.app.state.config.IMAGE_GENERATION_MODEL
|
|
|
|
if request.app.state.config.IMAGE_GENERATION_MODEL
|
2024-12-12 02:53:38 +00:00
|
|
|
else "dall-e-2"
|
|
|
|
)
|
2024-12-13 04:24:36 +00:00
|
|
|
elif request.app.state.config.IMAGE_GENERATION_ENGINE == "comfyui":
|
2024-12-13 04:26:28 +00:00
|
|
|
return (
|
|
|
|
request.app.state.config.IMAGE_GENERATION_MODEL
|
|
|
|
if request.app.state.config.IMAGE_GENERATION_MODEL
|
|
|
|
else ""
|
|
|
|
)
|
2024-12-12 02:53:38 +00:00
|
|
|
elif (
|
2024-12-13 04:24:36 +00:00
|
|
|
request.app.state.config.IMAGE_GENERATION_ENGINE == "automatic1111"
|
|
|
|
or request.app.state.config.IMAGE_GENERATION_ENGINE == ""
|
2024-12-12 02:53:38 +00:00
|
|
|
):
|
2024-08-20 22:35:42 +00:00
|
|
|
try:
|
|
|
|
r = requests.get(
|
2024-12-12 02:53:38 +00:00
|
|
|
url=f"{request.app.state.config.AUTOMATIC1111_BASE_URL}/sdapi/v1/options",
|
2024-08-20 22:35:42 +00:00
|
|
|
headers={"authorization": get_automatic1111_api_auth()},
|
|
|
|
)
|
|
|
|
options = r.json()
|
|
|
|
return options["sd_model_checkpoint"]
|
|
|
|
except Exception as e:
|
2024-12-13 04:22:17 +00:00
|
|
|
request.app.state.config.ENABLE_IMAGE_GENERATION = False
|
2024-08-20 22:35:42 +00:00
|
|
|
raise HTTPException(status_code=400, detail=ERROR_MESSAGES.DEFAULT(e))
|
2024-03-09 01:38:10 +00:00
|
|
|
|
|
|
|
|
2024-08-20 22:35:42 +00:00
|
|
|
class ImageConfigForm(BaseModel):
|
2024-08-20 23:21:03 +00:00
|
|
|
MODEL: str
|
|
|
|
IMAGE_SIZE: str
|
|
|
|
IMAGE_STEPS: int
|
2024-03-09 01:38:10 +00:00
|
|
|
|
2024-04-23 11:14:31 +00:00
|
|
|
|
2024-12-12 02:53:38 +00:00
|
|
|
@router.get("/image/config")
|
|
|
|
async def get_image_config(request: Request, user=Depends(get_admin_user)):
|
2024-03-09 01:38:10 +00:00
|
|
|
return {
|
2024-12-13 04:26:28 +00:00
|
|
|
"MODEL": request.app.state.config.IMAGE_GENERATION_MODEL,
|
2024-12-12 02:53:38 +00:00
|
|
|
"IMAGE_SIZE": request.app.state.config.IMAGE_SIZE,
|
|
|
|
"IMAGE_STEPS": request.app.state.config.IMAGE_STEPS,
|
2024-03-09 01:38:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2024-12-12 02:53:38 +00:00
|
|
|
@router.post("/image/config/update")
|
|
|
|
async def update_image_config(
|
|
|
|
request: Request, form_data: ImageConfigForm, user=Depends(get_admin_user)
|
|
|
|
):
|
2024-09-04 13:25:31 +00:00
|
|
|
|
2024-12-12 02:53:38 +00:00
|
|
|
set_image_model(request, form_data.MODEL)
|
2024-02-23 03:32:36 +00:00
|
|
|
|
2024-08-20 22:35:42 +00:00
|
|
|
pattern = r"^\d+x\d+$"
|
2024-08-20 23:21:03 +00:00
|
|
|
if re.match(pattern, form_data.IMAGE_SIZE):
|
2024-12-12 02:53:38 +00:00
|
|
|
request.app.state.config.IMAGE_SIZE = form_data.IMAGE_SIZE
|
2024-02-23 03:32:36 +00:00
|
|
|
else:
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=400,
|
|
|
|
detail=ERROR_MESSAGES.INCORRECT_FORMAT(" (e.g., 512x512)."),
|
|
|
|
)
|
2024-02-25 02:08:35 +00:00
|
|
|
|
2024-08-20 23:21:03 +00:00
|
|
|
if form_data.IMAGE_STEPS >= 0:
|
2024-12-12 02:53:38 +00:00
|
|
|
request.app.state.config.IMAGE_STEPS = form_data.IMAGE_STEPS
|
2024-02-24 15:44:08 +00:00
|
|
|
else:
|
|
|
|
raise HTTPException(
|
|
|
|
status_code=400,
|
|
|
|
detail=ERROR_MESSAGES.INCORRECT_FORMAT(" (e.g., 50)."),
|
|
|
|
)
|
2024-02-23 03:32:36 +00:00
|
|
|
|
2024-08-20 22:35:42 +00:00
|
|
|
return {
|
2024-12-13 04:26:28 +00:00
|
|
|
"MODEL": request.app.state.config.IMAGE_GENERATION_MODEL,
|
2024-12-12 02:53:38 +00:00
|
|
|
"IMAGE_SIZE": request.app.state.config.IMAGE_SIZE,
|
|
|
|
"IMAGE_STEPS": request.app.state.config.IMAGE_STEPS,
|
2024-08-20 22:35:42 +00:00
|
|
|
}
|
|
|
|
|
2024-02-23 03:32:36 +00:00
|
|
|
|
2024-12-12 02:53:38 +00:00
|
|
|
@router.get("/models")
|
|
|
|
def get_models(request: Request, user=Depends(get_verified_user)):
|
2024-02-22 02:12:01 +00:00
|
|
|
try:
|
2024-12-13 04:24:36 +00:00
|
|
|
if request.app.state.config.IMAGE_GENERATION_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"},
|
|
|
|
]
|
2024-12-13 04:24:36 +00:00
|
|
|
elif request.app.state.config.IMAGE_GENERATION_ENGINE == "comfyui":
|
2024-08-20 22:35:42 +00:00
|
|
|
# TODO - get models from comfyui
|
2024-12-17 07:29:00 +00:00
|
|
|
headers = {"Authorization": f"Bearer {request.app.state.config.COMFYUI_API_KEY}"}
|
|
|
|
r = requests.get(url=f"{request.app.state.config.COMFYUI_BASE_URL}/object_info", headers=headers)
|
2024-03-23 22:38:59 +00:00
|
|
|
info = r.json()
|
|
|
|
|
2024-12-12 02:53:38 +00:00
|
|
|
workflow = json.loads(request.app.state.config.COMFYUI_WORKFLOW)
|
2024-08-21 12:44:47 +00:00
|
|
|
model_node_id = None
|
|
|
|
|
2024-12-12 02:53:38 +00:00
|
|
|
for node in request.app.state.config.COMFYUI_WORKFLOW_NODES:
|
2024-08-21 12:44:47 +00:00
|
|
|
if node["type"] == "model":
|
2024-08-21 22:25:43 +00:00
|
|
|
if node["node_ids"]:
|
|
|
|
model_node_id = node["node_ids"][0]
|
2024-08-21 12:44:47 +00:00
|
|
|
break
|
|
|
|
|
|
|
|
if model_node_id:
|
|
|
|
model_list_key = None
|
|
|
|
|
|
|
|
print(workflow[model_node_id]["class_type"])
|
|
|
|
for key in info[workflow[model_node_id]["class_type"]]["input"][
|
|
|
|
"required"
|
|
|
|
]:
|
|
|
|
if "_name" in key:
|
|
|
|
model_list_key = key
|
|
|
|
break
|
|
|
|
|
|
|
|
if model_list_key:
|
|
|
|
return list(
|
|
|
|
map(
|
|
|
|
lambda model: {"id": model, "name": model},
|
|
|
|
info[workflow[model_node_id]["class_type"]]["input"][
|
|
|
|
"required"
|
|
|
|
][model_list_key][0],
|
|
|
|
)
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
return list(
|
|
|
|
map(
|
|
|
|
lambda model: {"id": model, "name": model},
|
|
|
|
info["CheckpointLoaderSimple"]["input"]["required"][
|
|
|
|
"ckpt_name"
|
|
|
|
][0],
|
|
|
|
)
|
2024-03-23 22:38:59 +00:00
|
|
|
)
|
2024-08-20 22:35:42 +00:00
|
|
|
elif (
|
2024-12-13 04:24:36 +00:00
|
|
|
request.app.state.config.IMAGE_GENERATION_ENGINE == "automatic1111"
|
|
|
|
or request.app.state.config.IMAGE_GENERATION_ENGINE == ""
|
2024-08-20 22:35:42 +00:00
|
|
|
):
|
2024-03-09 01:38:10 +00:00
|
|
|
r = requests.get(
|
2024-12-12 02:53:38 +00:00
|
|
|
url=f"{request.app.state.config.AUTOMATIC1111_BASE_URL}/sdapi/v1/sd-models",
|
2024-06-25 15:01:05 +00:00
|
|
|
headers={"authorization": get_automatic1111_api_auth()},
|
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:
|
2024-12-13 04:22:17 +00:00
|
|
|
request.app.state.config.ENABLE_IMAGE_GENERATION = 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 GenerateImageForm(BaseModel):
|
|
|
|
model: Optional[str] = None
|
|
|
|
prompt: str
|
2024-03-09 03:54:47 +00:00
|
|
|
size: Optional[str] = None
|
2024-08-20 22:35:42 +00:00
|
|
|
n: int = 1
|
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:
|
2024-04-28 04:00:52 +00:00
|
|
|
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()
|
2024-04-28 04:00:52 +00:00
|
|
|
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
|
|
|
|
2024-04-28 04:00:52 +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}")
|
2024-04-28 04:00:52 +00:00
|
|
|
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
|
2024-04-28 04:00:52 +00:00
|
|
|
else:
|
2024-08-27 22:10:27 +00:00
|
|
|
log.error("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:
|
2024-03-31 19:17:29 +00:00
|
|
|
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-12-12 02:53:38 +00:00
|
|
|
@router.post("/generations")
|
2024-07-15 14:25:00 +00:00
|
|
|
async def image_generations(
|
2024-12-12 02:53:38 +00:00
|
|
|
request: Request,
|
2024-06-25 15:01:05 +00:00
|
|
|
form_data: GenerateImageForm,
|
2024-06-27 18:29:59 +00:00
|
|
|
user=Depends(get_verified_user),
|
2024-02-22 02:12:01 +00:00
|
|
|
):
|
2024-12-12 02:53:38 +00:00
|
|
|
width, height = tuple(map(int, request.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:
|
2024-12-13 04:24:36 +00:00
|
|
|
if request.app.state.config.IMAGE_GENERATION_ENGINE == "openai":
|
2024-03-09 01:38:10 +00:00
|
|
|
headers = {}
|
2024-12-12 02:53:38 +00:00
|
|
|
headers["Authorization"] = (
|
2024-12-13 04:24:36 +00:00
|
|
|
f"Bearer {request.app.state.config.IMAGES_OPENAI_API_KEY}"
|
2024-12-12 02:53:38 +00:00
|
|
|
)
|
2024-03-09 01:38:10 +00:00
|
|
|
headers["Content-Type"] = "application/json"
|
2024-02-22 02:36:40 +00:00
|
|
|
|
2024-11-01 15:23:18 +00:00
|
|
|
if ENABLE_FORWARD_USER_INFO_HEADERS:
|
2024-10-30 15:02:56 +00:00
|
|
|
headers["X-OpenWebUI-User-Name"] = user.name
|
|
|
|
headers["X-OpenWebUI-User-Id"] = user.id
|
|
|
|
headers["X-OpenWebUI-User-Email"] = user.email
|
|
|
|
headers["X-OpenWebUI-User-Role"] = user.role
|
|
|
|
|
2024-03-09 01:38:10 +00:00
|
|
|
data = {
|
2024-05-10 07:03:24 +00:00
|
|
|
"model": (
|
2024-12-13 04:26:28 +00:00
|
|
|
request.app.state.config.IMAGE_GENERATION_MODEL
|
|
|
|
if request.app.state.config.IMAGE_GENERATION_MODEL != ""
|
2024-05-10 07:03:24 +00:00
|
|
|
else "dall-e-2"
|
|
|
|
),
|
2024-03-09 01:38:10 +00:00
|
|
|
"prompt": form_data.prompt,
|
|
|
|
"n": form_data.n,
|
2024-05-10 05:36:10 +00:00
|
|
|
"size": (
|
2024-12-12 02:53:38 +00:00
|
|
|
form_data.size
|
|
|
|
if form_data.size
|
|
|
|
else request.app.state.config.IMAGE_SIZE
|
2024-05-10 05:36:10 +00:00
|
|
|
),
|
2024-03-09 01:38:10 +00:00
|
|
|
"response_format": "b64_json",
|
|
|
|
}
|
2024-03-12 20:35:30 +00:00
|
|
|
|
2024-09-20 01:16:08 +00:00
|
|
|
# Use asyncio.to_thread for the requests.post call
|
|
|
|
r = await asyncio.to_thread(
|
|
|
|
requests.post,
|
2024-12-13 04:24:36 +00:00
|
|
|
url=f"{request.app.state.config.IMAGES_OPENAI_API_BASE_URL}/images/generations",
|
2024-03-09 01:38:10 +00:00
|
|
|
json=data,
|
|
|
|
headers=headers,
|
|
|
|
)
|
2024-02-24 15:44:08 +00:00
|
|
|
|
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
|
|
|
|
|
2024-12-13 04:24:36 +00:00
|
|
|
elif request.app.state.config.IMAGE_GENERATION_ENGINE == "comfyui":
|
2024-03-24 00:01:13 +00:00
|
|
|
data = {
|
|
|
|
"prompt": form_data.prompt,
|
|
|
|
"width": width,
|
|
|
|
"height": height,
|
|
|
|
"n": form_data.n,
|
|
|
|
}
|
|
|
|
|
2024-12-12 02:53:38 +00:00
|
|
|
if request.app.state.config.IMAGE_STEPS is not None:
|
|
|
|
data["steps"] = request.app.state.config.IMAGE_STEPS
|
2024-03-24 00:01:13 +00:00
|
|
|
|
2024-05-10 05:36:10 +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
|
|
|
|
|
2024-08-21 12:49:17 +00:00
|
|
|
form_data = ComfyUIGenerateImageForm(
|
2024-08-21 12:49:54 +00:00
|
|
|
**{
|
2024-08-21 12:49:17 +00:00
|
|
|
"workflow": ComfyUIWorkflow(
|
2024-08-21 12:49:54 +00:00
|
|
|
**{
|
2024-12-12 02:53:38 +00:00
|
|
|
"workflow": request.app.state.config.COMFYUI_WORKFLOW,
|
|
|
|
"nodes": request.app.state.config.COMFYUI_WORKFLOW_NODES,
|
2024-08-21 12:49:17 +00:00
|
|
|
}
|
|
|
|
),
|
|
|
|
**data,
|
|
|
|
}
|
|
|
|
)
|
2024-08-08 00:15:20 +00:00
|
|
|
res = await comfyui_generate_image(
|
2024-12-13 04:26:28 +00:00
|
|
|
request.app.state.config.IMAGE_GENERATION_MODEL,
|
2024-08-20 16:17:15 +00:00
|
|
|
form_data,
|
2024-03-24 00:01:13 +00:00
|
|
|
user.id,
|
2024-12-12 02:53:38 +00:00
|
|
|
request.app.state.config.COMFYUI_BASE_URL,
|
2024-12-17 07:29:00 +00:00
|
|
|
request.app.state.config.COMFYUI_API_KEY,
|
2024-03-24 00:01:13 +00:00
|
|
|
)
|
2024-03-31 19:17:29 +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:
|
2024-08-21 13:15:29 +00:00
|
|
|
json.dump(form_data.model_dump(exclude_none=True), f)
|
2024-03-24 00:01:13 +00:00
|
|
|
|
2024-03-31 19:17:29 +00:00
|
|
|
log.debug(f"images: {images}")
|
2024-03-24 00:01:13 +00:00
|
|
|
return images
|
2024-08-20 22:35:42 +00:00
|
|
|
elif (
|
2024-12-13 04:24:36 +00:00
|
|
|
request.app.state.config.IMAGE_GENERATION_ENGINE == "automatic1111"
|
|
|
|
or request.app.state.config.IMAGE_GENERATION_ENGINE == ""
|
2024-08-20 22:35:42 +00:00
|
|
|
):
|
2024-03-09 01:38:10 +00:00
|
|
|
if form_data.model:
|
2024-08-20 22:35:42 +00:00
|
|
|
set_image_model(form_data.model)
|
2024-03-09 01:38:10 +00:00
|
|
|
|
|
|
|
data = {
|
|
|
|
"prompt": form_data.prompt,
|
|
|
|
"batch_size": form_data.n,
|
|
|
|
"width": width,
|
|
|
|
"height": height,
|
|
|
|
}
|
|
|
|
|
2024-12-12 02:53:38 +00:00
|
|
|
if request.app.state.config.IMAGE_STEPS is not None:
|
|
|
|
data["steps"] = request.app.state.config.IMAGE_STEPS
|
2024-03-09 01:38:10 +00:00
|
|
|
|
2024-05-10 05:36: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
|
2024-09-13 04:49:23 +00:00
|
|
|
|
2024-12-12 02:53:38 +00:00
|
|
|
if request.app.state.config.AUTOMATIC1111_CFG_SCALE:
|
|
|
|
data["cfg_scale"] = request.app.state.config.AUTOMATIC1111_CFG_SCALE
|
2024-09-07 15:21:17 +00:00
|
|
|
|
2024-12-12 02:53:38 +00:00
|
|
|
if request.app.state.config.AUTOMATIC1111_SAMPLER:
|
|
|
|
data["sampler_name"] = request.app.state.config.AUTOMATIC1111_SAMPLER
|
2024-09-07 15:21:17 +00:00
|
|
|
|
2024-12-12 02:53:38 +00:00
|
|
|
if request.app.state.config.AUTOMATIC1111_SCHEDULER:
|
|
|
|
data["scheduler"] = request.app.state.config.AUTOMATIC1111_SCHEDULER
|
2024-03-09 01:38:10 +00:00
|
|
|
|
2024-08-23 14:51:34 +00:00
|
|
|
# Use asyncio.to_thread for the requests.post call
|
|
|
|
r = await asyncio.to_thread(
|
|
|
|
requests.post,
|
2024-12-12 02:53:38 +00:00
|
|
|
url=f"{request.app.state.config.AUTOMATIC1111_BASE_URL}/sdapi/v1/txt2img",
|
2024-03-09 01:38:10 +00:00
|
|
|
json=data,
|
2024-06-20 06:15:49 +00:00
|
|
|
headers={"authorization": get_automatic1111_api_auth()},
|
2024-03-09 01:38:10 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
res = r.json()
|
2024-03-20 23:11:36 +00:00
|
|
|
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))
|