|
import inspect |
|
import os |
|
from collections import namedtuple |
|
from typing import Optional, Dict, Any |
|
|
|
from fastapi import FastAPI |
|
from gradio import Blocks |
|
|
|
from modules import errors, timer |
|
|
|
|
|
def report_exception(c, job): |
|
errors.report(f"Error executing callback {job} for {c.script}", exc_info=True) |
|
|
|
|
|
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'""" |
|
|
|
|
|
class CFGDenoiserParams: |
|
def __init__(self, x, image_cond, sigma, sampling_step, total_sampling_steps, text_cond, text_uncond): |
|
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""" |
|
|
|
self.sampling_step = sampling_step |
|
"""Current Sampling step number""" |
|
|
|
self.total_sampling_steps = total_sampling_steps |
|
"""Total number of sampling steps planned""" |
|
|
|
self.text_cond = text_cond |
|
""" Encoder hidden states of text conditioning from prompt""" |
|
|
|
self.text_uncond = text_uncond |
|
""" Encoder hidden states of text conditioning from negative prompt""" |
|
|
|
|
|
class CFGDenoisedParams: |
|
def __init__(self, x, sampling_step, total_sampling_steps, inner_model): |
|
self.x = x |
|
"""Latent image representation in the process of being denoised""" |
|
|
|
self.sampling_step = sampling_step |
|
"""Current Sampling step number""" |
|
|
|
self.total_sampling_steps = total_sampling_steps |
|
"""Total number of sampling steps planned""" |
|
|
|
self.inner_model = inner_model |
|
"""Inner model reference used for denoising""" |
|
|
|
|
|
class AfterCFGCallbackParams: |
|
def __init__(self, x, sampling_step, total_sampling_steps): |
|
self.x = x |
|
"""Latent image representation in the process of being denoised""" |
|
|
|
self.sampling_step = sampling_step |
|
"""Current Sampling step number""" |
|
|
|
self.total_sampling_steps = total_sampling_steps |
|
"""Total number of sampling steps planned""" |
|
|
|
|
|
class UiTrainTabParams: |
|
def __init__(self, txt2img_preview_params): |
|
self.txt2img_preview_params = txt2img_preview_params |
|
|
|
|
|
class ImageGridLoopParams: |
|
def __init__(self, imgs, cols, rows): |
|
self.imgs = imgs |
|
self.cols = cols |
|
self.rows = rows |
|
|
|
|
|
ScriptCallback = namedtuple("ScriptCallback", ["script", "callback"]) |
|
callback_map = dict( |
|
callbacks_app_started=[], |
|
callbacks_model_loaded=[], |
|
callbacks_ui_tabs=[], |
|
callbacks_ui_train_tabs=[], |
|
callbacks_ui_settings=[], |
|
callbacks_before_image_saved=[], |
|
callbacks_image_saved=[], |
|
callbacks_cfg_denoiser=[], |
|
callbacks_cfg_denoised=[], |
|
callbacks_cfg_after_cfg=[], |
|
callbacks_before_component=[], |
|
callbacks_after_component=[], |
|
callbacks_image_grid=[], |
|
callbacks_infotext_pasted=[], |
|
callbacks_script_unloaded=[], |
|
callbacks_before_ui=[], |
|
callbacks_on_reload=[], |
|
callbacks_list_optimizers=[], |
|
callbacks_list_unets=[], |
|
) |
|
|
|
|
|
def clear_callbacks(): |
|
for callback_list in callback_map.values(): |
|
callback_list.clear() |
|
|
|
|
|
def app_started_callback(demo: Optional[Blocks], app: FastAPI): |
|
for c in callback_map['callbacks_app_started']: |
|
try: |
|
c.callback(demo, app) |
|
timer.startup_timer.record(os.path.basename(c.script)) |
|
except Exception: |
|
report_exception(c, 'app_started_callback') |
|
|
|
|
|
def app_reload_callback(): |
|
for c in callback_map['callbacks_on_reload']: |
|
try: |
|
c.callback() |
|
except Exception: |
|
report_exception(c, 'callbacks_on_reload') |
|
|
|
|
|
def model_loaded_callback(sd_model): |
|
for c in callback_map['callbacks_model_loaded']: |
|
try: |
|
c.callback(sd_model) |
|
except Exception: |
|
report_exception(c, 'model_loaded_callback') |
|
|
|
|
|
def ui_tabs_callback(): |
|
res = [] |
|
|
|
for c in callback_map['callbacks_ui_tabs']: |
|
try: |
|
res += c.callback() or [] |
|
except Exception: |
|
report_exception(c, 'ui_tabs_callback') |
|
|
|
return res |
|
|
|
|
|
def ui_train_tabs_callback(params: UiTrainTabParams): |
|
for c in callback_map['callbacks_ui_train_tabs']: |
|
try: |
|
c.callback(params) |
|
except Exception: |
|
report_exception(c, 'callbacks_ui_train_tabs') |
|
|
|
|
|
def ui_settings_callback(): |
|
for c in callback_map['callbacks_ui_settings']: |
|
try: |
|
c.callback() |
|
except Exception: |
|
report_exception(c, 'ui_settings_callback') |
|
|
|
|
|
def before_image_saved_callback(params: ImageSaveParams): |
|
for c in callback_map['callbacks_before_image_saved']: |
|
try: |
|
c.callback(params) |
|
except Exception: |
|
report_exception(c, 'before_image_saved_callback') |
|
|
|
|
|
def image_saved_callback(params: ImageSaveParams): |
|
for c in callback_map['callbacks_image_saved']: |
|
try: |
|
c.callback(params) |
|
except Exception: |
|
report_exception(c, 'image_saved_callback') |
|
|
|
|
|
def cfg_denoiser_callback(params: CFGDenoiserParams): |
|
for c in callback_map['callbacks_cfg_denoiser']: |
|
try: |
|
c.callback(params) |
|
except Exception: |
|
report_exception(c, 'cfg_denoiser_callback') |
|
|
|
|
|
def cfg_denoised_callback(params: CFGDenoisedParams): |
|
for c in callback_map['callbacks_cfg_denoised']: |
|
try: |
|
c.callback(params) |
|
except Exception: |
|
report_exception(c, 'cfg_denoised_callback') |
|
|
|
|
|
def cfg_after_cfg_callback(params: AfterCFGCallbackParams): |
|
for c in callback_map['callbacks_cfg_after_cfg']: |
|
try: |
|
c.callback(params) |
|
except Exception: |
|
report_exception(c, 'cfg_after_cfg_callback') |
|
|
|
|
|
def before_component_callback(component, **kwargs): |
|
for c in callback_map['callbacks_before_component']: |
|
try: |
|
c.callback(component, **kwargs) |
|
except Exception: |
|
report_exception(c, 'before_component_callback') |
|
|
|
|
|
def after_component_callback(component, **kwargs): |
|
for c in callback_map['callbacks_after_component']: |
|
try: |
|
c.callback(component, **kwargs) |
|
except Exception: |
|
report_exception(c, 'after_component_callback') |
|
|
|
|
|
def image_grid_callback(params: ImageGridLoopParams): |
|
for c in callback_map['callbacks_image_grid']: |
|
try: |
|
c.callback(params) |
|
except Exception: |
|
report_exception(c, 'image_grid') |
|
|
|
|
|
def infotext_pasted_callback(infotext: str, params: Dict[str, Any]): |
|
for c in callback_map['callbacks_infotext_pasted']: |
|
try: |
|
c.callback(infotext, params) |
|
except Exception: |
|
report_exception(c, 'infotext_pasted') |
|
|
|
|
|
def script_unloaded_callback(): |
|
for c in reversed(callback_map['callbacks_script_unloaded']): |
|
try: |
|
c.callback() |
|
except Exception: |
|
report_exception(c, 'script_unloaded') |
|
|
|
|
|
def before_ui_callback(): |
|
for c in reversed(callback_map['callbacks_before_ui']): |
|
try: |
|
c.callback() |
|
except Exception: |
|
report_exception(c, 'before_ui') |
|
|
|
|
|
def list_optimizers_callback(): |
|
res = [] |
|
|
|
for c in callback_map['callbacks_list_optimizers']: |
|
try: |
|
c.callback(res) |
|
except Exception: |
|
report_exception(c, 'list_optimizers') |
|
|
|
return res |
|
|
|
|
|
def list_unets_callback(): |
|
res = [] |
|
|
|
for c in callback_map['callbacks_list_unets']: |
|
try: |
|
c.callback(res) |
|
except Exception: |
|
report_exception(c, 'list_unets') |
|
|
|
return res |
|
|
|
|
|
def add_callback(callbacks, fun): |
|
stack = [x for x in inspect.stack() if x.filename != __file__] |
|
filename = stack[0].filename if stack else 'unknown file' |
|
|
|
callbacks.append(ScriptCallback(filename, fun)) |
|
|
|
|
|
def remove_current_script_callbacks(): |
|
stack = [x for x in inspect.stack() if x.filename != __file__] |
|
filename = stack[0].filename if stack else 'unknown file' |
|
if filename == 'unknown file': |
|
return |
|
for callback_list in callback_map.values(): |
|
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): |
|
for callback_list in callback_map.values(): |
|
for callback_to_remove in [cb for cb in callback_list if cb.callback == callback_func]: |
|
callback_list.remove(callback_to_remove) |
|
|
|
|
|
def on_app_started(callback): |
|
"""register a function to be called when the webui started, the gradio `Block` component and |
|
fastapi `FastAPI` object are passed as the arguments""" |
|
add_callback(callback_map['callbacks_app_started'], callback) |
|
|
|
|
|
def on_before_reload(callback): |
|
"""register a function to be called just before the server reloads.""" |
|
add_callback(callback_map['callbacks_on_reload'], callback) |
|
|
|
|
|
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; this function is also called when the script is reloaded. """ |
|
add_callback(callback_map['callbacks_model_loaded'], callback) |
|
|
|
|
|
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 |
|
""" |
|
add_callback(callback_map['callbacks_ui_tabs'], callback) |
|
|
|
|
|
def on_ui_train_tabs(callback): |
|
"""register a function to be called when the UI is creating new tabs for the train tab. |
|
Create your new tabs with gr.Tab. |
|
""" |
|
add_callback(callback_map['callbacks_ui_train_tabs'], callback) |
|
|
|
|
|
def on_ui_settings(callback): |
|
"""register a function to be called before UI settings are populated; add your settings |
|
by using shared.opts.add_option(shared.OptionInfo(...)) """ |
|
add_callback(callback_map['callbacks_ui_settings'], callback) |
|
|
|
|
|
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. |
|
""" |
|
add_callback(callback_map['callbacks_before_image_saved'], callback) |
|
|
|
|
|
def on_image_saved(callback): |
|
"""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. |
|
""" |
|
add_callback(callback_map['callbacks_image_saved'], callback) |
|
|
|
|
|
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: |
|
- params: CFGDenoiserParams - parameters to be passed to the inner model and sampling state details. |
|
""" |
|
add_callback(callback_map['callbacks_cfg_denoiser'], callback) |
|
|
|
|
|
def on_cfg_denoised(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: |
|
- params: CFGDenoisedParams - parameters to be passed to the inner model and sampling state details. |
|
""" |
|
add_callback(callback_map['callbacks_cfg_denoised'], callback) |
|
|
|
|
|
def on_cfg_after_cfg(callback): |
|
"""register a function to be called in the kdiffussion cfg_denoiser method after cfg calculations are completed. |
|
The callback is called with one argument: |
|
- params: AfterCFGCallbackParams - parameters to be passed to the script for post-processing after cfg calculation. |
|
""" |
|
add_callback(callback_map['callbacks_cfg_after_cfg'], callback) |
|
|
|
|
|
def on_before_component(callback): |
|
"""register a function to be called before a component is created. |
|
The callback is called with arguments: |
|
- component - gradio component that is about to be created. |
|
- **kwargs - args to gradio.components.IOComponent.__init__ function |
|
|
|
Use elem_id/label fields of kwargs to figure out which component it is. |
|
This can be useful to inject your own components somewhere in the middle of vanilla UI. |
|
""" |
|
add_callback(callback_map['callbacks_before_component'], callback) |
|
|
|
|
|
def on_after_component(callback): |
|
"""register a function to be called after a component is created. See on_before_component for more.""" |
|
add_callback(callback_map['callbacks_after_component'], callback) |
|
|
|
|
|
def on_image_grid(callback): |
|
"""register a function to be called before making an image grid. |
|
The callback is called with one argument: |
|
- params: ImageGridLoopParams - parameters to be used for grid creation. Can be modified. |
|
""" |
|
add_callback(callback_map['callbacks_image_grid'], callback) |
|
|
|
|
|
def on_infotext_pasted(callback): |
|
"""register a function to be called before applying an infotext. |
|
The callback is called with two arguments: |
|
- infotext: str - raw infotext. |
|
- result: Dict[str, any] - parsed infotext parameters. |
|
""" |
|
add_callback(callback_map['callbacks_infotext_pasted'], callback) |
|
|
|
|
|
def on_script_unloaded(callback): |
|
"""register a function to be called before the script is unloaded. Any hooks/hijacks/monkeying about that |
|
the script did should be reverted here""" |
|
|
|
add_callback(callback_map['callbacks_script_unloaded'], callback) |
|
|
|
|
|
def on_before_ui(callback): |
|
"""register a function to be called before the UI is created.""" |
|
|
|
add_callback(callback_map['callbacks_before_ui'], callback) |
|
|
|
|
|
def on_list_optimizers(callback): |
|
"""register a function to be called when UI is making a list of cross attention optimization options. |
|
The function will be called with one argument, a list, and shall add objects of type modules.sd_hijack_optimizations.SdOptimization |
|
to it.""" |
|
|
|
add_callback(callback_map['callbacks_list_optimizers'], callback) |
|
|
|
|
|
def on_list_unets(callback): |
|
"""register a function to be called when UI is making a list of alternative options for unet. |
|
The function will be called with one argument, a list, and shall add objects of type modules.sd_unet.SdUnetOption to it.""" |
|
|
|
add_callback(callback_map['callbacks_list_unets'], callback) |
|
|