api.py 7.4 KB
Newer Older
1
import time
2
import uvicorn
B
Bruno Seoane 已提交
3
from gradio.processing_utils import encode_pil_to_base64, decode_base64_to_file, decode_base64_to_image
B
Bruno Seoane 已提交
4
from fastapi import APIRouter, HTTPException
5
import modules.shared as shared
E
evshiron 已提交
6
from modules import devices
7
from modules.api.models import *
8 9
from modules.processing import StableDiffusionProcessingTxt2Img, StableDiffusionProcessingImg2Img, process_images
from modules.sd_samplers import all_samplers
B
Bruno Seoane 已提交
10
from modules.extras import run_extras, run_pnginfo
E
evshiron 已提交
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

# copy from wrap_gradio_gpu_call of webui.py
# because queue lock will be acquired in api handlers
# and time start needs to be set
# the function has been modified into two parts

def before_gpu_call():
    devices.torch_gc()

    shared.state.sampling_step = 0
    shared.state.job_count = -1
    shared.state.job_no = 0
    shared.state.job_timestamp = shared.state.get_job_timestamp()
    shared.state.current_latent = None
    shared.state.current_image = None
    shared.state.current_image_sampling_step = 0
    shared.state.skipped = False
    shared.state.interrupted = False
    shared.state.textinfo = None
    shared.state.time_start = time.time()

def after_gpu_call():
    shared.state.job = ""
    shared.state.job_count = 0

    devices.torch_gc()
37

B
Bruno Seoane 已提交
38 39 40 41
def upscaler_to_index(name: str):
    try:
        return [x.name.lower() for x in shared.sd_upscalers].index(name.lower())
    except:
B
Bruno Seoane 已提交
42
        raise HTTPException(status_code=400, detail=f"Invalid upscaler, needs to be on of these: {' , '.join([x.name for x in sd_upscalers])}")
43

A
arcticfaded 已提交
44
sampler_to_index = lambda name: next(filter(lambda row: name.lower() == row[1].name.lower(), enumerate(all_samplers)), None)
A
arcticfaded 已提交
45

B
Bruno Seoane 已提交
46 47 48 49 50 51 52 53
def setUpscalers(req: dict):
    reqDict = vars(req)
    reqDict['extras_upscaler_1'] = upscaler_to_index(req.upscaler_1)
    reqDict['extras_upscaler_2'] = upscaler_to_index(req.upscaler_2)
    reqDict.pop('upscaler_1')
    reqDict.pop('upscaler_2')
    return reqDict

54
class Api:
A
arcticfaded 已提交
55
    def __init__(self, app, queue_lock):
56
        self.router = APIRouter()
A
arcticfaded 已提交
57 58
        self.app = app
        self.queue_lock = queue_lock
B
Bruno Seoane 已提交
59
        self.app.add_api_route("/sdapi/v1/txt2img", self.text2imgapi, methods=["POST"], response_model=TextToImageResponse)
60
        self.app.add_api_route("/sdapi/v1/img2img", self.img2imgapi, methods=["POST"], response_model=ImageToImageResponse)
B
Bruno Seoane 已提交
61
        self.app.add_api_route("/sdapi/v1/extra-single-image", self.extras_single_image_api, methods=["POST"], response_model=ExtrasSingleImageResponse)
62
        self.app.add_api_route("/sdapi/v1/extra-batch-images", self.extras_batch_images_api, methods=["POST"], response_model=ExtrasBatchImagesResponse)
B
Bruno Seoane 已提交
63
        self.app.add_api_route("/sdapi/v1/png-info", self.pnginfoapi, methods=["POST"], response_model=PNGInfoResponse)
E
evshiron 已提交
64
        self.app.add_api_route("/sdapi/v1/progress", self.progressapi, methods=["GET"], response_model=ProgressResponse)
65

66
    def text2imgapi(self, txt2imgreq: StableDiffusionTxt2ImgProcessingAPI):
A
arcticfaded 已提交
67
        sampler_index = sampler_to_index(txt2imgreq.sampler_index)
E
evshiron 已提交
68

A
arcticfaded 已提交
69
        if sampler_index is None:
E
evshiron 已提交
70 71
            raise HTTPException(status_code=404, detail="Sampler not found")

A
arcticfaded 已提交
72
        populate = txt2imgreq.copy(update={ # Override __init__ params
E
evshiron 已提交
73
            "sd_model": shared.sd_model,
A
arcticfaded 已提交
74
            "sampler_index": sampler_index[0],
A
arcticfaded 已提交
75 76
            "do_not_save_samples": True,
            "do_not_save_grid": True
A
arcticfaded 已提交
77 78 79 80
            }
        )
        p = StableDiffusionProcessingTxt2Img(**vars(populate))
        # Override object param
E
evshiron 已提交
81
        before_gpu_call()
A
arcticfaded 已提交
82 83
        with self.queue_lock:
            processed = process_images(p)
E
evshiron 已提交
84 85
        after_gpu_call()

B
Bruno Seoane 已提交
86
        b64images = list(map(encode_pil_to_base64, processed.images))
E
evshiron 已提交
87

88
        return TextToImageResponse(images=b64images, parameters=vars(txt2imgreq), info=processed.js())
89

90 91
    def img2imgapi(self, img2imgreq: StableDiffusionImg2ImgProcessingAPI):
        sampler_index = sampler_to_index(img2imgreq.sampler_index)
E
evshiron 已提交
92

93
        if sampler_index is None:
E
evshiron 已提交
94
            raise HTTPException(status_code=404, detail="Sampler not found")
95 96 97 98


        init_images = img2imgreq.init_images
        if init_images is None:
E
evshiron 已提交
99
            raise HTTPException(status_code=404, detail="Init image not found")
100

S
Stephen 已提交
101 102
        mask = img2imgreq.mask
        if mask:
B
Bruno Seoane 已提交
103
            mask = decode_base64_to_image(mask)
S
Stephen 已提交
104

E
evshiron 已提交
105

106
        populate = img2imgreq.copy(update={ # Override __init__ params
E
evshiron 已提交
107
            "sd_model": shared.sd_model,
108 109
            "sampler_index": sampler_index[0],
            "do_not_save_samples": True,
E
evshiron 已提交
110
            "do_not_save_grid": True,
S
Stephen 已提交
111
            "mask": mask
112 113 114 115 116 117
            }
        )
        p = StableDiffusionProcessingImg2Img(**vars(populate))

        imgs = []
        for img in init_images:
B
Bruno Seoane 已提交
118
            img = decode_base64_to_image(img)
119 120 121 122
            imgs = [img] * p.batch_size

        p.init_images = imgs
        # Override object param
E
evshiron 已提交
123
        before_gpu_call()
124 125
        with self.queue_lock:
            processed = process_images(p)
E
evshiron 已提交
126 127
        after_gpu_call()

B
Bruno Seoane 已提交
128
        b64images = list(map(encode_pil_to_base64, processed.images))
129

130 131 132 133
        if (not img2imgreq.include_init_images):
            img2imgreq.init_images = None
            img2imgreq.mask = None

134
        return ImageToImageResponse(images=b64images, parameters=vars(img2imgreq), info=processed.js())
135

B
Bruno Seoane 已提交
136
    def extras_single_image_api(self, req: ExtrasSingleImageRequest):
B
Bruno Seoane 已提交
137
        reqDict = setUpscalers(req)
B
Bruno Seoane 已提交
138

B
Bruno Seoane 已提交
139
        reqDict['image'] = decode_base64_to_image(reqDict['image'])
B
Bruno Seoane 已提交
140 141

        with self.queue_lock:
B
Bruno Seoane 已提交
142
            result = run_extras(extras_mode=0, image_folder="", input_dir="", output_dir="", **reqDict)
B
Bruno Seoane 已提交
143

B
Bruno Seoane 已提交
144
        return ExtrasSingleImageResponse(image=encode_pil_to_base64(result[0][0]), html_info=result[1])
145 146

    def extras_batch_images_api(self, req: ExtrasBatchImagesRequest):
B
Bruno Seoane 已提交
147
        reqDict = setUpscalers(req)
148

B
Bruno Seoane 已提交
149 150 151 152 153 154
        def prepareFiles(file):
            file = decode_base64_to_file(file.data, file_path=file.name)
            file.orig_name = file.name
            return file

        reqDict['image_folder'] = list(map(prepareFiles, reqDict['imageList']))
155 156 157
        reqDict.pop('imageList')

        with self.queue_lock:
B
Bruno Seoane 已提交
158
            result = run_extras(extras_mode=1, image="", input_dir="", output_dir="", **reqDict)
159

B
Bruno Seoane 已提交
160
        return ExtrasBatchImagesResponse(images=list(map(encode_pil_to_base64, result[0])), html_info=result[1])
161

B
Bruno Seoane 已提交
162
    def pnginfoapi(self, req: PNGInfoRequest):
B
Bruno Seoane 已提交
163 164 165 166 167 168
        if(not req.image.strip()):
            return PNGInfoResponse(info="")

        result = run_pnginfo(decode_base64_to_image(req.image.strip()))

        return PNGInfoResponse(info=result[1])
169

E
evshiron 已提交
170 171 172 173
    def progressapi(self):
        # copy from check_progress_call of ui.py

        if shared.state.job_count == 0:
E
evshiron 已提交
174
            return ProgressResponse(progress=0, eta_relative=0, state=shared.state.dict())
E
evshiron 已提交
175 176 177 178 179 180 181 182 183 184 185 186 187 188 189

        # avoid dividing zero
        progress = 0.01

        if shared.state.job_count > 0:
            progress += shared.state.job_no / shared.state.job_count
        if shared.state.sampling_steps > 0:
            progress += 1 / shared.state.job_count * shared.state.sampling_step / shared.state.sampling_steps

        time_since_start = time.time() - shared.state.time_start
        eta = (time_since_start/progress)
        eta_relative = eta-time_since_start

        progress = min(progress, 1)

E
evshiron 已提交
190
        return ProgressResponse(progress=progress, eta_relative=eta_relative, state=shared.state.dict())
E
evshiron 已提交
191

192
    def launch(self, server_name, port):
A
arcticfaded 已提交
193 194
        self.app.include_router(self.router)
        uvicorn.run(self.app, host=server_name, port=port)