#!/usr/bin/env python3
"""
Plugin for generating images using self-hosted Stable Diffusion and sending them to a Matrix chat room.
Now fully asynchronous (uses aiohttp). All original parameters and help text are preserved.
"""
import aiohttp
import base64
import tempfile
import os
import argparse
import simplematrixbotlib as botlib
async def handle_command(room, message, bot, prefix, config):
match = botlib.MessageMatch(room, message, bot, prefix)
if not (match.prefix() and match.command("sd")):
return
# Check if API is reachable
try:
async with aiohttp.ClientSession() as session:
async with session.get("http://127.0.0.1:7860/docs", timeout=3) as resp:
if resp.status != 200:
await bot.api.send_text_message(room.room_id, "Stable Diffusion API is not running!")
return
except Exception:
await bot.api.send_text_message(room.room_id, "Could not reach Stable Diffusion API!")
return
try:
parser = argparse.ArgumentParser(description='Generate images using self-hosted Stable Diffusion')
parser.add_argument('--steps', type=int, default=4, help='Number of steps, default=4')
parser.add_argument('--cfg', type=int, default=2, help='CFG scale, default=2')
parser.add_argument('--h', type=int, default=512, help='Height of the image, default=512')
parser.add_argument('--w', type=int, default=512, help='Width of the image, default=512')
parser.add_argument('--neg', type=str, nargs='+',
default=['((((ugly)))), (((duplicate))), ((morbid)), ((mutilated)), out of frame, extra fingers, mutated hands, ((poorly drawn hands)), ((poorly drawn face)), (((mutation))), (((deformed))), ((ugly)), blurry, ((bad anatomy)), (((bad proportions))), ((extra limbs)), cloned face, (((disfigured))), out of frame, ugly, extra limbs, (bad anatomy), gross proportions, (malformed limbs), ((missing arms)), ((missing legs)), (((extra arms))), (((extra legs))), mutated hands, (fused fingers), (too many fingers), (((long neck)))'],
help='Negative prompt')
parser.add_argument('--sampler', type=str, nargs='*', default=['DPM++', 'SDE Karras'],
help='Sampler name, default=DPM++ SDE')
parser.add_argument('--seed', type=int, default=None,
help='Seed for deterministic generation (omit for random)')
parser.add_argument('prompt', type=str, nargs='*', help='Prompt for the image')
args = parser.parse_args(message.body.split()[1:]) # skip command prefix
if not args.prompt:
raise argparse.ArgumentError(None, "Prompt is required.")
prompt = ' '.join(args.prompt)
sampler_name = ' '.join(args.sampler)
neg_prompt = ' '.join(args.neg)
payload = {
"prompt": prompt,
"steps": args.steps,
"negative_prompt": neg_prompt,
"sampler_name": sampler_name,
"cfg_scale": args.cfg,
"width": args.w,
"height": args.h,
}
if args.seed is not None:
payload["seed"] = args.seed
async with aiohttp.ClientSession() as session:
async with session.post("http://127.0.0.1:7860/sdapi/v1/txt2img", json=payload, timeout=600) as response:
response.raise_for_status()
r = await response.json()
# Save and send image
image_data = base64.b64decode(r['images'][0])
with tempfile.NamedTemporaryFile(suffix='.png', delete=False) as temp_file:
filename = temp_file.name
temp_file.write(image_data)
await bot.api.send_image_message(room_id=room.room_id, image_filepath=filename)
# Optional info message (commented out to avoid spam, but can be enabled)
# neg_prompt_clean = neg_prompt.replace(" ", "")
# seed_info = f"
Seed: {args.seed}" if args.seed is not None else ""
# info_msg = f"🔍 Image Info
Prompt: {prompt[:100]}
...Stable Diffusion Help
" + print_help() + "
Generate images using self-hosted Stable Diffusion
Positional arguments:
Optional arguments:
LORA List:
Load LORAs like this:
!sd [options] <prompt>
--steps N – Sampling steps (default 4)--cfg scale – CFG scale (default 2)--h H --w W – Image dimensions (default 512)--neg <negative prompt>--sampler SAMPLER – Sampler name (default DPM++ SDE)--seed SEED – Deterministic seed (optional)LORAs: <lora:filename:weight>
Requires a locally running Stable Diffusion API.