api.py 8.4 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29
# import time

# from modules.api.models import StableDiffusionTxt2ImgProcessingAPI, StableDiffusionImg2ImgProcessingAPI
# from modules.processing import StableDiffusionProcessingTxt2Img, StableDiffusionProcessingImg2Img, process_images
# from modules.sd_samplers import all_samplers
# from modules.extras import run_pnginfo
# import modules.shared as shared
# from modules import devices
# import uvicorn
# from fastapi import Body, APIRouter, HTTPException
# from fastapi.responses import JSONResponse
# from pydantic import BaseModel, Field, Json
# from typing import List
# import json
# import io
# import base64
# from PIL import Image

# sampler_to_index = lambda name: next(filter(lambda row: name.lower() == row[1].name.lower(), enumerate(all_samplers)), None)

# class TextToImageResponse(BaseModel):
#     images: List[str] = Field(default=None, title="Image", description="The generated image in base64 format.")
#     parameters: Json
#     info: Json

# class ImageToImageResponse(BaseModel):
#     images: List[str] = Field(default=None, title="Image", description="The generated image in base64 format.")
#     parameters: Json
#     info: Json
E
evshiron 已提交
30

31
import time
32
import uvicorn
B
Bruno Seoane 已提交
33
from gradio.processing_utils import encode_pil_to_base64, decode_base64_to_file, decode_base64_to_image
B
Bruno Seoane 已提交
34
from fastapi import APIRouter, HTTPException
35
import modules.shared as shared
E
evshiron 已提交
36
from modules import devices
37
from modules.api.models import *
38 39
from modules.processing import StableDiffusionProcessingTxt2Img, StableDiffusionProcessingImg2Img, process_images
from modules.sd_samplers import all_samplers
B
Bruno Seoane 已提交
40
from modules.extras import run_extras, run_pnginfo
E
evshiron 已提交
41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66

# 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()
67

B
Bruno Seoane 已提交
68 69 70 71
def upscaler_to_index(name: str):
    try:
        return [x.name.lower() for x in shared.sd_upscalers].index(name.lower())
    except:
B
Bruno Seoane 已提交
72
        raise HTTPException(status_code=400, detail=f"Invalid upscaler, needs to be on of these: {' , '.join([x.name for x in sd_upscalers])}")
73

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

B
Bruno Seoane 已提交
76 77 78 79 80 81 82 83
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

84
class Api:
A
arcticfaded 已提交
85
    def __init__(self, app, queue_lock):
86
        self.router = APIRouter()
A
arcticfaded 已提交
87 88
        self.app = app
        self.queue_lock = queue_lock
B
Bruno Seoane 已提交
89
        self.app.add_api_route("/sdapi/v1/txt2img", self.text2imgapi, methods=["POST"], response_model=TextToImageResponse)
90
        self.app.add_api_route("/sdapi/v1/img2img", self.img2imgapi, methods=["POST"], response_model=ImageToImageResponse)
B
Bruno Seoane 已提交
91
        self.app.add_api_route("/sdapi/v1/extra-single-image", self.extras_single_image_api, methods=["POST"], response_model=ExtrasSingleImageResponse)
92
        self.app.add_api_route("/sdapi/v1/extra-batch-images", self.extras_batch_images_api, methods=["POST"], response_model=ExtrasBatchImagesResponse)
B
Bruno Seoane 已提交
93
        self.app.add_api_route("/sdapi/v1/png-info", self.pnginfoapi, methods=["POST"], response_model=PNGInfoResponse)
E
evshiron 已提交
94
        self.app.add_api_route("/sdapi/v1/progress", self.progressapi, methods=["GET"])
95

96
    def text2imgapi(self, txt2imgreq: StableDiffusionTxt2ImgProcessingAPI):
A
arcticfaded 已提交
97
        sampler_index = sampler_to_index(txt2imgreq.sampler_index)
E
evshiron 已提交
98

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

A
arcticfaded 已提交
102
        populate = txt2imgreq.copy(update={ # Override __init__ params
E
evshiron 已提交
103
            "sd_model": shared.sd_model,
A
arcticfaded 已提交
104
            "sampler_index": sampler_index[0],
A
arcticfaded 已提交
105 106
            "do_not_save_samples": True,
            "do_not_save_grid": True
A
arcticfaded 已提交
107 108 109 110
            }
        )
        p = StableDiffusionProcessingTxt2Img(**vars(populate))
        # Override object param
E
evshiron 已提交
111
        before_gpu_call()
A
arcticfaded 已提交
112 113
        with self.queue_lock:
            processed = process_images(p)
E
evshiron 已提交
114 115
        after_gpu_call()

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

118
        return TextToImageResponse(images=b64images, parameters=vars(txt2imgreq), info=processed.js())
119

120 121
    def img2imgapi(self, img2imgreq: StableDiffusionImg2ImgProcessingAPI):
        sampler_index = sampler_to_index(img2imgreq.sampler_index)
E
evshiron 已提交
122

123
        if sampler_index is None:
E
evshiron 已提交
124
            raise HTTPException(status_code=404, detail="Sampler not found")
125 126 127 128


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

S
Stephen 已提交
131 132
        mask = img2imgreq.mask
        if mask:
B
Bruno Seoane 已提交
133
            mask = decode_base64_to_image(mask)
S
Stephen 已提交
134

E
evshiron 已提交
135

136
        populate = img2imgreq.copy(update={ # Override __init__ params
E
evshiron 已提交
137
            "sd_model": shared.sd_model,
138 139
            "sampler_index": sampler_index[0],
            "do_not_save_samples": True,
E
evshiron 已提交
140
            "do_not_save_grid": True,
S
Stephen 已提交
141
            "mask": mask
142 143 144 145 146 147
            }
        )
        p = StableDiffusionProcessingImg2Img(**vars(populate))

        imgs = []
        for img in init_images:
B
Bruno Seoane 已提交
148
            img = decode_base64_to_image(img)
149 150 151 152
            imgs = [img] * p.batch_size

        p.init_images = imgs
        # Override object param
E
evshiron 已提交
153
        before_gpu_call()
154 155
        with self.queue_lock:
            processed = process_images(p)
E
evshiron 已提交
156 157
        after_gpu_call()

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

160 161 162 163
        if (not img2imgreq.include_init_images):
            img2imgreq.init_images = None
            img2imgreq.mask = None

164
        return ImageToImageResponse(images=b64images, parameters=vars(img2imgreq), info=processed.js())
165

B
Bruno Seoane 已提交
166
    def extras_single_image_api(self, req: ExtrasSingleImageRequest):
B
Bruno Seoane 已提交
167
        reqDict = setUpscalers(req)
B
Bruno Seoane 已提交
168

B
Bruno Seoane 已提交
169
        reqDict['image'] = decode_base64_to_image(reqDict['image'])
B
Bruno Seoane 已提交
170 171

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

B
Bruno Seoane 已提交
174
        return ExtrasSingleImageResponse(image=encode_pil_to_base64(result[0][0]), html_info=result[1])
175 176

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

B
Bruno Seoane 已提交
179 180 181 182 183 184
        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']))
185 186 187
        reqDict.pop('imageList')

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

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

B
Bruno Seoane 已提交
192
    def pnginfoapi(self, req: PNGInfoRequest):
B
Bruno Seoane 已提交
193 194 195 196 197 198
        if(not req.image.strip()):
            return PNGInfoResponse(info="")

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

        return PNGInfoResponse(info=result[1])
199

E
evshiron 已提交
200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221
    def progressapi(self):
        # copy from check_progress_call of ui.py

        if shared.state.job_count == 0:
            return ProgressResponse(progress=0, eta_relative=0, state=shared.state.js())

        # 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)

        return ProgressResponse(progress=progress, eta_relative=eta_relative, state=shared.state.js())

222
    def launch(self, server_name, port):
A
arcticfaded 已提交
223 224
        self.app.include_router(self.router)
        uvicorn.run(self.app, host=server_name, port=port)