mirror of
https://github.com/karl0ss/ai_image_frame_server.git
synced 2025-10-24 04:04:02 +01:00
retry logic, create endpoint now non blocking
This commit is contained in:
parent
42c8a2b988
commit
d32e9038f8
@ -9,6 +9,7 @@ from flask import (
|
|||||||
)
|
)
|
||||||
import os
|
import os
|
||||||
import time
|
import time
|
||||||
|
import threading
|
||||||
from apscheduler.schedulers.background import BackgroundScheduler
|
from apscheduler.schedulers.background import BackgroundScheduler
|
||||||
from lib import create_image, load_config
|
from lib import create_image, load_config
|
||||||
|
|
||||||
@ -58,6 +59,7 @@ def images(filename: str) -> None:
|
|||||||
return send_from_directory(image_folder, filename)
|
return send_from_directory(image_folder, filename)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@app.route("/create", methods=["GET", "POST"])
|
@app.route("/create", methods=["GET", "POST"])
|
||||||
def create() -> str:
|
def create() -> str:
|
||||||
"""Handles image creation requests.
|
"""Handles image creation requests.
|
||||||
@ -67,10 +69,12 @@ def create() -> str:
|
|||||||
str: Redirect to the main page or a JSON response.
|
str: Redirect to the main page or a JSON response.
|
||||||
"""
|
"""
|
||||||
prompt = request.form.get("prompt") if request.method == "POST" else None
|
prompt = request.form.get("prompt") if request.method == "POST" else None
|
||||||
create_image(prompt)
|
|
||||||
if request.method == "POST":
|
def create_image_in_background():
|
||||||
return jsonify({"message": "Image created", "prompt": prompt}), 200
|
create_image(prompt)
|
||||||
return redirect(url_for("index"))
|
|
||||||
|
threading.Thread(target=create_image_in_background).start()
|
||||||
|
return jsonify({"message": "Image creation started", "prompt": prompt if prompt else "Prompt will be generated"}), 200
|
||||||
|
|
||||||
|
|
||||||
def scheduled_task() -> None:
|
def scheduled_task() -> None:
|
||||||
|
55
lib.py
55
lib.py
@ -7,6 +7,11 @@ import time
|
|||||||
import os
|
import os
|
||||||
import requests
|
import requests
|
||||||
from comfy_api_simplified import ComfyApiWrapper, ComfyWorkflowWrapper
|
from comfy_api_simplified import ComfyApiWrapper, ComfyWorkflowWrapper
|
||||||
|
from tenacity import retry, stop_after_attempt, wait_fixed, before_log, retry_if_exception_type
|
||||||
|
import nest_asyncio
|
||||||
|
nest_asyncio.apply()
|
||||||
|
|
||||||
|
logging.basicConfig(level=logging.INFO)
|
||||||
|
|
||||||
|
|
||||||
def get_available_models() -> list:
|
def get_available_models() -> list:
|
||||||
@ -59,23 +64,54 @@ def send_prompt_to_openwebui(prompt: str) -> str:
|
|||||||
api_base=user_config["openwebui"]["base_url"],
|
api_base=user_config["openwebui"]["base_url"],
|
||||||
model="openai/" + user_config["openwebui"]["model"],
|
model="openai/" + user_config["openwebui"]["model"],
|
||||||
messages=[
|
messages=[
|
||||||
|
{
|
||||||
|
"role": "system",
|
||||||
|
"content": (
|
||||||
|
"You are a prompt generator for Stable Diffusion. "
|
||||||
|
"Generate a detailed and imaginative prompt with a strong visual theme. "
|
||||||
|
"Focus on lighting, atmosphere, and artistic style. "
|
||||||
|
"Keep the prompt concise, no extra commentary or formatting."
|
||||||
|
),
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"role": "user",
|
"role": "user",
|
||||||
"content": prompt,
|
"content": prompt,
|
||||||
}
|
},
|
||||||
],
|
],
|
||||||
api_key=user_config["openwebui"]["api_key"],
|
api_key=user_config["openwebui"]["api_key"],
|
||||||
)
|
)
|
||||||
|
|
||||||
return response["choices"][0]["message"]["content"].strip('"')
|
prompt = response["choices"][0]["message"]["content"].strip('"')
|
||||||
|
# response = litellm.completion(
|
||||||
|
# api_base=user_config["openwebui"]["base_url"],
|
||||||
|
# model="openai/brxce/stable-diffusion-prompt-generator:latest",
|
||||||
|
# messages=[
|
||||||
|
# {
|
||||||
|
# "role": "user",
|
||||||
|
# "content": prompt,
|
||||||
|
# },
|
||||||
|
# ],
|
||||||
|
# api_key=user_config["openwebui"]["api_key"],
|
||||||
|
# )
|
||||||
|
# prompt = response["choices"][0]["message"]["content"].strip('"')
|
||||||
|
logging.debug(prompt)
|
||||||
|
return prompt
|
||||||
|
|
||||||
|
|
||||||
|
# Define the retry logic using Tenacity
|
||||||
|
@retry(
|
||||||
|
stop=stop_after_attempt(3),
|
||||||
|
wait=wait_fixed(5),
|
||||||
|
before=before_log(logging.getLogger(), logging.DEBUG),
|
||||||
|
retry=retry_if_exception_type(Exception)
|
||||||
|
)
|
||||||
def generate_image(file_name: str, comfy_prompt: str) -> None:
|
def generate_image(file_name: str, comfy_prompt: str) -> None:
|
||||||
"""Generates an image using the Comfy API."""
|
"""Generates an image using the Comfy API with retry logic."""
|
||||||
try:
|
try:
|
||||||
# Initialize ComfyUI API and workflow
|
# Initialize ComfyUI API and workflow
|
||||||
api = ComfyApiWrapper(user_config["comfyui"]["comfyui_url"])
|
api = ComfyApiWrapper(user_config["comfyui"]["comfyui_url"])
|
||||||
wf = ComfyWorkflowWrapper("./workflow_api.json")
|
wf = ComfyWorkflowWrapper("./workflow_api.json")
|
||||||
|
|
||||||
# Set workflow parameters
|
# Set workflow parameters
|
||||||
wf.set_node_param(
|
wf.set_node_param(
|
||||||
"KSampler", "seed", random.getrandbits(32)
|
"KSampler", "seed", random.getrandbits(32)
|
||||||
@ -92,6 +128,7 @@ def generate_image(file_name: str, comfy_prompt: str) -> None:
|
|||||||
wf.set_node_param(
|
wf.set_node_param(
|
||||||
"Empty Latent Image", "height", user_config["comfyui"]["height"]
|
"Empty Latent Image", "height", user_config["comfyui"]["height"]
|
||||||
)
|
)
|
||||||
|
|
||||||
# Validate available models and choose a random one
|
# Validate available models and choose a random one
|
||||||
valid_models = list(
|
valid_models = list(
|
||||||
set(get_available_models()) # Get all available models from ComfyUI
|
set(get_available_models()) # Get all available models from ComfyUI
|
||||||
@ -103,12 +140,14 @@ def generate_image(file_name: str, comfy_prompt: str) -> None:
|
|||||||
wf.set_node_param(
|
wf.set_node_param(
|
||||||
"Load Checkpoint", "ckpt_name", model
|
"Load Checkpoint", "ckpt_name", model
|
||||||
) # Set the model to be used for image generation
|
) # Set the model to be used for image generation
|
||||||
|
|
||||||
# Generate the image using the workflow and wait for completion
|
# Generate the image using the workflow and wait for completion
|
||||||
logging.debug(f"Generating image: {file_name}")
|
logging.debug(f"Generating image: {file_name}")
|
||||||
results = api.queue_and_wait_images(
|
results = api.queue_and_wait_images(
|
||||||
wf, "Save Image"
|
wf, "Save Image"
|
||||||
) # Queue the workflow and wait for image generation to complete
|
) # Queue the workflow and wait for image generation to complete
|
||||||
rename_image() # Rename the generated image file if it exists
|
rename_image() # Rename the generated image file if it exists
|
||||||
|
|
||||||
# Save the generated image to disk
|
# Save the generated image to disk
|
||||||
for filename, image_data in results.items():
|
for filename, image_data in results.items():
|
||||||
with open(
|
with open(
|
||||||
@ -116,16 +155,22 @@ def generate_image(file_name: str, comfy_prompt: str) -> None:
|
|||||||
) as f:
|
) as f:
|
||||||
f.write(image_data)
|
f.write(image_data)
|
||||||
logging.debug(f"Image generated successfully for UID: {file_name}")
|
logging.debug(f"Image generated successfully for UID: {file_name}")
|
||||||
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
logging.error(f"Failed to generate image for UID: {file_name}. Error: {e}")
|
logging.error(f"Failed to generate image for UID: {file_name}. Error: {e}")
|
||||||
|
raise # Re-raise the exception for Tenacity to handle retries
|
||||||
|
|
||||||
|
|
||||||
def create_image(prompt: str | None = None) -> None:
|
def create_image(prompt: str | None = None) -> None:
|
||||||
"""Main function for generating images."""
|
"""Main function for generating images."""
|
||||||
if prompt is None:
|
if prompt is None:
|
||||||
prompt = send_prompt_to_openwebui(user_config["comfyui"]["prompt"])
|
prompt = send_prompt_to_openwebui(user_config["comfyui"]["prompt"])
|
||||||
print(f"Generated prompt: {prompt}")
|
if prompt:
|
||||||
generate_image("image", prompt)
|
logging.info(f"Generated prompt: {prompt}") # Log generated prompt
|
||||||
|
generate_image("image", prompt)
|
||||||
|
print(f"Image generation started with prompt: {prompt}")
|
||||||
|
else:
|
||||||
|
logging.error("No prompt generated.")
|
||||||
|
|
||||||
|
|
||||||
user_config = load_config()
|
user_config = load_config()
|
||||||
|
BIN
requirements.txt
BIN
requirements.txt
Binary file not shown.
Loading…
x
Reference in New Issue
Block a user