File size: 3,080 Bytes
3dabe4a
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
import os
import re
from PIL import Image

from scripts.iib.parsers.model import ImageGenerationInfo, ImageGenerationParams
from scripts.iib.tool import omit, parse_generation_parameters, unique_by


def remove_extra_spaces(text):
    return re.sub(r"\s+", " ", text)


def get_log_file(file_path: str):
    dir = os.path.dirname(file_path)
    with open(os.path.join(dir, "log.html")) as f:
        return f.read()

lora_re = re.compile("LoRA \d+", re.IGNORECASE)

class FooocusParser:
    def __init__(self):
        pass

    @classmethod
    def parse(clz, img: Image, file_path):
        if not clz.test(img, file_path):
            raise Exception("The input image does not match the current parser.")
        from lxml import etree

        log = get_log_file(file_path)
        root = etree.HTML(log)
        id = str(os.path.basename(file_path)).replace(".", "_")
        metadata = root.xpath(f'//div[@id="{id}"]/descendant::table[@class="metadata"]')
        tr_elements = metadata[0].xpath(".//tr")
        lora_list = []
        # As a workaround to bypass parsing errors in the parser.
        # https://github.com/jiw0220/stable-diffusion-image-metadata/blob/00b8d42d4d1a536862bba0b07c332bdebb2a0ce5/src/index.ts#L130
        metadata_list_str = "Steps: Unknown , Source Identifier: Fooocus ,"
        params = {"meta": {"Source Identifier": "Fooocus"}}
        for tr in tr_elements:
            label = tr.xpath('.//td[@class="label" or @class="key"]/text()')
            value = tr.xpath('.//td[@class="value"]/text()')
            if label:
                k = label[0]
                v = value[0] if value else "None"
                if k == "Fooocus V2 Expansion":
                    continue
                if k == "Prompt" or k == "Negative Prompt":
                    params[k] = remove_extra_spaces(v.replace("\n", "").strip())
                else:
                    v = v.replace(",", ",")
                    params["meta"][k] = v
                    metadata_list_str += f" {k}: {v},"
                if lora_re.search(k):
                    lora_list.append({ "name": v.strip(), "value": 1 })
        params["meta"]["Model"] = params["meta"]["Base Model"]
        params["meta"]["Size"] = str(params["meta"]["Resolution"]).replace("(", "").replace(")", "").replace(",", " * ")
        metadata_list_str = metadata_list_str.strip()
        info = f"""{params['Prompt']}\nNegative prompt: {params['Negative Prompt']}\n{metadata_list_str}""".strip()
        return ImageGenerationInfo(
            info,
            ImageGenerationParams(
                meta=params["meta"],
                pos_prompt=parse_generation_parameters(info)["pos_prompt"],
                extra={
                    "lora": unique_by(lora_list, lambda x: x["name"].lower())
                }
            ),
        )

    @classmethod
    def test(clz, img: Image, file_path: str):
        filename = os.path.basename(file_path)
        try:
            return get_log_file(file_path).find(filename) != -1
        except Exception as e:
            return False