2022-09-16 17:18:15 +00:00
|
|
|
import mimetypes
|
|
|
|
import transformers
|
|
|
|
import json
|
|
|
|
import os
|
|
|
|
import traceback
|
|
|
|
import eventlet
|
|
|
|
import glob
|
|
|
|
import shlex
|
2022-09-18 07:33:09 +00:00
|
|
|
import math
|
|
|
|
import shutil
|
2022-09-21 14:32:26 +00:00
|
|
|
import sys
|
|
|
|
|
|
|
|
sys.path.append(".")
|
|
|
|
|
|
|
|
from argparse import ArgumentTypeError
|
|
|
|
from modules.create_cmd_parser import create_cmd_parser
|
|
|
|
|
|
|
|
parser = create_cmd_parser()
|
|
|
|
opt = parser.parse_args()
|
|
|
|
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
from flask_socketio import SocketIO
|
|
|
|
from flask import Flask, send_from_directory, url_for, jsonify
|
|
|
|
from pathlib import Path
|
|
|
|
from PIL import Image
|
|
|
|
from pytorch_lightning import logging
|
|
|
|
from threading import Event
|
|
|
|
from uuid import uuid4
|
2022-09-17 07:19:09 +00:00
|
|
|
from send2trash import send2trash
|
2022-09-16 17:18:15 +00:00
|
|
|
|
2022-09-21 14:32:26 +00:00
|
|
|
|
2022-09-16 17:18:15 +00:00
|
|
|
from ldm.generate import Generate
|
2022-10-08 15:37:23 +00:00
|
|
|
from ldm.invoke.restoration import Restoration
|
|
|
|
from ldm.invoke.pngwriter import PngWriter, retrieve_metadata
|
|
|
|
from ldm.invoke.args import APP_ID, APP_VERSION, calculate_init_img_hash
|
2022-10-24 09:16:52 +00:00
|
|
|
from ldm.invoke.prompt_parser import split_weighted_subprompts
|
2022-09-16 20:58:16 +00:00
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
from modules.parameters import parameters_to_command
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
USER CONFIG
|
|
|
|
"""
|
2022-09-21 14:32:26 +00:00
|
|
|
if opt.cors and "*" in opt.cors:
|
|
|
|
raise ArgumentTypeError('"*" is not an allowed CORS origin')
|
|
|
|
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
output_dir = "outputs/" # Base output directory for images
|
2022-09-21 14:32:26 +00:00
|
|
|
host = opt.host # Web & socket.io host
|
|
|
|
port = opt.port # Web & socket.io port
|
|
|
|
verbose = opt.verbose # enables copious socket.io logging
|
|
|
|
precision = opt.precision
|
2022-09-27 17:38:07 +00:00
|
|
|
free_gpu_mem = opt.free_gpu_mem
|
2022-09-21 14:32:26 +00:00
|
|
|
embedding_path = opt.embedding_path
|
|
|
|
additional_allowed_origins = (
|
|
|
|
opt.cors if opt.cors else []
|
|
|
|
) # additional CORS allowed origins
|
2022-09-20 13:17:31 +00:00
|
|
|
model = "stable-diffusion-1.4"
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
"""
|
|
|
|
END USER CONFIG
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
2022-09-21 14:32:26 +00:00
|
|
|
print("* Initializing, be patient...\n")
|
|
|
|
|
|
|
|
|
2022-09-16 17:18:15 +00:00
|
|
|
"""
|
|
|
|
SERVER SETUP
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
|
|
# fix missing mimetypes on windows due to registry wonkiness
|
2022-09-20 13:17:31 +00:00
|
|
|
mimetypes.add_type("application/javascript", ".js")
|
|
|
|
mimetypes.add_type("text/css", ".css")
|
2022-09-16 17:18:15 +00:00
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
app = Flask(__name__, static_url_path="", static_folder="../frontend/dist/")
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
app.config["OUTPUTS_FOLDER"] = "../outputs"
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
@app.route("/outputs/<path:filename>")
|
2022-09-16 17:18:15 +00:00
|
|
|
def outputs(filename):
|
2022-09-20 13:17:31 +00:00
|
|
|
return send_from_directory(app.config["OUTPUTS_FOLDER"], filename)
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
@app.route("/", defaults={"path": ""})
|
2022-09-16 17:18:15 +00:00
|
|
|
def serve(path):
|
2022-09-20 13:17:31 +00:00
|
|
|
return send_from_directory(app.static_folder, "index.html")
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
logger = True if verbose else False
|
|
|
|
engineio_logger = True if verbose else False
|
|
|
|
|
|
|
|
# default 1,000,000, needs to be higher for socketio to accept larger images
|
|
|
|
max_http_buffer_size = 10000000
|
|
|
|
|
|
|
|
cors_allowed_origins = [f"http://{host}:{port}"] + additional_allowed_origins
|
|
|
|
|
|
|
|
socketio = SocketIO(
|
2022-09-20 13:17:31 +00:00
|
|
|
app,
|
|
|
|
logger=logger,
|
|
|
|
engineio_logger=engineio_logger,
|
|
|
|
max_http_buffer_size=max_http_buffer_size,
|
|
|
|
cors_allowed_origins=cors_allowed_origins,
|
2022-09-26 01:50:47 +00:00
|
|
|
ping_interval=(50, 50),
|
|
|
|
ping_timeout=60,
|
2022-09-20 13:17:31 +00:00
|
|
|
)
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
END SERVER SETUP
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
APP SETUP
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
|
|
class CanceledException(Exception):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2022-09-21 14:32:26 +00:00
|
|
|
try:
|
|
|
|
gfpgan, codeformer, esrgan = None, None, None
|
2022-10-08 15:37:23 +00:00
|
|
|
from ldm.invoke.restoration.base import Restoration
|
2022-09-21 14:32:26 +00:00
|
|
|
|
|
|
|
restoration = Restoration()
|
|
|
|
gfpgan, codeformer = restoration.load_face_restore_models()
|
|
|
|
esrgan = restoration.load_esrgan()
|
|
|
|
|
|
|
|
# coreformer.process(self, image, strength, device, seed=None, fidelity=0.75)
|
|
|
|
|
|
|
|
except (ModuleNotFoundError, ImportError):
|
|
|
|
print(traceback.format_exc(), file=sys.stderr)
|
|
|
|
print(">> You may need to install the ESRGAN and/or GFPGAN modules")
|
|
|
|
|
2022-09-16 17:18:15 +00:00
|
|
|
canceled = Event()
|
|
|
|
|
|
|
|
# reduce logging outputs to error
|
|
|
|
transformers.logging.set_verbosity_error()
|
2022-09-20 13:17:31 +00:00
|
|
|
logging.getLogger("pytorch_lightning").setLevel(logging.ERROR)
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
# Initialize and load model
|
2022-09-21 14:32:26 +00:00
|
|
|
generate = Generate(
|
|
|
|
model,
|
|
|
|
precision=precision,
|
|
|
|
embedding_path=embedding_path,
|
|
|
|
)
|
2022-09-27 17:38:07 +00:00
|
|
|
generate.free_gpu_mem = free_gpu_mem
|
2022-09-20 13:17:31 +00:00
|
|
|
generate.load_model()
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
# location for "finished" images
|
2022-09-20 13:17:31 +00:00
|
|
|
result_path = os.path.join(output_dir, "img-samples/")
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
# temporary path for intermediates
|
2022-09-20 13:17:31 +00:00
|
|
|
intermediate_path = os.path.join(result_path, "intermediates/")
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
# path for user-uploaded init images and masks
|
2022-09-20 13:17:31 +00:00
|
|
|
init_image_path = os.path.join(result_path, "init-images/")
|
|
|
|
mask_image_path = os.path.join(result_path, "mask-images/")
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
# txt log
|
2022-10-08 15:37:23 +00:00
|
|
|
log_path = os.path.join(result_path, "invoke_log.txt")
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
# make all output paths
|
2022-09-20 13:17:31 +00:00
|
|
|
[
|
|
|
|
os.makedirs(path, exist_ok=True)
|
|
|
|
for path in [result_path, intermediate_path, init_image_path, mask_image_path]
|
|
|
|
]
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
END APP SETUP
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
SOCKET.IO LISTENERS
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
@socketio.on("requestSystemConfig")
|
|
|
|
def handle_request_capabilities():
|
|
|
|
print(f">> System config requested")
|
|
|
|
config = get_system_config()
|
|
|
|
socketio.emit("systemConfig", config)
|
|
|
|
|
|
|
|
|
2022-09-26 01:50:47 +00:00
|
|
|
@socketio.on("requestImages")
|
|
|
|
def handle_request_images(page=1, offset=0, last_mtime=None):
|
|
|
|
chunk_size = 50
|
|
|
|
|
|
|
|
if last_mtime:
|
|
|
|
print(f">> Latest images requested")
|
|
|
|
else:
|
|
|
|
print(
|
|
|
|
f">> Page {page} of images requested (page size {chunk_size} offset {offset})"
|
|
|
|
)
|
|
|
|
|
|
|
|
paths = glob.glob(os.path.join(result_path, "*.png"))
|
|
|
|
sorted_paths = sorted(paths, key=lambda x: os.path.getmtime(x), reverse=True)
|
|
|
|
|
|
|
|
if last_mtime:
|
|
|
|
image_paths = filter(lambda x: os.path.getmtime(x) > last_mtime, sorted_paths)
|
|
|
|
else:
|
|
|
|
|
|
|
|
image_paths = sorted_paths[
|
|
|
|
slice(chunk_size * (page - 1) + offset, chunk_size * page + offset)
|
|
|
|
]
|
|
|
|
page = page + 1
|
|
|
|
|
2022-09-16 17:18:15 +00:00
|
|
|
image_array = []
|
2022-09-26 01:50:47 +00:00
|
|
|
|
|
|
|
for path in image_paths:
|
2022-09-20 14:25:49 +00:00
|
|
|
metadata = retrieve_metadata(path)
|
2022-09-26 01:50:47 +00:00
|
|
|
image_array.append(
|
|
|
|
{
|
|
|
|
"url": path,
|
|
|
|
"mtime": os.path.getmtime(path),
|
|
|
|
"metadata": metadata["sd-metadata"],
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
socketio.emit(
|
|
|
|
"galleryImages",
|
|
|
|
{
|
|
|
|
"images": image_array,
|
|
|
|
"nextPage": page,
|
|
|
|
"offset": offset,
|
|
|
|
"onlyNewImages": True if last_mtime else False,
|
|
|
|
},
|
|
|
|
)
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
@socketio.on("generateImage")
|
|
|
|
def handle_generate_image_event(
|
|
|
|
generation_parameters, esrgan_parameters, gfpgan_parameters
|
|
|
|
):
|
|
|
|
print(
|
|
|
|
f">> Image generation requested: {generation_parameters}\nESRGAN parameters: {esrgan_parameters}\nGFPGAN parameters: {gfpgan_parameters}"
|
2022-09-16 17:18:15 +00:00
|
|
|
)
|
2022-09-20 13:17:31 +00:00
|
|
|
generate_images(generation_parameters, esrgan_parameters, gfpgan_parameters)
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
@socketio.on("runESRGAN")
|
2022-09-16 17:18:15 +00:00
|
|
|
def handle_run_esrgan_event(original_image, esrgan_parameters):
|
2022-09-20 13:17:31 +00:00
|
|
|
print(
|
|
|
|
f'>> ESRGAN upscale requested for "{original_image["url"]}": {esrgan_parameters}'
|
|
|
|
)
|
2022-09-18 07:33:09 +00:00
|
|
|
progress = {
|
2022-09-20 13:17:31 +00:00
|
|
|
"currentStep": 1,
|
|
|
|
"totalSteps": 1,
|
|
|
|
"currentIteration": 1,
|
|
|
|
"totalIterations": 1,
|
|
|
|
"currentStatus": "Preparing",
|
|
|
|
"isProcessing": True,
|
|
|
|
"currentStatusHasSteps": False,
|
2022-09-18 07:33:09 +00:00
|
|
|
}
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
socketio.emit("progressUpdate", progress)
|
2022-09-18 07:33:09 +00:00
|
|
|
eventlet.sleep(0)
|
|
|
|
|
2022-09-16 17:18:15 +00:00
|
|
|
image = Image.open(original_image["url"])
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
seed = (
|
|
|
|
original_image["metadata"]["seed"]
|
|
|
|
if "seed" in original_image["metadata"]
|
|
|
|
else "unknown_seed"
|
|
|
|
)
|
2022-09-16 17:18:15 +00:00
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
progress["currentStatus"] = "Upscaling"
|
|
|
|
socketio.emit("progressUpdate", progress)
|
2022-09-18 07:33:09 +00:00
|
|
|
eventlet.sleep(0)
|
|
|
|
|
2022-09-21 14:32:26 +00:00
|
|
|
image = esrgan.process(
|
2022-09-16 17:18:15 +00:00
|
|
|
image=image,
|
2022-09-20 13:17:31 +00:00
|
|
|
upsampler_scale=esrgan_parameters["upscale"][0],
|
|
|
|
strength=esrgan_parameters["upscale"][1],
|
|
|
|
seed=seed,
|
2022-09-16 17:18:15 +00:00
|
|
|
)
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
progress["currentStatus"] = "Saving image"
|
|
|
|
socketio.emit("progressUpdate", progress)
|
2022-09-18 07:33:09 +00:00
|
|
|
eventlet.sleep(0)
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
esrgan_parameters["seed"] = seed
|
|
|
|
metadata = parameters_to_post_processed_image_metadata(
|
|
|
|
parameters=esrgan_parameters,
|
|
|
|
original_image_path=original_image["url"],
|
|
|
|
type="esrgan",
|
|
|
|
)
|
2022-09-16 17:18:15 +00:00
|
|
|
command = parameters_to_command(esrgan_parameters)
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
path = save_image(image, command, metadata, result_path, postprocessing="esrgan")
|
|
|
|
|
2022-09-16 17:18:15 +00:00
|
|
|
write_log_message(f'[Upscaled] "{original_image["url"]}" > "{path}": {command}')
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
progress["currentStatus"] = "Finished"
|
|
|
|
progress["currentStep"] = 0
|
|
|
|
progress["totalSteps"] = 0
|
|
|
|
progress["currentIteration"] = 0
|
|
|
|
progress["totalIterations"] = 0
|
|
|
|
progress["isProcessing"] = False
|
|
|
|
socketio.emit("progressUpdate", progress)
|
2022-09-18 07:33:09 +00:00
|
|
|
eventlet.sleep(0)
|
|
|
|
|
2022-09-16 17:18:15 +00:00
|
|
|
socketio.emit(
|
2022-09-20 13:17:31 +00:00
|
|
|
"esrganResult",
|
|
|
|
{
|
|
|
|
"url": os.path.relpath(path),
|
2022-09-26 01:50:47 +00:00
|
|
|
"mtime": os.path.getmtime(path),
|
2022-09-20 13:17:31 +00:00
|
|
|
"metadata": metadata,
|
|
|
|
},
|
|
|
|
)
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
@socketio.on("runGFPGAN")
|
2022-09-16 17:18:15 +00:00
|
|
|
def handle_run_gfpgan_event(original_image, gfpgan_parameters):
|
2022-09-20 13:17:31 +00:00
|
|
|
print(
|
|
|
|
f'>> GFPGAN face fix requested for "{original_image["url"]}": {gfpgan_parameters}'
|
|
|
|
)
|
2022-09-18 07:33:09 +00:00
|
|
|
progress = {
|
2022-09-20 13:17:31 +00:00
|
|
|
"currentStep": 1,
|
|
|
|
"totalSteps": 1,
|
|
|
|
"currentIteration": 1,
|
|
|
|
"totalIterations": 1,
|
|
|
|
"currentStatus": "Preparing",
|
|
|
|
"isProcessing": True,
|
|
|
|
"currentStatusHasSteps": False,
|
2022-09-18 07:33:09 +00:00
|
|
|
}
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
socketio.emit("progressUpdate", progress)
|
2022-09-18 07:33:09 +00:00
|
|
|
eventlet.sleep(0)
|
|
|
|
|
2022-09-16 17:18:15 +00:00
|
|
|
image = Image.open(original_image["url"])
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
seed = (
|
|
|
|
original_image["metadata"]["seed"]
|
|
|
|
if "seed" in original_image["metadata"]
|
|
|
|
else "unknown_seed"
|
|
|
|
)
|
2022-09-16 17:18:15 +00:00
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
progress["currentStatus"] = "Fixing faces"
|
|
|
|
socketio.emit("progressUpdate", progress)
|
2022-09-18 07:33:09 +00:00
|
|
|
eventlet.sleep(0)
|
|
|
|
|
2022-09-21 14:32:26 +00:00
|
|
|
image = gfpgan.process(
|
2022-10-13 13:14:21 +00:00
|
|
|
image=image, strength=gfpgan_parameters["facetool_strength"], seed=seed
|
2022-09-16 17:18:15 +00:00
|
|
|
)
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
progress["currentStatus"] = "Saving image"
|
|
|
|
socketio.emit("progressUpdate", progress)
|
2022-09-18 07:33:09 +00:00
|
|
|
eventlet.sleep(0)
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
gfpgan_parameters["seed"] = seed
|
|
|
|
metadata = parameters_to_post_processed_image_metadata(
|
|
|
|
parameters=gfpgan_parameters,
|
|
|
|
original_image_path=original_image["url"],
|
|
|
|
type="gfpgan",
|
|
|
|
)
|
2022-09-16 17:18:15 +00:00
|
|
|
command = parameters_to_command(gfpgan_parameters)
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
path = save_image(image, command, metadata, result_path, postprocessing="gfpgan")
|
|
|
|
|
2022-09-16 17:18:15 +00:00
|
|
|
write_log_message(f'[Fixed faces] "{original_image["url"]}" > "{path}": {command}')
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
progress["currentStatus"] = "Finished"
|
|
|
|
progress["currentStep"] = 0
|
|
|
|
progress["totalSteps"] = 0
|
|
|
|
progress["currentIteration"] = 0
|
|
|
|
progress["totalIterations"] = 0
|
|
|
|
progress["isProcessing"] = False
|
|
|
|
socketio.emit("progressUpdate", progress)
|
2022-09-18 07:33:09 +00:00
|
|
|
eventlet.sleep(0)
|
|
|
|
|
2022-09-16 17:18:15 +00:00
|
|
|
socketio.emit(
|
2022-09-20 13:17:31 +00:00
|
|
|
"gfpganResult",
|
|
|
|
{
|
|
|
|
"url": os.path.relpath(path),
|
2022-09-26 01:50:47 +00:00
|
|
|
"mtime": os.path.mtime(path),
|
2022-09-20 13:17:31 +00:00
|
|
|
"metadata": metadata,
|
|
|
|
},
|
|
|
|
)
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
@socketio.on("cancel")
|
2022-09-16 17:18:15 +00:00
|
|
|
def handle_cancel():
|
2022-09-20 13:17:31 +00:00
|
|
|
print(f">> Cancel processing requested")
|
2022-09-16 17:18:15 +00:00
|
|
|
canceled.set()
|
2022-09-20 13:17:31 +00:00
|
|
|
socketio.emit("processingCanceled")
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
# TODO: I think this needs a safety mechanism.
|
2022-09-20 13:17:31 +00:00
|
|
|
@socketio.on("deleteImage")
|
2022-09-18 07:33:09 +00:00
|
|
|
def handle_delete_image(path, uuid):
|
2022-09-16 17:18:15 +00:00
|
|
|
print(f'>> Delete requested "{path}"')
|
2022-09-17 07:19:09 +00:00
|
|
|
send2trash(path)
|
2022-09-20 13:17:31 +00:00
|
|
|
socketio.emit("imageDeleted", {"url": path, "uuid": uuid})
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
# TODO: I think this needs a safety mechanism.
|
2022-09-20 13:17:31 +00:00
|
|
|
@socketio.on("uploadInitialImage")
|
2022-09-16 17:18:15 +00:00
|
|
|
def handle_upload_initial_image(bytes, name):
|
|
|
|
print(f'>> Init image upload requested "{name}"')
|
|
|
|
uuid = uuid4().hex
|
|
|
|
split = os.path.splitext(name)
|
2022-09-20 13:17:31 +00:00
|
|
|
name = f"{split[0]}.{uuid}{split[1]}"
|
2022-09-18 07:33:09 +00:00
|
|
|
file_path = os.path.join(init_image_path, name)
|
2022-09-16 17:18:15 +00:00
|
|
|
os.makedirs(os.path.dirname(file_path), exist_ok=True)
|
|
|
|
newFile = open(file_path, "wb")
|
|
|
|
newFile.write(bytes)
|
2022-09-20 13:17:31 +00:00
|
|
|
socketio.emit("initialImageUploaded", {"url": file_path, "uuid": ""})
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
# TODO: I think this needs a safety mechanism.
|
2022-09-20 13:17:31 +00:00
|
|
|
@socketio.on("uploadMaskImage")
|
2022-09-16 17:18:15 +00:00
|
|
|
def handle_upload_mask_image(bytes, name):
|
|
|
|
print(f'>> Mask image upload requested "{name}"')
|
|
|
|
uuid = uuid4().hex
|
|
|
|
split = os.path.splitext(name)
|
2022-09-20 13:17:31 +00:00
|
|
|
name = f"{split[0]}.{uuid}{split[1]}"
|
2022-09-18 07:33:09 +00:00
|
|
|
file_path = os.path.join(mask_image_path, name)
|
2022-09-16 17:18:15 +00:00
|
|
|
os.makedirs(os.path.dirname(file_path), exist_ok=True)
|
|
|
|
newFile = open(file_path, "wb")
|
|
|
|
newFile.write(bytes)
|
2022-09-20 13:17:31 +00:00
|
|
|
socketio.emit("maskImageUploaded", {"url": file_path, "uuid": ""})
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
END SOCKET.IO LISTENERS
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
ADDITIONAL FUNCTIONS
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
def get_system_config():
|
|
|
|
return {
|
|
|
|
"model": "stable diffusion",
|
|
|
|
"model_id": model,
|
|
|
|
"model_hash": generate.model_hash,
|
|
|
|
"app_id": APP_ID,
|
|
|
|
"app_version": APP_VERSION,
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
def parameters_to_post_processed_image_metadata(parameters, original_image_path, type):
|
|
|
|
# top-level metadata minus `image` or `images`
|
|
|
|
metadata = get_system_config()
|
|
|
|
|
|
|
|
orig_hash = calculate_init_img_hash(original_image_path)
|
|
|
|
|
|
|
|
image = {"orig_path": original_image_path, "orig_hash": orig_hash}
|
|
|
|
|
|
|
|
if type == "esrgan":
|
|
|
|
image["type"] = "esrgan"
|
|
|
|
image["scale"] = parameters["upscale"][0]
|
|
|
|
image["strength"] = parameters["upscale"][1]
|
|
|
|
elif type == "gfpgan":
|
|
|
|
image["type"] = "gfpgan"
|
2022-10-13 13:14:21 +00:00
|
|
|
image["strength"] = parameters["facetool_strength"]
|
2022-09-20 13:17:31 +00:00
|
|
|
else:
|
|
|
|
raise TypeError(f"Invalid type: {type}")
|
|
|
|
|
|
|
|
metadata["image"] = image
|
|
|
|
return metadata
|
|
|
|
|
|
|
|
|
|
|
|
def parameters_to_generated_image_metadata(parameters):
|
|
|
|
# top-level metadata minus `image` or `images`
|
|
|
|
|
|
|
|
metadata = get_system_config()
|
|
|
|
# remove any image keys not mentioned in RFC #266
|
|
|
|
rfc266_img_fields = [
|
|
|
|
"type",
|
|
|
|
"postprocessing",
|
|
|
|
"sampler",
|
|
|
|
"prompt",
|
|
|
|
"seed",
|
|
|
|
"variations",
|
|
|
|
"steps",
|
|
|
|
"cfg_scale",
|
2022-09-28 23:47:36 +00:00
|
|
|
"threshold",
|
|
|
|
"perlin",
|
2022-09-20 13:17:31 +00:00
|
|
|
"step_number",
|
|
|
|
"width",
|
|
|
|
"height",
|
|
|
|
"extra",
|
|
|
|
"seamless",
|
2022-10-07 20:56:38 +00:00
|
|
|
"hires_fix",
|
2022-09-20 13:17:31 +00:00
|
|
|
]
|
|
|
|
|
|
|
|
rfc_dict = {}
|
|
|
|
|
|
|
|
for item in parameters.items():
|
|
|
|
key, value = item
|
|
|
|
if key in rfc266_img_fields:
|
|
|
|
rfc_dict[key] = value
|
|
|
|
|
|
|
|
postprocessing = []
|
|
|
|
|
|
|
|
# 'postprocessing' is either null or an
|
2022-10-13 13:14:21 +00:00
|
|
|
if "facetool_strength" in parameters:
|
2022-09-20 13:17:31 +00:00
|
|
|
|
|
|
|
postprocessing.append(
|
2022-10-13 13:14:21 +00:00
|
|
|
{"type": "gfpgan", "strength": float(parameters["facetool_strength"])}
|
2022-09-20 13:17:31 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
if "upscale" in parameters:
|
|
|
|
postprocessing.append(
|
|
|
|
{
|
|
|
|
"type": "esrgan",
|
|
|
|
"scale": int(parameters["upscale"][0]),
|
|
|
|
"strength": float(parameters["upscale"][1]),
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
rfc_dict["postprocessing"] = postprocessing if len(postprocessing) > 0 else None
|
|
|
|
|
|
|
|
# semantic drift
|
|
|
|
rfc_dict["sampler"] = parameters["sampler_name"]
|
|
|
|
|
|
|
|
# display weighted subprompts (liable to change)
|
2022-10-21 13:07:11 +00:00
|
|
|
subprompts = split_weighted_subprompts(parameters["prompt"])
|
2022-09-20 13:17:31 +00:00
|
|
|
subprompts = [{"prompt": x[0], "weight": x[1]} for x in subprompts]
|
|
|
|
rfc_dict["prompt"] = subprompts
|
|
|
|
|
|
|
|
# 'variations' should always exist and be an array, empty or consisting of {'seed': seed, 'weight': weight} pairs
|
|
|
|
variations = []
|
|
|
|
|
|
|
|
if "with_variations" in parameters:
|
|
|
|
variations = [
|
|
|
|
{"seed": x[0], "weight": x[1]} for x in parameters["with_variations"]
|
|
|
|
]
|
|
|
|
|
|
|
|
rfc_dict["variations"] = variations
|
|
|
|
|
|
|
|
if "init_img" in parameters:
|
|
|
|
rfc_dict["type"] = "img2img"
|
|
|
|
rfc_dict["strength"] = parameters["strength"]
|
|
|
|
rfc_dict["fit"] = parameters["fit"] # TODO: Noncompliant
|
|
|
|
rfc_dict["orig_hash"] = calculate_init_img_hash(parameters["init_img"])
|
|
|
|
rfc_dict["init_image_path"] = parameters["init_img"] # TODO: Noncompliant
|
|
|
|
rfc_dict["sampler"] = "ddim" # TODO: FIX ME WHEN IMG2IMG SUPPORTS ALL SAMPLERS
|
|
|
|
if "init_mask" in parameters:
|
|
|
|
rfc_dict["mask_hash"] = calculate_init_img_hash(
|
|
|
|
parameters["init_mask"]
|
|
|
|
) # TODO: Noncompliant
|
|
|
|
rfc_dict["mask_image_path"] = parameters["init_mask"] # TODO: Noncompliant
|
|
|
|
else:
|
|
|
|
rfc_dict["type"] = "txt2img"
|
|
|
|
|
|
|
|
metadata["image"] = rfc_dict
|
|
|
|
|
|
|
|
return metadata
|
|
|
|
|
|
|
|
|
2022-09-18 07:33:09 +00:00
|
|
|
def make_unique_init_image_filename(name):
|
|
|
|
uuid = uuid4().hex
|
|
|
|
split = os.path.splitext(name)
|
2022-09-20 13:17:31 +00:00
|
|
|
name = f"{split[0]}.{uuid}{split[1]}"
|
2022-09-18 07:33:09 +00:00
|
|
|
return name
|
|
|
|
|
|
|
|
|
2022-09-16 17:18:15 +00:00
|
|
|
def write_log_message(message, log_path=log_path):
|
|
|
|
"""Logs the filename and parameters used to generate or process that image to log file"""
|
2022-09-20 13:17:31 +00:00
|
|
|
message = f"{message}\n"
|
|
|
|
with open(log_path, "a", encoding="utf-8") as file:
|
2022-09-16 17:18:15 +00:00
|
|
|
file.writelines(message)
|
|
|
|
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
def save_image(
|
|
|
|
image, command, metadata, output_dir, step_index=None, postprocessing=False
|
|
|
|
):
|
2022-09-16 17:18:15 +00:00
|
|
|
pngwriter = PngWriter(output_dir)
|
|
|
|
prefix = pngwriter.unique_prefix()
|
|
|
|
|
2022-09-20 14:44:57 +00:00
|
|
|
seed = "unknown_seed"
|
|
|
|
|
|
|
|
if "image" in metadata:
|
|
|
|
if "seed" in metadata["image"]:
|
|
|
|
seed = metadata["image"]["seed"]
|
|
|
|
|
|
|
|
filename = f"{prefix}.{seed}"
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
if step_index:
|
2022-09-20 13:17:31 +00:00
|
|
|
filename += f".{step_index}"
|
2022-09-16 17:18:15 +00:00
|
|
|
if postprocessing:
|
2022-09-20 13:17:31 +00:00
|
|
|
filename += f".postprocessed"
|
2022-09-16 17:18:15 +00:00
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
filename += ".png"
|
2022-09-16 17:18:15 +00:00
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
path = pngwriter.save_image_and_prompt_to_png(
|
|
|
|
image=image, dream_prompt=command, metadata=metadata, name=filename
|
|
|
|
)
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
return path
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
|
2022-09-18 07:33:09 +00:00
|
|
|
def calculate_real_steps(steps, strength, has_init_image):
|
|
|
|
return math.floor(strength * steps) if has_init_image else steps
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
|
2022-09-16 17:18:15 +00:00
|
|
|
def generate_images(generation_parameters, esrgan_parameters, gfpgan_parameters):
|
|
|
|
canceled.clear()
|
|
|
|
|
|
|
|
step_index = 1
|
2022-09-21 14:32:26 +00:00
|
|
|
prior_variations = (
|
|
|
|
generation_parameters["with_variations"]
|
|
|
|
if "with_variations" in generation_parameters
|
|
|
|
else []
|
|
|
|
)
|
2022-09-18 07:33:09 +00:00
|
|
|
"""
|
|
|
|
If a result image is used as an init image, and then deleted, we will want to be
|
|
|
|
able to use it as an init image in the future. Need to copy it.
|
|
|
|
|
|
|
|
If the init/mask image doesn't exist in the init_image_path/mask_image_path,
|
|
|
|
make a unique filename for it and copy it there.
|
|
|
|
"""
|
2022-09-20 13:17:31 +00:00
|
|
|
if "init_img" in generation_parameters:
|
|
|
|
filename = os.path.basename(generation_parameters["init_img"])
|
2022-09-18 07:33:09 +00:00
|
|
|
if not os.path.exists(os.path.join(init_image_path, filename)):
|
|
|
|
unique_filename = make_unique_init_image_filename(filename)
|
|
|
|
new_path = os.path.join(init_image_path, unique_filename)
|
2022-09-20 13:17:31 +00:00
|
|
|
shutil.copy(generation_parameters["init_img"], new_path)
|
|
|
|
generation_parameters["init_img"] = new_path
|
|
|
|
if "init_mask" in generation_parameters:
|
|
|
|
filename = os.path.basename(generation_parameters["init_mask"])
|
2022-09-18 07:33:09 +00:00
|
|
|
if not os.path.exists(os.path.join(mask_image_path, filename)):
|
|
|
|
unique_filename = make_unique_init_image_filename(filename)
|
|
|
|
new_path = os.path.join(init_image_path, unique_filename)
|
2022-09-20 13:17:31 +00:00
|
|
|
shutil.copy(generation_parameters["init_img"], new_path)
|
|
|
|
generation_parameters["init_mask"] = new_path
|
2022-09-18 07:33:09 +00:00
|
|
|
|
|
|
|
totalSteps = calculate_real_steps(
|
2022-09-20 13:17:31 +00:00
|
|
|
steps=generation_parameters["steps"],
|
|
|
|
strength=generation_parameters["strength"]
|
|
|
|
if "strength" in generation_parameters
|
|
|
|
else None,
|
|
|
|
has_init_image="init_img" in generation_parameters,
|
|
|
|
)
|
2022-09-18 07:33:09 +00:00
|
|
|
|
|
|
|
progress = {
|
2022-09-20 13:17:31 +00:00
|
|
|
"currentStep": 1,
|
|
|
|
"totalSteps": totalSteps,
|
|
|
|
"currentIteration": 1,
|
|
|
|
"totalIterations": generation_parameters["iterations"],
|
|
|
|
"currentStatus": "Preparing",
|
|
|
|
"isProcessing": True,
|
|
|
|
"currentStatusHasSteps": False,
|
2022-09-18 07:33:09 +00:00
|
|
|
}
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
socketio.emit("progressUpdate", progress)
|
2022-09-18 07:33:09 +00:00
|
|
|
eventlet.sleep(0)
|
|
|
|
|
2022-09-16 17:18:15 +00:00
|
|
|
def image_progress(sample, step):
|
|
|
|
if canceled.is_set():
|
|
|
|
raise CanceledException
|
2022-09-18 07:33:09 +00:00
|
|
|
|
2022-09-16 17:18:15 +00:00
|
|
|
nonlocal step_index
|
|
|
|
nonlocal generation_parameters
|
2022-09-18 07:33:09 +00:00
|
|
|
nonlocal progress
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
progress["currentStep"] = step + 1
|
|
|
|
progress["currentStatus"] = "Generating"
|
|
|
|
progress["currentStatusHasSteps"] = True
|
|
|
|
|
|
|
|
if (
|
|
|
|
generation_parameters["progress_images"]
|
|
|
|
and step % 5 == 0
|
|
|
|
and step < generation_parameters["steps"] - 1
|
|
|
|
):
|
|
|
|
image = generate.sample_to_image(sample)
|
2022-09-27 04:25:27 +00:00
|
|
|
|
|
|
|
metadata = parameters_to_generated_image_metadata(generation_parameters)
|
|
|
|
command = parameters_to_command(generation_parameters)
|
|
|
|
path = save_image(image, command, metadata, intermediate_path, step_index=step_index, postprocessing=False)
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
step_index += 1
|
2022-09-20 13:17:31 +00:00
|
|
|
socketio.emit(
|
|
|
|
"intermediateResult",
|
2022-09-26 01:50:47 +00:00
|
|
|
{
|
|
|
|
"url": os.path.relpath(path),
|
|
|
|
"mtime": os.path.getmtime(path),
|
2022-09-27 05:29:27 +00:00
|
|
|
"metadata": metadata,
|
2022-09-26 01:50:47 +00:00
|
|
|
},
|
2022-09-20 13:17:31 +00:00
|
|
|
)
|
|
|
|
socketio.emit("progressUpdate", progress)
|
2022-09-16 17:18:15 +00:00
|
|
|
eventlet.sleep(0)
|
|
|
|
|
2022-09-21 14:32:26 +00:00
|
|
|
def image_done(image, seed, first_seed):
|
2022-09-16 17:18:15 +00:00
|
|
|
nonlocal generation_parameters
|
|
|
|
nonlocal esrgan_parameters
|
|
|
|
nonlocal gfpgan_parameters
|
2022-09-18 07:33:09 +00:00
|
|
|
nonlocal progress
|
|
|
|
|
|
|
|
step_index = 1
|
2022-09-21 14:32:26 +00:00
|
|
|
nonlocal prior_variations
|
2022-09-18 07:33:09 +00:00
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
progress["currentStatus"] = "Generation complete"
|
|
|
|
socketio.emit("progressUpdate", progress)
|
2022-09-18 07:33:09 +00:00
|
|
|
eventlet.sleep(0)
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
all_parameters = generation_parameters
|
|
|
|
postprocessing = False
|
|
|
|
|
2022-09-21 14:32:26 +00:00
|
|
|
if (
|
|
|
|
"variation_amount" in all_parameters
|
|
|
|
and all_parameters["variation_amount"] > 0
|
|
|
|
):
|
|
|
|
first_seed = first_seed or seed
|
|
|
|
this_variation = [[seed, all_parameters["variation_amount"]]]
|
|
|
|
all_parameters["with_variations"] = prior_variations + this_variation
|
2022-09-22 09:25:21 +00:00
|
|
|
all_parameters["seed"] = first_seed
|
|
|
|
elif ("with_variations" in all_parameters):
|
|
|
|
all_parameters["seed"] = first_seed
|
|
|
|
else:
|
|
|
|
all_parameters["seed"] = seed
|
2022-09-21 14:32:26 +00:00
|
|
|
|
2022-09-16 17:18:15 +00:00
|
|
|
if esrgan_parameters:
|
2022-09-20 13:17:31 +00:00
|
|
|
progress["currentStatus"] = "Upscaling"
|
|
|
|
progress["currentStatusHasSteps"] = False
|
|
|
|
socketio.emit("progressUpdate", progress)
|
2022-09-18 07:33:09 +00:00
|
|
|
eventlet.sleep(0)
|
|
|
|
|
2022-09-21 14:32:26 +00:00
|
|
|
image = esrgan.process(
|
2022-09-16 17:18:15 +00:00
|
|
|
image=image,
|
2022-09-20 13:17:31 +00:00
|
|
|
upsampler_scale=esrgan_parameters["level"],
|
2022-09-21 14:32:26 +00:00
|
|
|
strength=esrgan_parameters["strength"],
|
2022-09-20 13:17:31 +00:00
|
|
|
seed=seed,
|
2022-09-16 17:18:15 +00:00
|
|
|
)
|
2022-09-21 14:32:26 +00:00
|
|
|
|
2022-09-16 17:18:15 +00:00
|
|
|
postprocessing = True
|
2022-09-20 13:17:31 +00:00
|
|
|
all_parameters["upscale"] = [
|
|
|
|
esrgan_parameters["level"],
|
|
|
|
esrgan_parameters["strength"],
|
|
|
|
]
|
2022-09-16 17:18:15 +00:00
|
|
|
|
|
|
|
if gfpgan_parameters:
|
2022-09-20 13:17:31 +00:00
|
|
|
progress["currentStatus"] = "Fixing faces"
|
|
|
|
progress["currentStatusHasSteps"] = False
|
|
|
|
socketio.emit("progressUpdate", progress)
|
2022-09-18 07:33:09 +00:00
|
|
|
eventlet.sleep(0)
|
|
|
|
|
2022-09-21 14:32:26 +00:00
|
|
|
image = gfpgan.process(
|
|
|
|
image=image, strength=gfpgan_parameters["strength"], seed=seed
|
2022-09-16 17:18:15 +00:00
|
|
|
)
|
|
|
|
postprocessing = True
|
2022-10-13 13:14:21 +00:00
|
|
|
all_parameters["facetool_strength"] = gfpgan_parameters["strength"]
|
2022-09-16 17:18:15 +00:00
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
progress["currentStatus"] = "Saving image"
|
|
|
|
socketio.emit("progressUpdate", progress)
|
2022-09-18 07:33:09 +00:00
|
|
|
eventlet.sleep(0)
|
2022-09-16 17:18:15 +00:00
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
metadata = parameters_to_generated_image_metadata(all_parameters)
|
2022-09-16 17:18:15 +00:00
|
|
|
command = parameters_to_command(all_parameters)
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
path = save_image(
|
|
|
|
image, command, metadata, result_path, postprocessing=postprocessing
|
|
|
|
)
|
|
|
|
|
|
|
|
print(f'>> Image generated: "{path}"')
|
2022-09-16 17:18:15 +00:00
|
|
|
write_log_message(f'[Generated] "{path}": {command}')
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
if progress["totalIterations"] > progress["currentIteration"]:
|
|
|
|
progress["currentStep"] = 1
|
|
|
|
progress["currentIteration"] += 1
|
|
|
|
progress["currentStatus"] = "Iteration finished"
|
|
|
|
progress["currentStatusHasSteps"] = False
|
2022-09-18 07:33:09 +00:00
|
|
|
else:
|
2022-09-20 13:17:31 +00:00
|
|
|
progress["currentStep"] = 0
|
|
|
|
progress["totalSteps"] = 0
|
|
|
|
progress["currentIteration"] = 0
|
|
|
|
progress["totalIterations"] = 0
|
|
|
|
progress["currentStatus"] = "Finished"
|
|
|
|
progress["isProcessing"] = False
|
|
|
|
|
|
|
|
socketio.emit("progressUpdate", progress)
|
2022-09-18 07:33:09 +00:00
|
|
|
eventlet.sleep(0)
|
|
|
|
|
2022-09-16 17:18:15 +00:00
|
|
|
socketio.emit(
|
2022-09-20 13:17:31 +00:00
|
|
|
"generationResult",
|
2022-09-26 01:50:47 +00:00
|
|
|
{
|
|
|
|
"url": os.path.relpath(path),
|
|
|
|
"mtime": os.path.getmtime(path),
|
|
|
|
"metadata": metadata,
|
|
|
|
},
|
2022-09-20 13:17:31 +00:00
|
|
|
)
|
2022-09-16 17:18:15 +00:00
|
|
|
eventlet.sleep(0)
|
|
|
|
|
|
|
|
try:
|
2022-09-20 13:17:31 +00:00
|
|
|
generate.prompt2image(
|
2022-09-16 17:18:15 +00:00
|
|
|
**generation_parameters,
|
|
|
|
step_callback=image_progress,
|
2022-09-20 13:17:31 +00:00
|
|
|
image_callback=image_done,
|
2022-09-16 17:18:15 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
except KeyboardInterrupt:
|
|
|
|
raise
|
|
|
|
except CanceledException:
|
|
|
|
pass
|
|
|
|
except Exception as e:
|
2022-09-20 13:17:31 +00:00
|
|
|
socketio.emit("error", {"message": (str(e))})
|
2022-09-16 17:18:15 +00:00
|
|
|
print("\n")
|
|
|
|
traceback.print_exc()
|
|
|
|
print("\n")
|
|
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
END ADDITIONAL FUNCTIONS
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
2022-09-20 13:17:31 +00:00
|
|
|
if __name__ == "__main__":
|
|
|
|
print(f">> Starting server at http://{host}:{port}")
|
2022-09-16 17:18:15 +00:00
|
|
|
socketio.run(app, host=host, port=port)
|