deepbooru.py 6.3 KB
Newer Older
G
Greendayle 已提交
1 2
import os.path
from concurrent.futures import ProcessPoolExecutor
3
import multiprocessing
4
import time
5 6 7
import re

re_special = re.compile(r'([\\()])')
G
Greendayle 已提交
8

9
def get_deepbooru_tags(pil_image):
10 11 12 13 14
    """
    This method is for running only one image at a time for simple use.  Used to the img2img interrogate.
    """
    from modules import shared  # prevents circular reference

15 16 17 18 19 20 21
    try:
        create_deepbooru_process(shared.opts.interrogate_deepbooru_score_threshold, create_deepbooru_opts())
        return get_tags_from_process(pil_image)
    finally:
        release_process()


22
OPT_INCLUDE_RANKS = "include_ranks"
23 24
def create_deepbooru_opts():
    from modules import shared
25

26 27 28 29
    return {
        "use_spaces": shared.opts.deepbooru_use_spaces,
        "use_escape": shared.opts.deepbooru_escape,
        "alpha_sort": shared.opts.deepbooru_sort_alpha,
30
        OPT_INCLUDE_RANKS: shared.opts.interrogate_return_ranks,
31 32 33 34
    }


def deepbooru_process(queue, deepbooru_process_return, threshold, deepbooru_opts):
35 36 37 38 39 40
    model, tags = get_deepbooru_tags_model()
    while True: # while process is running, keep monitoring queue for new image
        pil_image = queue.get()
        if pil_image == "QUIT":
            break
        else:
41
            deepbooru_process_return["value"] = get_deepbooru_tags_from_model(model, tags, pil_image, threshold, deepbooru_opts)
42 43


44
def create_deepbooru_process(threshold, deepbooru_opts):
45 46 47 48 49 50 51 52 53 54 55 56
    """
    Creates deepbooru process.  A queue is created to send images into the process.  This enables multiple images
    to be processed in a row without reloading the model or creating a new process.  To return the data, a shared
    dictionary is created to hold the tags created.  To wait for tags to be returned, a value of -1 is assigned
    to the dictionary and the method adding the image to the queue should wait for this value to be updated with
    the tags.
    """
    from modules import shared  # prevents circular reference
    shared.deepbooru_process_manager = multiprocessing.Manager()
    shared.deepbooru_process_queue = shared.deepbooru_process_manager.Queue()
    shared.deepbooru_process_return = shared.deepbooru_process_manager.dict()
    shared.deepbooru_process_return["value"] = -1
57
    shared.deepbooru_process = multiprocessing.Process(target=deepbooru_process, args=(shared.deepbooru_process_queue, shared.deepbooru_process_return, threshold, deepbooru_opts))
58 59 60
    shared.deepbooru_process.start()


61 62 63 64 65 66 67 68 69 70 71 72 73
def get_tags_from_process(image):
    from modules import shared

    shared.deepbooru_process_return["value"] = -1
    shared.deepbooru_process_queue.put(image)
    while shared.deepbooru_process_return["value"] == -1:
        time.sleep(0.2)
    caption = shared.deepbooru_process_return["value"]
    shared.deepbooru_process_return["value"] = -1

    return caption


74 75 76 77 78 79 80 81 82 83 84 85 86
def release_process():
    """
    Stops the deepbooru process to return used memory
    """
    from modules import shared  # prevents circular reference
    shared.deepbooru_process_queue.put("QUIT")
    shared.deepbooru_process.join()
    shared.deepbooru_process_queue = None
    shared.deepbooru_process = None
    shared.deepbooru_process_return = None
    shared.deepbooru_process_manager = None

def get_deepbooru_tags_model():
87 88 89
    import deepdanbooru as dd
    import tensorflow as tf
    import numpy as np
G
Greendayle 已提交
90
    this_folder = os.path.dirname(__file__)
91 92 93 94 95
    model_path = os.path.abspath(os.path.join(this_folder, '..', 'models', 'deepbooru'))
    if not os.path.exists(os.path.join(model_path, 'project.json')):
        # there is no point importing these every time
        import zipfile
        from basicsr.utils.download_util import load_file_from_url
96 97 98
        load_file_from_url(
            r"https://github.com/KichangKim/DeepDanbooru/releases/download/v3-20211112-sgd-e28/deepdanbooru-v3-20211112-sgd-e28.zip",
            model_path)
99 100 101
        with zipfile.ZipFile(os.path.join(model_path, "deepdanbooru-v3-20211112-sgd-e28.zip"), "r") as zip_ref:
            zip_ref.extractall(model_path)
        os.remove(os.path.join(model_path, "deepdanbooru-v3-20211112-sgd-e28.zip"))
G
Greendayle 已提交
102 103 104 105 106

    tags = dd.project.load_tags_from_project(model_path)
    model = dd.project.load_model_from_project(
        model_path, compile_model=True
    )
107
    return model, tags
G
Greendayle 已提交
108

109

110
def get_deepbooru_tags_from_model(model, tags, pil_image, threshold, deepbooru_opts):
111 112 113
    import deepdanbooru as dd
    import tensorflow as tf
    import numpy as np
114 115 116 117

    alpha_sort = deepbooru_opts['alpha_sort']
    use_spaces = deepbooru_opts['use_spaces']
    use_escape = deepbooru_opts['use_escape']
118
    include_ranks = deepbooru_opts['include_ranks']
119

G
Greendayle 已提交
120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140
    width = model.input_shape[2]
    height = model.input_shape[1]
    image = np.array(pil_image)
    image = tf.image.resize(
        image,
        size=(height, width),
        method=tf.image.ResizeMethod.AREA,
        preserve_aspect_ratio=True,
    )
    image = image.numpy()  # EagerTensor to np.array
    image = dd.image.transform_and_pad_image(image, width, height)
    image = image / 255.0
    image_shape = image.shape
    image = image.reshape((1, image_shape[0], image_shape[1], image_shape[2]))

    y = model.predict(image)[0]

    result_dict = {}

    for i, tag in enumerate(tags):
        result_dict[tag] = y[i]
141 142

    unsorted_tags_in_theshold = []
G
Greendayle 已提交
143 144 145
    result_tags_print = []
    for tag in tags:
        if result_dict[tag] >= threshold:
G
Greendayle 已提交
146 147
            if tag.startswith("rating:"):
                continue
148
            unsorted_tags_in_theshold.append((result_dict[tag], tag))
G
Greendayle 已提交
149 150
            result_tags_print.append(f'{result_dict[tag]} {tag}')

151 152 153 154 155 156
    # sort tags
    result_tags_out = []
    sort_ndx = 0
    if alpha_sort:
        sort_ndx = 1

157
    # sort by reverse by likelihood and normal for alpha, and format tag text as requested
158 159
    unsorted_tags_in_theshold.sort(key=lambda y: y[sort_ndx], reverse=(not alpha_sort))
    for weight, tag in unsorted_tags_in_theshold:
160 161 162 163 164 165 166
        # note: tag_outformat will still have a colon if include_ranks is True
        tag_outformat = tag.replace(':', ' ')
        if use_spaces:
            tag_outformat = tag_outformat.replace('_', ' ')
        if use_escape:
            tag_outformat = re.sub(re_special, r'\\\1', tag_outformat)
        if include_ranks:
167
            tag_outformat = f"({tag_outformat}:{weight:.3f})"
168

169
        result_tags_out.append(tag_outformat)
170

171
    print('\n'.join(sorted(result_tags_print, reverse=True)))
172

173
    return ', '.join(result_tags_out)