2022-09-08 13:37:13 +00:00
import sys
2022-09-03 09:08:45 +00:00
import argparse
import json
import os
2022-09-07 09:32:28 +00:00
2022-09-03 09:08:45 +00:00
import gradio as gr
import torch
2022-09-08 13:37:13 +00:00
import tqdm
2022-09-03 09:08:45 +00:00
2022-09-05 20:08:06 +00:00
import modules . artists
2022-09-03 09:08:45 +00:00
from modules . paths import script_path , sd_path
2022-09-09 20:16:02 +00:00
import modules . styles
2022-09-03 09:08:45 +00:00
config_filename = " config.json "
sd_model_file = os . path . join ( script_path , ' model.ckpt ' )
if not os . path . exists ( sd_model_file ) :
sd_model_file = " models/ldm/stable-diffusion-v1/model.ckpt "
parser = argparse . ArgumentParser ( )
parser . add_argument ( " --config " , type = str , default = os . path . join ( sd_path , " configs/stable-diffusion/v1-inference.yaml " ) , help = " path to config which constructs model " , )
parser . add_argument ( " --ckpt " , type = str , default = os . path . join ( sd_path , sd_model_file ) , help = " path to checkpoint of model " , )
parser . add_argument ( " --gfpgan-dir " , type = str , help = " GFPGAN directory " , default = ( ' ./src/gfpgan ' if os . path . exists ( ' ./src/gfpgan ' ) else ' ./GFPGAN ' ) )
parser . add_argument ( " --gfpgan-model " , type = str , help = " GFPGAN model file name " , default = ' GFPGANv1.3.pth ' )
parser . add_argument ( " --no-half " , action = ' store_true ' , help = " do not switch the model to 16-bit floats " )
parser . add_argument ( " --no-progressbar-hiding " , action = ' store_true ' , help = " do not hide progressbar in gradio UI (we hide it because it slows down ML if you have hardware accleration in browser) " )
parser . add_argument ( " --max-batch-count " , type = int , default = 16 , help = " maximum batch count value for the UI " )
2022-09-08 23:17:38 +00:00
parser . add_argument ( " --embeddings-dir " , type = str , default = ' embeddings ' , help = " embeddings directory for textual inversion (default: embeddings) " )
2022-09-03 09:08:45 +00:00
parser . add_argument ( " --allow-code " , action = ' store_true ' , help = " allow custom script execution from webui " )
2022-09-08 23:17:38 +00:00
parser . add_argument ( " --medvram " , action = ' store_true ' , help = " enable stable diffusion model optimizations for sacrificing a little speed for low VRM usage " )
parser . add_argument ( " --lowvram " , action = ' store_true ' , help = " enable stable diffusion model optimizations for sacrificing a lot of speed for very low VRM usage " )
parser . add_argument ( " --always-batch-cond-uncond " , action = ' store_true ' , help = " a workaround test; may help with speed if you use --lowvram " )
2022-09-03 14:28:30 +00:00
parser . add_argument ( " --unload-gfpgan " , action = ' store_true ' , help = " unload GFPGAN every time after processing images. Warning: seems to cause memory leaks " )
2022-09-03 09:08:45 +00:00
parser . add_argument ( " --precision " , type = str , help = " evaluate at this precision " , choices = [ " full " , " autocast " ] , default = " autocast " )
parser . add_argument ( " --share " , action = ' store_true ' , help = " use share=True for gradio and make the UI accessible through their site (doesn ' t work for me but you might have better luck) " )
2022-09-04 15:54:12 +00:00
parser . add_argument ( " --esrgan-models-path " , type = str , help = " path to directory with ESRGAN models " , default = os . path . join ( script_path , ' ESRGAN ' ) )
2022-09-08 23:17:38 +00:00
parser . add_argument ( " --opt-split-attention " , action = ' store_true ' , help = " enable optimization that reduce vram usage by a lot for about 10 %% decrease in performance " )
2022-09-05 16:37:11 +00:00
parser . add_argument ( " --listen " , action = ' store_true ' , help = " launch gradio with 0.0.0.0 as server name, allowing to respond to network requests " )
2022-09-08 08:42:21 +00:00
parser . add_argument ( " --port " , type = int , help = " launch gradio with given server port, you need root/admin rights for ports < 1024, defaults to 7860 if available " , default = None )
2022-09-09 06:15:36 +00:00
parser . add_argument ( " --show-negative-prompt " , action = ' store_true ' , help = " enable the field that lets you input negative prompt " , default = False )
2022-09-10 05:18:54 +00:00
parser . add_argument ( " --ui-config-file " , type = str , help = " filename to use for ui configuration " , default = os . path . join ( script_path , ' ui-config.json ' ) )
2022-09-09 06:15:36 +00:00
2022-09-06 07:58:58 +00:00
cmd_opts = parser . parse_args ( )
2022-09-03 09:08:45 +00:00
2022-09-07 13:58:25 +00:00
if torch . has_cuda :
device = torch . device ( " cuda " )
elif torch . has_mps :
device = torch . device ( " mps " )
else :
device = torch . device ( " cpu " )
2022-09-03 09:08:45 +00:00
batch_cond_uncond = cmd_opts . always_batch_cond_uncond or not ( cmd_opts . lowvram or cmd_opts . medvram )
2022-09-06 20:10:12 +00:00
parallel_processing_allowed = not cmd_opts . lowvram and not cmd_opts . medvram
2022-09-06 16:33:51 +00:00
2022-09-07 09:32:28 +00:00
2022-09-03 09:08:45 +00:00
class State :
interrupted = False
job = " "
2022-09-05 23:09:01 +00:00
job_no = 0
job_count = 0
sampling_step = 0
sampling_steps = 0
2022-09-06 16:33:51 +00:00
current_latent = None
current_image = None
2022-09-06 20:10:12 +00:00
current_image_sampling_step = 0
2022-09-03 09:08:45 +00:00
def interrupt ( self ) :
self . interrupted = True
2022-09-05 23:09:01 +00:00
def nextjob ( self ) :
self . job_no + = 1
self . sampling_step = 0
2022-09-06 20:10:12 +00:00
self . current_image_sampling_step = 0
2022-09-05 23:09:01 +00:00
2022-09-06 07:11:25 +00:00
2022-09-03 09:08:45 +00:00
state = State ( )
2022-09-05 20:08:06 +00:00
artist_db = modules . artists . ArtistsDatabase ( os . path . join ( script_path , ' artists.csv ' ) )
2022-09-09 20:16:02 +00:00
styles_filename = os . path . join ( script_path , ' styles.csv ' )
prompt_styles = modules . styles . load_styles ( styles_filename )
2022-09-03 09:08:45 +00:00
2022-09-09 20:16:02 +00:00
face_restorers = [ ]
2022-09-09 06:15:36 +00:00
2022-09-03 09:08:45 +00:00
class Options :
class OptionInfo :
def __init__ ( self , default = None , label = " " , component = None , component_args = None ) :
self . default = default
self . label = label
self . component = component
self . component_args = component_args
data = None
data_labels = {
2022-09-08 23:17:38 +00:00
" outdir_samples " : OptionInfo ( " " , " Output directory for images; if empty, defaults to two directories below " ) ,
" outdir_txt2img_samples " : OptionInfo ( " outputs/txt2img-images " , ' Output directory for txt2img images ' ) ,
" outdir_img2img_samples " : OptionInfo ( " outputs/img2img-images " , ' Output directory for img2img images ' ) ,
" outdir_extras_samples " : OptionInfo ( " outputs/extras-images " , ' Output directory for images from extras tab ' ) ,
" outdir_grids " : OptionInfo ( " " , " Output directory for grids; if empty, defaults to two directories below " ) ,
" outdir_txt2img_grids " : OptionInfo ( " outputs/txt2img-grids " , ' Output directory for txt2img grids ' ) ,
" outdir_img2img_grids " : OptionInfo ( " outputs/img2img-grids " , ' Output directory for img2img grids ' ) ,
2022-09-10 10:36:16 +00:00
" save_to_dirs " : OptionInfo ( False , " When writing images, create a directory with name derived from the prompt " ) ,
" grid_save_to_dirs " : OptionInfo ( False , " When writing grids, create a directory with name derived from the prompt " ) ,
2022-09-03 09:08:45 +00:00
" save_to_dirs_prompt_len " : OptionInfo ( 10 , " When using above, how many words from prompt to put into directory name " , gr . Slider , { " minimum " : 1 , " maximum " : 32 , " step " : 1 } ) ,
" outdir_save " : OptionInfo ( " log/images " , " Directory for saving images using the Save button " ) ,
" samples_save " : OptionInfo ( True , " Save indiviual samples " ) ,
2022-09-10 08:10:00 +00:00
" samples_format " : OptionInfo ( ' png ' , ' File format for individual samples ' ) ,
2022-09-03 09:08:45 +00:00
" grid_save " : OptionInfo ( True , " Save image grids " ) ,
" return_grid " : OptionInfo ( True , " Show grid in results for web " ) ,
" grid_format " : OptionInfo ( ' png ' , ' File format for grids ' ) ,
" grid_extended_filename " : OptionInfo ( False , " Add extended info (seed, prompt) to filename when saving grid " ) ,
" grid_only_if_multiple " : OptionInfo ( True , " Do not save grids consisting of one picture " ) ,
" n_rows " : OptionInfo ( - 1 , " Grid row count; use -1 for autodetect and 0 for it to be same as batch size " , gr . Slider , { " minimum " : - 1 , " maximum " : 16 , " step " : 1 } ) ,
" jpeg_quality " : OptionInfo ( 80 , " Quality for saved jpeg images " , gr . Slider , { " minimum " : 1 , " maximum " : 100 , " step " : 1 } ) ,
" export_for_4chan " : OptionInfo ( True , " If PNG image is larger than 4MB or any dimension is larger than 4000, downscale and save copy as JPG " ) ,
" enable_pnginfo " : OptionInfo ( True , " Save text information about generation parameters as chunks to png files " ) ,
2022-09-09 17:13:52 +00:00
" font " : OptionInfo ( " " , " Font for image grids that have text " ) ,
2022-09-03 09:08:45 +00:00
" enable_emphasis " : OptionInfo ( True , " Use (text) to make model pay more attention to text text and [text] to make it pay less attention " ) ,
" save_txt " : OptionInfo ( False , " Create a text file next to every image with generation parameters. " ) ,
2022-09-08 12:19:36 +00:00
" ESRGAN_tile " : OptionInfo ( 192 , " Tile size for upscaling. 0 = no tiling. " , gr . Slider , { " minimum " : 0 , " maximum " : 512 , " step " : 16 } ) ,
" ESRGAN_tile_overlap " : OptionInfo ( 8 , " Tile overlap, in pixels for upscaling. Low values = visible seam. " , gr . Slider , { " minimum " : 0 , " maximum " : 48 , " step " : 1 } ) ,
2022-09-05 20:08:06 +00:00
" random_artist_categories " : OptionInfo ( [ ] , " Allowed categories for random artists selection when using the Roll button " , gr . CheckboxGroup , { " choices " : artist_db . categories ( ) } ) ,
" upscale_at_full_resolution_padding " : OptionInfo ( 16 , " Inpainting at full resolution: padding, in pixels, for the masked region. " , gr . Slider , { " minimum " : 0 , " maximum " : 128 , " step " : 4 } ) ,
2022-09-05 23:09:01 +00:00
" show_progressbar " : OptionInfo ( True , " Show progressbar " ) ,
2022-09-06 20:10:12 +00:00
" show_progress_every_n_steps " : OptionInfo ( 0 , " Show show image creation progress every N sampling steps. Set 0 to disable. " , gr . Slider , { " minimum " : 0 , " maximum " : 32 , " step " : 1 } ) ,
2022-09-08 13:37:13 +00:00
" multiple_tqdm " : OptionInfo ( True , " Add a second progress bar to the console that shows progress for an entire job. Broken in PyCharm console. " ) ,
2022-09-07 09:32:28 +00:00
" face_restoration_model " : OptionInfo ( None , " Face restoration model " , gr . Radio , lambda : { " choices " : [ x . name ( ) for x in face_restorers ] } ) ,
2022-09-07 10:46:08 +00:00
" code_former_weight " : OptionInfo ( 0.5 , " CodeFormer weight parameter; 0 = maximum effect; 1 = minimum effect " , gr . Slider , { " minimum " : 0 , " maximum " : 1 , " step " : 0.01 } ) ,
2022-09-03 09:08:45 +00:00
}
def __init__ ( self ) :
self . data = { k : v . default for k , v in self . data_labels . items ( ) }
def __setattr__ ( self , key , value ) :
if self . data is not None :
if key in self . data :
self . data [ key ] = value
return super ( Options , self ) . __setattr__ ( key , value )
def __getattr__ ( self , item ) :
if self . data is not None :
if item in self . data :
return self . data [ item ]
if item in self . data_labels :
return self . data_labels [ item ] . default
return super ( Options , self ) . __getattribute__ ( item )
def save ( self , filename ) :
with open ( filename , " w " , encoding = " utf8 " ) as file :
json . dump ( self . data , file )
def load ( self , filename ) :
with open ( filename , " r " , encoding = " utf8 " ) as file :
self . data = json . load ( file )
opts = Options ( )
if os . path . exists ( config_filename ) :
opts . load ( config_filename )
2022-09-04 15:54:12 +00:00
sd_upscalers = [ ]
2022-09-03 09:08:45 +00:00
sd_model = None
2022-09-05 20:08:06 +00:00
2022-09-08 13:37:13 +00:00
progress_print_out = sys . stdout
2022-09-05 20:08:06 +00:00
2022-09-08 13:37:13 +00:00
class TotalTQDM :
def __init__ ( self ) :
self . _tqdm = None
def reset ( self ) :
self . _tqdm = tqdm . tqdm (
desc = " Total progress " ,
total = state . job_count * state . sampling_steps ,
position = 1 ,
file = progress_print_out
)
def update ( self ) :
if not opts . multiple_tqdm :
return
if self . _tqdm is None :
self . reset ( )
self . _tqdm . update ( )
def clear ( self ) :
if self . _tqdm is not None :
self . _tqdm . close ( )
self . _tqdm = None
total_tqdm = TotalTQDM ( )