|
@@ -18,6 +18,8 @@ from utils.utils import (
|
|
|
get_current_user,
|
|
|
get_admin_user,
|
|
|
)
|
|
|
+
|
|
|
+from apps.images.utils.comfyui import ImageGenerationPayload, comfyui_generate_image
|
|
|
from utils.misc import calculate_sha256
|
|
|
from typing import Optional
|
|
|
from pydantic import BaseModel
|
|
@@ -26,7 +28,7 @@ import uuid
|
|
|
import base64
|
|
|
import json
|
|
|
|
|
|
-from config import CACHE_DIR, AUTOMATIC1111_BASE_URL
|
|
|
+from config import CACHE_DIR, AUTOMATIC1111_BASE_URL, COMFYUI_BASE_URL
|
|
|
|
|
|
|
|
|
IMAGE_CACHE_DIR = Path(CACHE_DIR).joinpath("./image/generations/")
|
|
@@ -49,6 +51,8 @@ app.state.MODEL = ""
|
|
|
|
|
|
|
|
|
app.state.AUTOMATIC1111_BASE_URL = AUTOMATIC1111_BASE_URL
|
|
|
+app.state.COMFYUI_BASE_URL = COMFYUI_BASE_URL
|
|
|
+
|
|
|
|
|
|
app.state.IMAGE_SIZE = "512x512"
|
|
|
app.state.IMAGE_STEPS = 50
|
|
@@ -71,32 +75,48 @@ async def update_config(form_data: ConfigUpdateForm, user=Depends(get_admin_user
|
|
|
return {"engine": app.state.ENGINE, "enabled": app.state.ENABLED}
|
|
|
|
|
|
|
|
|
-class UrlUpdateForm(BaseModel):
|
|
|
- url: str
|
|
|
+class EngineUrlUpdateForm(BaseModel):
|
|
|
+ AUTOMATIC1111_BASE_URL: Optional[str] = None
|
|
|
+ COMFYUI_BASE_URL: Optional[str] = None
|
|
|
|
|
|
|
|
|
@app.get("/url")
|
|
|
-async def get_automatic1111_url(user=Depends(get_admin_user)):
|
|
|
- return {"AUTOMATIC1111_BASE_URL": app.state.AUTOMATIC1111_BASE_URL}
|
|
|
+async def get_engine_url(user=Depends(get_admin_user)):
|
|
|
+ return {
|
|
|
+ "AUTOMATIC1111_BASE_URL": app.state.AUTOMATIC1111_BASE_URL,
|
|
|
+ "COMFYUI_BASE_URL": app.state.COMFYUI_BASE_URL,
|
|
|
+ }
|
|
|
|
|
|
|
|
|
@app.post("/url/update")
|
|
|
-async def update_automatic1111_url(
|
|
|
- form_data: UrlUpdateForm, user=Depends(get_admin_user)
|
|
|
+async def update_engine_url(
|
|
|
+ form_data: EngineUrlUpdateForm, user=Depends(get_admin_user)
|
|
|
):
|
|
|
|
|
|
- if form_data.url == "":
|
|
|
+ if form_data.AUTOMATIC1111_BASE_URL == None:
|
|
|
app.state.AUTOMATIC1111_BASE_URL = AUTOMATIC1111_BASE_URL
|
|
|
else:
|
|
|
- url = form_data.url.strip("/")
|
|
|
+ url = form_data.AUTOMATIC1111_BASE_URL.strip("/")
|
|
|
try:
|
|
|
r = requests.head(url)
|
|
|
app.state.AUTOMATIC1111_BASE_URL = url
|
|
|
except Exception as e:
|
|
|
raise HTTPException(status_code=400, detail=ERROR_MESSAGES.DEFAULT(e))
|
|
|
|
|
|
+ if form_data.COMFYUI_BASE_URL == None:
|
|
|
+ app.state.COMFYUI_BASE_URL = COMFYUI_BASE_URL
|
|
|
+ else:
|
|
|
+ url = form_data.COMFYUI_BASE_URL.strip("/")
|
|
|
+
|
|
|
+ try:
|
|
|
+ r = requests.head(url)
|
|
|
+ app.state.COMFYUI_BASE_URL = url
|
|
|
+ except Exception as e:
|
|
|
+ raise HTTPException(status_code=400, detail=ERROR_MESSAGES.DEFAULT(e))
|
|
|
+
|
|
|
return {
|
|
|
"AUTOMATIC1111_BASE_URL": app.state.AUTOMATIC1111_BASE_URL,
|
|
|
+ "COMFYUI_BASE_URL": app.state.COMFYUI_BASE_URL,
|
|
|
"status": True,
|
|
|
}
|
|
|
|
|
@@ -186,6 +206,18 @@ def get_models(user=Depends(get_current_user)):
|
|
|
{"id": "dall-e-2", "name": "DALL·E 2"},
|
|
|
{"id": "dall-e-3", "name": "DALL·E 3"},
|
|
|
]
|
|
|
+ elif app.state.ENGINE == "comfyui":
|
|
|
+
|
|
|
+ r = requests.get(url=f"{app.state.COMFYUI_BASE_URL}/object_info")
|
|
|
+ info = r.json()
|
|
|
+
|
|
|
+ return list(
|
|
|
+ map(
|
|
|
+ lambda model: {"id": model, "name": model},
|
|
|
+ info["CheckpointLoaderSimple"]["input"]["required"]["ckpt_name"][0],
|
|
|
+ )
|
|
|
+ )
|
|
|
+
|
|
|
else:
|
|
|
r = requests.get(
|
|
|
url=f"{app.state.AUTOMATIC1111_BASE_URL}/sdapi/v1/sd-models"
|
|
@@ -207,6 +239,8 @@ async def get_default_model(user=Depends(get_admin_user)):
|
|
|
try:
|
|
|
if app.state.ENGINE == "openai":
|
|
|
return {"model": app.state.MODEL if app.state.MODEL else "dall-e-2"}
|
|
|
+ elif app.state.ENGINE == "comfyui":
|
|
|
+ return {"model": app.state.MODEL if app.state.MODEL else ""}
|
|
|
else:
|
|
|
r = requests.get(url=f"{app.state.AUTOMATIC1111_BASE_URL}/sdapi/v1/options")
|
|
|
options = r.json()
|
|
@@ -221,10 +255,12 @@ class UpdateModelForm(BaseModel):
|
|
|
|
|
|
|
|
|
def set_model_handler(model: str):
|
|
|
-
|
|
|
if app.state.ENGINE == "openai":
|
|
|
app.state.MODEL = model
|
|
|
return app.state.MODEL
|
|
|
+ if app.state.ENGINE == "comfyui":
|
|
|
+ app.state.MODEL = model
|
|
|
+ return app.state.MODEL
|
|
|
else:
|
|
|
r = requests.get(url=f"{app.state.AUTOMATIC1111_BASE_URL}/sdapi/v1/options")
|
|
|
options = r.json()
|
|
@@ -272,12 +308,31 @@ def save_b64_image(b64_str):
|
|
|
return None
|
|
|
|
|
|
|
|
|
+def save_url_image(url):
|
|
|
+ image_id = str(uuid.uuid4())
|
|
|
+ file_path = IMAGE_CACHE_DIR.joinpath(f"{image_id}.png")
|
|
|
+
|
|
|
+ try:
|
|
|
+ r = requests.get(url)
|
|
|
+ r.raise_for_status()
|
|
|
+
|
|
|
+ with open(file_path, "wb") as image_file:
|
|
|
+ image_file.write(r.content)
|
|
|
+
|
|
|
+ return image_id
|
|
|
+ except Exception as e:
|
|
|
+ print(f"Error saving image: {e}")
|
|
|
+ return None
|
|
|
+
|
|
|
+
|
|
|
@app.post("/generations")
|
|
|
def generate_image(
|
|
|
form_data: GenerateImageForm,
|
|
|
user=Depends(get_current_user),
|
|
|
):
|
|
|
|
|
|
+ width, height = tuple(map(int, app.state.IMAGE_SIZE.split("x")))
|
|
|
+
|
|
|
r = None
|
|
|
try:
|
|
|
if app.state.ENGINE == "openai":
|
|
@@ -315,12 +370,47 @@ def generate_image(
|
|
|
|
|
|
return images
|
|
|
|
|
|
+ elif app.state.ENGINE == "comfyui":
|
|
|
+
|
|
|
+ data = {
|
|
|
+ "prompt": form_data.prompt,
|
|
|
+ "width": width,
|
|
|
+ "height": height,
|
|
|
+ "n": form_data.n,
|
|
|
+ }
|
|
|
+
|
|
|
+ if app.state.IMAGE_STEPS != None:
|
|
|
+ data["steps"] = app.state.IMAGE_STEPS
|
|
|
+
|
|
|
+ if form_data.negative_prompt != None:
|
|
|
+ data["negative_prompt"] = form_data.negative_prompt
|
|
|
+
|
|
|
+ data = ImageGenerationPayload(**data)
|
|
|
+
|
|
|
+ res = comfyui_generate_image(
|
|
|
+ app.state.MODEL,
|
|
|
+ data,
|
|
|
+ user.id,
|
|
|
+ app.state.COMFYUI_BASE_URL,
|
|
|
+ )
|
|
|
+ print(res)
|
|
|
+
|
|
|
+ images = []
|
|
|
+
|
|
|
+ for image in res["data"]:
|
|
|
+ image_id = save_url_image(image["url"])
|
|
|
+ images.append({"url": f"/cache/image/generations/{image_id}.png"})
|
|
|
+ file_body_path = IMAGE_CACHE_DIR.joinpath(f"{image_id}.json")
|
|
|
+
|
|
|
+ with open(file_body_path, "w") as f:
|
|
|
+ json.dump(data.model_dump(exclude_none=True), f)
|
|
|
+
|
|
|
+ print(images)
|
|
|
+ return images
|
|
|
else:
|
|
|
if form_data.model:
|
|
|
set_model_handler(form_data.model)
|
|
|
|
|
|
- width, height = tuple(map(int, app.state.IMAGE_SIZE.split("x")))
|
|
|
-
|
|
|
data = {
|
|
|
"prompt": form_data.prompt,
|
|
|
"batch_size": form_data.n,
|