main.py 6.07 KB
Newer Older
comfyanonymous's avatar
comfyanonymous committed
1
2
import os
import sys
3
import shutil
4

comfyanonymous's avatar
comfyanonymous committed
5
import threading
6
import asyncio
comfyanonymous's avatar
comfyanonymous committed
7

pythongosssss's avatar
pythongosssss committed
8
9
10
11
if os.name == "nt":
    import logging
    logging.getLogger("xformers").addFilter(lambda record: 'A matching Triton is not available' not in record.getMessage())

comfyanonymous's avatar
comfyanonymous committed
12
13
if __name__ == "__main__":
    if '--help' in sys.argv:
14
        print()
comfyanonymous's avatar
comfyanonymous committed
15
        print("Valid Command line Arguments:")
16
        print("\t--listen [ip]\t\t\tListen on ip or 0.0.0.0 if none given so the UI can be accessed from other computers.")
comfyanonymous's avatar
comfyanonymous committed
17
        print("\t--port 8188\t\t\tSet the listen port.")
18
19
        print()
        print("\t--extra-model-paths-config file.yaml\tload an extra_model_paths.yaml file.")
20
        print("\t--output-directory path/to/output\tSet the ComfyUI output directory.")
21
22
        print()
        print()
comfyanonymous's avatar
comfyanonymous committed
23
24
        print("\t--dont-upcast-attention\t\tDisable upcasting of attention \n\t\t\t\t\tcan boost speed but increase the chances of black images.\n")
        print("\t--use-split-cross-attention\tUse the split cross attention optimization instead of the sub-quadratic one.\n\t\t\t\t\tIgnored when xformers is used.")
25
        print("\t--use-pytorch-cross-attention\tUse the new pytorch 2.0 cross attention function.")
26
        print("\t--disable-xformers\t\tdisables xformers")
27
        print("\t--cuda-device 1\t\tSet the id of the cuda device this instance will use.")
comfyanonymous's avatar
comfyanonymous committed
28
        print()
29
        print("\t--highvram\t\t\tBy default models will be unloaded to CPU memory after being used.\n\t\t\t\t\tThis option keeps them in GPU memory.\n")
30
        print("\t--normalvram\t\t\tUsed to force normal vram use if lowvram gets automatically enabled.")
31
32
33
        print("\t--lowvram\t\t\tSplit the unet in parts to use less vram.")
        print("\t--novram\t\t\tWhen lowvram isn't enough.")
        print()
34
        print("\t--cpu\t\t\tTo use the CPU for everything (slow).")
comfyanonymous's avatar
comfyanonymous committed
35
36
        exit()

pythongosssss's avatar
pythongosssss committed
37
38
39
    if '--dont-upcast-attention' in sys.argv:
        print("disabling upcasting of attention")
        os.environ['ATTN_PRECISION'] = "fp16"
40

41
42
43
44
45
46
47
48
    try:
        index = sys.argv.index('--cuda-device')
        device = sys.argv[index + 1]
        os.environ['CUDA_VISIBLE_DEVICES'] = device
        print("Set cuda device to:", device)
    except:
        pass

49
from nodes import init_custom_nodes
50
51
import execution
import server
52
53
import folder_paths
import yaml
54

pythongosssss's avatar
pythongosssss committed
55
def prompt_worker(q, server):
56
    e = execution.PromptExecutor(server)
comfyanonymous's avatar
comfyanonymous committed
57
    while True:
58
        item, item_id = q.get()
comfyanonymous's avatar
comfyanonymous committed
59
        e.execute(item[-2], item[-1])
pythongosssss's avatar
pythongosssss committed
60
        q.task_done(item_id, e.outputs)
comfyanonymous's avatar
comfyanonymous committed
61

62
63
async def run(server, address='', port=8188, verbose=True, call_on_start=None):
    await asyncio.gather(server.start(address, port, verbose, call_on_start), server.publish_loop())
comfyanonymous's avatar
comfyanonymous committed
64

pythongosssss's avatar
pythongosssss committed
65
66
67
68
69
70
71
72
73
def hijack_progress(server):
    from tqdm.auto import tqdm
    orig_func = getattr(tqdm, "update")
    def wrapped_func(*args, **kwargs):
        pbar = args[0]
        v = orig_func(*args, **kwargs)
        server.send_sync("progress", { "value": pbar.n, "max": pbar.total}, server.client_id)            
        return v
    setattr(tqdm, "update", wrapped_func)
comfyanonymous's avatar
comfyanonymous committed
74

75
76
77
def cleanup_temp():
    temp_dir = os.path.join(os.path.dirname(os.path.realpath(__file__)), "temp")
    if os.path.exists(temp_dir):
78
        shutil.rmtree(temp_dir, ignore_errors=True)
79

80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
def load_extra_path_config(yaml_path):
    with open(yaml_path, 'r') as stream:
        config = yaml.safe_load(stream)
    for c in config:
        conf = config[c]
        if conf is None:
            continue
        base_path = None
        if "base_path" in conf:
            base_path = conf.pop("base_path")
        for x in conf:
            for y in conf[x].split("\n"):
                if len(y) == 0:
                    continue
                full_path = y
                if base_path is not None:
                    full_path = os.path.join(base_path, full_path)
                print("Adding extra search path", x, full_path)
                folder_paths.add_model_folder_path(x, full_path)

comfyanonymous's avatar
comfyanonymous committed
100
if __name__ == "__main__":
101
102
    cleanup_temp()

103
104
    loop = asyncio.new_event_loop()
    asyncio.set_event_loop(loop)
pythongosssss's avatar
pythongosssss committed
105
    server = server.PromptServer(loop)
106
    q = execution.PromptQueue(server)
107

108
109
    init_custom_nodes()
    server.add_routes()
pythongosssss's avatar
pythongosssss committed
110
111
112
    hijack_progress(server)

    threading.Thread(target=prompt_worker, daemon=True, args=(q,server,)).start()
113
    try:
114
        address = '0.0.0.0'
115
116
117
118
119
120
121
122
        p_index = sys.argv.index('--listen')
        try:
            ip = sys.argv[p_index + 1]
            if ip[:2] != '--':
                address = ip
        except:
            pass
    except:
123
        address = '127.0.0.1'
124

comfyanonymous's avatar
comfyanonymous committed
125
126
127
128
    dont_print = False
    if '--dont-print-server' in sys.argv:
        dont_print = True

129
130
131
132
133
134
135
136
137
    extra_model_paths_config_path = os.path.join(os.path.dirname(os.path.realpath(__file__)), "extra_model_paths.yaml")
    if os.path.isfile(extra_model_paths_config_path):
        load_extra_path_config(extra_model_paths_config_path)

    if '--extra-model-paths-config' in sys.argv:
        indices = [(i + 1) for i in range(len(sys.argv) - 1) if sys.argv[i] == '--extra-model-paths-config']
        for i in indices:
            load_extra_path_config(sys.argv[i])

138
139
140
141
142
143
144
145
    try:
        output_dir = sys.argv[sys.argv.index('--output-directory') + 1]
        output_dir = os.path.abspath(output_dir)
        print("setting output directory to:", output_dir)
        folder_paths.set_output_directory(output_dir)
    except:
        pass

146
147
148
149
150
151
152
    port = 8188
    try:
        p_index = sys.argv.index('--port')
        port = int(sys.argv[p_index + 1])
    except:
        pass

153
154
155
    if '--quick-test-for-ci' in sys.argv:
        exit(0)

156
157
158
159
160
161
162
    call_on_start = None
    if "--windows-standalone-build" in sys.argv:
        def startup_server(address, port):
            import webbrowser
            webbrowser.open("http://{}:{}".format(address, port))
        call_on_start = startup_server

pythongosssss's avatar
pythongosssss committed
163
164
    if os.name == "nt":
        try:
165
            loop.run_until_complete(run(server, address=address, port=port, verbose=not dont_print, call_on_start=call_on_start))
pythongosssss's avatar
pythongosssss committed
166
167
168
        except KeyboardInterrupt:
            pass
    else:
169
        loop.run_until_complete(run(server, address=address, port=port, verbose=not dont_print, call_on_start=call_on_start))
comfyanonymous's avatar
comfyanonymous committed
170

171
    cleanup_temp()