File size: 23,629 Bytes
6f65703
 
8721ec2
 
 
009432d
c276a99
7223d40
6f65703
009432d
 
b9247d3
009432d
6f65703
 
46c5397
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
009432d
 
 
46c5397
009432d
 
 
 
 
 
 
 
46c5397
 
 
 
 
 
 
 
009432d
46c5397
 
 
 
 
 
 
009432d
46c5397
 
 
 
 
 
 
009432d
 
 
 
 
 
46c5397
009432d
 
 
 
46c5397
 
 
 
 
 
 
009432d
 
 
 
 
6f65703
46c5397
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
7223d40
 
 
 
 
 
 
 
6f65703
 
 
 
0d79725
 
 
 
 
 
 
 
 
6f65703
0d79725
 
 
 
 
 
 
 
 
 
 
 
 
6f65703
7223d40
 
8721ec2
7223d40
8721ec2
 
57ab548
7223d40
c276a99
009432d
 
 
 
46c5397
009432d
46c5397
c276a99
46c5397
 
 
 
 
 
 
57ab548
c276a99
8721ec2
7223d40
 
 
6f65703
 
 
 
 
 
0d79725
 
 
 
 
 
 
 
6f65703
 
 
c276a99
6f65703
 
 
 
 
c276a99
0d79725
 
 
 
 
6f65703
 
c276a99
6f65703
 
 
0d79725
6f65703
 
0d79725
b9247d3
 
 
 
 
0d79725
 
 
 
 
 
 
 
 
b9247d3
 
0d79725
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
b9247d3
 
 
 
 
 
 
 
c276a99
b9247d3
 
 
 
 
 
c276a99
b9247d3
 
 
 
 
 
 
c276a99
b9247d3
 
 
 
 
 
 
 
 
 
0d79725
 
 
 
 
 
 
 
b9247d3
 
c276a99
b9247d3
 
 
 
 
c276a99
0d79725
 
 
 
 
b9247d3
 
c276a99
b9247d3
 
 
0d79725
b9247d3
 
6f65703
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
b9247d3
 
 
 
0d79725
 
 
 
 
 
 
 
 
6f65703
 
 
0d79725
 
b9247d3
6f65703
 
 
 
 
 
 
b9247d3
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
0d79725
b9247d3
 
 
 
0d79725
 
 
 
6f65703
 
0d79725
 
 
 
 
 
b9247d3
0d79725
b9247d3
0d79725
6f65703
 
b9247d3
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
0d79725
b9247d3
 
 
 
0d79725
 
 
 
6f65703
 
0d79725
 
 
 
 
 
 
 
 
 
 
 
b9247d3
0d79725
b9247d3
0d79725
6f65703
 
 
 
 
 
0d79725
 
 
 
 
 
 
 
 
 
 
 
fce53a0
0d79725
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
30
31
32
33
34
35
36
37
38
39
40
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
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
import gradio as gr
import threading
import os
import shutil
import tempfile
import time
from util import process_image_edit, process_local_image_edit
from nfsw import NSFWDetector

# 配置参数
NSFW_TIME_WINDOW = 5  # 时间窗口:5分钟
NSFW_LIMIT = 10        # 限制次数:6次

IP_Dict = {}
NSFW_Dict = {}  # 记录每个IP的NSFW违规次数
NSFW_Time_Dict = {}  # 记录每个IP在特定时间窗口的NSFW检测次数,键格式: "ip_timestamp"

def get_current_time_window():
    """
    获取当前的整点时间窗口
    
    Returns:
        tuple: (窗口开始时间戳, 窗口结束时间戳)
    """
    current_time = time.time()
    # 获取当前时间的分钟数
    current_struct = time.localtime(current_time)
    current_minute = current_struct.tm_min
    
    # 计算当前5分钟时间窗口的开始分钟
    window_start_minute = (current_minute // NSFW_TIME_WINDOW) * NSFW_TIME_WINDOW
    
    # 构建窗口开始时间
    window_start_struct = time.struct_time((
        current_struct.tm_year, current_struct.tm_mon, current_struct.tm_mday,
        current_struct.tm_hour, window_start_minute, 0,
        current_struct.tm_wday, current_struct.tm_yday, current_struct.tm_isdst
    ))
    
    window_start_time = time.mktime(window_start_struct)
    window_end_time = window_start_time + (NSFW_TIME_WINDOW * 60)
    
    return window_start_time, window_end_time

def check_nsfw_rate_limit(client_ip):
    """
    检查IP的NSFW检测频率限制(基于整点时间窗口)
    
    Args:
        client_ip (str): 客户端IP地址
    
    Returns:
        tuple: (是否超过限制, 剩余等待时间)
    """
    current_time = time.time()
    window_start_time, window_end_time = get_current_time_window()
    
    # 清理不在当前时间窗口的记录
    current_window_key = f"{client_ip}_{int(window_start_time)}"
    
    # 如果没有当前窗口的记录,创建新的
    if current_window_key not in NSFW_Time_Dict:
        NSFW_Time_Dict[current_window_key] = 0
    
    # 清理旧的窗口记录(保持内存清洁)
    keys_to_remove = []
    for key in NSFW_Time_Dict:
        if key.startswith(client_ip + "_"):
            window_time = int(key.split("_")[1])
            if window_time < window_start_time:
                keys_to_remove.append(key)
    
    for key in keys_to_remove:
        del NSFW_Time_Dict[key]
    
    # 检查当前窗口是否超过限制
    if NSFW_Time_Dict[current_window_key] >= NSFW_LIMIT:
        # 计算到下一个时间窗口的等待时间
        wait_time = window_end_time - current_time
        return True, max(0, wait_time)
    
    return False, 0

def record_nsfw_detection(client_ip):
    """
    记录IP的NSFW检测时间(基于整点时间窗口)
    
    Args:
        client_ip (str): 客户端IP地址
    """
    window_start_time, _ = get_current_time_window()
    current_window_key = f"{client_ip}_{int(window_start_time)}"
    
    # 增加当前窗口的计数
    if current_window_key not in NSFW_Time_Dict:
        NSFW_Time_Dict[current_window_key] = 0
    NSFW_Time_Dict[current_window_key] += 1
    
    # 记录到NSFW_Dict中(兼容现有逻辑)
    if client_ip not in NSFW_Dict:
        NSFW_Dict[client_ip] = 0
    NSFW_Dict[client_ip] += 1

def get_current_window_info(client_ip):
    """
    获取当前窗口的统计信息(用于调试)
    
    Args:
        client_ip (str): 客户端IP地址
        
    Returns:
        dict: 当前窗口的统计信息
    """
    window_start_time, window_end_time = get_current_time_window()
    current_window_key = f"{client_ip}_{int(window_start_time)}"
    
    current_count = NSFW_Time_Dict.get(current_window_key, 0)
    
    # 格式化时间显示
    start_time_str = time.strftime("%H:%M:%S", time.localtime(window_start_time))
    end_time_str = time.strftime("%H:%M:%S", time.localtime(window_end_time))
    
    return {
        "window_start": start_time_str,
        "window_end": end_time_str,
        "current_count": current_count,
        "limit": NSFW_LIMIT,
        "window_key": current_window_key
    }

# 初始化NSFW检测器(从Hugging Face下载)
try:
    nsfw_detector = NSFWDetector()  # 自动从Hugging Face下载falconsai_yolov9_nsfw_model_quantized.pt
    print("✅ NSFW检测器初始化成功")
except Exception as e:
    print(f"❌ NSFW检测器初始化失败: {e}")
    nsfw_detector = None

def edit_image_interface(input_image, prompt, request: gr.Request, progress=gr.Progress()):
    """
    Interface function for processing image editing
    """
    try:
        # 提取用户IP
        client_ip = request.client.host
        x_forwarded_for = dict(request.headers).get('x-forwarded-for')
        if x_forwarded_for:
            client_ip = x_forwarded_for   
        if client_ip not in IP_Dict:
            IP_Dict[client_ip] = 0
        IP_Dict[client_ip] += 1

        
        if input_image is None:
            return None, "Please upload an image first"
        
        if not prompt or prompt.strip() == "":
            return None, "Please enter editing prompt"
        
        # 检查prompt长度是否大于3个字符
        if len(prompt.strip()) <= 3:
            return None, "❌ Editing prompt must be more than 3 characters"
    except Exception as e:
        print(f"⚠️ Request preprocessing error: {e}")
        return None, "❌ Request processing error"
    
    # 检查图片是否包含NSFW内容
    nsfw_result = None
    if nsfw_detector is not None and input_image is not None:
        try:
            # 直接使用PIL Image对象进行检测,避免文件路径问题
            nsfw_result = nsfw_detector.predict_pil_label_only(input_image)
            
            if nsfw_result.lower() == "nsfw":
                print(f"🔍 NSFW检测结果: ❌❌❌ {nsfw_result} - IP: {client_ip}")
                # 检查NSFW频率限制
                is_rate_limited, wait_time = check_nsfw_rate_limit(client_ip)
                
                if is_rate_limited:
                    # 超过频率限制,显示等待提示并阻止继续
                    wait_minutes = int(wait_time / 60) + 1  # 向上取整到分钟
                    window_info = get_current_window_info(client_ip)
                    print(f"⚠️ NSFW频率限制 - IP: {client_ip}")
                    print(f"   时间窗口: {window_info['window_start']} - {window_info['window_end']}")
                    print(f"   当前计数: {window_info['current_count']}/{NSFW_LIMIT}, 需要等待 {wait_minutes} 分钟")
                    return None, f"❌ Please wait {wait_minutes} minutes before generating again"
                else:
                    # 未超过频率限制,记录此次检测但允许继续处理
                    record_nsfw_detection(client_ip)
                    window_info = get_current_window_info(client_ip)
            else:
                print(f"🔍 NSFW检测结果: ✅✅✅ {nsfw_result} - IP: {client_ip}")
                
        except Exception as e:
            print(f"⚠️ NSFW检测失败: {e}")
            # 检测失败时允许继续处理
    

    result_url = None
    status_message = ""
    
    def progress_callback(message):
        try:
            nonlocal status_message
            status_message = message
            # 增加错误处理,防止 progress 更新失败
            if progress is not None:
                progress(0.5, desc=message)
        except Exception as e:
            print(f"⚠️ Progress update failed: {e}")
    
    try:
        # 打印成功访问的信息
        print(f"✅ Processing started - IP: {client_ip}, count: {IP_Dict[client_ip]}, prompt: {prompt.strip()}", flush=True)
        
        # Call image editing processing function
        result_url, message = process_image_edit(input_image, prompt.strip(), progress_callback)
        
        if result_url:
            print(f"✅ Processing completed successfully - IP: {client_ip}, result_url: {result_url}", flush=True)
            try:
                if progress is not None:
                    progress(1.0, desc="Processing completed")
            except Exception as e:
                print(f"⚠️ Final progress update failed: {e}")
            return result_url, "✅ " + message
        else:
            print(f"❌ Processing failed - IP: {client_ip}, error: {message}", flush=True)
            return None, "❌ " + message
            
    except Exception as e:
        print(f"❌ Processing exception - IP: {client_ip}, error: {str(e)}")
        return None, f"❌ Error occurred during processing: {str(e)}"

# 不再需要复杂的状态管理函数,已简化为内联函数

def local_edit_interface(image_dict, prompt, request: gr.Request, progress=gr.Progress()):
    """
    处理局部编辑请求
    """
    try:
        # 提取用户IP
        client_ip = request.client.host
        x_forwarded_for = dict(request.headers).get('x-forwarded-for')
        if x_forwarded_for:
            client_ip = x_forwarded_for   
        if client_ip not in IP_Dict:
            IP_Dict[client_ip] = 0
        IP_Dict[client_ip] += 1

        
        if image_dict is None:
            return None, "Please upload an image and draw the area to edit"
        
        # Check if background and layers exist
        if "background" not in image_dict or "layers" not in image_dict:
            return None, "Please draw the area to edit on the image"
        
        base_image = image_dict["background"]
        layers = image_dict["layers"]
        
        if not layers:
            return None, "Please draw the area to edit on the image"
            
        if not prompt or prompt.strip() == "":
            return None, "Please enter editing prompt"
        
        # Check prompt length
        if len(prompt.strip()) <= 3:
            return None, "❌ Editing prompt must be more than 3 characters"
    except Exception as e:
        print(f"⚠️ Local edit request preprocessing error: {e}")
        return None, "❌ Request processing error"
    
    # 检查图片是否包含NSFW内容
    nsfw_result = None
    if nsfw_detector is not None and base_image is not None:
        try:
            nsfw_result = nsfw_detector.predict_pil_label_only(base_image)
            
            if nsfw_result.lower() == "nsfw":
                print(f"🔍 NSFW检测结果: ❌❌❌ {nsfw_result} - IP: {client_ip}")
                # 检查NSFW频率限制
                is_rate_limited, wait_time = check_nsfw_rate_limit(client_ip)
                
                if is_rate_limited:
                    wait_minutes = int(wait_time / 60) + 1
                    window_info = get_current_window_info(client_ip)
                    print(f"⚠️ NSFW频率限制 - IP: {client_ip}")
                    print(f"   时间窗口: {window_info['window_start']} - {window_info['window_end']}")
                    print(f"   当前计数: {window_info['current_count']}/{NSFW_LIMIT}, 需要等待 {wait_minutes} 分钟")
                    return None, f"❌ Please wait {wait_minutes} minutes before generating again"
                else:
                    record_nsfw_detection(client_ip)
                    window_info = get_current_window_info(client_ip)
            else:
                print(f"🔍 NSFW检测结果: ✅✅✅ {nsfw_result} - IP: {client_ip}")
                
        except Exception as e:
            print(f"⚠️ NSFW检测失败: {e}")
    
    # IP访问限制检查

    result_url = None
    status_message = ""
    
    def progress_callback(message):
        try:
            nonlocal status_message
            status_message = message
            # 增加错误处理,防止 progress 更新失败
            if progress is not None:
                progress(0.5, desc=message)
        except Exception as e:
            print(f"⚠️ Local edit progress update failed: {e}")
    
    try:
        print(f"✅ Local editing started - IP: {client_ip}, count: {IP_Dict[client_ip]}, prompt: {prompt.strip()}", flush=True)
        
        # 调用局部图像编辑处理函数
        result_url, message = process_local_image_edit(base_image, layers, prompt.strip(), progress_callback)
        
        if result_url:
            print(f"✅ Local editing completed successfully - IP: {client_ip}, result_url: {result_url}", flush=True)
            try:
                if progress is not None:
                    progress(1.0, desc="Processing completed")
            except Exception as e:
                print(f"⚠️ Local edit final progress update failed: {e}")
            return result_url, "✅ " + message
        else:
            print(f"❌ Local editing processing failed - IP: {client_ip}, error: {message}", flush=True)
            return None, "❌ " + message
            
    except Exception as e:
        print(f"❌ Local editing exception - IP: {client_ip}, error: {str(e)}")
        return None, f"❌ Error occurred during processing: {str(e)}"

# Create Gradio interface
def create_app():
    with gr.Blocks(
        title="AI Image Editor",
        theme=gr.themes.Soft(),
        css="""
        .main-container {
            max-width: 1200px;
            margin: 0 auto;
        }
        .upload-area {
            border: 2px dashed #ccc;
            border-radius: 10px;
            padding: 20px;
            text-align: center;
        }
        .result-area {
            margin-top: 20px;
            padding: 20px;
            border-radius: 10px;
            background-color: #f8f9fa;
        }
        .use-as-input-btn {
            margin-top: 10px;
            width: 100%;
        }
        """,
        # 改善并发性能的配置
        head="""
        <script>
        // 减少客户端状态更新频率,避免过度的 SSE 连接
        if (window.gradio) {
            window.gradio.update_frequency = 2000; // 2秒更新一次
        }
        </script>
        """
    ) as app:
        
        # 减少State组件,只保留必要的
        # 移除了大部分State组件以减少状态管理复杂度
        
        gr.Markdown(
            """
            # 🎨 AI Image Editor
            """,
            elem_classes=["main-container"]
        )
        
        with gr.Tabs():
            # Global editing tab
            with gr.Tab("🌍 Global Editing"):
                with gr.Row():
                    with gr.Column(scale=1):
                        gr.Markdown("### 📸 Upload Image")
                        input_image = gr.Image(
                            label="Select image to edit",
                            type="pil",
                            height=512,
                            elem_classes=["upload-area"]
                        )
                        
                        gr.Markdown("### ✍️ Editing Instructions")
                        prompt_input = gr.Textbox(
                            label="Enter editing prompt",
                            placeholder="For example: change background to beach, add rainbow, remove background, etc...",
                            lines=3,
                            max_lines=5
                        )
                        
                        edit_button = gr.Button(
                            "🚀 Start Editing",
                            variant="primary",
                            size="lg"
                        )
                    
                    with gr.Column(scale=1):
                        gr.Markdown("### 🎯 Editing Result")
                        output_image = gr.Image(
                            label="Edited image",
                            height=320,
                            elem_classes=["result-area"]
                        )
                        
                        # 添加 "Use as Input" 按钮
                        use_as_input_btn = gr.Button(
                            "🔄 Use as Input",
                            variant="secondary",
                            size="sm",
                            elem_classes=["use-as-input-btn"]
                        )
                        
                        status_output = gr.Textbox(
                            label="Processing status",
                            lines=2,
                            max_lines=3,
                            interactive=False
                        )
                
                # Example area
                gr.Markdown("### 💡 Prompt Examples")
                with gr.Row():
                    example_prompts = [
                        "Change the character's background to a sunny seaside with blue waves",
                        "Change the character's background to New York at night with neon lights",
                        "Change the character's background to a fairytale castle with bright colors",
                        "Change background to forest",
                        "Change background to snow mountain"
                    ]
                    
                    for prompt in example_prompts:
                        gr.Button(
                            prompt,
                            size="sm"
                        ).click(
                            lambda p=prompt: p,
                            outputs=prompt_input
                        )
                
                # 绑定按钮点击事件 - 简化,移除状态管理
                edit_button.click(
                    fn=edit_image_interface,
                    inputs=[input_image, prompt_input],
                    outputs=[output_image, status_output],
                    show_progress=True,
                    # 增加并发设置
                    concurrency_limit=10,  # 限制并发数
                    api_name="global_edit"
                )
                
                # 简化 "Use as Input" 按钮,直接复制图片
                def simple_use_as_input(output_img):
                    if output_img is not None:
                        return output_img
                    return None
                
                use_as_input_btn.click(
                    fn=simple_use_as_input,
                    inputs=[output_image],
                    outputs=[input_image]
                )
            
            # Local editing tab  
            with gr.Tab("🖌️ Local Editing"):
                with gr.Row():
                    with gr.Column(scale=1):
                        gr.Markdown("### 📸 Upload Image and Draw Edit Area")
                        local_input_image = gr.ImageEditor(
                            label="Upload image and draw mask",
                            type="pil",
                            height=512,
                            brush=gr.Brush(colors=["#ff0000"], default_size=60),
                            elem_classes=["upload-area"]
                        )
                        
                        gr.Markdown("### ✍️ Editing Instructions")
                        local_prompt_input = gr.Textbox(
                            label="Enter local editing prompt",
                            placeholder="For example: change selected area hair to golden, add patterns to selected object, change selected area color, etc...",
                            lines=3,
                            max_lines=5
                        )
                        
                        local_edit_button = gr.Button(
                            "🎯 Start Local Editing",
                            variant="primary",
                            size="lg"
                        )
                    
                    with gr.Column(scale=1):
                        gr.Markdown("### 🎯 Editing Result")
                        local_output_image = gr.Image(
                            label="Local edited image",
                            height=320,
                            elem_classes=["result-area"]
                        )
                        
                        # 添加 "Use as Input" 按钮
                        local_use_as_input_btn = gr.Button(
                            "🔄 Use as Input",
                            variant="secondary",
                            size="sm",
                            elem_classes=["use-as-input-btn"]
                        )
                        
                        local_status_output = gr.Textbox(
                            label="Processing status",
                            lines=2,
                            max_lines=3,
                            interactive=False
                        )
                
                # Local editing examples
                gr.Markdown("### 💡 Local Editing Prompt Examples")
                with gr.Row():
                    local_example_prompts = [
                        "Change selected area hair to golden",
                        "Add pattern designs to selected clothing",
                        "Change selected area to different material",
                        "Add decorations to selected object",
                        "Change selected area color and style"
                    ]
                    
                    for prompt in local_example_prompts:
                        gr.Button(
                            prompt,
                            size="sm"
                        ).click(
                            lambda p=prompt: p,
                            outputs=local_prompt_input
                        )
                
                # 绑定局部编辑按钮点击事件 - 简化,移除状态管理
                local_edit_button.click(
                    fn=local_edit_interface,
                    inputs=[local_input_image, local_prompt_input],
                    outputs=[local_output_image, local_status_output],
                    show_progress=True,
                    # 增加并发设置
                    concurrency_limit=8,  # 局部编辑更复杂,限制更少的并发
                    api_name="local_edit"
                )
                
                # 简化局部编辑 "Use as Input" 按钮
                def simple_local_use_as_input(output_img):
                    if output_img is not None:
                        # 创建简单的 ImageEditor 格式
                        editor_data = {
                            "background": output_img, 
                            "layers": [],
                            "composite": output_img
                        }
                        return editor_data
                    return None
                
                local_use_as_input_btn.click(
                    fn=simple_local_use_as_input,
                    inputs=[local_output_image],
                    outputs=[local_input_image]
                )
    
    return app

if __name__ == "__main__":
    app = create_app()
    # 改善队列配置以处理高并发和防止 SSE 连接问题
    app.queue(
        default_concurrency_limit=20,  # 默认并发限制
        max_size=50,  # 队列最大大小
        api_open=False  # 关闭 API 访问,减少资源消耗
    )
    app.launch(
        server_name="0.0.0.0",
        show_error=True,  # 显示详细错误信息
        quiet=False,  # 保持日志输出
        max_threads=40,  # 增加线程池大小
        height=800,
        favicon_path=None  # 减少资源加载
    )