scripts.py 21.0 KB
Newer Older
1
import os
2
import re
3 4
import sys
import traceback
5
from collections import namedtuple
6 7 8

import gradio as gr

A
AUTOMATIC 已提交
9
from modules import shared, paths, script_callbacks, extensions, script_loading, scripts_postprocessing, timer
10 11 12

AlwaysVisible = object()

A
AUTOMATIC 已提交
13

14 15 16 17 18
class PostprocessImageArgs:
    def __init__(self, image):
        self.image = image


19
class Script:
A
AUTOMATIC 已提交
20 21 22
    name = None
    """script's internal name derived from title"""

23
    filename = None
A
AUTOMATIC 已提交
24 25
    args_from = None
    args_to = None
26 27
    alwayson = False

28 29 30
    is_txt2img = False
    is_img2img = False

31
    group = None
A
AUTOMATIC 已提交
32
    """A gr.Group component that has all script's UI inside it"""
33

34 35 36 37
    infotext_fields = None
    """if set in ui(), this is a list of pairs of gradio component + text; the text will be used when
    parsing infotext to set the value for the component; see ui.py's txt2img_paste_fields for an example
    """
38

39 40 41 42 43
    paste_field_names = None
    """if set in ui(), this is a list of names of infotext fields; the fields will be sent through the
    various "Send to <X>" buttons when clicked
    """

A
AUTOMATIC 已提交
44 45 46
    api_info = None
    """Generated value of type modules.api.models.ScriptInfo with information about the script for API"""

47
    def title(self):
48 49
        """this function should return the title of the script. This is what will be displayed in the dropdown menu."""

50 51
        raise NotImplementedError()

A
AUTOMATIC 已提交
52
    def ui(self, is_img2img):
53 54
        """this function should create gradio UI elements. See https://gradio.app/docs/#components
        The return value should be an array of all components that are used in processing.
J
Jim Hays 已提交
55
        Values of those returned components will be passed to run() and process() functions.
56 57
        """

A
AUTOMATIC 已提交
58 59
        pass

A
AUTOMATIC 已提交
60
    def show(self, is_img2img):
61 62 63 64 65
        """
        is_img2img is True if this function is called for the img2img interface, and Fasle otherwise

        This function should return:
         - False if the script should not be shown in UI at all
J
Jim Hays 已提交
66
         - True if the script should be shown in UI if it's selected in the scripts dropdown
67 68 69
         - script.AlwaysVisible if the script should be shown in UI at all times
         """

A
AUTOMATIC 已提交
70 71
        return True

72 73 74 75 76 77 78 79 80 81 82
    def run(self, p, *args):
        """
        This function is called if the script has been selected in the script dropdown.
        It must do all processing and return the Processed object with results, same as
        one returned by processing.process_images.

        Usually the processing is done by calling the processing.process_images function.

        args contains all values returned by components from ui()
        """

83
        pass
A
AUTOMATIC 已提交
84

85 86 87
    def process(self, p, *args):
        """
        This function is called before processing begins for AlwaysVisible scripts.
A
AUTOMATIC 已提交
88 89 90 91 92 93
        You can modify the processing object (p) here, inject hooks, etc.
        args contains all values returned by components from ui()
        """

        pass

94 95 96 97 98 99 100 101 102 103 104 105 106 107
    def before_process_batch(self, p, *args, **kwargs):
        """
        Called before extra networks are parsed from the prompt, so you can add
        new extra network keywords to the prompt with this callback.

        **kwargs will have those items:
          - batch_number - index of current batch, from 0 to number of batches-1
          - prompts - list of prompts for current batch; you can change contents of this list but changing the number of entries will likely break things
          - seeds - list of seeds for current batch
          - subseeds - list of subseeds for current batch
        """

        pass

108
    def process_batch(self, p, *args, **kwargs):
A
Artem Zagidulin 已提交
109
        """
110 111 112 113 114 115 116
        Same as process(), but called for every batch.

        **kwargs will have those items:
          - batch_number - index of current batch, from 0 to number of batches-1
          - prompts - list of prompts for current batch; you can change contents of this list but changing the number of entries will likely break things
          - seeds - list of seeds for current batch
          - subseeds - list of subseeds for current batch
A
Artem Zagidulin 已提交
117 118 119 120
        """

        pass

121 122 123 124 125 126 127 128 129 130 131
    def postprocess_batch(self, p, *args, **kwargs):
        """
        Same as process_batch(), but called for every batch after it has been generated.

        **kwargs will have same items as process_batch, and also:
          - batch_number - index of current batch, from 0 to number of batches-1
          - images - torch tensor with all generated images, with values ranging from 0 to 1;
        """

        pass

132 133 134 135 136 137 138
    def postprocess_image(self, p, pp: PostprocessImageArgs, *args):
        """
        Called for every image after it has been generated.
        """

        pass

A
AUTOMATIC 已提交
139 140 141 142
    def postprocess(self, p, processed, *args):
        """
        This function is called after processing ends for AlwaysVisible scripts.
        args contains all values returned by components from ui()
143 144 145 146
        """

        pass

147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163
    def before_component(self, component, **kwargs):
        """
        Called before a component is created.
        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.
        You can return created components in the ui() function to add them to the list of arguments for your processing functions
        """

        pass

    def after_component(self, component, **kwargs):
        """
        Called after a component is created. Same as above.
        """

        pass

A
AUTOMATIC 已提交
164
    def describe(self):
165
        """unused"""
A
AUTOMATIC 已提交
166 167
        return ""

168 169 170 171
    def elem_id(self, item_id):
        """helper function to generate id for a HTML element, constructs final id out of script name, tab and user-supplied item_id"""

        need_tabname = self.show(True) == self.show(False)
172 173
        tabkind = 'img2img' if self.is_img2img else 'txt2txt'
        tabname = f"{tabkind}_" if need_tabname else ""
174 175 176 177
        title = re.sub(r'[^a-z_0-9]', '', re.sub(r'\s', '_', self.title().lower()))

        return f'script_{tabname}{title}_{item_id}'

178

179 180 181 182 183 184 185 186 187 188 189 190
current_basedir = paths.script_path


def basedir():
    """returns the base directory for the current script. For scripts in the main scripts directory,
    this is the main directory (where webui.py resides), and for scripts in extensions directory
    (ie extensions/aesthetic/script/aesthetic.py), this is extension's directory (extensions/aesthetic)
    """
    return current_basedir


ScriptFile = namedtuple("ScriptFile", ["basedir", "filename", "path"])
191 192 193

scripts_data = []
postprocessing_scripts_data = []
194
ScriptClassData = namedtuple("ScriptClassData", ["script_class", "path", "basedir", "module"])
195 196 197 198 199 200 201 202 203 204


def list_scripts(scriptdirname, extension):
    scripts_list = []

    basedir = os.path.join(paths.script_path, scriptdirname)
    if os.path.exists(basedir):
        for filename in sorted(os.listdir(basedir)):
            scripts_list.append(ScriptFile(paths.script_path, filename, os.path.join(basedir, filename)))

205 206
    for ext in extensions.active():
        scripts_list += ext.list_files(scriptdirname, extension)
207

208
    scripts_list = [x for x in scripts_list if os.path.splitext(x.path)[1].lower() == extension and os.path.isfile(x.path)]
209

210
    return scripts_list
211

212

A
AUTOMATIC 已提交
213 214 215
def list_files_with_name(filename):
    res = []

216
    dirs = [paths.script_path] + [ext.path for ext in extensions.active()]
A
AUTOMATIC 已提交
217 218 219 220 221 222

    for dirpath in dirs:
        if not os.path.isdir(dirpath):
            continue

        path = os.path.join(dirpath, filename)
T
Tong Zeng 已提交
223
        if os.path.isfile(path):
A
AUTOMATIC 已提交
224 225 226 227 228
            res.append(path)

    return res


229 230 231
def load_scripts():
    global current_basedir
    scripts_data.clear()
232
    postprocessing_scripts_data.clear()
233 234 235 236 237
    script_callbacks.clear_callbacks()

    scripts_list = list_scripts("scripts", ".py")

    syspath = sys.path
238

239
    def register_scripts_from_module(module):
A
AUTOMATIC 已提交
240
        for script_class in module.__dict__.values():
241 242 243 244 245 246 247 248
            if type(script_class) != type:
                continue

            if issubclass(script_class, Script):
                scripts_data.append(ScriptClassData(script_class, scriptfile.path, scriptfile.basedir, module))
            elif issubclass(script_class, scripts_postprocessing.ScriptPostprocessing):
                postprocessing_scripts_data.append(ScriptClassData(script_class, scriptfile.path, scriptfile.basedir, module))

S
sumof2primes 已提交
249 250 251 252 253 254 255 256
    def orderby(basedir):
        # 1st webui, 2nd extensions-builtin, 3rd extensions
        priority = {os.path.join(paths.script_path, "extensions-builtin"):1, paths.script_path:0}
        for key in priority:
            if basedir.startswith(key):
                return priority[key]
        return 9999

S
sumof2primes 已提交
257
    for scriptfile in sorted(scripts_list, key=lambda x: [orderby(x.basedir), x]):
258
        try:
259 260 261 262
            if scriptfile.basedir != paths.script_path:
                sys.path = [scriptfile.basedir] + sys.path
            current_basedir = scriptfile.basedir

263 264
            script_module = script_loading.load_module(scriptfile.path)
            register_scripts_from_module(script_module)
265 266

        except Exception:
267
            print(f"Error loading script: {scriptfile.filename}", file=sys.stderr)
268
            print(traceback.format_exc(), file=sys.stderr)
269

270 271 272
        finally:
            sys.path = syspath
            current_basedir = paths.script_path
A
AUTOMATIC 已提交
273
            timer.startup_timer.record(scriptfile.filename)
274

275 276 277 278 279 280
    global scripts_txt2img, scripts_img2img, scripts_postproc

    scripts_txt2img = ScriptRunner()
    scripts_img2img = ScriptRunner()
    scripts_postproc = scripts_postprocessing.ScriptPostprocessingRunner()

281 282 283

def wrap_call(func, filename, funcname, *args, default=None, **kwargs):
    try:
A
AUTOMATIC 已提交
284
        res = func(*args, **kwargs)
285 286
        return res
    except Exception:
A
AUTOMATIC 已提交
287
        print(f"Error calling: {filename}/{funcname}", file=sys.stderr)
288 289 290 291 292
        print(traceback.format_exc(), file=sys.stderr)

    return default


A
AUTOMATIC 已提交
293 294 295
class ScriptRunner:
    def __init__(self):
        self.scripts = []
296 297
        self.selectable_scripts = []
        self.alwayson_scripts = []
ふぁ 已提交
298
        self.titles = []
299
        self.infotext_fields = []
300
        self.paste_field_names = []
A
AUTOMATIC 已提交
301

302
    def initialize_scripts(self, is_img2img):
303 304
        from modules import scripts_auto_postprocessing

305 306 307 308
        self.scripts.clear()
        self.alwayson_scripts.clear()
        self.selectable_scripts.clear()

309 310
        auto_processing_scripts = scripts_auto_postprocessing.create_auto_preprocessing_script_data()

A
AUTOMATIC 已提交
311 312 313
        for script_data in auto_processing_scripts + scripts_data:
            script = script_data.script_class()
            script.filename = script_data.path
314 315
            script.is_txt2img = not is_img2img
            script.is_img2img = is_img2img
A
AUTOMATIC 已提交
316

317
            visibility = script.show(script.is_img2img)
A
AUTOMATIC 已提交
318

319 320 321 322
            if visibility == AlwaysVisible:
                self.scripts.append(script)
                self.alwayson_scripts.append(script)
                script.alwayson = True
A
AUTOMATIC 已提交
323

324 325 326
            elif visibility:
                self.scripts.append(script)
                self.selectable_scripts.append(script)
A
AUTOMATIC 已提交
327

328
    def setup_ui(self):
A
AUTOMATIC 已提交
329 330
        import modules.api.models as api_models

331 332 333 334
        self.titles = [wrap_call(script.title, script.filename, "title") or f"{script.filename} [error]" for script in self.selectable_scripts]

        inputs = [None]
        inputs_alwayson = [True]
A
AUTOMATIC 已提交
335

336
        def create_script_ui(script, inputs, inputs_alwayson):
A
AUTOMATIC 已提交
337
            script.args_from = len(inputs)
O
OWKenobi 已提交
338
            script.args_to = len(inputs)
A
AUTOMATIC 已提交
339

340
            controls = wrap_call(script.ui, script.filename, "ui", script.is_img2img)
A
AUTOMATIC 已提交
341 342

            if controls is None:
343
                return
A
AUTOMATIC 已提交
344

A
AUTOMATIC 已提交
345 346 347
            script.name = wrap_call(script.title, script.filename, "title", default=script.filename).lower()
            api_args = []

A
AUTOMATIC 已提交
348
            for control in controls:
D
DepFA 已提交
349
                control.custom_script_source = os.path.basename(script.filename)
350

A
AUTOMATIC 已提交
351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366
                arg_info = api_models.ScriptArg(label=control.label or "")

                for field in ("value", "minimum", "maximum", "step", "choices"):
                    v = getattr(control, field, None)
                    if v is not None:
                        setattr(arg_info, field, v)

                api_args.append(arg_info)

            script.api_info = api_models.ScriptInfo(
                name=script.name,
                is_img2img=script.is_img2img,
                is_alwayson=script.alwayson,
                args=api_args,
            )

367 368
            if script.infotext_fields is not None:
                self.infotext_fields += script.infotext_fields
A
AUTOMATIC 已提交
369

370 371 372
            if script.paste_field_names is not None:
                self.paste_field_names += script.paste_field_names

A
AUTOMATIC 已提交
373
            inputs += controls
374
            inputs_alwayson += [script.alwayson for _ in controls]
A
AUTOMATIC 已提交
375
            script.args_to = len(inputs)
A
AUTOMATIC 已提交
376

377
        for script in self.alwayson_scripts:
378
            with gr.Group() as group:
379 380
                create_script_ui(script, inputs, inputs_alwayson)

381 382
            script.group = group

X
xmodar 已提交
383
        dropdown = gr.Dropdown(label="Script", elem_id="script_list", choices=["None"] + self.titles, value="None", type="index")
384 385 386
        inputs[0] = dropdown

        for script in self.selectable_scripts:
387 388 389 390
            with gr.Group(visible=False) as group:
                create_script_ui(script, inputs, inputs_alwayson)

            script.group = group
391

A
AUTOMATIC 已提交
392
        def select_script(script_index):
393
            selected_script = self.selectable_scripts[script_index - 1] if script_index>0 else None
A
AUTOMATIC 已提交
394

395
            return [gr.update(visible=selected_script == s) for s in self.selectable_scripts]
A
AUTOMATIC 已提交
396

ふぁ 已提交
397
        def init_field(title):
398 399
            """called when an initial value is set from ui-config.json to show script's UI components"""

ふぁ 已提交
400
            if title == 'None':
ふぁ 已提交
401
                return
402

ふぁ 已提交
403
            script_index = self.titles.index(title)
404
            self.selectable_scripts[script_index].group.visible = True
ふぁ 已提交
405 406

        dropdown.init_field = init_field
407

A
AUTOMATIC 已提交
408 409 410
        dropdown.change(
            fn=select_script,
            inputs=[dropdown],
411
            outputs=[script.group for script in self.selectable_scripts]
A
AUTOMATIC 已提交
412
        )
A
AUTOMATIC 已提交
413

E
EllangoK 已提交
414 415 416 417 418 419 420 421 422 423 424 425 426 427
        self.script_load_ctr = 0
        def onload_script_visibility(params):
            title = params.get('Script', None)
            if title:
                title_index = self.titles.index(title)
                visibility = title_index == self.script_load_ctr
                self.script_load_ctr = (self.script_load_ctr + 1) % len(self.titles)
                return gr.update(visible=visibility)
            else:
                return gr.update(visible=False)

        self.infotext_fields.append( (dropdown, lambda x: gr.update(value=x.get('Script', 'None'))) )
        self.infotext_fields.extend( [(script.group, onload_script_visibility) for script in self.selectable_scripts] )

A
AUTOMATIC 已提交
428
        return inputs
A
AUTOMATIC 已提交
429

430
    def run(self, p, *args):
A
AUTOMATIC 已提交
431
        script_index = args[0]
A
AUTOMATIC 已提交
432

A
AUTOMATIC 已提交
433 434
        if script_index == 0:
            return None
A
AUTOMATIC 已提交
435

436
        script = self.selectable_scripts[script_index-1]
A
AUTOMATIC 已提交
437

A
AUTOMATIC 已提交
438 439
        if script is None:
            return None
A
AUTOMATIC 已提交
440

A
AUTOMATIC 已提交
441 442
        script_args = args[script.args_from:script.args_to]
        processed = script.run(p, *script_args)
A
AUTOMATIC 已提交
443

444 445
        shared.total_tqdm.clear()

A
AUTOMATIC 已提交
446
        return processed
A
AUTOMATIC 已提交
447

A
AUTOMATIC 已提交
448
    def process(self, p):
449 450 451 452 453
        for script in self.alwayson_scripts:
            try:
                script_args = p.script_args[script.args_from:script.args_to]
                script.process(p, *script_args)
            except Exception:
A
AUTOMATIC 已提交
454 455 456
                print(f"Error running process: {script.filename}", file=sys.stderr)
                print(traceback.format_exc(), file=sys.stderr)

457 458 459 460 461 462 463 464 465
    def before_process_batch(self, p, **kwargs):
        for script in self.alwayson_scripts:
            try:
                script_args = p.script_args[script.args_from:script.args_to]
                script.before_process_batch(p, *script_args, **kwargs)
            except Exception:
                print(f"Error running before_process_batch: {script.filename}", file=sys.stderr)
                print(traceback.format_exc(), file=sys.stderr)

466
    def process_batch(self, p, **kwargs):
A
Artem Zagidulin 已提交
467 468 469
        for script in self.alwayson_scripts:
            try:
                script_args = p.script_args[script.args_from:script.args_to]
470
                script.process_batch(p, *script_args, **kwargs)
A
Artem Zagidulin 已提交
471
            except Exception:
472
                print(f"Error running process_batch: {script.filename}", file=sys.stderr)
A
Artem Zagidulin 已提交
473 474
                print(traceback.format_exc(), file=sys.stderr)

A
AUTOMATIC 已提交
475 476 477 478 479 480 481
    def postprocess(self, p, processed):
        for script in self.alwayson_scripts:
            try:
                script_args = p.script_args[script.args_from:script.args_to]
                script.postprocess(p, processed, *script_args)
            except Exception:
                print(f"Error running postprocess: {script.filename}", file=sys.stderr)
482 483
                print(traceback.format_exc(), file=sys.stderr)

484 485 486 487 488 489 490 491 492
    def postprocess_batch(self, p, images, **kwargs):
        for script in self.alwayson_scripts:
            try:
                script_args = p.script_args[script.args_from:script.args_to]
                script.postprocess_batch(p, *script_args, images=images, **kwargs)
            except Exception:
                print(f"Error running postprocess_batch: {script.filename}", file=sys.stderr)
                print(traceback.format_exc(), file=sys.stderr)

493 494 495 496 497 498 499 500 501
    def postprocess_image(self, p, pp: PostprocessImageArgs):
        for script in self.alwayson_scripts:
            try:
                script_args = p.script_args[script.args_from:script.args_to]
                script.postprocess_image(p, pp, *script_args)
            except Exception:
                print(f"Error running postprocess_batch: {script.filename}", file=sys.stderr)
                print(traceback.format_exc(), file=sys.stderr)

502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517
    def before_component(self, component, **kwargs):
        for script in self.scripts:
            try:
                script.before_component(component, **kwargs)
            except Exception:
                print(f"Error running before_component: {script.filename}", file=sys.stderr)
                print(traceback.format_exc(), file=sys.stderr)

    def after_component(self, component, **kwargs):
        for script in self.scripts:
            try:
                script.after_component(component, **kwargs)
            except Exception:
                print(f"Error running after_component: {script.filename}", file=sys.stderr)
                print(traceback.format_exc(), file=sys.stderr)

A
AUTOMATIC 已提交
518
    def reload_sources(self, cache):
D
DepFA 已提交
519
        for si, script in list(enumerate(self.scripts)):
520 521 522 523 524 525 526 527 528
            args_from = script.args_from
            args_to = script.args_to
            filename = script.filename

            module = cache.get(filename, None)
            if module is None:
                module = script_loading.load_module(script.filename)
                cache[filename] = module

A
AUTOMATIC 已提交
529
            for script_class in module.__dict__.values():
530 531 532 533 534
                if type(script_class) == type and issubclass(script_class, Script):
                    self.scripts[si] = script_class()
                    self.scripts[si].filename = filename
                    self.scripts[si].args_from = args_from
                    self.scripts[si].args_to = args_to
A
AUTOMATIC 已提交
535

536

537 538 539
scripts_txt2img: ScriptRunner = None
scripts_img2img: ScriptRunner = None
scripts_postproc: scripts_postprocessing.ScriptPostprocessingRunner = None
540
scripts_current: ScriptRunner = None
D
DepFA 已提交
541

542

D
DepFA 已提交
543
def reload_script_body_only():
A
AUTOMATIC 已提交
544 545 546
    cache = {}
    scripts_txt2img.reload_sources(cache)
    scripts_img2img.reload_sources(cache)
D
DepFA 已提交
547

D
DepFA 已提交
548

549
reload_scripts = load_scripts  # compatibility alias
550

551

552 553 554 555 556
def add_classes_to_gradio_component(comp):
    """
    this adds gradio-* to the component for css styling (ie gradio-button to gr.Button), as well as some others
    """

557
    comp.elem_classes = [f"gradio-{comp.get_block_name()}", *(comp.elem_classes or [])]
558 559 560 561 562 563

    if getattr(comp, 'multiselect', False):
        comp.elem_classes.append('multiselect')



564 565 566 567 568 569 570 571
def IOComponent_init(self, *args, **kwargs):
    if scripts_current is not None:
        scripts_current.before_component(self, **kwargs)

    script_callbacks.before_component_callback(self, **kwargs)

    res = original_IOComponent_init(self, *args, **kwargs)

572
    add_classes_to_gradio_component(self)
A
AUTOMATIC 已提交
573

574 575 576 577 578 579 580 581 582 583
    script_callbacks.after_component_callback(self, **kwargs)

    if scripts_current is not None:
        scripts_current.after_component(self, **kwargs)

    return res


original_IOComponent_init = gr.components.IOComponent.__init__
gr.components.IOComponent.__init__ = IOComponent_init
584 585 586 587 588 589 590 591 592 593 594 595


def BlockContext_init(self, *args, **kwargs):
    res = original_BlockContext_init(self, *args, **kwargs)

    add_classes_to_gradio_component(self)

    return res


original_BlockContext_init = gr.blocks.BlockContext.__init__
gr.blocks.BlockContext.__init__ = BlockContext_init