2022-10-24 14:23:51 +00:00
|
|
|
import sys
|
|
|
|
import traceback
|
|
|
|
from collections import namedtuple
|
|
|
|
import inspect
|
2022-11-02 07:04:35 +00:00
|
|
|
from typing import Optional
|
2022-10-22 09:23:45 +00:00
|
|
|
|
2022-10-30 14:46:43 +00:00
|
|
|
from fastapi import FastAPI
|
|
|
|
from gradio import Blocks
|
2022-10-24 14:23:51 +00:00
|
|
|
|
|
|
|
def report_exception(c, job):
|
|
|
|
print(f"Error executing callback {job} for {c.script}", file=sys.stderr)
|
|
|
|
print(traceback.format_exc(), file=sys.stderr)
|
|
|
|
|
|
|
|
|
2022-10-26 10:12:44 +00:00
|
|
|
class ImageSaveParams:
|
|
|
|
def __init__(self, image, p, filename, pnginfo):
|
|
|
|
self.image = image
|
|
|
|
"""the PIL image itself"""
|
|
|
|
|
|
|
|
self.p = p
|
|
|
|
"""p object with processing parameters; either StableDiffusionProcessing or an object with same fields"""
|
|
|
|
|
|
|
|
self.filename = filename
|
|
|
|
"""name of file that the image would be saved to"""
|
|
|
|
|
|
|
|
self.pnginfo = pnginfo
|
|
|
|
"""dictionary with parameters for image's PNG info data; infotext will have the key 'parameters'"""
|
|
|
|
|
|
|
|
|
2022-11-02 00:34:58 +00:00
|
|
|
class CFGDenoiserParams:
|
|
|
|
def __init__(self, x, image_cond, sigma, sampling_step, total_sampling_steps):
|
|
|
|
self.x = x
|
|
|
|
"""Latent image representation in the process of being denoised"""
|
|
|
|
|
|
|
|
self.image_cond = image_cond
|
|
|
|
"""Conditioning image"""
|
|
|
|
|
|
|
|
self.sigma = sigma
|
|
|
|
"""Current sigma noise step value"""
|
|
|
|
|
2022-10-30 23:45:52 +00:00
|
|
|
self.sampling_step = sampling_step
|
2022-11-02 00:34:58 +00:00
|
|
|
"""Current Sampling step number"""
|
|
|
|
|
2022-10-30 23:45:52 +00:00
|
|
|
self.total_sampling_steps = total_sampling_steps
|
2022-11-02 00:34:58 +00:00
|
|
|
"""Total number of sampling steps planned"""
|
2022-10-30 23:45:52 +00:00
|
|
|
|
|
|
|
|
2022-10-24 14:23:51 +00:00
|
|
|
ScriptCallback = namedtuple("ScriptCallback", ["script", "callback"])
|
2022-11-04 12:19:16 +00:00
|
|
|
_callback_map = dict(
|
2022-11-02 16:59:10 +00:00
|
|
|
callbacks_app_started=[],
|
|
|
|
callbacks_model_loaded=[],
|
|
|
|
callbacks_ui_tabs=[],
|
|
|
|
callbacks_ui_settings=[],
|
|
|
|
callbacks_before_image_saved=[],
|
|
|
|
callbacks_image_saved=[],
|
|
|
|
callbacks_cfg_denoiser=[]
|
|
|
|
)
|
2022-10-22 09:23:45 +00:00
|
|
|
|
2022-10-26 10:12:44 +00:00
|
|
|
|
2022-10-22 09:23:45 +00:00
|
|
|
def clear_callbacks():
|
2022-11-04 12:19:16 +00:00
|
|
|
for callback_list in _callback_map.values():
|
2022-11-02 16:59:10 +00:00
|
|
|
callback_list.clear()
|
2022-10-22 09:23:45 +00:00
|
|
|
|
2022-11-02 07:04:35 +00:00
|
|
|
def app_started_callback(demo: Optional[Blocks], app: FastAPI):
|
2022-11-04 12:19:16 +00:00
|
|
|
for c in _callback_map['callbacks_app_started']:
|
2022-10-30 14:46:43 +00:00
|
|
|
try:
|
|
|
|
c.callback(demo, app)
|
|
|
|
except Exception:
|
|
|
|
report_exception(c, 'app_started_callback')
|
|
|
|
|
|
|
|
|
2022-10-22 09:23:45 +00:00
|
|
|
def model_loaded_callback(sd_model):
|
2022-11-04 12:19:16 +00:00
|
|
|
for c in _callback_map['callbacks_model_loaded']:
|
2022-10-24 14:23:51 +00:00
|
|
|
try:
|
|
|
|
c.callback(sd_model)
|
|
|
|
except Exception:
|
|
|
|
report_exception(c, 'model_loaded_callback')
|
2022-10-22 09:23:45 +00:00
|
|
|
|
|
|
|
|
|
|
|
def ui_tabs_callback():
|
|
|
|
res = []
|
2022-10-24 02:10:33 +00:00
|
|
|
|
2022-11-04 12:19:16 +00:00
|
|
|
for c in _callback_map['callbacks_ui_tabs']:
|
2022-10-24 14:23:51 +00:00
|
|
|
try:
|
|
|
|
res += c.callback() or []
|
|
|
|
except Exception:
|
|
|
|
report_exception(c, 'ui_tabs_callback')
|
2022-10-22 09:23:45 +00:00
|
|
|
|
|
|
|
return res
|
|
|
|
|
|
|
|
|
2022-10-22 16:18:56 +00:00
|
|
|
def ui_settings_callback():
|
2022-11-04 12:19:16 +00:00
|
|
|
for c in _callback_map['callbacks_ui_settings']:
|
2022-10-24 14:23:51 +00:00
|
|
|
try:
|
|
|
|
c.callback()
|
|
|
|
except Exception:
|
|
|
|
report_exception(c, 'ui_settings_callback')
|
|
|
|
|
|
|
|
|
2022-10-26 10:12:44 +00:00
|
|
|
def before_image_saved_callback(params: ImageSaveParams):
|
2022-11-04 12:19:16 +00:00
|
|
|
for c in _callback_map['callbacks_before_image_saved']:
|
2022-10-25 09:16:17 +00:00
|
|
|
try:
|
2022-10-26 10:12:44 +00:00
|
|
|
c.callback(params)
|
|
|
|
except Exception:
|
|
|
|
report_exception(c, 'before_image_saved_callback')
|
|
|
|
|
|
|
|
|
|
|
|
def image_saved_callback(params: ImageSaveParams):
|
2022-11-04 12:19:16 +00:00
|
|
|
for c in _callback_map['callbacks_image_saved']:
|
2022-10-26 10:12:44 +00:00
|
|
|
try:
|
|
|
|
c.callback(params)
|
2022-10-25 09:16:17 +00:00
|
|
|
except Exception:
|
|
|
|
report_exception(c, 'image_saved_callback')
|
|
|
|
|
|
|
|
|
2022-11-02 00:34:58 +00:00
|
|
|
def cfg_denoiser_callback(params: CFGDenoiserParams):
|
2022-11-04 12:19:16 +00:00
|
|
|
for c in _callback_map['callbacks_cfg_denoiser']:
|
2022-10-30 23:45:52 +00:00
|
|
|
try:
|
|
|
|
c.callback(params)
|
|
|
|
except Exception:
|
|
|
|
report_exception(c, 'cfg_denoiser_callback')
|
|
|
|
|
|
|
|
|
2022-10-24 14:23:51 +00:00
|
|
|
def add_callback(callbacks, fun):
|
|
|
|
stack = [x for x in inspect.stack() if x.filename != __file__]
|
|
|
|
filename = stack[0].filename if len(stack) > 0 else 'unknown file'
|
|
|
|
|
|
|
|
callbacks.append(ScriptCallback(filename, fun))
|
2022-10-22 16:18:56 +00:00
|
|
|
|
2022-11-02 16:59:10 +00:00
|
|
|
|
|
|
|
def remove_current_script_callbacks():
|
|
|
|
stack = [x for x in inspect.stack() if x.filename != __file__]
|
|
|
|
filename = stack[0].filename if len(stack) > 0 else 'unknown file'
|
|
|
|
if filename == 'unknown file':
|
|
|
|
return
|
2022-11-04 12:19:16 +00:00
|
|
|
for callback_list in _callback_map.values():
|
2022-11-02 16:59:10 +00:00
|
|
|
for callback_to_remove in [cb for cb in callback_list if cb.script == filename]:
|
|
|
|
callback_list.remove(callback_to_remove)
|
|
|
|
|
|
|
|
|
|
|
|
def remove_callbacks_for_function(callback_func):
|
2022-11-04 12:19:16 +00:00
|
|
|
for callback_list in _callback_map.values():
|
2022-11-02 16:59:10 +00:00
|
|
|
for callback_to_remove in [cb for cb in callback_list if cb.callback == callback_func]:
|
|
|
|
callback_list.remove(callback_to_remove)
|
|
|
|
|
2022-10-22 16:18:56 +00:00
|
|
|
|
2022-10-30 14:46:43 +00:00
|
|
|
def on_app_started(callback):
|
2022-10-31 00:47:43 +00:00
|
|
|
"""register a function to be called when the webui started, the gradio `Block` component and
|
|
|
|
fastapi `FastAPI` object are passed as the arguments"""
|
2022-11-04 12:19:16 +00:00
|
|
|
add_callback(_callback_map['callbacks_app_started'], callback)
|
2022-10-30 14:46:43 +00:00
|
|
|
|
|
|
|
|
2022-10-22 09:23:45 +00:00
|
|
|
def on_model_loaded(callback):
|
|
|
|
"""register a function to be called when the stable diffusion model is created; the model is
|
|
|
|
passed as an argument"""
|
2022-11-04 12:19:16 +00:00
|
|
|
add_callback(_callback_map['callbacks_model_loaded'], callback)
|
2022-10-22 09:23:45 +00:00
|
|
|
|
|
|
|
|
|
|
|
def on_ui_tabs(callback):
|
|
|
|
"""register a function to be called when the UI is creating new tabs.
|
|
|
|
The function must either return a None, which means no new tabs to be added, or a list, where
|
|
|
|
each element is a tuple:
|
|
|
|
(gradio_component, title, elem_id)
|
|
|
|
|
|
|
|
gradio_component is a gradio component to be used for contents of the tab (usually gr.Blocks)
|
|
|
|
title is tab text displayed to user in the UI
|
|
|
|
elem_id is HTML id for the tab
|
|
|
|
"""
|
2022-11-04 12:19:16 +00:00
|
|
|
add_callback(_callback_map['callbacks_ui_tabs'], callback)
|
2022-10-22 09:23:45 +00:00
|
|
|
|
2022-10-22 16:18:56 +00:00
|
|
|
|
|
|
|
def on_ui_settings(callback):
|
2022-10-22 16:19:17 +00:00
|
|
|
"""register a function to be called before UI settings are populated; add your settings
|
2022-10-22 16:18:56 +00:00
|
|
|
by using shared.opts.add_option(shared.OptionInfo(...)) """
|
2022-11-04 12:19:16 +00:00
|
|
|
add_callback(_callback_map['callbacks_ui_settings'], callback)
|
2022-10-24 06:17:09 +00:00
|
|
|
|
|
|
|
|
2022-10-26 10:12:44 +00:00
|
|
|
def on_before_image_saved(callback):
|
|
|
|
"""register a function to be called before an image is saved to a file.
|
|
|
|
The callback is called with one argument:
|
|
|
|
- params: ImageSaveParams - parameters the image is to be saved with. You can change fields in this object.
|
|
|
|
"""
|
2022-11-04 12:19:16 +00:00
|
|
|
add_callback(_callback_map['callbacks_before_image_saved'], callback)
|
2022-10-26 10:12:44 +00:00
|
|
|
|
|
|
|
|
2022-10-26 06:56:25 +00:00
|
|
|
def on_image_saved(callback):
|
2022-10-26 10:12:44 +00:00
|
|
|
"""register a function to be called after an image is saved to a file.
|
|
|
|
The callback is called with one argument:
|
|
|
|
- params: ImageSaveParams - parameters the image was saved with. Changing fields in this object does nothing.
|
2022-10-25 09:16:17 +00:00
|
|
|
"""
|
2022-11-04 12:19:16 +00:00
|
|
|
add_callback(_callback_map['callbacks_image_saved'], callback)
|
2022-10-30 23:45:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
def on_cfg_denoiser(callback):
|
|
|
|
"""register a function to be called in the kdiffussion cfg_denoiser method after building the inner model inputs.
|
|
|
|
The callback is called with one argument:
|
2022-11-02 00:34:58 +00:00
|
|
|
- params: CFGDenoiserParams - parameters to be passed to the inner model and sampling state details.
|
2022-10-30 23:45:52 +00:00
|
|
|
"""
|
2022-11-04 12:19:16 +00:00
|
|
|
add_callback(_callback_map['callbacks_cfg_denoiser'], callback)
|