#!/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 InfoPrompt: {prompt[:100]}
...
" # await bot.api.send_markdown_message(room.room_id, info_msg) # Clean up temp file os.remove(filename) except argparse.ArgumentError as e: await bot.api.send_text_message(room.room_id, f"Argument Error: {e}") await bot.api.send_markdown_message(room.room_id, "
Stable Diffusion Help" + print_help() + "
") except Exception as e: await bot.api.send_text_message(room.room_id, f"Error processing the command: {str(e)}") def print_help(): """ Generates the full help text for the 'sd' command, including LORA list. """ return """

Generate images using self-hosted Stable Diffusion

Positional arguments:

Optional arguments:

LORA List:

Load LORAs like this:

""" # --------------------------------------------------------------------------- # Plugin Metadata # --------------------------------------------------------------------------- __version__ = "1.1.2" __author__ = "Funguy Bot" __description__ = "Stable Diffusion image generation (async, LORA support)" __help__ = """
!sd – Generate images via Stable Diffusion

!sd [options] <prompt>

LORAs: <lora:filename:weight>

Requires a locally running Stable Diffusion API.

"""