scripts.py 22.9 KB
Newer Older
1
import os
2
import re
3
import sys
H
huchenlei 已提交
4
import inspect
5
from collections import namedtuple
6 7 8

import gradio as gr

9
from modules import shared, paths, script_callbacks, extensions, script_loading, scripts_postprocessing, errors, 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 24 25
    section = None
    """name of UI section that the script's controls will be placed into"""

26
    filename = None
A
AUTOMATIC 已提交
27 28
    args_from = None
    args_to = None
29 30
    alwayson = False

31 32 33
    is_txt2img = False
    is_img2img = False

34
    group = None
A
AUTOMATIC 已提交
35
    """A gr.Group component that has all script's UI inside it"""
36

37 38 39 40
    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
    """
41

42 43 44 45 46
    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 已提交
47 48 49
    api_info = None
    """Generated value of type modules.api.models.ScriptInfo with information about the script for API"""

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

53 54
        raise NotImplementedError()

A
AUTOMATIC 已提交
55
    def ui(self, is_img2img):
56 57
        """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 已提交
58
        Values of those returned components will be passed to run() and process() functions.
59 60
        """

A
AUTOMATIC 已提交
61 62
        pass

A
AUTOMATIC 已提交
63
    def show(self, is_img2img):
64 65 66 67 68
        """
        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 已提交
69
         - True if the script should be shown in UI if it's selected in the scripts dropdown
70 71 72
         - script.AlwaysVisible if the script should be shown in UI at all times
         """

A
AUTOMATIC 已提交
73 74
        return True

75 76 77 78 79 80 81 82 83 84 85
    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()
        """

86
        pass
A
AUTOMATIC 已提交
87

88 89 90 91 92 93 94 95 96
    def before_process(self, p, *args):
        """
        This function is called very early before processing begins for AlwaysVisible scripts.
        You can modify the processing object (p) here, inject hooks, etc.
        args contains all values returned by components from ui()
        """

        pass

97 98 99
    def process(self, p, *args):
        """
        This function is called before processing begins for AlwaysVisible scripts.
A
AUTOMATIC 已提交
100 101 102 103 104 105
        You can modify the processing object (p) here, inject hooks, etc.
        args contains all values returned by components from ui()
        """

        pass

106 107 108 109 110 111 112 113 114 115 116 117 118 119
    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

120 121
    def after_extra_networks_activate(self, p, *args, **kwargs):
        """
122
        Called after extra networks activation, before conds calculation
123 124 125 126 127 128 129 130 131 132 133 134
        allow modification of the network after extra networks activation been applied
        won't be call if p.disable_extra_networks

        **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
          - extra_network_data - list of ExtraNetworkParams for current stage
        """
        pass

135
    def process_batch(self, p, *args, **kwargs):
A
Artem Zagidulin 已提交
136
        """
137 138 139 140 141 142 143
        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 已提交
144 145 146 147
        """

        pass

148 149 150 151 152 153 154 155 156 157 158
    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

159 160 161 162 163 164 165
    def postprocess_image(self, p, pp: PostprocessImageArgs, *args):
        """
        Called for every image after it has been generated.
        """

        pass

A
AUTOMATIC 已提交
166 167 168 169
    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()
170 171 172 173
        """

        pass

174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190
    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 已提交
191
    def describe(self):
192
        """unused"""
A
AUTOMATIC 已提交
193 194
        return ""

195 196 197 198
    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)
199 200
        tabkind = 'img2img' if self.is_img2img else 'txt2txt'
        tabname = f"{tabkind}_" if need_tabname else ""
201 202 203 204
        title = re.sub(r'[^a-z_0-9]', '', re.sub(r'\s', '_', self.title().lower()))

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

A
AUTOMATIC1111 已提交
205
    def before_hr(self, p, *args):
206 207 208 209
        """
        This function is called before hires fix start.
        """
        pass
210

211 212 213 214 215 216 217 218 219 220 221 222
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"])
223 224 225

scripts_data = []
postprocessing_scripts_data = []
226
ScriptClassData = namedtuple("ScriptClassData", ["script_class", "path", "basedir", "module"])
227 228 229 230 231 232 233 234 235 236


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)))

237 238
    for ext in extensions.active():
        scripts_list += ext.list_files(scriptdirname, extension)
239

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

242
    return scripts_list
243

244

A
AUTOMATIC 已提交
245 246 247
def list_files_with_name(filename):
    res = []

248
    dirs = [paths.script_path] + [ext.path for ext in extensions.active()]
A
AUTOMATIC 已提交
249 250 251 252 253 254

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

        path = os.path.join(dirpath, filename)
T
Tong Zeng 已提交
255
        if os.path.isfile(path):
A
AUTOMATIC 已提交
256 257 258 259 260
            res.append(path)

    return res


261 262 263
def load_scripts():
    global current_basedir
    scripts_data.clear()
264
    postprocessing_scripts_data.clear()
265 266 267 268 269
    script_callbacks.clear_callbacks()

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

    syspath = sys.path
270

271
    def register_scripts_from_module(module):
A
AUTOMATIC 已提交
272
        for script_class in module.__dict__.values():
H
huchenlei 已提交
273
            if not inspect.isclass(script_class):
274 275 276 277 278 279 280
                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 已提交
281 282 283 284 285 286 287 288
    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 已提交
289
    for scriptfile in sorted(scripts_list, key=lambda x: [orderby(x.basedir), x]):
290
        try:
291 292 293 294
            if scriptfile.basedir != paths.script_path:
                sys.path = [scriptfile.basedir] + sys.path
            current_basedir = scriptfile.basedir

295 296
            script_module = script_loading.load_module(scriptfile.path)
            register_scripts_from_module(script_module)
297 298

        except Exception:
299
            errors.report(f"Error loading script: {scriptfile.filename}", exc_info=True)
300

301 302 303
        finally:
            sys.path = syspath
            current_basedir = paths.script_path
A
AUTOMATIC 已提交
304
            timer.startup_timer.record(scriptfile.filename)
305

306 307 308 309 310 311
    global scripts_txt2img, scripts_img2img, scripts_postproc

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

312 313 314

def wrap_call(func, filename, funcname, *args, default=None, **kwargs):
    try:
315
        return func(*args, **kwargs)
316
    except Exception:
317
        errors.report(f"Error calling: {filename}/{funcname}", exc_info=True)
318 319 320 321

    return default


A
AUTOMATIC 已提交
322 323 324
class ScriptRunner:
    def __init__(self):
        self.scripts = []
325 326
        self.selectable_scripts = []
        self.alwayson_scripts = []
ふぁ 已提交
327
        self.titles = []
328
        self.infotext_fields = []
329
        self.paste_field_names = []
330
        self.inputs = [None]
A
AUTOMATIC 已提交
331

332
    def initialize_scripts(self, is_img2img):
333 334
        from modules import scripts_auto_postprocessing

335 336 337 338
        self.scripts.clear()
        self.alwayson_scripts.clear()
        self.selectable_scripts.clear()

339 340
        auto_processing_scripts = scripts_auto_postprocessing.create_auto_preprocessing_script_data()

A
AUTOMATIC 已提交
341 342 343
        for script_data in auto_processing_scripts + scripts_data:
            script = script_data.script_class()
            script.filename = script_data.path
344 345
            script.is_txt2img = not is_img2img
            script.is_img2img = is_img2img
A
AUTOMATIC 已提交
346

347
            visibility = script.show(script.is_img2img)
A
AUTOMATIC 已提交
348

349 350 351 352
            if visibility == AlwaysVisible:
                self.scripts.append(script)
                self.alwayson_scripts.append(script)
                script.alwayson = True
A
AUTOMATIC 已提交
353

354 355 356
            elif visibility:
                self.scripts.append(script)
                self.selectable_scripts.append(script)
A
AUTOMATIC 已提交
357

358
    def create_script_ui(self, script):
A
AUTOMATIC 已提交
359 360
        import modules.api.models as api_models

361 362
        script.args_from = len(self.inputs)
        script.args_to = len(self.inputs)
363

364
        controls = wrap_call(script.ui, script.filename, "ui", script.is_img2img)
A
AUTOMATIC 已提交
365

366 367
        if controls is None:
            return
A
AUTOMATIC 已提交
368

369 370
        script.name = wrap_call(script.title, script.filename, "title", default=script.filename).lower()
        api_args = []
A
AUTOMATIC 已提交
371

372 373
        for control in controls:
            control.custom_script_source = os.path.basename(script.filename)
A
AUTOMATIC 已提交
374

375
            arg_info = api_models.ScriptArg(label=control.label or "")
A
AUTOMATIC 已提交
376

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

382
            api_args.append(arg_info)
A
AUTOMATIC 已提交
383

384 385 386 387 388 389
        script.api_info = api_models.ScriptInfo(
            name=script.name,
            is_img2img=script.is_img2img,
            is_alwayson=script.alwayson,
            args=api_args,
        )
A
AUTOMATIC 已提交
390

391 392
        if script.infotext_fields is not None:
            self.infotext_fields += script.infotext_fields
A
AUTOMATIC 已提交
393

394 395
        if script.paste_field_names is not None:
            self.paste_field_names += script.paste_field_names
A
AUTOMATIC 已提交
396

397 398
        self.inputs += controls
        script.args_to = len(self.inputs)
A
AUTOMATIC 已提交
399

400 401 402
    def setup_ui_for_section(self, section, scriptlist=None):
        if scriptlist is None:
            scriptlist = self.alwayson_scripts
403

404 405 406
        for script in scriptlist:
            if script.alwayson and script.section != section:
                continue
A
AUTOMATIC 已提交
407

408 409
            with gr.Group(visible=script.alwayson) as group:
                self.create_script_ui(script)
410

411 412
            script.group = group

413 414
    def prepare_ui(self):
        self.inputs = [None]
415

416 417
    def setup_ui(self):
        self.titles = [wrap_call(script.title, script.filename, "title") or f"{script.filename} [error]" for script in self.selectable_scripts]
418

419 420 421 422 423 424
        self.setup_ui_for_section(None)

        dropdown = gr.Dropdown(label="Script", elem_id="script_list", choices=["None"] + self.titles, value="None", type="index")
        self.inputs[0] = dropdown

        self.setup_ui_for_section(None, self.selectable_scripts)
425

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

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

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

ふぁ 已提交
434
            if title == 'None':
ふぁ 已提交
435
                return
436

ふぁ 已提交
437
            script_index = self.titles.index(title)
438
            self.selectable_scripts[script_index].group.visible = True
ふぁ 已提交
439 440

        dropdown.init_field = init_field
441

A
AUTOMATIC 已提交
442 443 444
        dropdown.change(
            fn=select_script,
            inputs=[dropdown],
445
            outputs=[script.group for script in self.selectable_scripts]
A
AUTOMATIC 已提交
446
        )
A
AUTOMATIC 已提交
447

E
EllangoK 已提交
448
        self.script_load_ctr = 0
449

E
EllangoK 已提交
450 451 452 453 454 455 456 457 458 459
        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)

460 461
        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])
E
EllangoK 已提交
462

463
        return self.inputs
A
AUTOMATIC 已提交
464

465
    def run(self, p, *args):
A
AUTOMATIC 已提交
466
        script_index = args[0]
A
AUTOMATIC 已提交
467

A
AUTOMATIC 已提交
468 469
        if script_index == 0:
            return None
A
AUTOMATIC 已提交
470

471
        script = self.selectable_scripts[script_index-1]
A
AUTOMATIC 已提交
472

A
AUTOMATIC 已提交
473 474
        if script is None:
            return None
A
AUTOMATIC 已提交
475

A
AUTOMATIC 已提交
476 477
        script_args = args[script.args_from:script.args_to]
        processed = script.run(p, *script_args)
A
AUTOMATIC 已提交
478

479 480
        shared.total_tqdm.clear()

A
AUTOMATIC 已提交
481
        return processed
A
AUTOMATIC 已提交
482

483 484 485 486 487 488 489 490
    def before_process(self, p):
        for script in self.alwayson_scripts:
            try:
                script_args = p.script_args[script.args_from:script.args_to]
                script.before_process(p, *script_args)
            except Exception:
                errors.report(f"Error running before_process: {script.filename}", exc_info=True)

A
AUTOMATIC 已提交
491
    def process(self, p):
492 493 494 495 496
        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:
497
                errors.report(f"Error running process: {script.filename}", exc_info=True)
A
AUTOMATIC 已提交
498

499 500 501 502 503 504
    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:
505
                errors.report(f"Error running before_process_batch: {script.filename}", exc_info=True)
506

507 508 509 510 511 512 513 514
    def after_extra_networks_activate(self, p, **kwargs):
        for script in self.alwayson_scripts:
            try:
                script_args = p.script_args[script.args_from:script.args_to]
                script.after_extra_networks_activate(p, *script_args, **kwargs)
            except Exception:
                errors.report(f"Error running after_extra_networks_activate: {script.filename}", exc_info=True)

515
    def process_batch(self, p, **kwargs):
A
Artem Zagidulin 已提交
516 517 518
        for script in self.alwayson_scripts:
            try:
                script_args = p.script_args[script.args_from:script.args_to]
519
                script.process_batch(p, *script_args, **kwargs)
A
Artem Zagidulin 已提交
520
            except Exception:
521
                errors.report(f"Error running process_batch: {script.filename}", exc_info=True)
A
Artem Zagidulin 已提交
522

A
AUTOMATIC 已提交
523 524 525 526 527 528
    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:
529
                errors.report(f"Error running postprocess: {script.filename}", exc_info=True)
530

531 532 533 534 535 536
    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:
537
                errors.report(f"Error running postprocess_batch: {script.filename}", exc_info=True)
538

539 540 541 542 543 544
    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:
545
                errors.report(f"Error running postprocess_image: {script.filename}", exc_info=True)
546

547 548 549 550 551
    def before_component(self, component, **kwargs):
        for script in self.scripts:
            try:
                script.before_component(component, **kwargs)
            except Exception:
552
                errors.report(f"Error running before_component: {script.filename}", exc_info=True)
553 554 555 556 557 558

    def after_component(self, component, **kwargs):
        for script in self.scripts:
            try:
                script.after_component(component, **kwargs)
            except Exception:
559
                errors.report(f"Error running after_component: {script.filename}", exc_info=True)
560

A
AUTOMATIC 已提交
561
    def reload_sources(self, cache):
D
DepFA 已提交
562
        for si, script in list(enumerate(self.scripts)):
563 564 565 566 567 568 569 570 571
            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 已提交
572
            for script_class in module.__dict__.values():
573 574 575 576 577
                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 已提交
578

579

580 581 582 583 584 585 586 587 588
    def before_hr(self, p):
        for script in self.alwayson_scripts:
            try:
                script_args = p.script_args[script.args_from:script.args_to]
                script.before_hr(p, *script_args)
            except Exception:
                errors.report(f"Error running before_hr: {script.filename}", exc_info=True)


589 590 591
scripts_txt2img: ScriptRunner = None
scripts_img2img: ScriptRunner = None
scripts_postproc: scripts_postprocessing.ScriptPostprocessingRunner = None
592
scripts_current: ScriptRunner = None
D
DepFA 已提交
593

594

D
DepFA 已提交
595
def reload_script_body_only():
A
AUTOMATIC 已提交
596 597 598
    cache = {}
    scripts_txt2img.reload_sources(cache)
    scripts_img2img.reload_sources(cache)
D
DepFA 已提交
599

D
DepFA 已提交
600

601
reload_scripts = load_scripts  # compatibility alias
602

603

604 605 606 607 608
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
    """

609
    comp.elem_classes = [f"gradio-{comp.get_block_name()}", *(comp.elem_classes or [])]
610 611 612 613 614 615

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



616 617 618 619 620 621 622 623
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)

624
    add_classes_to_gradio_component(self)
A
AUTOMATIC 已提交
625

626 627 628 629 630 631 632 633 634 635
    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
636 637 638 639 640 641 642 643 644 645 646 647


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