- New Checkpoint model (slug, name, checkpoint_path, data JSON, image_path) - sync_checkpoints() loads metadata from data/checkpoints/*.json and falls back to template defaults for models without a JSON file - _apply_checkpoint_settings() applies per-checkpoint steps, CFG, sampler, base positive/negative prompts, and VAE (with dynamic VAELoader node injection for non-integrated VAEs) to the ComfyUI workflow - Bulk Create from Checkpoints: scans Illustrious/Noob model directories, reads matching HTML files, uses LLM to populate metadata, falls back to template defaults when no HTML is present - Gallery index with batch cover generation and WebSocket progress bar - Detail page showing Generation Settings and Base Prompts cards - Checkpoints nav link added to layout - New data/prompts/checkpoint_system.txt LLM system prompt - Updated README with all current galleries and file structure - Also includes accumulated action/scene JSON updates, new actions, and other template/generator improvements from prior sessions Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
124 lines
5.2 KiB
Python
124 lines
5.2 KiB
Python
from flask_sqlalchemy import SQLAlchemy
|
|
|
|
db = SQLAlchemy()
|
|
|
|
class Character(db.Model):
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
character_id = db.Column(db.String(100), unique=True, nullable=False)
|
|
slug = db.Column(db.String(100), unique=True, nullable=False)
|
|
filename = db.Column(db.String(255), nullable=True)
|
|
name = db.Column(db.String(100), nullable=False)
|
|
data = db.Column(db.JSON, nullable=False)
|
|
default_fields = db.Column(db.JSON, nullable=True)
|
|
image_path = db.Column(db.String(255), nullable=True)
|
|
active_outfit = db.Column(db.String(100), default='default')
|
|
|
|
def get_active_wardrobe(self):
|
|
"""Get the currently active wardrobe outfit."""
|
|
wardrobe = self.data.get('wardrobe', {})
|
|
# Check if wardrobe is nested (new format) or flat (legacy)
|
|
if 'default' in wardrobe and isinstance(wardrobe.get('default'), dict):
|
|
# New nested format - return active outfit
|
|
return wardrobe.get(self.active_outfit or 'default', wardrobe.get('default', {}))
|
|
else:
|
|
# Legacy flat format - return as-is
|
|
return wardrobe
|
|
|
|
def get_available_outfits(self):
|
|
"""Get list of available outfit names."""
|
|
wardrobe = self.data.get('wardrobe', {})
|
|
if 'default' in wardrobe and isinstance(wardrobe.get('default'), dict):
|
|
return list(wardrobe.keys())
|
|
return ['default']
|
|
|
|
def __repr__(self):
|
|
return f'<Character {self.character_id}>'
|
|
|
|
class Outfit(db.Model):
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
outfit_id = db.Column(db.String(100), unique=True, nullable=False)
|
|
slug = db.Column(db.String(100), unique=True, nullable=False)
|
|
filename = db.Column(db.String(255), nullable=True)
|
|
name = db.Column(db.String(100), nullable=False)
|
|
data = db.Column(db.JSON, nullable=False)
|
|
default_fields = db.Column(db.JSON, nullable=True)
|
|
image_path = db.Column(db.String(255), nullable=True)
|
|
|
|
def __repr__(self):
|
|
return f'<Outfit {self.outfit_id}>'
|
|
|
|
class Action(db.Model):
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
action_id = db.Column(db.String(100), unique=True, nullable=False)
|
|
slug = db.Column(db.String(100), unique=True, nullable=False)
|
|
filename = db.Column(db.String(255), nullable=True)
|
|
name = db.Column(db.String(100), nullable=False)
|
|
data = db.Column(db.JSON, nullable=False)
|
|
default_fields = db.Column(db.JSON, nullable=True)
|
|
image_path = db.Column(db.String(255), nullable=True)
|
|
|
|
def __repr__(self):
|
|
return f'<Action {self.action_id}>'
|
|
|
|
class Style(db.Model):
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
style_id = db.Column(db.String(100), unique=True, nullable=False)
|
|
slug = db.Column(db.String(100), unique=True, nullable=False)
|
|
filename = db.Column(db.String(255), nullable=True)
|
|
name = db.Column(db.String(100), nullable=False)
|
|
data = db.Column(db.JSON, nullable=False)
|
|
default_fields = db.Column(db.JSON, nullable=True)
|
|
image_path = db.Column(db.String(255), nullable=True)
|
|
|
|
def __repr__(self):
|
|
return f'<Style {self.style_id}>'
|
|
|
|
class Scene(db.Model):
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
scene_id = db.Column(db.String(100), unique=True, nullable=False)
|
|
slug = db.Column(db.String(100), unique=True, nullable=False)
|
|
filename = db.Column(db.String(255), nullable=True)
|
|
name = db.Column(db.String(100), nullable=False)
|
|
data = db.Column(db.JSON, nullable=False)
|
|
default_fields = db.Column(db.JSON, nullable=True)
|
|
image_path = db.Column(db.String(255), nullable=True)
|
|
|
|
def __repr__(self):
|
|
return f'<Scene {self.scene_id}>'
|
|
|
|
class Detailer(db.Model):
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
detailer_id = db.Column(db.String(100), unique=True, nullable=False)
|
|
slug = db.Column(db.String(100), unique=True, nullable=False)
|
|
filename = db.Column(db.String(255), nullable=True)
|
|
name = db.Column(db.String(100), nullable=False)
|
|
data = db.Column(db.JSON, nullable=False)
|
|
default_fields = db.Column(db.JSON, nullable=True)
|
|
image_path = db.Column(db.String(255), nullable=True)
|
|
|
|
def __repr__(self):
|
|
return f'<Detailer {self.detailer_id}>'
|
|
|
|
class Checkpoint(db.Model):
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
checkpoint_id = db.Column(db.String(255), unique=True, nullable=False)
|
|
slug = db.Column(db.String(255), unique=True, nullable=False)
|
|
name = db.Column(db.String(255), nullable=False)
|
|
checkpoint_path = db.Column(db.String(255), nullable=False) # e.g. "Illustrious/model.safetensors"
|
|
data = db.Column(db.JSON, nullable=True)
|
|
image_path = db.Column(db.String(255), nullable=True)
|
|
|
|
def __repr__(self):
|
|
return f'<Checkpoint {self.checkpoint_id}>'
|
|
|
|
class Settings(db.Model):
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
llm_provider = db.Column(db.String(50), default='openrouter') # 'openrouter', 'ollama', 'lmstudio'
|
|
openrouter_api_key = db.Column(db.String(255), nullable=True)
|
|
openrouter_model = db.Column(db.String(100), default='google/gemini-2.0-flash-001')
|
|
local_base_url = db.Column(db.String(255), nullable=True)
|
|
local_model = db.Column(db.String(100), nullable=True)
|
|
|
|
def __repr__(self):
|
|
return '<Settings>'
|