open-webui/backend/open_webui/routers/images.py

1045 lines
40 KiB
Python
Raw Normal View History

2024-08-27 22:10:27 +00:00
import asyncio
2024-03-09 01:38:10 +00:00
import base64
2025-11-05 08:31:37 +00:00
import uuid
import io
2024-03-09 01:38:10 +00:00
import json
import logging
2025-02-06 23:22:20 +00:00
import mimetypes
2024-08-20 22:35:42 +00:00
import re
2024-08-27 22:10:27 +00:00
from pathlib import Path
from typing import Optional
2024-08-20 22:35:42 +00:00
from urllib.parse import quote
2024-08-27 22:10:27 +00:00
import requests
2025-11-05 08:31:37 +00:00
from fastapi import APIRouter, Depends, HTTPException, Request, UploadFile
from fastapi.responses import FileResponse
2025-08-22 12:58:25 +00:00
2024-12-10 08:54:13 +00:00
from open_webui.config import CACHE_DIR
from open_webui.constants import ERROR_MESSAGES
from open_webui.env import ENABLE_FORWARD_USER_INFO_HEADERS, SRC_LOG_LEVELS
2025-11-05 08:31:37 +00:00
from open_webui.routers.files import upload_file_handler, get_file_content_by_id
2024-12-09 00:01:56 +00:00
from open_webui.utils.auth import get_admin_user, get_verified_user
2025-11-04 18:30:59 +00:00
from open_webui.utils.headers import include_user_info_headers
2024-12-12 02:53:38 +00:00
from open_webui.utils.images.comfyui import (
2025-11-04 18:30:59 +00:00
ComfyUICreateImageForm,
2025-11-06 08:43:59 +00:00
ComfyUIEditImageForm,
2024-12-12 02:53:38 +00:00
ComfyUIWorkflow,
2025-11-06 08:43:59 +00:00
comfyui_upload_image,
2025-11-04 18:30:59 +00:00
comfyui_create_image,
2025-11-06 08:43:59 +00:00
comfyui_edit_image,
2024-12-12 02:53:38 +00:00
)
from pydantic import BaseModel
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 = CACHE_DIR / "image" / "generations"
2024-03-09 01:38:10 +00:00
IMAGE_CACHE_DIR.mkdir(parents=True, exist_ok=True)
2024-02-22 02:12:01 +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
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-12-25 20:26:13 +00:00
api_auth = get_automatic1111_api_auth(request)
2025-11-15 20:43:23 +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": api_auth},
)
2025-11-15 20:43:23 +00:00
options = r.json()
if model != options["sd_model_checkpoint"]:
options["sd_model_checkpoint"] = model
r = requests.post(
url=f"{request.app.state.config.AUTOMATIC1111_BASE_URL}/sdapi/v1/options",
json=options,
headers={"authorization": api_auth},
)
except Exception as e:
log.debug(f"{e}")
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"
)
elif request.app.state.config.IMAGE_GENERATION_ENGINE == "gemini":
return (
request.app.state.config.IMAGE_GENERATION_MODEL
if request.app.state.config.IMAGE_GENERATION_MODEL
else "imagen-3.0-generate-002"
)
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-12-26 07:34:54 +00:00
headers={"authorization": get_automatic1111_api_auth(request)},
2024-08-20 22:35:42 +00:00
)
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
2025-11-04 18:30:59 +00:00
class ImagesConfig(BaseModel):
ENABLE_IMAGE_GENERATION: bool
ENABLE_IMAGE_PROMPT_GENERATION: bool
IMAGE_GENERATION_ENGINE: str
2025-11-04 18:30:59 +00:00
IMAGE_GENERATION_MODEL: str
IMAGE_SIZE: Optional[str]
IMAGE_STEPS: Optional[int]
2024-03-09 01:38:10 +00:00
2025-11-04 18:30:59 +00:00
IMAGES_OPENAI_API_BASE_URL: str
IMAGES_OPENAI_API_KEY: str
IMAGES_OPENAI_API_VERSION: str
2025-11-04 18:30:59 +00:00
AUTOMATIC1111_BASE_URL: str
2025-11-15 20:43:23 +00:00
AUTOMATIC1111_API_AUTH: Optional[dict | str]
2025-11-04 18:30:59 +00:00
AUTOMATIC1111_PARAMS: Optional[dict | str]
2025-11-04 18:30:59 +00:00
COMFYUI_BASE_URL: str
COMFYUI_API_KEY: str
COMFYUI_WORKFLOW: str
COMFYUI_WORKFLOW_NODES: list[dict]
2025-11-04 18:30:59 +00:00
IMAGES_GEMINI_API_BASE_URL: str
IMAGES_GEMINI_API_KEY: str
IMAGES_GEMINI_ENDPOINT_METHOD: str
2025-11-04 18:30:59 +00:00
2025-11-05 08:31:37 +00:00
IMAGE_EDIT_ENGINE: str
IMAGE_EDIT_MODEL: str
IMAGE_EDIT_SIZE: Optional[str]
IMAGES_EDIT_OPENAI_API_BASE_URL: str
IMAGES_EDIT_OPENAI_API_KEY: str
IMAGES_EDIT_OPENAI_API_VERSION: str
IMAGES_EDIT_GEMINI_API_BASE_URL: str
IMAGES_EDIT_GEMINI_API_KEY: str
2025-11-06 08:43:59 +00:00
IMAGES_EDIT_COMFYUI_BASE_URL: str
IMAGES_EDIT_COMFYUI_API_KEY: str
IMAGES_EDIT_COMFYUI_WORKFLOW: str
IMAGES_EDIT_COMFYUI_WORKFLOW_NODES: list[dict]
2025-11-05 08:31:37 +00:00
2024-04-23 11:14:31 +00:00
2025-11-04 18:30:59 +00:00
@router.get("/config", response_model=ImagesConfig)
async def get_config(request: Request, user=Depends(get_admin_user)):
2024-03-09 01:38:10 +00:00
return {
2025-11-04 18:30:59 +00:00
"ENABLE_IMAGE_GENERATION": request.app.state.config.ENABLE_IMAGE_GENERATION,
"ENABLE_IMAGE_PROMPT_GENERATION": request.app.state.config.ENABLE_IMAGE_PROMPT_GENERATION,
"IMAGE_GENERATION_ENGINE": request.app.state.config.IMAGE_GENERATION_ENGINE,
2025-11-04 18:30:59 +00:00
"IMAGE_GENERATION_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,
2025-11-04 18:30:59 +00:00
"IMAGES_OPENAI_API_BASE_URL": request.app.state.config.IMAGES_OPENAI_API_BASE_URL,
"IMAGES_OPENAI_API_KEY": request.app.state.config.IMAGES_OPENAI_API_KEY,
"IMAGES_OPENAI_API_VERSION": request.app.state.config.IMAGES_OPENAI_API_VERSION,
"AUTOMATIC1111_BASE_URL": request.app.state.config.AUTOMATIC1111_BASE_URL,
"AUTOMATIC1111_API_AUTH": request.app.state.config.AUTOMATIC1111_API_AUTH,
"AUTOMATIC1111_PARAMS": request.app.state.config.AUTOMATIC1111_PARAMS,
"COMFYUI_BASE_URL": request.app.state.config.COMFYUI_BASE_URL,
"COMFYUI_API_KEY": request.app.state.config.COMFYUI_API_KEY,
"COMFYUI_WORKFLOW": request.app.state.config.COMFYUI_WORKFLOW,
"COMFYUI_WORKFLOW_NODES": request.app.state.config.COMFYUI_WORKFLOW_NODES,
"IMAGES_GEMINI_API_BASE_URL": request.app.state.config.IMAGES_GEMINI_API_BASE_URL,
"IMAGES_GEMINI_API_KEY": request.app.state.config.IMAGES_GEMINI_API_KEY,
"IMAGES_GEMINI_ENDPOINT_METHOD": request.app.state.config.IMAGES_GEMINI_ENDPOINT_METHOD,
2025-11-05 08:31:37 +00:00
"IMAGE_EDIT_ENGINE": request.app.state.config.IMAGE_EDIT_ENGINE,
"IMAGE_EDIT_MODEL": request.app.state.config.IMAGE_EDIT_MODEL,
"IMAGE_EDIT_SIZE": request.app.state.config.IMAGE_EDIT_SIZE,
"IMAGES_EDIT_OPENAI_API_BASE_URL": request.app.state.config.IMAGES_EDIT_OPENAI_API_BASE_URL,
"IMAGES_EDIT_OPENAI_API_KEY": request.app.state.config.IMAGES_EDIT_OPENAI_API_KEY,
"IMAGES_EDIT_OPENAI_API_VERSION": request.app.state.config.IMAGES_EDIT_OPENAI_API_VERSION,
"IMAGES_EDIT_GEMINI_API_BASE_URL": request.app.state.config.IMAGES_EDIT_GEMINI_API_BASE_URL,
"IMAGES_EDIT_GEMINI_API_KEY": request.app.state.config.IMAGES_EDIT_GEMINI_API_KEY,
2025-11-06 08:43:59 +00:00
"IMAGES_EDIT_COMFYUI_BASE_URL": request.app.state.config.IMAGES_EDIT_COMFYUI_BASE_URL,
"IMAGES_EDIT_COMFYUI_API_KEY": request.app.state.config.IMAGES_EDIT_COMFYUI_API_KEY,
"IMAGES_EDIT_COMFYUI_WORKFLOW": request.app.state.config.IMAGES_EDIT_COMFYUI_WORKFLOW,
"IMAGES_EDIT_COMFYUI_WORKFLOW_NODES": request.app.state.config.IMAGES_EDIT_COMFYUI_WORKFLOW_NODES,
2024-03-09 01:38:10 +00:00
}
2025-11-04 18:30:59 +00:00
@router.post("/config/update")
async def update_config(
request: Request, form_data: ImagesConfig, user=Depends(get_admin_user)
2024-12-12 02:53:38 +00:00
):
2025-11-04 18:30:59 +00:00
request.app.state.config.ENABLE_IMAGE_GENERATION = form_data.ENABLE_IMAGE_GENERATION
2025-11-05 08:31:37 +00:00
# Create Image
2025-11-04 18:30:59 +00:00
request.app.state.config.ENABLE_IMAGE_PROMPT_GENERATION = (
form_data.ENABLE_IMAGE_PROMPT_GENERATION
)
2024-02-23 03:32:36 +00:00
request.app.state.config.IMAGE_GENERATION_ENGINE = form_data.IMAGE_GENERATION_ENGINE
2025-11-04 18:30:59 +00:00
set_image_model(request, form_data.IMAGE_GENERATION_MODEL)
if (
form_data.IMAGE_SIZE == "auto"
and form_data.IMAGE_GENERATION_MODEL != "gpt-image-1"
):
2025-06-22 20:55:29 +00:00
raise HTTPException(
status_code=400,
2025-06-28 11:21:20 +00:00
detail=ERROR_MESSAGES.INCORRECT_FORMAT(
" (auto is only allowed with gpt-image-1)."
),
2025-06-22 20:55:29 +00:00
)
2024-08-20 22:35:42 +00:00
pattern = r"^\d+x\d+$"
if (
form_data.IMAGE_SIZE == "auto"
or form_data.IMAGE_SIZE == ""
or 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
else:
raise HTTPException(
status_code=400,
detail=ERROR_MESSAGES.INCORRECT_FORMAT(" (e.g., 50)."),
)
2024-02-23 03:32:36 +00:00
2025-11-04 18:30:59 +00:00
request.app.state.config.IMAGES_OPENAI_API_BASE_URL = (
form_data.IMAGES_OPENAI_API_BASE_URL
)
request.app.state.config.IMAGES_OPENAI_API_KEY = form_data.IMAGES_OPENAI_API_KEY
request.app.state.config.IMAGES_OPENAI_API_VERSION = (
form_data.IMAGES_OPENAI_API_VERSION
)
request.app.state.config.AUTOMATIC1111_BASE_URL = form_data.AUTOMATIC1111_BASE_URL
request.app.state.config.AUTOMATIC1111_API_AUTH = form_data.AUTOMATIC1111_API_AUTH
request.app.state.config.AUTOMATIC1111_PARAMS = form_data.AUTOMATIC1111_PARAMS
request.app.state.config.COMFYUI_BASE_URL = form_data.COMFYUI_BASE_URL.strip("/")
request.app.state.config.COMFYUI_API_KEY = form_data.COMFYUI_API_KEY
request.app.state.config.COMFYUI_WORKFLOW = form_data.COMFYUI_WORKFLOW
request.app.state.config.COMFYUI_WORKFLOW_NODES = form_data.COMFYUI_WORKFLOW_NODES
request.app.state.config.IMAGES_GEMINI_API_BASE_URL = (
form_data.IMAGES_GEMINI_API_BASE_URL
)
request.app.state.config.IMAGES_GEMINI_API_KEY = form_data.IMAGES_GEMINI_API_KEY
request.app.state.config.IMAGES_GEMINI_ENDPOINT_METHOD = (
form_data.IMAGES_GEMINI_ENDPOINT_METHOD
)
2025-11-04 18:30:59 +00:00
2025-11-05 08:31:37 +00:00
# Edit Image
request.app.state.config.IMAGE_EDIT_ENGINE = form_data.IMAGE_EDIT_ENGINE
request.app.state.config.IMAGE_EDIT_MODEL = form_data.IMAGE_EDIT_MODEL
request.app.state.config.IMAGE_EDIT_SIZE = form_data.IMAGE_EDIT_SIZE
request.app.state.config.IMAGES_EDIT_OPENAI_API_BASE_URL = (
form_data.IMAGES_EDIT_OPENAI_API_BASE_URL
2025-11-05 08:31:37 +00:00
)
request.app.state.config.IMAGES_EDIT_OPENAI_API_KEY = (
form_data.IMAGES_EDIT_OPENAI_API_KEY
2025-11-05 08:31:37 +00:00
)
request.app.state.config.IMAGES_EDIT_OPENAI_API_VERSION = (
form_data.IMAGES_EDIT_OPENAI_API_VERSION
)
request.app.state.config.IMAGES_EDIT_GEMINI_API_BASE_URL = (
form_data.IMAGES_EDIT_GEMINI_API_BASE_URL
)
request.app.state.config.IMAGES_EDIT_GEMINI_API_KEY = (
form_data.IMAGES_EDIT_GEMINI_API_KEY
)
2025-11-06 08:43:59 +00:00
request.app.state.config.IMAGES_EDIT_COMFYUI_BASE_URL = (
form_data.IMAGES_EDIT_COMFYUI_BASE_URL.strip("/")
)
request.app.state.config.IMAGES_EDIT_COMFYUI_API_KEY = (
form_data.IMAGES_EDIT_COMFYUI_API_KEY
)
request.app.state.config.IMAGES_EDIT_COMFYUI_WORKFLOW = (
form_data.IMAGES_EDIT_COMFYUI_WORKFLOW
)
request.app.state.config.IMAGES_EDIT_COMFYUI_WORKFLOW_NODES = (
form_data.IMAGES_EDIT_COMFYUI_WORKFLOW_NODES
)
2024-08-20 22:35:42 +00:00
return {
2025-11-04 18:30:59 +00:00
"ENABLE_IMAGE_GENERATION": request.app.state.config.ENABLE_IMAGE_GENERATION,
"ENABLE_IMAGE_PROMPT_GENERATION": request.app.state.config.ENABLE_IMAGE_PROMPT_GENERATION,
"IMAGE_GENERATION_ENGINE": request.app.state.config.IMAGE_GENERATION_ENGINE,
2025-11-04 18:30:59 +00:00
"IMAGE_GENERATION_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,
2025-11-04 18:30:59 +00:00
"IMAGES_OPENAI_API_BASE_URL": request.app.state.config.IMAGES_OPENAI_API_BASE_URL,
"IMAGES_OPENAI_API_KEY": request.app.state.config.IMAGES_OPENAI_API_KEY,
"IMAGES_OPENAI_API_VERSION": request.app.state.config.IMAGES_OPENAI_API_VERSION,
"AUTOMATIC1111_BASE_URL": request.app.state.config.AUTOMATIC1111_BASE_URL,
"AUTOMATIC1111_API_AUTH": request.app.state.config.AUTOMATIC1111_API_AUTH,
"AUTOMATIC1111_PARAMS": request.app.state.config.AUTOMATIC1111_PARAMS,
"COMFYUI_BASE_URL": request.app.state.config.COMFYUI_BASE_URL,
"COMFYUI_API_KEY": request.app.state.config.COMFYUI_API_KEY,
"COMFYUI_WORKFLOW": request.app.state.config.COMFYUI_WORKFLOW,
"COMFYUI_WORKFLOW_NODES": request.app.state.config.COMFYUI_WORKFLOW_NODES,
"IMAGES_GEMINI_API_BASE_URL": request.app.state.config.IMAGES_GEMINI_API_BASE_URL,
"IMAGES_GEMINI_API_KEY": request.app.state.config.IMAGES_GEMINI_API_KEY,
"IMAGES_GEMINI_ENDPOINT_METHOD": request.app.state.config.IMAGES_GEMINI_ENDPOINT_METHOD,
2025-11-05 08:31:37 +00:00
"IMAGE_EDIT_ENGINE": request.app.state.config.IMAGE_EDIT_ENGINE,
"IMAGE_EDIT_MODEL": request.app.state.config.IMAGE_EDIT_MODEL,
"IMAGE_EDIT_SIZE": request.app.state.config.IMAGE_EDIT_SIZE,
"IMAGES_EDIT_OPENAI_API_BASE_URL": request.app.state.config.IMAGES_EDIT_OPENAI_API_BASE_URL,
"IMAGES_EDIT_OPENAI_API_KEY": request.app.state.config.IMAGES_EDIT_OPENAI_API_KEY,
"IMAGES_EDIT_OPENAI_API_VERSION": request.app.state.config.IMAGES_EDIT_OPENAI_API_VERSION,
"IMAGES_EDIT_GEMINI_API_BASE_URL": request.app.state.config.IMAGES_EDIT_GEMINI_API_BASE_URL,
"IMAGES_EDIT_GEMINI_API_KEY": request.app.state.config.IMAGES_EDIT_GEMINI_API_KEY,
2025-11-06 08:43:59 +00:00
"IMAGES_EDIT_COMFYUI_BASE_URL": request.app.state.config.IMAGES_EDIT_COMFYUI_BASE_URL,
"IMAGES_EDIT_COMFYUI_API_KEY": request.app.state.config.IMAGES_EDIT_COMFYUI_API_KEY,
"IMAGES_EDIT_COMFYUI_WORKFLOW": request.app.state.config.IMAGES_EDIT_COMFYUI_WORKFLOW,
"IMAGES_EDIT_COMFYUI_WORKFLOW_NODES": request.app.state.config.IMAGES_EDIT_COMFYUI_WORKFLOW_NODES,
2024-08-20 22:35:42 +00:00
}
2024-02-23 03:32:36 +00:00
2025-11-04 18:30:59 +00:00
def get_automatic1111_api_auth(request: Request):
if request.app.state.config.AUTOMATIC1111_API_AUTH is None:
return ""
else:
auth1111_byte_string = request.app.state.config.AUTOMATIC1111_API_AUTH.encode(
"utf-8"
)
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}"
@router.get("/config/url/verify")
async def verify_url(request: Request, user=Depends(get_admin_user)):
if request.app.state.config.IMAGE_GENERATION_ENGINE == "automatic1111":
try:
r = requests.get(
url=f"{request.app.state.config.AUTOMATIC1111_BASE_URL}/sdapi/v1/options",
headers={"authorization": get_automatic1111_api_auth(request)},
)
r.raise_for_status()
return True
except Exception:
request.app.state.config.ENABLE_IMAGE_GENERATION = False
raise HTTPException(status_code=400, detail=ERROR_MESSAGES.INVALID_URL)
elif request.app.state.config.IMAGE_GENERATION_ENGINE == "comfyui":
headers = None
if request.app.state.config.COMFYUI_API_KEY:
headers = {
"Authorization": f"Bearer {request.app.state.config.COMFYUI_API_KEY}"
}
try:
r = requests.get(
url=f"{request.app.state.config.COMFYUI_BASE_URL}/object_info",
headers=headers,
)
r.raise_for_status()
return True
except Exception:
request.app.state.config.ENABLE_IMAGE_GENERATION = False
raise HTTPException(status_code=400, detail=ERROR_MESSAGES.INVALID_URL)
else:
return True
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"},
{"id": "gpt-image-1", "name": "GPT-IMAGE 1"},
2024-03-09 01:38:10 +00:00
]
elif request.app.state.config.IMAGE_GENERATION_ENGINE == "gemini":
return [
{"id": "imagen-3.0-generate-002", "name": "imagen-3.0 generate-002"},
]
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 21:51:29 +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)
model_node_id = None
2024-12-12 02:53:38 +00:00
for node in request.app.state.config.COMFYUI_WORKFLOW_NODES:
if node["type"] == "model":
2024-08-21 22:25:43 +00:00
if node["node_ids"]:
model_node_id = node["node_ids"][0]
break
if model_node_id:
model_list_key = None
log.info(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-12-26 07:34:54 +00:00
headers={"authorization": get_automatic1111_api_auth(request)},
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
2025-11-04 18:30:59 +00:00
class CreateImageForm(BaseModel):
2024-02-22 02:12:01 +00:00
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
2025-11-04 22:58:51 +00:00
GenerateImageForm = CreateImageForm # Alias for backward compatibility
2025-11-04 18:30:59 +00:00
def get_image_data(data: str, headers=None):
2024-03-09 01:38:10 +00:00
try:
if data.startswith("http://") or data.startswith("https://"):
2025-11-04 18:30:59 +00:00
if headers:
r = requests.get(data, headers=headers)
else:
r = requests.get(data)
2025-01-16 19:17:37 +00:00
2025-11-04 18:30:59 +00:00
r.raise_for_status()
if r.headers["content-type"].split("/")[0] == "image":
mime_type = r.headers["content-type"]
return r.content, mime_type
else:
log.error("Url does not point to an image.")
return None
else:
2025-11-04 18:30:59 +00:00
if "," in data:
header, encoded = data.split(",", 1)
mime_type = header.split(";")[0].lstrip("data:")
img_data = base64.b64decode(encoded)
else:
mime_type = "image/png"
img_data = base64.b64decode(data)
return img_data, mime_type
2024-03-24 00:01:13 +00:00
except Exception as e:
2025-11-04 18:30:59 +00:00
log.exception(f"Error loading image data: {e}")
return None, None
2024-03-24 00:01:13 +00:00
2025-08-22 13:19:57 +00:00
def upload_image(request, image_data, content_type, metadata, user):
2025-02-06 23:22:20 +00:00
image_format = mimetypes.guess_extension(content_type)
2025-02-06 23:12:04 +00:00
file = UploadFile(
file=io.BytesIO(image_data),
2025-02-12 05:13:42 +00:00
filename=f"generated-image{image_format}", # will be converted to a unique ID on upload_file
2025-02-06 23:12:04 +00:00
headers={
"content-type": content_type,
},
)
2025-08-22 13:19:57 +00:00
file_item = upload_file_handler(
2025-08-22 12:58:25 +00:00
request,
file=file,
metadata=metadata,
process=False,
user=user,
2025-08-19 20:36:13 +00:00
)
2025-02-06 23:12:04 +00:00
url = request.app.url_path_for("get_file_content_by_id", id=file_item.id)
return url
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,
2025-11-04 18:30:59 +00:00
form_data: CreateImageForm,
2024-06-27 18:29:59 +00:00
user=Depends(get_verified_user),
2024-02-22 02:12:01 +00:00
):
2025-06-22 20:55:29 +00:00
# if IMAGE_SIZE = 'auto', default WidthxHeight to the 512x512 default
# This is only relevant when the user has set IMAGE_SIZE to 'auto' with an
# image model other than gpt-image-1, which is warned about on settings save
size = "512x512"
2025-08-22 09:25:23 +00:00
if (
request.app.state.config.IMAGE_SIZE
and "x" in request.app.state.config.IMAGE_SIZE
):
size = request.app.state.config.IMAGE_SIZE
2025-08-22 09:25:23 +00:00
if form_data.size and "x" in form_data.size:
size = form_data.size
width, height = tuple(map(int, size.split("x")))
2025-09-24 12:17:41 +00:00
model = get_image_model(request)
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":
2025-11-04 18:30:59 +00:00
headers = {
"Authorization": f"Bearer {request.app.state.config.IMAGES_OPENAI_API_KEY}",
"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:
2025-11-04 18:30:59 +00:00
headers = include_user_info_headers(headers, user)
2024-03-09 01:38:10 +00:00
data = {
2025-09-24 12:17:41 +00:00
"model": model,
2024-03-09 01:38:10 +00:00
"prompt": form_data.prompt,
"n": form_data.n,
"size": (
2024-12-12 02:53:38 +00:00
form_data.size
if form_data.size
else request.app.state.config.IMAGE_SIZE
),
2025-04-29 15:34:00 +00:00
**(
{}
2025-04-29 15:34:00 +00:00
if "gpt-image-1" in request.app.state.config.IMAGE_GENERATION_MODEL
else {"response_format": "b64_json"}
2025-04-29 15:34:00 +00:00
),
2024-03-09 01:38:10 +00:00
}
2025-09-02 17:15:32 +00:00
api_version_query_param = ""
2025-09-02 17:15:32 +00:00
if request.app.state.config.IMAGES_OPENAI_API_VERSION:
api_version_query_param = (
f"?api-version={request.app.state.config.IMAGES_OPENAI_API_VERSION}"
)
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,
url=f"{request.app.state.config.IMAGES_OPENAI_API_BASE_URL}/images/generations{api_version_query_param}",
2024-03-09 01:38:10 +00:00
json=data,
2025-02-06 22:37:18 +00:00
headers=headers,
2024-03-09 01:38:10 +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"]:
2025-03-11 15:40:31 +00:00
if image_url := image.get("url", None):
2025-11-04 18:30:59 +00:00
image_data, content_type = get_image_data(image_url, headers)
2025-03-04 04:07:59 +00:00
else:
2025-11-04 18:30:59 +00:00
image_data, content_type = get_image_data(image["b64_json"])
2025-03-04 04:07:59 +00:00
2025-08-22 13:19:57 +00:00
url = upload_image(request, image_data, content_type, data, user)
images.append({"url": url})
2024-03-09 01:38:10 +00:00
return images
elif request.app.state.config.IMAGE_GENERATION_ENGINE == "gemini":
headers = {
"Content-Type": "application/json",
"x-goog-api-key": request.app.state.config.IMAGES_GEMINI_API_KEY,
}
data = {}
if (
request.app.state.config.IMAGES_GEMINI_ENDPOINT_METHOD == ""
or request.app.state.config.IMAGES_GEMINI_ENDPOINT_METHOD == "predict"
):
model = f"{model}:predict"
data = {
"instances": {"prompt": form_data.prompt},
"parameters": {
"sampleCount": form_data.n,
"outputOptions": {"mimeType": "image/png"},
},
}
elif (
request.app.state.config.IMAGES_GEMINI_ENDPOINT_METHOD
== "generateContent"
):
model = f"{model}:generateContent"
data = {"contents": [{"parts": [{"text": form_data.prompt}]}]}
# Use asyncio.to_thread for the requests.post call
r = await asyncio.to_thread(
requests.post,
url=f"{request.app.state.config.IMAGES_GEMINI_API_BASE_URL}/models/{model}",
json=data,
headers=headers,
)
r.raise_for_status()
res = r.json()
images = []
if model.endswith(":predict"):
for image in res["predictions"]:
image_data, content_type = get_image_data(
image["bytesBase64Encoded"]
)
url = upload_image(request, image_data, content_type, data, user)
images.append({"url": url})
elif model.endswith(":generateContent"):
for image in res["candidates"]:
for part in image["content"]["parts"]:
if part.get("inlineData", {}).get("data"):
image_data, content_type = get_image_data(
part["inlineData"]["data"]
)
url = upload_image(
request, image_data, content_type, data, user
)
images.append({"url": url})
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
if form_data.negative_prompt is not None:
2024-03-24 00:01:13 +00:00
data["negative_prompt"] = form_data.negative_prompt
2025-11-04 18:30:59 +00:00
form_data = ComfyUICreateImageForm(
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,
}
)
2025-11-04 18:30:59 +00:00
res = await comfyui_create_image(
2025-09-24 12:17:41 +00:00
model,
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
)
log.debug(f"res: {res}")
2024-03-24 00:01:13 +00:00
images = []
for image in res["data"]:
2025-01-16 19:17:37 +00:00
headers = None
if request.app.state.config.COMFYUI_API_KEY:
headers = {
"Authorization": f"Bearer {request.app.state.config.COMFYUI_API_KEY}"
}
2025-11-04 18:30:59 +00:00
image_data, content_type = get_image_data(image["url"], headers)
2025-02-06 23:12:04 +00:00
url = upload_image(
request,
image_data,
content_type,
2025-05-26 22:48:22 +00:00
form_data.model_dump(exclude_none=True),
2025-02-06 23:12:04 +00:00
user,
)
images.append({"url": url})
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:
set_image_model(request, 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
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
2025-11-04 18:30:59 +00:00
if request.app.state.config.AUTOMATIC1111_PARAMS:
data = {**data, **request.app.state.config.AUTOMATIC1111_PARAMS}
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-12-26 07:34:54 +00:00
headers={"authorization": get_automatic1111_api_auth(request)},
2024-03-09 01:38:10 +00:00
)
res = r.json()
log.debug(f"res: {res}")
2024-03-09 01:38:10 +00:00
images = []
for image in res["images"]:
2025-11-04 18:30:59 +00:00
image_data, content_type = get_image_data(image)
2025-02-06 23:12:04 +00:00
url = upload_image(
request,
image_data,
content_type,
2025-05-26 22:48:22 +00:00
{**data, "info": res["info"]},
2025-02-06 23:12:04 +00:00
user,
)
images.append({"url": url})
2024-03-09 01:38:10 +00:00
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))
2025-11-05 08:31:37 +00:00
class EditImageForm(BaseModel):
image: str | list[str] # base64-encoded image(s) or URL(s)
prompt: str
model: Optional[str] = None
size: Optional[str] = None
n: Optional[int] = None
negative_prompt: Optional[str] = None
@router.post("/edit")
async def image_edits(
request: Request,
form_data: EditImageForm,
user=Depends(get_verified_user),
):
size = None
width, height = None, None
if (
request.app.state.config.IMAGE_EDIT_SIZE
and "x" in request.app.state.config.IMAGE_EDIT_SIZE
) or (form_data.size and "x" in form_data.size):
size = (
form_data.size
if form_data.size
else request.app.state.config.IMAGE_EDIT_SIZE
)
width, height = tuple(map(int, size.split("x")))
model = (
request.app.state.config.IMAGE_EDIT_MODEL
if form_data.model is None
else form_data.model
)
2025-11-05 08:42:34 +00:00
try:
async def load_url_image(data):
if data.startswith("http://") or data.startswith("https://"):
r = await asyncio.to_thread(requests.get, data)
r.raise_for_status()
image_data = base64.b64encode(r.content).decode("utf-8")
return f"data:{r.headers['content-type']};base64,{image_data}"
elif data.startswith("/api/v1/files"):
file_id = data.split("/api/v1/files/")[1].split("/content")[0]
file_response = await get_file_content_by_id(file_id, user)
if isinstance(file_response, FileResponse):
file_path = file_response.path
with open(file_path, "rb") as f:
file_bytes = f.read()
image_data = base64.b64encode(file_bytes).decode("utf-8")
mime_type, _ = mimetypes.guess_type(file_path)
return f"data:{mime_type};base64,{image_data}"
return data
# Load image(s) from URL(s) if necessary
if isinstance(form_data.image, str):
form_data.image = await load_url_image(form_data.image)
elif isinstance(form_data.image, list):
form_data.image = [await load_url_image(img) for img in form_data.image]
except Exception as e:
raise HTTPException(status_code=400, detail=ERROR_MESSAGES.DEFAULT(e))
2025-11-05 08:31:37 +00:00
2025-11-06 08:43:59 +00:00
def get_image_file_item(base64_string):
data = base64_string
header, encoded = data.split(",", 1)
mime_type = header.split(";")[0].lstrip("data:")
image_data = base64.b64decode(encoded)
return (
"image",
(
f"{uuid.uuid4()}.png",
io.BytesIO(image_data),
mime_type if mime_type else "image/png",
),
)
2025-11-05 08:31:37 +00:00
r = None
try:
if request.app.state.config.IMAGE_EDIT_ENGINE == "openai":
headers = {
"Authorization": f"Bearer {request.app.state.config.IMAGES_EDIT_OPENAI_API_KEY}",
}
if ENABLE_FORWARD_USER_INFO_HEADERS:
headers = include_user_info_headers(headers, user)
data = {
"model": model,
"prompt": form_data.prompt,
**({"n": form_data.n} if form_data.n else {}),
**({"size": size} if size else {}),
**(
{}
2025-11-06 08:43:59 +00:00
if "gpt-image-1" in request.app.state.config.IMAGE_EDIT_MODEL
2025-11-05 08:31:37 +00:00
else {"response_format": "b64_json"}
),
}
files = []
if isinstance(form_data.image, str):
files = [get_image_file_item(form_data.image)]
elif isinstance(form_data.image, list):
for img in form_data.image:
files.append(get_image_file_item(img))
url_search_params = ""
if request.app.state.config.IMAGES_EDIT_OPENAI_API_VERSION:
url_search_params += f"?api-version={request.app.state.config.IMAGES_EDIT_OPENAI_API_VERSION}"
# Use asyncio.to_thread for the requests.post call
r = await asyncio.to_thread(
requests.post,
2025-11-06 08:43:59 +00:00
url=f"{request.app.state.config.IMAGES_EDIT_OPENAI_API_BASE_URL}/images/edits{url_search_params}",
2025-11-05 08:31:37 +00:00
headers=headers,
files=files,
data=data,
)
r.raise_for_status()
res = r.json()
images = []
for image in res["data"]:
if image_url := image.get("url", None):
image_data, content_type = get_image_data(image_url, headers)
else:
image_data, content_type = get_image_data(image["b64_json"])
url = upload_image(request, image_data, content_type, data, user)
images.append({"url": url})
return images
2025-11-06 08:43:59 +00:00
elif request.app.state.config.IMAGE_EDIT_ENGINE == "gemini":
2025-11-05 08:31:37 +00:00
headers = {
"Content-Type": "application/json",
2025-11-06 08:43:59 +00:00
"x-goog-api-key": request.app.state.config.IMAGES_EDIT_GEMINI_API_KEY,
2025-11-05 08:31:37 +00:00
}
model = f"{model}:generateContent"
data = {"contents": [{"parts": [{"text": form_data.prompt}]}]}
if isinstance(form_data.image, str):
data["contents"][0]["parts"].append(
{
"inline_data": {
"mime_type": "image/png",
"data": form_data.image.split(",", 1)[1],
}
}
)
elif isinstance(form_data.image, list):
data["contents"][0]["parts"].extend(
[
{
"inline_data": {
"mime_type": "image/png",
"data": image.split(",", 1)[1],
}
}
for image in form_data.image
]
)
# Use asyncio.to_thread for the requests.post call
r = await asyncio.to_thread(
requests.post,
2025-11-06 08:43:59 +00:00
url=f"{request.app.state.config.IMAGES_EDIT_GEMINI_API_BASE_URL}/models/{model}",
2025-11-05 08:31:37 +00:00
json=data,
headers=headers,
)
r.raise_for_status()
res = r.json()
images = []
for image in res["candidates"]:
for part in image["content"]["parts"]:
if part.get("inlineData", {}).get("data"):
image_data, content_type = get_image_data(
part["inlineData"]["data"]
)
url = upload_image(
request, image_data, content_type, data, user
)
images.append({"url": url})
return images
2025-11-06 08:43:59 +00:00
elif request.app.state.config.IMAGE_EDIT_ENGINE == "comfyui":
try:
files = []
if isinstance(form_data.image, str):
files = [get_image_file_item(form_data.image)]
elif isinstance(form_data.image, list):
for img in form_data.image:
files.append(get_image_file_item(img))
# Upload images to ComfyUI and get their names
comfyui_images = []
for file_item in files:
res = await comfyui_upload_image(
file_item,
request.app.state.config.IMAGES_EDIT_COMFYUI_BASE_URL,
request.app.state.config.IMAGES_EDIT_COMFYUI_API_KEY,
)
comfyui_images.append(res.get("name", file_item[1][0]))
except Exception as e:
log.debug(f"Error uploading images to ComfyUI: {e}")
raise Exception("Failed to upload images to ComfyUI.")
2025-11-05 08:31:37 +00:00
data = {
2025-11-06 08:43:59 +00:00
"image": comfyui_images,
2025-11-05 08:31:37 +00:00
"prompt": form_data.prompt,
2025-11-06 08:43:59 +00:00
**({"width": width} if width is not None else {}),
**({"height": height} if height is not None else {}),
**({"n": form_data.n} if form_data.n else {}),
2025-11-05 08:31:37 +00:00
}
2025-11-06 08:43:59 +00:00
form_data = ComfyUIEditImageForm(
2025-11-05 08:31:37 +00:00
**{
"workflow": ComfyUIWorkflow(
**{
2025-11-06 08:43:59 +00:00
"workflow": request.app.state.config.IMAGES_EDIT_COMFYUI_WORKFLOW,
"nodes": request.app.state.config.IMAGES_EDIT_COMFYUI_WORKFLOW_NODES,
2025-11-05 08:31:37 +00:00
}
),
**data,
}
)
2025-11-06 08:43:59 +00:00
res = await comfyui_edit_image(
2025-11-05 08:31:37 +00:00
model,
form_data,
user.id,
2025-11-06 08:43:59 +00:00
request.app.state.config.IMAGES_EDIT_COMFYUI_BASE_URL,
request.app.state.config.IMAGES_EDIT_COMFYUI_API_KEY,
2025-11-05 08:31:37 +00:00
)
log.debug(f"res: {res}")
2025-11-06 08:43:59 +00:00
image_urls = set()
for image in res["data"]:
image_urls.add(image["url"])
image_urls = list(image_urls)
# Prioritize output type URLs if available
output_type_urls = [url for url in image_urls if "type=output" in url]
if output_type_urls:
image_urls = output_type_urls
log.debug(f"Image URLs: {image_urls}")
2025-11-05 08:31:37 +00:00
images = []
2025-11-06 08:43:59 +00:00
for image_url in image_urls:
2025-11-05 08:31:37 +00:00
headers = None
2025-11-06 08:43:59 +00:00
if request.app.state.config.IMAGES_EDIT_COMFYUI_API_KEY:
2025-11-05 08:31:37 +00:00
headers = {
2025-11-06 08:43:59 +00:00
"Authorization": f"Bearer {request.app.state.config.IMAGES_EDIT_COMFYUI_API_KEY}"
2025-11-05 08:31:37 +00:00
}
2025-11-06 08:43:59 +00:00
image_data, content_type = get_image_data(image_url, headers)
2025-11-05 08:31:37 +00:00
url = upload_image(
request,
image_data,
content_type,
form_data.model_dump(exclude_none=True),
user,
)
images.append({"url": url})
2025-11-06 08:43:59 +00:00
2025-11-05 08:31:37 +00:00
return images
except Exception as e:
error = e
if r != None:
data = r.text
try:
data = json.loads(data)
if "error" in data:
error = data["error"]["message"]
except Exception:
error = data
raise HTTPException(status_code=400, detail=ERROR_MESSAGES.DEFAULT(error))