2023-04-06 00:32:59 +00:00
import argparse
import asyncio
2023-01-03 06:53:32 +00:00
import os
2023-03-13 19:34:05 +00:00
import shutil
2023-04-06 00:32:59 +00:00
import sys
2023-01-03 06:53:32 +00:00
import threading
2023-02-16 18:19:26 +00:00
if os . name == " nt " :
import logging
logging . getLogger ( " xformers " ) . addFilter ( lambda record : ' A matching Triton is not available ' not in record . getMessage ( ) )
2023-02-08 03:12:56 +00:00
if __name__ == " __main__ " :
2023-04-06 00:32:59 +00:00
parser = argparse . ArgumentParser ( description = " Script Arguments " )
parser . add_argument ( " --listen " , type = str , default = " 127.0.0.1 " , help = " Listen on IP or 0.0.0.0 if none given so the UI can be accessed from other computers. " )
parser . add_argument ( " --port " , type = int , default = 8188 , help = " Set the listen port. " )
parser . add_argument ( " --extra-model-paths-config " , type = str , default = None , help = " Load an extra_model_paths.yaml file. " )
parser . add_argument ( " --output-directory " , type = str , default = None , help = " Set the ComfyUI output directory. " )
parser . add_argument ( " --dont-upcast-attention " , action = " store_true " , help = " Disable upcasting of attention. Can boost speed but increase the chances of black images. " )
parser . add_argument ( " --use-split-cross-attention " , action = " store_true " , help = " Use the split cross attention optimization instead of the sub-quadratic one. Ignored when xformers is used. " )
parser . add_argument ( " --use-pytorch-cross-attention " , action = " store_true " , help = " Use the new pytorch 2.0 cross attention function. " )
parser . add_argument ( " --disable-xformers " , action = " store_true " , help = " Disable xformers. " )
parser . add_argument ( " --cuda-device " , type = int , default = None , help = " Set the id of the cuda device this instance will use. " )
parser . add_argument ( " --highvram " , action = " store_true " , help = " By default models will be unloaded to CPU memory after being used. This option keeps them in GPU memory. " )
parser . add_argument ( " --normalvram " , action = " store_true " , help = " Used to force normal vram use if lowvram gets automatically enabled. " )
parser . add_argument ( " --lowvram " , action = " store_true " , help = " Split the unet in parts to use less vram. " )
parser . add_argument ( " --novram " , action = " store_true " , help = " When lowvram isn ' t enough. " )
parser . add_argument ( " --cpu " , action = " store_true " , help = " To use the CPU for everything (slow). " )
parser . add_argument ( " --dont-print-server " , action = " store_true " , help = " Don ' t print server output. " )
parser . add_argument ( " --quick-test-for-ci " , action = " store_true " , help = " Quick test for CI. " )
parser . add_argument ( " --windows-standalone-build " , action = " store_true " , help = " Windows standalone build. " )
args = parser . parse_args ( )
if args . dont_upcast_attention :
2023-02-21 19:29:49 +00:00
print ( " disabling upcasting of attention " )
os . environ [ ' ATTN_PRECISION ' ] = " fp16 "
2023-01-29 18:12:22 +00:00
2023-04-06 00:32:59 +00:00
if args . cuda_device is not None :
os . environ [ ' CUDA_VISIBLE_DEVICES ' ] = str ( args . cuda_device )
print ( " Set cuda device to: " , args . cuda_device )
import yaml
2023-03-30 03:28:21 +00:00
2023-03-13 15:36:48 +00:00
import execution
2023-03-18 06:52:43 +00:00
import folder_paths
2023-04-06 00:32:59 +00:00
import server
from nodes import init_custom_nodes
2023-03-13 15:36:48 +00:00
2023-02-21 19:29:49 +00:00
def prompt_worker ( q , server ) :
2023-02-28 00:43:55 +00:00
e = execution . PromptExecutor ( server )
2023-01-03 06:53:32 +00:00
while True :
2023-02-02 03:33:10 +00:00
item , item_id = q . get ( )
2023-01-03 06:53:32 +00:00
e . execute ( item [ - 2 ] , item [ - 1 ] )
2023-02-23 20:12:57 +00:00
q . task_done ( item_id , e . outputs )
2023-01-03 06:53:32 +00:00
2023-03-12 19:44:16 +00:00
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 ( ) )
2023-01-03 06:53:32 +00:00
2023-02-21 19:29:49 +00:00
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 )
2023-01-03 06:53:32 +00:00
2023-03-13 19:34:05 +00:00
def cleanup_temp ( ) :
temp_dir = os . path . join ( os . path . dirname ( os . path . realpath ( __file__ ) ) , " temp " )
if os . path . exists ( temp_dir ) :
2023-03-14 22:07:09 +00:00
shutil . rmtree ( temp_dir , ignore_errors = True )
2023-03-13 19:34:05 +00:00
2023-03-18 06:52:43 +00:00
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 )
2023-01-03 06:53:32 +00:00
if __name__ == " __main__ " :
2023-03-13 19:34:05 +00:00
cleanup_temp ( )
2023-02-12 15:53:48 +00:00
loop = asyncio . new_event_loop ( )
asyncio . set_event_loop ( loop )
2023-02-21 19:29:49 +00:00
server = server . PromptServer ( loop )
2023-02-28 00:43:55 +00:00
q = execution . PromptQueue ( server )
2023-02-12 15:53:48 +00:00
2023-04-01 11:44:29 +00:00
init_custom_nodes ( )
server . add_routes ( )
2023-02-21 19:29:49 +00:00
hijack_progress ( server )
threading . Thread ( target = prompt_worker , daemon = True , args = ( q , server , ) ) . start ( )
2023-02-08 02:57:17 +00:00
2023-04-06 00:32:59 +00:00
address = args . listen
dont_print = args . dont_print_server
2023-02-26 03:49:22 +00:00
2023-03-18 06:52:43 +00:00
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 )
2023-04-06 00:32:59 +00:00
if args . extra_model_paths_config :
load_extra_path_config ( args . extra_model_paths_config )
2023-03-18 06:52:43 +00:00
2023-04-06 00:32:59 +00:00
if args . output_directory :
output_dir = os . path . abspath ( args . output_directory )
2023-04-05 18:01:01 +00:00
print ( " setting output directory to: " , output_dir )
folder_paths . set_output_directory ( output_dir )
2023-04-06 00:32:59 +00:00
port = args . port
2023-02-08 02:57:17 +00:00
2023-04-06 00:32:59 +00:00
if args . quick_test_for_ci :
2023-03-15 03:02:57 +00:00
exit ( 0 )
2023-03-12 19:44:16 +00:00
call_on_start = None
2023-04-06 00:32:59 +00:00
if args . windows_standalone_build :
2023-03-12 19:44:16 +00:00
def startup_server ( address , port ) :
import webbrowser
webbrowser . open ( " http:// {} : {} " . format ( address , port ) )
call_on_start = startup_server
2023-02-21 19:29:49 +00:00
if os . name == " nt " :
try :
2023-03-12 19:44:16 +00:00
loop . run_until_complete ( run ( server , address = address , port = port , verbose = not dont_print , call_on_start = call_on_start ) )
2023-02-21 19:29:49 +00:00
except KeyboardInterrupt :
pass
else :
2023-03-12 19:44:16 +00:00
loop . run_until_complete ( run ( server , address = address , port = port , verbose = not dont_print , call_on_start = call_on_start ) )
2023-01-03 06:53:32 +00:00
2023-03-13 19:34:05 +00:00
cleanup_temp ( )