mirror of
				https://github.com/karl0ss/ai_image_frame_server.git
				synced 2025-10-25 04:34:01 +01:00 
			
		
		
		
	Compare commits
	
		
			9 Commits
		
	
	
		
			d80cf9473a
			...
			57bb0fed5b
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 57bb0fed5b | |||
| 6e39c34a58 | |||
| e2acd2dcd6 | |||
| aa75646d5f | |||
| ba2b943c0d | |||
| 5c45b8b832 | |||
| 9462888701 | |||
| bd1bb98160 | |||
| 76e33ea523 | 
| @ -1,5 +1,5 @@ | ||||
| [tool.bumpversion] | ||||
| current_version = "0.2.19" | ||||
| current_version = "0.3.1" | ||||
| parse = "(?P<major>\\d+)\\.(?P<minor>\\d+)\\.(?P<patch>\\d+)" | ||||
| serialize = ["{major}.{minor}.{patch}"] | ||||
| replace = "{new_version}" | ||||
|  | ||||
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -11,3 +11,4 @@ publish.sh | ||||
| test.py | ||||
| .vscode/launch.json | ||||
| favourites.json | ||||
| .vscode/launch.json | ||||
|  | ||||
							
								
								
									
										2
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							| @ -8,7 +8,7 @@ | ||||
|             "name": "Python Debugger: Current File", | ||||
|             "type": "debugpy", | ||||
|             "request": "launch", | ||||
|             "program": "${file}", | ||||
|             "program": "ai_frame_image_server.py", | ||||
|             "console": "integratedTerminal", | ||||
|             "justMyCode": false, | ||||
|             "env": {"SECRET_KEY":"dkdkdk"} | ||||
|  | ||||
| @ -4,7 +4,7 @@ FROM python:3.11-slim | ||||
| # Set the working directory in the container | ||||
| WORKDIR /app | ||||
| # Set version label | ||||
| ARG VERSION="0.2.19" | ||||
| ARG VERSION="0.3.1" | ||||
| LABEL version=$VERSION | ||||
| 
 | ||||
| # Copy project files into the container | ||||
|  | ||||
| @ -15,7 +15,7 @@ from tenacity import ( | ||||
| import nest_asyncio | ||||
| from libs.generic import rename_image, load_config, save_prompt | ||||
| from libs.create_thumbnail import generate_thumbnail | ||||
| from libs.ollama import create_prompt_on_openwebui | ||||
| from libs.openwebui import create_prompt_on_openwebui | ||||
| nest_asyncio.apply() | ||||
| 
 | ||||
| logging.basicConfig(level=logging.INFO) | ||||
| @ -123,14 +123,14 @@ def select_model(model: str) -> tuple[str, str]: | ||||
|     use_flux = json.loads(user_config["comfyui"].get("FLUX", "false").lower()) | ||||
|     only_flux = json.loads(user_config["comfyui"].get("ONLY_FLUX", "false").lower()) | ||||
| 
 | ||||
|     if model == "Random": | ||||
|     if model == "Random Image Model": | ||||
|         selected_workflow = "FLUX" if (use_flux and (only_flux or random.choice([True, False]))) else "SDXL" | ||||
|     elif "flux" in model.lower(): | ||||
|         selected_workflow = "FLUX" | ||||
|     else: | ||||
|         selected_workflow = "SDXL" | ||||
| 
 | ||||
|     if model == "Random": | ||||
|     if model == "Random Image Model": | ||||
|         if selected_workflow == "FLUX": | ||||
|             valid_models = user_config["comfyui:flux"]["models"].split(",") | ||||
|         else:  # SDXL | ||||
| @ -141,11 +141,12 @@ def select_model(model: str) -> tuple[str, str]: | ||||
|     return selected_workflow, model | ||||
| 
 | ||||
| 
 | ||||
| def create_image(prompt: str | None = None, model: str = "Random") -> None: | ||||
| def create_image(prompt: str | None = None, model: str = "Random Image Model") -> None: | ||||
|     """Generate an image with a chosen workflow (Random, FLUX*, or SDXL*).""" | ||||
| 
 | ||||
|     if prompt is None: | ||||
|         prompt = create_prompt_on_openwebui(user_config["comfyui"]["prompt"]) | ||||
|         logging.error("No prompt provided.") | ||||
|         return | ||||
| 
 | ||||
|     if not prompt: | ||||
|         logging.error("No prompt generated.") | ||||
|  | ||||
| @ -4,6 +4,7 @@ import logging | ||||
| import sys | ||||
| import time | ||||
| import os | ||||
| import random | ||||
| from PIL import Image | ||||
| import nest_asyncio | ||||
| import json | ||||
| @ -124,5 +125,59 @@ def load_topics_from_config(): | ||||
|     sorted_topics = sorted(topics, key=str.lower) | ||||
|     return sorted_topics | ||||
| 
 | ||||
| def load_openrouter_models_from_config(): | ||||
|     config = load_config() | ||||
|     if config["openrouter"].get("enabled", "False").lower() == "true": | ||||
|         models = config["openrouter"]["models"].split(",") | ||||
|         return sorted([model.strip() for model in models if model.strip()], key=str.lower) | ||||
|     return [] | ||||
| 
 | ||||
| def load_openwebui_models_from_config(): | ||||
|     config = load_config() | ||||
|     if "openwebui" in config and "models" in config["openwebui"]: | ||||
|         models = config["openwebui"]["models"].split(",") | ||||
|         return sorted([model.strip() for model in models if model.strip()], key=str.lower) | ||||
|     return [] | ||||
| 
 | ||||
| def load_prompt_models_from_config(): | ||||
|     """Load and return a list of available prompt generation models (both OpenWebUI and OpenRouter).""" | ||||
|     config = load_config() | ||||
|     prompt_models = [] | ||||
|      | ||||
|     # Add OpenWebUI models if configured | ||||
|     if "openwebui" in config and "models" in config["openwebui"]: | ||||
|         openwebui_models = config["openwebui"]["models"].split(",") | ||||
|         prompt_models.extend([("openwebui", model.strip()) for model in openwebui_models if model.strip()]) | ||||
|      | ||||
|     # Add OpenRouter models if enabled and configured | ||||
|     if config["openrouter"].get("enabled", "False").lower() == "true" and "models" in config["openrouter"]: | ||||
|         openrouter_models = config["openrouter"]["models"].split(",") | ||||
|         prompt_models.extend([("openrouter", model.strip()) for model in openrouter_models if model.strip()]) | ||||
|      | ||||
|     return prompt_models | ||||
| 
 | ||||
| 
 | ||||
| def create_prompt_with_random_model(base_prompt: str, topic: str = "random"): | ||||
|     """Create a prompt using a randomly selected model from OpenWebUI or OpenRouter.""" | ||||
|     prompt_models = load_prompt_models_from_config() | ||||
|      | ||||
|     if not prompt_models: | ||||
|         logging.warning("No prompt generation models configured.") | ||||
|         return None | ||||
|      | ||||
|     # Randomly select a model | ||||
|     service, model = random.choice(prompt_models) | ||||
|      | ||||
|     if service == "openwebui": | ||||
|         # Import here to avoid circular imports | ||||
|         from libs.openwebui import create_prompt_on_openwebui | ||||
|         return create_prompt_on_openwebui(base_prompt, topic) | ||||
|     elif service == "openrouter": | ||||
|         # Import here to avoid circular imports | ||||
|         from libs.openrouter import create_prompt_on_openrouter | ||||
|         return create_prompt_on_openrouter(base_prompt, topic) | ||||
|      | ||||
|     return None | ||||
| 
 | ||||
| user_config = load_config() | ||||
| output_folder = user_config["comfyui"]["output_dir"] | ||||
							
								
								
									
										95
									
								
								libs/openrouter.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										95
									
								
								libs/openrouter.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,95 @@ | ||||
| import random | ||||
| import logging | ||||
| from openai import OpenAI | ||||
| import nest_asyncio | ||||
| from libs.generic import load_recent_prompts, load_config | ||||
| import re | ||||
| nest_asyncio.apply() | ||||
| 
 | ||||
| logging.basicConfig(level=logging.INFO) | ||||
| 
 | ||||
| LOG_FILE = "./prompts_log.jsonl" | ||||
| 
 | ||||
| user_config = load_config() | ||||
| output_folder = user_config["comfyui"]["output_dir"] | ||||
| 
 | ||||
| def create_prompt_on_openrouter(prompt: str, topic: str = "random", model: str = None) -> str: | ||||
|     """Sends prompt to OpenRouter and returns the generated response.""" | ||||
|     # Check if OpenRouter is enabled | ||||
|     if user_config["openrouter"].get("enabled", "False").lower() != "true": | ||||
|         logging.warning("OpenRouter is not enabled in the configuration.") | ||||
|         return "" | ||||
|      | ||||
|     topic_instruction = "" | ||||
|     selected_topic = "" | ||||
|     # Unique list of recent prompts | ||||
|     recent_prompts = list(set(load_recent_prompts())) | ||||
|     if topic == "random": | ||||
|         topics = [t.strip() for t in user_config["comfyui"]["topics"].split(",") if t.strip()] | ||||
|         selected_topic = random.choice(topics) if topics else "" | ||||
|     elif topic != "": | ||||
|         selected_topic = topic | ||||
|     else: | ||||
|         # Decide on whether to include a topic (e.g., 30% chance to include) | ||||
|         topics = [t.strip() for t in user_config["comfyui"]["topics"].split(",") if t.strip()] | ||||
|         if random.random() < 0.3 and topics: | ||||
|             selected_topic = random.choice(topics) | ||||
|     if selected_topic != "": | ||||
|         topic_instruction = f" Incorporate the theme of '{selected_topic}' into the new prompt." | ||||
| 
 | ||||
|     user_content = ( | ||||
|         "Can you generate me a really random image idea, Do not exceed 10 words. Use clear language, not poetic metaphors." | ||||
|         + topic_instruction | ||||
|         + "Avoid prompts similar to the following:" | ||||
|         + "\n".join(f"{i+1}. {p}" for i, p in enumerate(recent_prompts)) | ||||
|     ) | ||||
| 
 | ||||
|     # Use the specified model or select a random model from the configured OpenRouter models | ||||
|     if model: | ||||
|         # Use the specified model | ||||
|         model = model | ||||
|     else: | ||||
|         # Select a random model from the configured OpenRouter models | ||||
|         models = [m.strip() for m in user_config["openrouter"]["models"].split(",") if m.strip()] | ||||
|         if not models: | ||||
|             logging.error("No OpenRouter models configured.") | ||||
|             return "" | ||||
|          | ||||
|         model = random.choice(models) | ||||
|      | ||||
|     try: | ||||
|         client = OpenAI( | ||||
|             base_url="https://openrouter.ai/api/v1", | ||||
|             api_key=user_config["openrouter"]["api_key"], | ||||
|         ) | ||||
| 
 | ||||
|         completion = client.chat.completions.create( | ||||
|             model=model, | ||||
|             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", | ||||
|                     "content": user_content, | ||||
|                 }, | ||||
|             ] | ||||
|         ) | ||||
| 
 | ||||
|         prompt = completion.choices[0].message.content.strip('"') | ||||
|         match = re.search(r'"([^"]+)"', prompt) | ||||
|         if not match: | ||||
|             match = re.search(r":\s*\n*\s*(.+)", prompt) | ||||
|         if match: | ||||
|             prompt = match.group(1) | ||||
|         logging.debug(prompt) | ||||
|         return prompt | ||||
|     except Exception as e: | ||||
|         logging.error(f"Error generating prompt with OpenRouter: {e}") | ||||
|         return "" | ||||
| @ -1,9 +1,11 @@ | ||||
| import random | ||||
| import logging | ||||
| import litellm | ||||
| import nest_asyncio | ||||
| from libs.generic import load_recent_prompts, load_config | ||||
| import re | ||||
| from openwebui_chat_client import OpenWebUIClient | ||||
| from datetime import datetime | ||||
| 
 | ||||
| nest_asyncio.apply() | ||||
| 
 | ||||
| logging.basicConfig(level=logging.INFO) | ||||
| @ -13,7 +15,7 @@ LOG_FILE = "./prompts_log.jsonl" | ||||
| user_config = load_config() | ||||
| output_folder = user_config["comfyui"]["output_dir"] | ||||
| 
 | ||||
| def create_prompt_on_openwebui(prompt: str, topic: str = "random") -> str: | ||||
| def create_prompt_on_openwebui(prompt: str, topic: str = "random", model: str = None) -> str: | ||||
|     """Sends prompt to OpenWebui and returns the generated response.""" | ||||
|     topic_instruction = "" | ||||
|     selected_topic = "" | ||||
| @ -33,48 +35,56 @@ def create_prompt_on_openwebui(prompt: str, topic: str = "random") -> str: | ||||
|         topic_instruction = f" Incorporate the theme of '{selected_topic}' into the new prompt." | ||||
| 
 | ||||
|     user_content = ( | ||||
|         "Can you generate me a really random image idea, Do not exceed 10 words. Use clear language, not poetic metaphors.”" | ||||
|         "Can you generate me a really random image idea, Do not exceed 10 words. Use clear language, not poetic metaphors." | ||||
|         + topic_instruction | ||||
|         + "Avoid prompts similar to the following:" | ||||
|         + "\n".join(f"{i+1}. {p}" for i, p in enumerate(recent_prompts)) | ||||
|     ) | ||||
| 
 | ||||
|     if model: | ||||
|         # Use the specified model | ||||
|         model = model | ||||
|     else: | ||||
|         # Select a random model | ||||
|         model = random.choice(user_config["openwebui"]["models"].split(",")).strip() | ||||
| 
 | ||||
|     model = random.choice(user_config["openwebui"]["models"].split(",")) | ||||
|     response = litellm.completion( | ||||
|         api_base=user_config["openwebui"]["base_url"], | ||||
|         model="openai/" + model, | ||||
|         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", | ||||
|                 "content": user_content, | ||||
|             }, | ||||
|         ], | ||||
|         api_key=user_config["openwebui"]["api_key"], | ||||
|     # Create OpenWebUI client | ||||
|     client = OpenWebUIClient( | ||||
|         base_url=user_config["openwebui"]["base_url"], | ||||
|         token=user_config["openwebui"]["api_key"], | ||||
|         default_model_id=model | ||||
|     ) | ||||
| 
 | ||||
|     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('"') | ||||
|     # Prepare messages for the chat | ||||
|     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", | ||||
|             "content": user_content, | ||||
|         }, | ||||
|     ] | ||||
| 
 | ||||
|     # Send the chat request | ||||
|     result = client.chat( | ||||
|         question=user_content, | ||||
|         chat_title=datetime.now().strftime("%Y-%m-%d %H:%M"), | ||||
|         folder_name="ai-frame-image-server"  | ||||
|     ) | ||||
| 
 | ||||
|     if result: | ||||
|         prompt = result["response"].strip('"') | ||||
|     else: | ||||
|         # Fallback if the request fails | ||||
|         prompt = "A vibrant landscape" | ||||
| 
 | ||||
|     match = re.search(r'"([^"]+)"', prompt) | ||||
|     if not match: | ||||
|         match = re.search(r":\s*\n*\s*(.+)", prompt)  | ||||
							
								
								
									
										
											BIN
										
									
								
								requirements.txt
									
									
									
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								requirements.txt
									
									
									
									
									
								
							
										
											Binary file not shown.
										
									
								
							| @ -1,8 +1,8 @@ | ||||
| from flask import Blueprint, request, render_template, redirect, url_for, session | ||||
| import threading | ||||
| from libs.comfyui import create_image, select_model, get_available_models | ||||
| from libs.ollama import create_prompt_on_openwebui | ||||
| from libs.generic import load_models_from_config, load_topics_from_config | ||||
| from libs.openwebui import create_prompt_on_openwebui | ||||
| from libs.generic import load_models_from_config, load_topics_from_config, load_openrouter_models_from_config, load_openwebui_models_from_config, create_prompt_with_random_model | ||||
| import os | ||||
| 
 | ||||
| bp = Blueprint("create_routes", __name__) | ||||
| @ -12,28 +12,67 @@ user_config = None  # will be set in init_app | ||||
| def create(): | ||||
|     if request.method == "POST": | ||||
|         prompt = request.form.get("prompt") | ||||
|         selected_workflow, model = select_model(request.form.get("model") or "Random") | ||||
|         image_model = request.form.get("model") or "Random Image Model" | ||||
|         selected_workflow, model = select_model(image_model) | ||||
|         topic = request.form.get("topic") | ||||
| 
 | ||||
|         if not prompt: | ||||
|             prompt = create_prompt_on_openwebui(user_config["comfyui"]["prompt"], topic) | ||||
|             # Get the prompt model from the form data | ||||
|             prompt_model = request.form.get("prompt_model") or "" | ||||
|             if prompt_model and prompt_model != "Random Prompt Model": | ||||
|                 # Use the specified prompt model | ||||
|                 service, service_model = prompt_model.split(":", 1) if ":" in prompt_model else (prompt_model, "") | ||||
|                 if service == "openwebui": | ||||
|                     from libs.openwebui import create_prompt_on_openwebui | ||||
|                     prompt = create_prompt_on_openwebui(user_config["comfyui"]["prompt"], topic, service_model) | ||||
|                 elif service == "openrouter": | ||||
|                     from libs.openrouter import create_prompt_on_openrouter | ||||
|                     prompt = create_prompt_on_openrouter(user_config["comfyui"]["prompt"], topic, service_model) | ||||
|             else: | ||||
|                 # Use a random prompt model | ||||
|                 prompt = create_prompt_with_random_model(user_config["comfyui"]["prompt"], topic) | ||||
| 
 | ||||
|         threading.Thread(target=lambda: create_image(prompt, model)).start() | ||||
|         return redirect(url_for("create_routes.image_queued", prompt=prompt, model=model.split(".")[0])) | ||||
| 
 | ||||
|     return render_template("create_image.html", models=load_models_from_config()[0]+load_models_from_config()[1], topics=load_topics_from_config()) | ||||
|     # Load all models (SDXL and FLUX only) | ||||
|     sdxl_models, flux_models = load_models_from_config() | ||||
|     openwebui_models = load_openwebui_models_from_config() | ||||
|     openrouter_models = load_openrouter_models_from_config() | ||||
|      | ||||
|     return render_template("create_image.html", | ||||
|                          sdxl_models=sdxl_models, | ||||
|                          flux_models=flux_models, | ||||
|                          openwebui_models=openwebui_models, | ||||
|                          openrouter_models=openrouter_models, | ||||
|                          topics=load_topics_from_config()) | ||||
| 
 | ||||
| @bp.route("/image_queued") | ||||
| def image_queued(): | ||||
|     prompt = request.args.get("prompt", "No prompt provided.") | ||||
|     model = request.args.get("model", "No model selected.").split(".")[0] | ||||
|     model = request.args.get("model", "No model selected.") | ||||
|     if model == "Random Image Model": | ||||
|         model = "Random" | ||||
|     else: | ||||
|         model = model.split(".")[0] | ||||
|     return render_template("image_queued.html", prompt=prompt, model=model) | ||||
| 
 | ||||
| @bp.route("/create_image", methods=["GET"]) | ||||
| def create_image_page(): | ||||
|     if user_config["frame"]["create_requires_auth"] == "True" and not session.get("authenticated"): | ||||
|         return redirect(url_for("auth_routes.login", next=request.path)) | ||||
|     return render_template("create_image.html", models=load_models_from_config()[0]+load_models_from_config()[1], topics=load_topics_from_config()) | ||||
|      | ||||
|     # Load all models (SDXL and FLUX only) | ||||
|     sdxl_models, flux_models = load_models_from_config() | ||||
|     openwebui_models = load_openwebui_models_from_config() | ||||
|     openrouter_models = load_openrouter_models_from_config() | ||||
|      | ||||
|     return render_template("create_image.html", | ||||
|                          sdxl_models=sdxl_models, | ||||
|                          flux_models=flux_models, | ||||
|                          openwebui_models=openwebui_models, | ||||
|                          openrouter_models=openrouter_models, | ||||
|                          topics=load_topics_from_config()) | ||||
| 
 | ||||
| 
 | ||||
| def init_app(config): | ||||
|  | ||||
| @ -33,6 +33,28 @@ | ||||
|         align-items: center; | ||||
|     } | ||||
| 
 | ||||
|     .model-selection { | ||||
|         display: flex; | ||||
|         flex-wrap: wrap; | ||||
|         gap: 20px; | ||||
|         justify-content: center; | ||||
|         margin: 20px 0; | ||||
|         width: 100%; | ||||
|         max-width: 800px; | ||||
|     } | ||||
| 
 | ||||
|     .model-group { | ||||
|         display: flex; | ||||
|         flex-direction: column; | ||||
|         align-items: flex-start; | ||||
|         gap: 5px; | ||||
|     } | ||||
| 
 | ||||
|     .model-group label { | ||||
|         font-weight: bold; | ||||
|         color: #ddd; | ||||
|     } | ||||
| 
 | ||||
|     button, | ||||
|     select { | ||||
|         background: #333; | ||||
| @ -43,6 +65,7 @@ | ||||
|         font-size: 16px; | ||||
|         cursor: pointer; | ||||
|         transition: background 0.3s; | ||||
|         min-width: 150px; | ||||
|     } | ||||
| 
 | ||||
|     button:hover, | ||||
| @ -90,6 +113,15 @@ | ||||
|             width: 100%; | ||||
|         } | ||||
| 
 | ||||
|         .model-selection { | ||||
|             flex-direction: column; | ||||
|             align-items: stretch; | ||||
|         } | ||||
| 
 | ||||
|         .model-group { | ||||
|             align-items: stretch; | ||||
|         } | ||||
| 
 | ||||
|         button, | ||||
|         select { | ||||
|             width: 100%; | ||||
| @ -109,34 +141,65 @@ | ||||
| 
 | ||||
| <div class="button-group"> | ||||
|     <button onclick="showSpinner(); location.href='/'">Back</button> | ||||
| 
 | ||||
|     <button onclick="sendPrompt()">Send Prompt</button> | ||||
| 
 | ||||
|     <button onclick="randomPrompt()">Random Prompt</button> | ||||
| </div> | ||||
| 
 | ||||
|     <select id="model-select"> | ||||
|         <option value="" selected>Random</option> | ||||
|         <optgroup label="FLUX"> | ||||
|             {% for m in models if 'flux' in m|lower %} | ||||
|             <option value="{{ m }}">{{ m.rsplit('.', 1)[0] }}</option> | ||||
|             {% endfor %} | ||||
|         </optgroup> | ||||
|         <optgroup label="SDXL"> | ||||
|             {% for m in models if 'flux' not in m|lower %} | ||||
|             <option value="{{ m }}">{{ m.rsplit('.', 1)[0] }}</option> | ||||
|             {% endfor %} | ||||
|         </optgroup> | ||||
|     </select> | ||||
| <div class="model-selection"> | ||||
|     <div class="model-group"> | ||||
|         <label for="model-select">Image Model:</label> | ||||
|         <select id="model-select"> | ||||
|             <option value="" selected>Random Image Model</option> | ||||
|             {% if flux_models %} | ||||
|             <optgroup label="FLUX"> | ||||
|                 {% for m in flux_models %} | ||||
|                 <option value="{{ m }}">{{ m.rsplit('.', 1)[0] if '.' in m else m }}</option> | ||||
|                 {% endfor %} | ||||
|             </optgroup> | ||||
|             {% endif %} | ||||
|             {% if sdxl_models %} | ||||
|             <optgroup label="SDXL"> | ||||
|                 {% for m in sdxl_models %} | ||||
|                 <option value="{{ m }}">{{ m.rsplit('.', 1)[0] if '.' in m else m }}</option> | ||||
|                 {% endfor %} | ||||
|             </optgroup> | ||||
|             {% endif %} | ||||
|         </select> | ||||
|     </div> | ||||
|      | ||||
|     <select id="topic-select"> | ||||
|         <option value="">No Topic</option> | ||||
|         <option value="random">Random</option> | ||||
|         <optgroup label="Topics"> | ||||
|             {% for t in topics %} | ||||
|             <option value="{{ t }}">{{ t }}</option> | ||||
|             {% endfor %} | ||||
|         </optgroup> | ||||
|     </select> | ||||
|     <div class="model-group"> | ||||
|         <label for="prompt-model-select">Prompt Model:</label> | ||||
|         <select id="prompt-model-select"> | ||||
|             <option value="" selected>Random Prompt Model</option> | ||||
|             {% if openwebui_models %} | ||||
|             <optgroup label="OpenWebUI"> | ||||
|                 {% for m in openwebui_models %} | ||||
|                 <option value="openwebui:{{ m }}">{{ m }}</option> | ||||
|                 {% endfor %} | ||||
|             </optgroup> | ||||
|             {% endif %} | ||||
|             {% if openrouter_models %} | ||||
|             <optgroup label="OpenRouter"> | ||||
|                 {% for m in openrouter_models %} | ||||
|                 <option value="openrouter:{{ m }}">{{ m }}</option> | ||||
|                 {% endfor %} | ||||
|             </optgroup> | ||||
|             {% endif %} | ||||
|         </select> | ||||
|     </div> | ||||
|      | ||||
|     <div class="model-group"> | ||||
|         <label for="topic-select">Topic:</label> | ||||
|         <select id="topic-select"> | ||||
|             <option value="">No Topic</option> | ||||
|             <option value="random">Random</option> | ||||
|             <optgroup label="Topics"> | ||||
|                 {% for t in topics %} | ||||
|                 <option value="{{ t }}">{{ t }}</option> | ||||
|                 {% endfor %} | ||||
|             </optgroup> | ||||
|         </select> | ||||
|     </div> | ||||
| </div> | ||||
| 
 | ||||
| <div id="spinner-overlay"> | ||||
| @ -154,10 +217,12 @@ | ||||
|         showSpinner(); | ||||
|         const prompt = document.getElementById('prompt-box').value; | ||||
|         const model = document.getElementById('model-select').value; | ||||
|         const promptModel = document.getElementById('prompt-model-select').value; | ||||
| 
 | ||||
|         const formData = new URLSearchParams(); | ||||
|         formData.append('prompt', prompt); | ||||
|         formData.append('model', model); | ||||
|         formData.append('prompt_model', promptModel); | ||||
| 
 | ||||
|         fetch('/create', { | ||||
|             method: 'POST', | ||||
| @ -176,10 +241,12 @@ | ||||
|     function randomPrompt() { | ||||
|         showSpinner(); | ||||
|         const model = document.getElementById('model-select').value; | ||||
|         const promptModel = document.getElementById('prompt-model-select').value; | ||||
|         const topic = document.getElementById('topic-select').value; | ||||
| 
 | ||||
|         const formData = new URLSearchParams(); | ||||
|         formData.append('model', model); | ||||
|         formData.append('prompt_model', promptModel); | ||||
|         formData.append('topic', topic); | ||||
| 
 | ||||
|         fetch('/create', { | ||||
|  | ||||
| @ -25,3 +25,8 @@ models = flux1-dev-Q4_0.gguf,flux1-schnell-Q4_0.gguf | ||||
| base_url = https://openwebui | ||||
| api_key = sk- | ||||
| models = llama3:latest,cogito:14b,gemma3:12b | ||||
| 
 | ||||
| [openrouter] | ||||
| enabled = False | ||||
| api_key = | ||||
| models = mistralai/mistral-7b-instruct:free,google/gemma-7b-it:free,meta-llama/llama-3.1-8b-instruct:free | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user