2022-10-24 02:35:49 +08:00
import inspect
2023-07-18 16:48:40 +08:00
2022-10-17 15:02:08 +08:00
from pydantic import BaseModel , Field , create_model
2022-10-24 02:35:49 +08:00
from typing import Any , Optional
2022-10-23 10:13:32 +08:00
from typing_extensions import Literal
2022-10-24 02:35:49 +08:00
from inflection import underscore
2022-10-22 07:27:40 +08:00
from modules . processing import StableDiffusionProcessingTxt2Img , StableDiffusionProcessingImg2Img
2022-11-03 11:51:22 +08:00
from modules . shared import sd_upscalers , opts , parser
Python 3.8 typing compatibility
Solves problems with
```Traceback (most recent call last):
File "webui.py", line 201, in <module>
webui()
File "webui.py", line 178, in webui
create_api(app)
File "webui.py", line 117, in create_api
from modules.api.api import Api
File "H:\AIart\stable-diffusion\stable-diffusion-webui\modules\api\api.py", line 9, in <module>
from modules.api.models import *
File "H:\AIart\stable-diffusion\stable-diffusion-webui\modules\api\models.py", line 194, in <module>
class SamplerItem(BaseModel):
File "H:\AIart\stable-diffusion\stable-diffusion-webui\modules\api\models.py", line 196, in SamplerItem
aliases: list[str] = Field(title="Aliases")
TypeError: 'type' object is not subscriptable```
and
```Traceback (most recent call last):
File "webui.py", line 201, in <module>
webui()
File "webui.py", line 178, in webui
create_api(app)
File "webui.py", line 117, in create_api
from modules.api.api import Api
File "H:\AIart\stable-diffusion\stable-diffusion-webui\modules\api\api.py", line 9, in <module>
from modules.api.models import *
File "H:\AIart\stable-diffusion\stable-diffusion-webui\modules\api\models.py", line 194, in <module>
class SamplerItem(BaseModel):
File "H:\AIart\stable-diffusion\stable-diffusion-webui\modules\api\models.py", line 197, in SamplerItem
options: dict[str, str] = Field(title="Options")
TypeError: 'type' object is not subscriptable```
2022-11-05 22:06:56 +08:00
from typing import Dict , List
2022-10-17 15:02:08 +08:00
2022-10-18 03:10:36 +08:00
API_NOT_ALLOWED = [
" self " ,
" kwargs " ,
" sd_model " ,
" outpath_samples " ,
" outpath_grids " ,
" sampler_index " ,
2023-03-03 21:29:10 +08:00
# "do_not_save_samples",
# "do_not_save_grid",
2022-10-18 03:10:36 +08:00
" extra_generation_params " ,
" overlay_images " ,
" do_not_reload_embeddings " ,
" seed_enable_extras " ,
" prompt_for_display " ,
" sampler_noise_scheduler_override " ,
" ddim_discretize "
]
2022-10-17 15:02:08 +08:00
class ModelDef ( BaseModel ) :
""" Assistance Class for Pydantic Dynamic Model Generation """
field : str
field_alias : str
field_type : Any
field_value : Any
2022-10-24 23:16:07 +08:00
field_exclude : bool = False
2022-10-17 15:02:08 +08:00
2022-10-18 03:10:36 +08:00
class PydanticModelGenerator :
2022-10-17 15:02:08 +08:00
"""
2022-10-17 15:18:41 +08:00
Takes in created classes and stubs them out in a way FastAPI / Pydantic is happy about :
source_data is a snapshot of the default values produced by the class
params are the names of the actual keys required by __init__
2022-10-17 15:02:08 +08:00
"""
def __init__ (
self ,
model_name : str = None ,
2022-10-19 03:04:56 +08:00
class_instance = None ,
additional_fields = None ,
2022-10-17 15:02:08 +08:00
) :
2022-10-18 03:10:36 +08:00
def field_type_generator ( k , v ) :
# field_type = str if not overrides.get(k) else overrides[k]["type"]
# print(k, v.annotation, v.default)
field_type = v . annotation
2022-10-30 03:45:29 +08:00
2022-10-17 15:02:08 +08:00
return Optional [ field_type ]
2022-10-30 03:45:29 +08:00
2022-10-18 03:10:36 +08:00
def merge_class_params ( class_ ) :
all_classes = list ( filter ( lambda x : x is not object , inspect . getmro ( class_ ) ) )
parameters = { }
for classes in all_classes :
parameters = { * * parameters , * * inspect . signature ( classes . __init__ ) . parameters }
return parameters
2022-10-30 03:45:29 +08:00
2022-10-17 15:02:08 +08:00
self . _model_name = model_name
2022-10-31 23:50:33 +08:00
self . _class_data = merge_class_params ( class_instance )
2022-10-28 03:20:15 +08:00
2022-10-17 15:02:08 +08:00
self . _model_def = [
ModelDef (
field = underscore ( k ) ,
field_alias = k ,
2022-10-18 03:10:36 +08:00
field_type = field_type_generator ( k , v ) ,
2022-10-25 00:18:54 +08:00
field_value = v . default
2022-10-17 15:02:08 +08:00
)
2022-10-18 03:10:36 +08:00
for ( k , v ) in self . _class_data . items ( ) if k not in API_NOT_ALLOWED
2022-10-17 15:02:08 +08:00
]
2022-10-30 03:45:29 +08:00
2022-10-19 03:04:56 +08:00
for fields in additional_fields :
self . _model_def . append ( ModelDef (
2022-10-30 03:45:29 +08:00
field = underscore ( fields [ " key " ] ) ,
field_alias = fields [ " key " ] ,
2022-10-19 03:04:56 +08:00
field_type = fields [ " type " ] ,
2022-10-24 23:16:07 +08:00
field_value = fields [ " default " ] ,
field_exclude = fields [ " exclude " ] if " exclude " in fields else False ) )
2022-10-17 15:02:08 +08:00
def generate_model ( self ) :
"""
Creates a pydantic BaseModel
from the json and overrides provided at initialization
"""
fields = {
2022-10-24 23:16:07 +08:00
d . field : ( d . field_type , Field ( default = d . field_value , alias = d . field_alias , exclude = d . field_exclude ) ) for d in self . _model_def
2022-10-17 15:02:08 +08:00
}
DynamicModel = create_model ( self . _model_name , * * fields )
DynamicModel . __config__ . allow_population_by_field_name = True
2022-10-18 03:10:36 +08:00
DynamicModel . __config__ . allow_mutation = True
2022-10-17 15:02:08 +08:00
return DynamicModel
2022-10-30 03:45:29 +08:00
2022-10-22 07:27:40 +08:00
StableDiffusionTxt2ImgProcessingAPI = PydanticModelGenerator (
2022-10-30 03:45:29 +08:00
" StableDiffusionProcessingTxt2Img " ,
2022-10-19 03:04:56 +08:00
StableDiffusionProcessingTxt2Img ,
2023-03-03 22:00:52 +08:00
[
{ " key " : " sampler_index " , " type " : str , " default " : " Euler " } ,
{ " key " : " script_name " , " type " : str , " default " : None } ,
{ " key " : " script_args " , " type " : list , " default " : [ ] } ,
2023-03-11 18:22:59 +08:00
{ " key " : " send_images " , " type " : bool , " default " : True } ,
{ " key " : " save_images " , " type " : bool , " default " : False } ,
2023-03-12 01:33:35 +08:00
{ " key " : " alwayson_scripts " , " type " : dict , " default " : { } } ,
2023-03-03 22:00:52 +08:00
]
2022-10-22 07:27:40 +08:00
) . generate_model ( )
StableDiffusionImg2ImgProcessingAPI = PydanticModelGenerator (
2022-10-30 03:45:29 +08:00
" StableDiffusionProcessingImg2Img " ,
2022-10-22 07:27:40 +08:00
StableDiffusionProcessingImg2Img ,
2023-03-03 22:00:52 +08:00
[
{ " key " : " sampler_index " , " type " : str , " default " : " Euler " } ,
{ " key " : " init_images " , " type " : list , " default " : None } ,
{ " key " : " denoising_strength " , " type " : float , " default " : 0.75 } ,
{ " key " : " mask " , " type " : str , " default " : None } ,
{ " key " : " include_init_images " , " type " : bool , " default " : False , " exclude " : True } ,
{ " key " : " script_name " , " type " : str , " default " : None } ,
{ " key " : " script_args " , " type " : list , " default " : [ ] } ,
2023-03-11 18:22:59 +08:00
{ " key " : " send_images " , " type " : bool , " default " : True } ,
{ " key " : " save_images " , " type " : bool , " default " : False } ,
2023-03-12 01:33:35 +08:00
{ " key " : " alwayson_scripts " , " type " : dict , " default " : { } } ,
2023-03-03 22:00:52 +08:00
]
2022-10-28 03:20:15 +08:00
) . generate_model ( )
2022-10-23 07:24:04 +08:00
class TextToImageResponse ( BaseModel ) :
2022-11-04 06:55:03 +08:00
images : List [ str ] = Field ( default = None , title = " Image " , description = " The generated image in base64 format. " )
2022-10-24 02:35:49 +08:00
parameters : dict
info : str
class ImageToImageResponse ( BaseModel ) :
2022-11-04 06:55:03 +08:00
images : List [ str ] = Field ( default = None , title = " Image " , description = " The generated image in base64 format. " )
2022-10-24 02:35:49 +08:00
parameters : dict
2022-10-24 02:13:37 +08:00
info : str
2022-10-23 07:24:04 +08:00
2022-10-23 10:13:32 +08:00
class ExtrasBaseRequest ( BaseModel ) :
resize_mode : Literal [ 0 , 1 ] = Field ( default = 0 , title = " Resize Mode " , description = " Sets the resize mode: 0 to upscale by upscaling_resize amount, 1 to upscale up to upscaling_resize_h x upscaling_resize_w. " )
show_extras_results : bool = Field ( default = True , title = " Show results " , description = " Should the backend return the generated image? " )
gfpgan_visibility : float = Field ( default = 0 , title = " GFPGAN Visibility " , ge = 0 , le = 1 , allow_inf_nan = False , description = " Sets the visibility of GFPGAN, values should be between 0 and 1. " )
codeformer_visibility : float = Field ( default = 0 , title = " CodeFormer Visibility " , ge = 0 , le = 1 , allow_inf_nan = False , description = " Sets the visibility of CodeFormer, values should be between 0 and 1. " )
codeformer_weight : float = Field ( default = 0 , title = " CodeFormer Weight " , ge = 0 , le = 1 , allow_inf_nan = False , description = " Sets the weight of CodeFormer, values should be between 0 and 1. " )
2023-01-07 06:00:12 +08:00
upscaling_resize : float = Field ( default = 2 , title = " Upscaling Factor " , ge = 1 , le = 8 , description = " By how much to upscale the image, only used when resize_mode=0. " )
2022-10-23 10:13:32 +08:00
upscaling_resize_w : int = Field ( default = 512 , title = " Target Width " , ge = 1 , description = " Target width for the upscaler to hit. Only used when resize_mode=1. " )
upscaling_resize_h : int = Field ( default = 512 , title = " Target Height " , ge = 1 , description = " Target height for the upscaler to hit. Only used when resize_mode=1. " )
2022-12-15 10:01:32 +08:00
upscaling_crop : bool = Field ( default = True , title = " Crop to fit " , description = " Should the upscaler crop the image to fit in the chosen size? " )
2022-10-23 10:13:32 +08:00
upscaler_1 : str = Field ( default = " None " , title = " Main upscaler " , description = f " The name of the main upscaler to use, it has to be one of this list: { ' , ' . join ( [ x . name for x in sd_upscalers ] ) } " )
upscaler_2 : str = Field ( default = " None " , title = " Secondary upscaler " , description = f " The name of the secondary upscaler to use, it has to be one of this list: { ' , ' . join ( [ x . name for x in sd_upscalers ] ) } " )
extras_upscaler_2_visibility : float = Field ( default = 0 , title = " Secondary upscaler visibility " , ge = 0 , le = 1 , allow_inf_nan = False , description = " Sets the visibility of secondary upscaler, values should be between 0 and 1. " )
2022-11-04 08:35:18 +08:00
upscale_first : bool = Field ( default = False , title = " Upscale first " , description = " Should the upscaler run before restoring faces? " )
2022-10-23 10:13:32 +08:00
class ExtraBaseResponse ( BaseModel ) :
2022-10-24 03:03:30 +08:00
html_info : str = Field ( title = " HTML info " , description = " A series of HTML tags containing the process info. " )
2022-10-23 10:13:32 +08:00
class ExtrasSingleImageRequest ( ExtrasBaseRequest ) :
image : str = Field ( default = " " , title = " Image " , description = " Image to work on, must be a Base64 string containing the image ' s data. " )
class ExtrasSingleImageResponse ( ExtraBaseResponse ) :
2022-10-24 00:07:59 +08:00
image : str = Field ( default = None , title = " Image " , description = " The generated image in base64 format. " )
2022-10-24 19:32:18 +08:00
class FileData ( BaseModel ) :
data : str = Field ( title = " File data " , description = " Base64 representation of the file " )
name : str = Field ( title = " File name " )
2022-10-24 00:07:59 +08:00
class ExtrasBatchImagesRequest ( ExtrasBaseRequest ) :
2022-11-04 06:55:03 +08:00
imageList : List [ FileData ] = Field ( title = " Images " , description = " List of images to work on. Must be Base64 strings " )
2022-10-24 00:07:59 +08:00
class ExtrasBatchImagesResponse ( ExtraBaseResponse ) :
2022-11-04 06:55:03 +08:00
images : List [ str ] = Field ( title = " Images " , description = " The generated images in base64 format. " )
2022-10-30 03:45:29 +08:00
2022-10-30 03:09:19 +08:00
class PNGInfoRequest ( BaseModel ) :
image : str = Field ( title = " Image " , description = " The base64 encoded PNG image " )
class PNGInfoResponse ( BaseModel ) :
2023-01-05 04:36:30 +08:00
info : str = Field ( title = " Image info " , description = " A string with the parameters used to generate the image " )
items : dict = Field ( title = " Items " , description = " An object containing all the info the image had " )
2022-10-30 03:47:24 +08:00
2022-10-30 06:03:32 +08:00
class ProgressRequest ( BaseModel ) :
skip_current_image : bool = Field ( default = False , title = " Skip current image " , description = " Skip current image serialization " )
2022-10-30 03:45:29 +08:00
class ProgressResponse ( BaseModel ) :
2022-10-30 03:55:43 +08:00
progress : float = Field ( title = " Progress " , description = " The progress with a range of 0 to 1 " )
eta_relative : float = Field ( title = " ETA in secs " )
2022-10-30 05:04:29 +08:00
state : dict = Field ( title = " State " , description = " The current state snapshot " )
2022-10-30 05:19:17 +08:00
current_image : str = Field ( default = None , title = " Current image " , description = " The current image in base64 format. opts.show_progress_every_n_steps is required for this to work. " )
2023-01-11 23:23:51 +08:00
textinfo : str = Field ( default = None , title = " Info text " , description = " Info text used by WebUI. " )
2022-10-31 23:45:52 +08:00
class InterrogateRequest ( BaseModel ) :
image : str = Field ( default = " " , title = " Image " , description = " Image to work on, must be a Base64 string containing the image ' s data. " )
2022-11-07 02:32:06 +08:00
model : str = Field ( default = " clip " , title = " Model " , description = " The interrogate model used. " )
2022-10-31 23:45:52 +08:00
class InterrogateResponse ( BaseModel ) :
caption : str = Field ( default = None , title = " Caption " , description = " The generated caption for the image. " )
2022-11-06 16:27:54 +08:00
2022-12-25 07:02:22 +08:00
class TrainResponse ( BaseModel ) :
info : str = Field ( title = " Train info " , description = " Response string from train embedding or hypernetwork task. " )
class CreateResponse ( BaseModel ) :
info : str = Field ( title = " Create info " , description = " Response string from create embedding or hypernetwork task. " )
class PreprocessResponse ( BaseModel ) :
info : str = Field ( title = " Preprocess info " , description = " Response string from preprocessing task. " )
2022-11-03 11:51:22 +08:00
fields = { }
2022-11-06 05:46:47 +08:00
for key , metadata in opts . data_labels . items ( ) :
value = opts . data . get ( key )
2023-07-26 15:47:12 +08:00
optType = opts . typemap . get ( type ( metadata . default ) , type ( metadata . default ) ) if metadata . default else Any
2022-11-03 11:51:22 +08:00
2023-07-26 15:47:12 +08:00
if metadata is not None :
2023-07-18 16:48:40 +08:00
fields . update ( { key : ( Optional [ optType ] , Field ( default = metadata . default , description = metadata . label ) ) } )
2022-11-03 11:51:22 +08:00
else :
fields . update ( { key : ( Optional [ optType ] , Field ( ) ) } )
OptionsModel = create_model ( " Options " , * * fields )
flags = { }
_options = vars ( parser ) [ ' _option_string_actions ' ]
for key in _options :
if ( _options [ key ] . dest != ' help ' ) :
flag = _options [ key ]
2022-11-05 11:38:24 +08:00
_type = str
2023-05-10 13:25:25 +08:00
if _options [ key ] . default is not None :
_type = type ( _options [ key ] . default )
flags . update ( { flag . dest : ( _type , Field ( default = flag . default , description = flag . help ) ) } )
2022-11-03 11:51:22 +08:00
FlagsModel = create_model ( " Flags " , * * flags )
class SamplerItem ( BaseModel ) :
name : str = Field ( title = " Name " )
Python 3.8 typing compatibility
Solves problems with
```Traceback (most recent call last):
File "webui.py", line 201, in <module>
webui()
File "webui.py", line 178, in webui
create_api(app)
File "webui.py", line 117, in create_api
from modules.api.api import Api
File "H:\AIart\stable-diffusion\stable-diffusion-webui\modules\api\api.py", line 9, in <module>
from modules.api.models import *
File "H:\AIart\stable-diffusion\stable-diffusion-webui\modules\api\models.py", line 194, in <module>
class SamplerItem(BaseModel):
File "H:\AIart\stable-diffusion\stable-diffusion-webui\modules\api\models.py", line 196, in SamplerItem
aliases: list[str] = Field(title="Aliases")
TypeError: 'type' object is not subscriptable```
and
```Traceback (most recent call last):
File "webui.py", line 201, in <module>
webui()
File "webui.py", line 178, in webui
create_api(app)
File "webui.py", line 117, in create_api
from modules.api.api import Api
File "H:\AIart\stable-diffusion\stable-diffusion-webui\modules\api\api.py", line 9, in <module>
from modules.api.models import *
File "H:\AIart\stable-diffusion\stable-diffusion-webui\modules\api\models.py", line 194, in <module>
class SamplerItem(BaseModel):
File "H:\AIart\stable-diffusion\stable-diffusion-webui\modules\api\models.py", line 197, in SamplerItem
options: dict[str, str] = Field(title="Options")
TypeError: 'type' object is not subscriptable```
2022-11-05 22:06:56 +08:00
aliases : List [ str ] = Field ( title = " Aliases " )
options : Dict [ str , str ] = Field ( title = " Options " )
2022-11-03 11:51:22 +08:00
class UpscalerItem ( BaseModel ) :
name : str = Field ( title = " Name " )
2022-11-05 11:38:24 +08:00
model_name : Optional [ str ] = Field ( title = " Model Name " )
model_path : Optional [ str ] = Field ( title = " Path " )
2023-01-24 15:09:30 +08:00
model_url : Optional [ str ] = Field ( title = " URL " )
2023-01-24 15:05:45 +08:00
scale : Optional [ float ] = Field ( title = " Scale " )
2022-11-03 11:51:22 +08:00
2023-06-04 21:59:23 +08:00
class LatentUpscalerModeItem ( BaseModel ) :
name : str = Field ( title = " Name " )
2022-11-03 11:51:22 +08:00
class SDModelItem ( BaseModel ) :
title : str = Field ( title = " Title " )
model_name : str = Field ( title = " Model Name " )
2023-01-14 14:56:59 +08:00
hash : Optional [ str ] = Field ( title = " Short hash " )
sha256 : Optional [ str ] = Field ( title = " sha256 hash " )
2022-11-03 11:51:22 +08:00
filename : str = Field ( title = " Filename " )
2023-01-27 16:54:19 +08:00
config : Optional [ str ] = Field ( title = " Config file " )
2022-11-03 11:51:22 +08:00
2023-05-30 05:25:43 +08:00
class SDVaeItem ( BaseModel ) :
model_name : str = Field ( title = " Model Name " )
filename : str = Field ( title = " Filename " )
2022-11-03 11:51:22 +08:00
class HypernetworkItem ( BaseModel ) :
name : str = Field ( title = " Name " )
2022-11-05 11:38:24 +08:00
path : Optional [ str ] = Field ( title = " Path " )
2022-11-03 11:51:22 +08:00
class FaceRestorerItem ( BaseModel ) :
name : str = Field ( title = " Name " )
2022-11-05 11:38:24 +08:00
cmd_dir : Optional [ str ] = Field ( title = " Path " )
2022-11-03 11:51:22 +08:00
class RealesrganItem ( BaseModel ) :
name : str = Field ( title = " Name " )
2022-11-05 11:38:24 +08:00
path : Optional [ str ] = Field ( title = " Path " )
scale : Optional [ int ] = Field ( title = " Scale " )
2022-11-03 11:51:22 +08:00
class PromptStyleItem ( BaseModel ) :
name : str = Field ( title = " Name " )
2022-11-05 11:38:24 +08:00
prompt : Optional [ str ] = Field ( title = " Prompt " )
negative_prompt : Optional [ str ] = Field ( title = " Negative Prompt " )
2022-11-03 11:51:22 +08:00
2022-11-06 16:27:54 +08:00
2023-01-02 09:21:22 +08:00
class EmbeddingItem ( BaseModel ) :
step : Optional [ int ] = Field ( title = " Step " , description = " The number of steps that were used to train this embedding, if available " )
sd_checkpoint : Optional [ str ] = Field ( title = " SD Checkpoint " , description = " The hash of the checkpoint this embedding was trained on, if available " )
sd_checkpoint_name : Optional [ str ] = Field ( title = " SD Checkpoint Name " , description = " The name of the checkpoint this embedding was trained on, if available. Note that this is the name that was used by the trainer; for a stable identifier, use `sd_checkpoint` instead " )
shape : int = Field ( title = " Shape " , description = " The length of each individual vector in the embedding " )
vectors : int = Field ( title = " Vectors " , description = " The number of vectors in the embedding " )
2023-01-02 07:17:33 +08:00
class EmbeddingsResponse ( BaseModel ) :
2023-01-02 09:21:22 +08:00
loaded : Dict [ str , EmbeddingItem ] = Field ( title = " Loaded " , description = " Embeddings loaded for the current model " )
2023-01-05 04:36:30 +08:00
skipped : Dict [ str , EmbeddingItem ] = Field ( title = " Skipped " , description = " Embeddings skipped for the current model (likely due to architecture incompatibility) " )
2023-01-07 20:51:35 +08:00
class MemoryResponse ( BaseModel ) :
2023-01-10 10:23:58 +08:00
ram : dict = Field ( title = " RAM " , description = " System memory stats " )
cuda : dict = Field ( title = " CUDA " , description = " nVidia CUDA memory stats " )
2023-03-04 11:46:07 +08:00
2023-05-18 03:43:24 +08:00
2023-03-04 11:46:07 +08:00
class ScriptsList ( BaseModel ) :
2023-05-18 03:43:24 +08:00
txt2img : list = Field ( default = None , title = " Txt2img " , description = " Titles of scripts (txt2img) " )
img2img : list = Field ( default = None , title = " Img2img " , description = " Titles of scripts (img2img) " )
class ScriptArg ( BaseModel ) :
label : str = Field ( default = None , title = " Label " , description = " Name of the argument in UI " )
value : Optional [ Any ] = Field ( default = None , title = " Value " , description = " Default value of the argument " )
minimum : Optional [ Any ] = Field ( default = None , title = " Minimum " , description = " Minimum allowed value for the argumentin UI " )
maximum : Optional [ Any ] = Field ( default = None , title = " Minimum " , description = " Maximum allowed value for the argumentin UI " )
step : Optional [ Any ] = Field ( default = None , title = " Minimum " , description = " Step for changing value of the argumentin UI " )
choices : Optional [ List [ str ] ] = Field ( default = None , title = " Choices " , description = " Possible values for the argument " )
class ScriptInfo ( BaseModel ) :
name : str = Field ( default = None , title = " Name " , description = " Script name " )
is_alwayson : bool = Field ( default = None , title = " IsAlwayson " , description = " Flag specifying whether this script is an alwayson script " )
is_img2img : bool = Field ( default = None , title = " IsImg2img " , description = " Flag specifying whether this script is an img2img script " )
args : List [ ScriptArg ] = Field ( title = " Arguments " , description = " List of script ' s arguments " )