mirror of
				https://github.com/Minidoracat/mcp-feedback-enhanced.git
				synced 2025-10-27 01:16:00 +08:00 
			
		
		
		
	✨ 新增 desktop 模式
This commit is contained in:
		
							parent
							
								
									3ed676cf29
								
							
						
					
					
						commit
						a2d4180cf6
					
				
							
								
								
									
										51
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										51
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -10,13 +10,62 @@ wheels/ | |||||||
| .venv*/ | .venv*/ | ||||||
| venv*/ | venv*/ | ||||||
| 
 | 
 | ||||||
|  | # Node.js / Electron (Desktop App) | ||||||
|  | node_modules/ | ||||||
|  | npm-debug.log* | ||||||
|  | yarn-debug.log* | ||||||
|  | yarn-error.log* | ||||||
|  | package-lock.json | ||||||
|  | yarn.lock | ||||||
|  | .npm | ||||||
|  | .yarn-integrity | ||||||
|  | 
 | ||||||
|  | # Electron build outputs | ||||||
|  | dist-desktop/ | ||||||
|  | out/ | ||||||
|  | *.dmg | ||||||
|  | *.exe | ||||||
|  | *.deb | ||||||
|  | *.rpm | ||||||
|  | *.AppImage | ||||||
|  | *.zip | ||||||
|  | *.tar.gz | ||||||
|  | 
 | ||||||
|  | # Electron cache | ||||||
|  | .electron/ | ||||||
|  | .electron-gyp/ | ||||||
|  | 
 | ||||||
| # Logs | # Logs | ||||||
| *.log | *.log | ||||||
| 
 | 
 | ||||||
| #Others | # macOS | ||||||
| .DS_Store | .DS_Store | ||||||
|  | .AppleDouble | ||||||
|  | .LSOverride | ||||||
| 
 | 
 | ||||||
|  | # Windows | ||||||
|  | Thumbs.db | ||||||
|  | ehthumbs.db | ||||||
|  | Desktop.ini | ||||||
|  | 
 | ||||||
|  | # Linux | ||||||
|  | *~ | ||||||
|  | 
 | ||||||
|  | # IDE | ||||||
|  | .vscode/ | ||||||
|  | .idea/ | ||||||
|  | *.swp | ||||||
|  | *.swo | ||||||
|  | 
 | ||||||
|  | # Others | ||||||
| .cursor/rules/ | .cursor/rules/ | ||||||
| uv.lock | uv.lock | ||||||
| .mcp_feedback_settings.json | .mcp_feedback_settings.json | ||||||
| test_reports/ | test_reports/ | ||||||
|  | 
 | ||||||
|  | # Temporary test files | ||||||
|  | test_*.py | ||||||
|  | 
 | ||||||
|  | # User configuration files | ||||||
|  | ui_settings.json | ||||||
|  | .config/ | ||||||
							
								
								
									
										154
									
								
								electron-builder.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										154
									
								
								electron-builder.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,154 @@ | |||||||
|  | { | ||||||
|  |   "appId": "com.minidoracat.mcp-feedback-enhanced", | ||||||
|  |   "productName": "MCP Feedback Enhanced", | ||||||
|  |   "copyright": "Copyright © 2024 Minidoracat", | ||||||
|  |   "directories": { | ||||||
|  |     "app": "src/mcp_feedback_enhanced/desktop", | ||||||
|  |     "output": "dist-desktop", | ||||||
|  |     "buildResources": "src/mcp_feedback_enhanced/desktop/assets" | ||||||
|  |   }, | ||||||
|  |   "files": [ | ||||||
|  |     "main.js", | ||||||
|  |     "preload.js", | ||||||
|  |     "package.json", | ||||||
|  |     "node_modules/**/*" | ||||||
|  |   ], | ||||||
|  |   "extraResources": [ | ||||||
|  |     { | ||||||
|  |       "from": "src/mcp_feedback_enhanced", | ||||||
|  |       "to": "app", | ||||||
|  |       "filter": [ | ||||||
|  |         "**/*", | ||||||
|  |         "!desktop/node_modules/**/*", | ||||||
|  |         "!desktop/dist/**/*", | ||||||
|  |         "!**/*.pyc", | ||||||
|  |         "!**/__pycache__/**/*", | ||||||
|  |         "!**/test_*.py" | ||||||
|  |       ] | ||||||
|  |     } | ||||||
|  |   ], | ||||||
|  |   "compression": "normal", | ||||||
|  |   "removePackageScripts": true, | ||||||
|  |   "win": { | ||||||
|  |     "target": [ | ||||||
|  |       { | ||||||
|  |         "target": "nsis", | ||||||
|  |         "arch": ["x64", "ia32"] | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         "target": "portable", | ||||||
|  |         "arch": ["x64"] | ||||||
|  |       } | ||||||
|  |     ], | ||||||
|  |     "icon": "src/mcp_feedback_enhanced/desktop/assets/icon.ico", | ||||||
|  |     "requestedExecutionLevel": "asInvoker", | ||||||
|  |     "artifactName": "${productName}-${version}-${arch}.${ext}", | ||||||
|  |     "publisherName": "Minidoracat" | ||||||
|  |   }, | ||||||
|  |   "mac": { | ||||||
|  |     "target": [ | ||||||
|  |       { | ||||||
|  |         "target": "dmg", | ||||||
|  |         "arch": ["x64", "arm64"] | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         "target": "zip", | ||||||
|  |         "arch": ["x64", "arm64"] | ||||||
|  |       } | ||||||
|  |     ], | ||||||
|  |     "icon": "src/mcp_feedback_enhanced/desktop/assets/icon.icns", | ||||||
|  |     "category": "public.app-category.developer-tools", | ||||||
|  |     "hardenedRuntime": true, | ||||||
|  |     "gatekeeperAssess": false, | ||||||
|  |     "artifactName": "${productName}-${version}-${arch}.${ext}" | ||||||
|  |   }, | ||||||
|  |   "linux": { | ||||||
|  |     "target": [ | ||||||
|  |       { | ||||||
|  |         "target": "AppImage", | ||||||
|  |         "arch": ["x64"] | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         "target": "deb", | ||||||
|  |         "arch": ["x64"] | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         "target": "rpm", | ||||||
|  |         "arch": ["x64"] | ||||||
|  |       } | ||||||
|  |     ], | ||||||
|  |     "icon": "src/mcp_feedback_enhanced/desktop/assets/icon.png", | ||||||
|  |     "category": "Development", | ||||||
|  |     "artifactName": "${productName}-${version}-${arch}.${ext}", | ||||||
|  |     "desktop": { | ||||||
|  |       "StartupNotify": "true", | ||||||
|  |       "Encoding": "UTF-8", | ||||||
|  |       "MimeType": "x-scheme-handler/mcp-feedback", | ||||||
|  |       "Keywords": "mcp;feedback;ai;development" | ||||||
|  |     } | ||||||
|  |   }, | ||||||
|  |   "nsis": { | ||||||
|  |     "oneClick": false, | ||||||
|  |     "allowToChangeInstallationDirectory": true, | ||||||
|  |     "allowElevation": true, | ||||||
|  |     "createDesktopShortcut": true, | ||||||
|  |     "createStartMenuShortcut": true, | ||||||
|  |     "shortcutName": "MCP Feedback Enhanced", | ||||||
|  |     "include": "build/installer.nsh", | ||||||
|  |     "artifactName": "${productName}-Setup-${version}.${ext}", | ||||||
|  |     "deleteAppDataOnUninstall": false | ||||||
|  |   }, | ||||||
|  |   "dmg": { | ||||||
|  |     "title": "${productName} ${version}", | ||||||
|  |     "icon": "src/mcp_feedback_enhanced/desktop/assets/icon.icns", | ||||||
|  |     "background": "src/mcp_feedback_enhanced/desktop/assets/dmg-background.png", | ||||||
|  |     "contents": [ | ||||||
|  |       { | ||||||
|  |         "x": 130, | ||||||
|  |         "y": 220, | ||||||
|  |         "type": "file" | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         "x": 410, | ||||||
|  |         "y": 220, | ||||||
|  |         "type": "link", | ||||||
|  |         "path": "/Applications" | ||||||
|  |       } | ||||||
|  |     ], | ||||||
|  |     "window": { | ||||||
|  |       "width": 540, | ||||||
|  |       "height": 380 | ||||||
|  |     }, | ||||||
|  |     "artifactName": "${productName}-${version}.${ext}" | ||||||
|  |   }, | ||||||
|  |   "appImage": { | ||||||
|  |     "artifactName": "${productName}-${version}-${arch}.${ext}" | ||||||
|  |   }, | ||||||
|  |   "deb": { | ||||||
|  |     "artifactName": "${productName}-${version}-${arch}.${ext}", | ||||||
|  |     "priority": "optional", | ||||||
|  |     "depends": [ | ||||||
|  |       "gconf2", | ||||||
|  |       "gconf-service", | ||||||
|  |       "libnotify4", | ||||||
|  |       "libappindicator1", | ||||||
|  |       "libxtst6", | ||||||
|  |       "libnss3" | ||||||
|  |     ] | ||||||
|  |   }, | ||||||
|  |   "rpm": { | ||||||
|  |     "artifactName": "${productName}-${version}-${arch}.${ext}", | ||||||
|  |     "depends": [ | ||||||
|  |       "libXScrnSaver", | ||||||
|  |       "libnotify", | ||||||
|  |       "libnss3" | ||||||
|  |     ] | ||||||
|  |   }, | ||||||
|  |   "publish": { | ||||||
|  |     "provider": "github", | ||||||
|  |     "owner": "minidoracat", | ||||||
|  |     "repo": "mcp-feedback-enhanced", | ||||||
|  |     "releaseType": "draft" | ||||||
|  |   }, | ||||||
|  |   "buildVersion": "2.3.0" | ||||||
|  | } | ||||||
| @ -1,13 +1,13 @@ | |||||||
| [project] | [project] | ||||||
| name = "mcp-feedback-enhanced" | name = "mcp-feedback-enhanced" | ||||||
| version = "2.3.0" | version = "2.3.0" | ||||||
| description = "Enhanced MCP server for interactive user feedback and command execution in AI-assisted development, featuring Web UI with intelligent environment detection." | description = "Enhanced MCP server for interactive user feedback and command execution in AI-assisted development, featuring Web UI with intelligent environment detection and optional Electron desktop application." | ||||||
| readme = "README.md" | readme = "README.md" | ||||||
| requires-python = ">=3.11" | requires-python = ">=3.11" | ||||||
| authors = [ | authors = [ | ||||||
|     { name = "Minidoracat", email = "minidora0702@gmail.com" } |     { name = "Minidoracat", email = "minidora0702@gmail.com" } | ||||||
| ] | ] | ||||||
| keywords = ["mcp", "ai", "feedback", "web-ui", "interactive", "development"] | keywords = ["mcp", "ai", "feedback", "web-ui", "interactive", "development", "electron", "desktop", "cross-platform"] | ||||||
| classifiers = [ | classifiers = [ | ||||||
|     "Development Status :: 4 - Beta", |     "Development Status :: 4 - Beta", | ||||||
|     "Intended Audience :: Developers", |     "Intended Audience :: Developers", | ||||||
| @ -34,6 +34,9 @@ dev = [ | |||||||
|     "pytest>=7.0.0", |     "pytest>=7.0.0", | ||||||
|     "pytest-asyncio>=0.21.0", |     "pytest-asyncio>=0.21.0", | ||||||
| ] | ] | ||||||
|  | desktop = [ | ||||||
|  |     "nodejs>=16.0.0",  # Node.js 環境(需要系統安裝) | ||||||
|  | ] | ||||||
| 
 | 
 | ||||||
| [project.urls] | [project.urls] | ||||||
| Homepage = "https://github.com/Minidoracat/mcp-feedback-enhanced" | Homepage = "https://github.com/Minidoracat/mcp-feedback-enhanced" | ||||||
|  | |||||||
| @ -14,6 +14,19 @@ MCP Interactive Feedback Enhanced - 主程式入口 | |||||||
| import sys | import sys | ||||||
| import argparse | import argparse | ||||||
| import os | import os | ||||||
|  | import asyncio | ||||||
|  | import warnings | ||||||
|  | 
 | ||||||
|  | # 抑制 Windows 上的 asyncio ResourceWarning | ||||||
|  | if sys.platform == 'win32': | ||||||
|  |     warnings.filterwarnings("ignore", category=ResourceWarning, message=".*unclosed transport.*") | ||||||
|  |     warnings.filterwarnings("ignore", category=ResourceWarning, message=".*unclosed.*") | ||||||
|  | 
 | ||||||
|  |     # 設置 asyncio 事件循環策略以減少警告 | ||||||
|  |     try: | ||||||
|  |         asyncio.set_event_loop_policy(asyncio.WindowsProactorEventLoopPolicy()) | ||||||
|  |     except AttributeError: | ||||||
|  |         pass | ||||||
| 
 | 
 | ||||||
| def main(): | def main(): | ||||||
|     """主程式入口點""" |     """主程式入口點""" | ||||||
| @ -29,6 +42,9 @@ def main(): | |||||||
|     # 測試命令 |     # 測試命令 | ||||||
|     test_parser = subparsers.add_parser('test', help='執行測試') |     test_parser = subparsers.add_parser('test', help='執行測試') | ||||||
|     test_parser.add_argument('--web', action='store_true', help='測試 Web UI (自動持續運行)') |     test_parser.add_argument('--web', action='store_true', help='測試 Web UI (自動持續運行)') | ||||||
|  |     test_parser.add_argument('--desktop', action='store_true', help='測試桌面應用 (啟動 Electron 應用)') | ||||||
|  |     test_parser.add_argument('--full', action='store_true', help='完整整合測試 (Web + 桌面)') | ||||||
|  |     test_parser.add_argument('--electron-only', action='store_true', help='僅測試 Electron 環境') | ||||||
|     test_parser.add_argument('--timeout', type=int, default=60, help='測試超時時間 (秒)') |     test_parser.add_argument('--timeout', type=int, default=60, help='測試超時時間 (秒)') | ||||||
|      |      | ||||||
|     # 版本命令 |     # 版本命令 | ||||||
| @ -59,14 +75,37 @@ def run_tests(args): | |||||||
|     # 啟用調試模式以顯示測試過程 |     # 啟用調試模式以顯示測試過程 | ||||||
|     os.environ["MCP_DEBUG"] = "true" |     os.environ["MCP_DEBUG"] = "true" | ||||||
| 
 | 
 | ||||||
|  |     # 在 Windows 上抑制 asyncio 警告 | ||||||
|  |     if sys.platform == 'win32': | ||||||
|  |         os.environ["PYTHONWARNINGS"] = "ignore::ResourceWarning" | ||||||
|  | 
 | ||||||
|     if args.web: |     if args.web: | ||||||
|         print("🧪 執行 Web UI 測試...") |         print("🧪 執行 Web UI 測試...") | ||||||
|         success = test_web_ui_simple() |         success = test_web_ui_simple() | ||||||
|         if not success: |         if not success: | ||||||
|             sys.exit(1) |             sys.exit(1) | ||||||
|  |     elif args.desktop: | ||||||
|  |         print("🧪 執行桌面應用測試...") | ||||||
|  |         success = test_desktop_app() | ||||||
|  |         if not success: | ||||||
|  |             sys.exit(1) | ||||||
|  |     elif args.full: | ||||||
|  |         print("🧪 執行完整整合測試...") | ||||||
|  |         success = test_full_integration() | ||||||
|  |         if not success: | ||||||
|  |             sys.exit(1) | ||||||
|  |     elif args.electron_only: | ||||||
|  |         print("🧪 執行 Electron 環境測試...") | ||||||
|  |         success = test_electron_environment() | ||||||
|  |         if not success: | ||||||
|  |             sys.exit(1) | ||||||
|     else: |     else: | ||||||
|         print("❌ 測試功能已簡化") |         print("❌ 測試功能已簡化") | ||||||
|         print("💡 對於用戶:使用 'test --web' 測試 Web UI") |         print("💡 可用的測試選項:") | ||||||
|  |         print("  --web         測試 Web UI") | ||||||
|  |         print("  --desktop     測試桌面應用") | ||||||
|  |         print("  --full        完整整合測試") | ||||||
|  |         print("  --electron-only  僅測試 Electron 環境") | ||||||
|         print("💡 對於開發者:使用 'uv run pytest' 執行完整測試") |         print("💡 對於開發者:使用 'uv run pytest' 執行完整測試") | ||||||
|         sys.exit(1) |         sys.exit(1) | ||||||
| 
 | 
 | ||||||
| @ -135,6 +174,215 @@ def test_web_ui_simple(): | |||||||
|         return False |         return False | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | def test_desktop_app(): | ||||||
|  |     """測試桌面應用""" | ||||||
|  |     try: | ||||||
|  |         print("🔧 檢查桌面環境...") | ||||||
|  | 
 | ||||||
|  |         # 檢查桌面環境可用性 | ||||||
|  |         from .desktop import is_desktop_available | ||||||
|  |         if not is_desktop_available(): | ||||||
|  |             print("❌ 桌面環境不可用") | ||||||
|  |             print("💡 請確保 Node.js 已安裝且不在遠程環境中") | ||||||
|  |             return False | ||||||
|  | 
 | ||||||
|  |         print("✅ 桌面環境檢查通過") | ||||||
|  | 
 | ||||||
|  |         # 設置桌面模式 | ||||||
|  |         os.environ['MCP_FEEDBACK_MODE'] = 'desktop' | ||||||
|  | 
 | ||||||
|  |         print("🔧 創建 Electron 管理器...") | ||||||
|  |         from .desktop.electron_manager import ElectronManager | ||||||
|  |         import asyncio | ||||||
|  | 
 | ||||||
|  |         async def run_desktop_test(): | ||||||
|  |             print("🚀 啟動完整桌面應用測試...") | ||||||
|  |             print("💡 這將啟動 Web 服務器和 Electron 應用視窗") | ||||||
|  |             print("💡 請在應用中測試基本功能,然後關閉視窗") | ||||||
|  | 
 | ||||||
|  |             # 使用完整的桌面應用啟動函數 | ||||||
|  |             from .desktop import launch_desktop_app | ||||||
|  | 
 | ||||||
|  |             try: | ||||||
|  |                 # 這會啟動 Web 服務器和 Electron 應用 | ||||||
|  |                 result = await launch_desktop_app( | ||||||
|  |                     os.getcwd(), | ||||||
|  |                     "桌面應用測試 - 驗證 Electron 整合功能", | ||||||
|  |                     300  # 5分鐘超時 | ||||||
|  |                 ) | ||||||
|  | 
 | ||||||
|  |                 print("✅ 桌面應用測試完成") | ||||||
|  |                 print(f"收到回饋: {result.get('interactive_feedback', '無回饋')}") | ||||||
|  |                 return True | ||||||
|  | 
 | ||||||
|  |             except Exception as e: | ||||||
|  |                 print(f"❌ 桌面應用測試失敗: {e}") | ||||||
|  |                 return False | ||||||
|  | 
 | ||||||
|  |         return asyncio.run(run_desktop_test()) | ||||||
|  | 
 | ||||||
|  |     except Exception as e: | ||||||
|  |         print(f"❌ 桌面應用測試失敗: {e}") | ||||||
|  |         import traceback | ||||||
|  |         traceback.print_exc() | ||||||
|  |         return False | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | async def wait_for_process(process): | ||||||
|  |     """等待進程結束""" | ||||||
|  |     try: | ||||||
|  |         # 等待進程自然結束 | ||||||
|  |         await process.wait() | ||||||
|  | 
 | ||||||
|  |         # 確保管道正確關閉 | ||||||
|  |         try: | ||||||
|  |             if hasattr(process, 'stdout') and process.stdout: | ||||||
|  |                 process.stdout.close() | ||||||
|  |             if hasattr(process, 'stderr') and process.stderr: | ||||||
|  |                 process.stderr.close() | ||||||
|  |             if hasattr(process, 'stdin') and process.stdin: | ||||||
|  |                 process.stdin.close() | ||||||
|  |         except Exception as close_error: | ||||||
|  |             print(f"關閉進程管道時出錯: {close_error}") | ||||||
|  | 
 | ||||||
|  |     except Exception as e: | ||||||
|  |         print(f"等待進程時出錯: {e}") | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def test_electron_environment(): | ||||||
|  |     """測試 Electron 環境""" | ||||||
|  |     try: | ||||||
|  |         print("🔧 檢查 Electron 環境...") | ||||||
|  | 
 | ||||||
|  |         # 檢查 Node.js | ||||||
|  |         import subprocess | ||||||
|  |         try: | ||||||
|  |             result = subprocess.run(['node', '--version'], | ||||||
|  |                                   capture_output=True, text=True, timeout=10) | ||||||
|  |             if result.returncode == 0: | ||||||
|  |                 print(f"✅ Node.js 版本: {result.stdout.strip()}") | ||||||
|  |             else: | ||||||
|  |                 print("❌ Node.js 不可用") | ||||||
|  |                 return False | ||||||
|  |         except (subprocess.TimeoutExpired, FileNotFoundError): | ||||||
|  |             print("❌ Node.js 不可用") | ||||||
|  |             return False | ||||||
|  | 
 | ||||||
|  |         # 檢查桌面模組 | ||||||
|  |         from .desktop import is_desktop_available | ||||||
|  |         if is_desktop_available(): | ||||||
|  |             print("✅ 桌面環境可用") | ||||||
|  |         else: | ||||||
|  |             print("❌ 桌面環境不可用") | ||||||
|  |             return False | ||||||
|  | 
 | ||||||
|  |         # 檢查 Electron 管理器 | ||||||
|  |         from .desktop.electron_manager import ElectronManager | ||||||
|  |         manager = ElectronManager() | ||||||
|  | 
 | ||||||
|  |         if manager.is_electron_available(): | ||||||
|  |             print("✅ Electron 環境可用") | ||||||
|  |         else: | ||||||
|  |             print("❌ Electron 環境不可用") | ||||||
|  |             return False | ||||||
|  | 
 | ||||||
|  |         # 檢查文件結構 | ||||||
|  |         desktop_dir = manager.desktop_dir | ||||||
|  |         required_files = ['main.js', 'preload.js', 'package.json'] | ||||||
|  | 
 | ||||||
|  |         for file_name in required_files: | ||||||
|  |             file_path = desktop_dir / file_name | ||||||
|  |             if file_path.exists(): | ||||||
|  |                 print(f"✅ {file_name} 存在") | ||||||
|  |             else: | ||||||
|  |                 print(f"❌ {file_name} 不存在") | ||||||
|  |                 return False | ||||||
|  | 
 | ||||||
|  |         # 檢查 node_modules | ||||||
|  |         node_modules = desktop_dir / "node_modules" | ||||||
|  |         if node_modules.exists(): | ||||||
|  |             print("✅ node_modules 存在") | ||||||
|  |         else: | ||||||
|  |             print("❌ node_modules 不存在") | ||||||
|  |             return False | ||||||
|  | 
 | ||||||
|  |         print("🎉 Electron 環境測試完成,所有檢查通過") | ||||||
|  |         return True | ||||||
|  | 
 | ||||||
|  |     except Exception as e: | ||||||
|  |         print(f"❌ Electron 環境測試失敗: {e}") | ||||||
|  |         return False | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def test_full_integration(): | ||||||
|  |     """完整整合測試""" | ||||||
|  |     try: | ||||||
|  |         print("🧪 執行完整整合測試...") | ||||||
|  | 
 | ||||||
|  |         # 1. 環境變數測試 | ||||||
|  |         print("\n📋 1. 測試環境變數控制...") | ||||||
|  |         test_cases = [("auto", "auto"), ("web", "web"), ("desktop", "desktop")] | ||||||
|  | 
 | ||||||
|  |         for env_value, expected in test_cases: | ||||||
|  |             os.environ['MCP_FEEDBACK_MODE'] = env_value | ||||||
|  | 
 | ||||||
|  |             # 重新導入以獲取新的環境變數值 | ||||||
|  |             import sys | ||||||
|  |             if 'mcp_feedback_enhanced.server' in sys.modules: | ||||||
|  |                 del sys.modules['mcp_feedback_enhanced.server'] | ||||||
|  | 
 | ||||||
|  |             from .server import get_feedback_mode | ||||||
|  |             actual = get_feedback_mode().value | ||||||
|  | 
 | ||||||
|  |             if actual == expected: | ||||||
|  |                 print(f"  ✅ MCP_FEEDBACK_MODE='{env_value}' → {actual}") | ||||||
|  |             else: | ||||||
|  |                 print(f"  ❌ MCP_FEEDBACK_MODE='{env_value}' → {actual} (期望: {expected})") | ||||||
|  |                 return False | ||||||
|  | 
 | ||||||
|  |         # 2. Electron 環境測試 | ||||||
|  |         print("\n📋 2. 測試 Electron 環境...") | ||||||
|  |         if not test_electron_environment(): | ||||||
|  |             print("❌ Electron 環境測試失敗") | ||||||
|  |             return False | ||||||
|  | 
 | ||||||
|  |         # 3. Web UI 基本功能測試 | ||||||
|  |         print("\n📋 3. 測試 Web UI 基本功能...") | ||||||
|  |         from .web.main import WebUIManager | ||||||
|  |         import tempfile | ||||||
|  | 
 | ||||||
|  |         with tempfile.TemporaryDirectory() as temp_dir: | ||||||
|  |             manager = WebUIManager(host="127.0.0.1", port=8766)  # 使用不同端口避免衝突 | ||||||
|  |             session_id = manager.create_session(temp_dir, "整合測試會話") | ||||||
|  | 
 | ||||||
|  |             if session_id: | ||||||
|  |                 print("  ✅ Web UI 會話創建成功") | ||||||
|  |             else: | ||||||
|  |                 print("  ❌ Web UI 會話創建失敗") | ||||||
|  |                 return False | ||||||
|  | 
 | ||||||
|  |         # 4. 桌面模式檢測測試 | ||||||
|  |         print("\n📋 4. 測試桌面模式檢測...") | ||||||
|  |         os.environ['MCP_FEEDBACK_MODE'] = 'desktop' | ||||||
|  | 
 | ||||||
|  |         manager = WebUIManager() | ||||||
|  |         if manager.should_use_desktop_mode(): | ||||||
|  |             print("  ✅ 桌面模式檢測正常") | ||||||
|  |         else: | ||||||
|  |             print("  ❌ 桌面模式檢測失敗") | ||||||
|  |             return False | ||||||
|  | 
 | ||||||
|  |         print("\n🎉 完整整合測試通過!") | ||||||
|  |         print("💡 所有核心功能正常運作") | ||||||
|  |         return True | ||||||
|  | 
 | ||||||
|  |     except Exception as e: | ||||||
|  |         print(f"❌ 完整整合測試失敗: {e}") | ||||||
|  |         import traceback | ||||||
|  |         traceback.print_exc() | ||||||
|  |         return False | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| def show_version(): | def show_version(): | ||||||
|     """顯示版本資訊""" |     """顯示版本資訊""" | ||||||
|     from . import __version__, __author__ |     from . import __version__, __author__ | ||||||
|  | |||||||
							
								
								
									
										168
									
								
								src/mcp_feedback_enhanced/desktop/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										168
									
								
								src/mcp_feedback_enhanced/desktop/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,168 @@ | |||||||
|  | #!/usr/bin/env python3 | ||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  | """ | ||||||
|  | 桌面應用模組 | ||||||
|  | =========== | ||||||
|  | 
 | ||||||
|  | 此模組提供 Electron 桌面應用整合功能,實現零前端改動的桌面化方案。 | ||||||
|  | 
 | ||||||
|  | 主要功能: | ||||||
|  | - Electron 進程管理 | ||||||
|  | - 與現有 Web UI 的無縫整合 | ||||||
|  | - 跨平台桌面應用支援 | ||||||
|  | - 環境變數控制 | ||||||
|  | 
 | ||||||
|  | 作者: Augment Agent | ||||||
|  | 版本: 2.3.0 | ||||||
|  | """ | ||||||
|  | 
 | ||||||
|  | import os | ||||||
|  | import sys | ||||||
|  | from typing import Optional | ||||||
|  | 
 | ||||||
|  | from ..debug import web_debug_log as debug_log | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def is_desktop_available() -> bool: | ||||||
|  |     """ | ||||||
|  |     檢測桌面環境是否可用 | ||||||
|  |      | ||||||
|  |     Returns: | ||||||
|  |         bool: True 表示桌面環境可用 | ||||||
|  |     """ | ||||||
|  |     try: | ||||||
|  |         # 檢查是否有 Node.js 環境 | ||||||
|  |         import subprocess | ||||||
|  |         result = subprocess.run(['node', '--version'],  | ||||||
|  |                               capture_output=True,  | ||||||
|  |                               text=True,  | ||||||
|  |                               timeout=5) | ||||||
|  |         if result.returncode != 0: | ||||||
|  |             debug_log("Node.js 不可用,桌面模式不可用") | ||||||
|  |             return False | ||||||
|  |              | ||||||
|  |         # 檢查是否為遠程環境 | ||||||
|  |         from ..server import is_remote_environment | ||||||
|  |         if is_remote_environment(): | ||||||
|  |             debug_log("檢測到遠程環境,桌面模式不適用") | ||||||
|  |             return False | ||||||
|  |              | ||||||
|  |         debug_log("桌面環境檢測通過") | ||||||
|  |         return True | ||||||
|  |          | ||||||
|  |     except (subprocess.TimeoutExpired, FileNotFoundError, ImportError) as e: | ||||||
|  |         debug_log(f"桌面環境檢測失敗: {e}") | ||||||
|  |         return False | ||||||
|  |     except Exception as e: | ||||||
|  |         debug_log(f"桌面環境檢測出現未預期錯誤: {e}") | ||||||
|  |         return False | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | async def launch_desktop_app(project_dir: str, summary: str, timeout: int) -> dict: | ||||||
|  |     """ | ||||||
|  |     啟動桌面應用收集回饋 | ||||||
|  | 
 | ||||||
|  |     Args: | ||||||
|  |         project_dir: 專案目錄路徑 | ||||||
|  |         summary: AI 工作摘要 | ||||||
|  |         timeout: 超時時間(秒) | ||||||
|  | 
 | ||||||
|  |     Returns: | ||||||
|  |         dict: 收集到的回饋資料 | ||||||
|  |     """ | ||||||
|  |     debug_log("啟動桌面應用...") | ||||||
|  | 
 | ||||||
|  |     try: | ||||||
|  |         # 創建 Electron 管理器 | ||||||
|  |         from .electron_manager import ElectronManager | ||||||
|  |         manager = ElectronManager() | ||||||
|  | 
 | ||||||
|  |         # 首先啟動 Web 服務器(桌面應用需要載入 Web UI) | ||||||
|  |         from ..web import get_web_ui_manager | ||||||
|  |         web_manager = get_web_ui_manager() | ||||||
|  | 
 | ||||||
|  |         # 創建會話 | ||||||
|  |         session_id = web_manager.create_session(project_dir, summary) | ||||||
|  |         session = web_manager.get_current_session() | ||||||
|  | 
 | ||||||
|  |         if not session: | ||||||
|  |             raise RuntimeError("無法創建回饋會話") | ||||||
|  | 
 | ||||||
|  |         # 啟動 Web 服務器(如果尚未啟動) | ||||||
|  |         if not web_manager.server_thread or not web_manager.server_thread.is_alive(): | ||||||
|  |             debug_log("啟動 Web 服務器...") | ||||||
|  |             web_manager.start_server() | ||||||
|  | 
 | ||||||
|  |         # 等待 Web 服務器完全啟動 | ||||||
|  |         import time | ||||||
|  |         debug_log("等待 Web 服務器啟動...") | ||||||
|  |         time.sleep(5)  # 增加等待時間 | ||||||
|  | 
 | ||||||
|  |         # 驗證 Web 服務器是否正常運行 | ||||||
|  |         if web_manager.server_thread and web_manager.server_thread.is_alive(): | ||||||
|  |             debug_log(f"✅ Web 服務器成功啟動在端口: {web_manager.port}") | ||||||
|  |         else: | ||||||
|  |             raise RuntimeError(f"Web 服務器啟動失敗") | ||||||
|  | 
 | ||||||
|  |         # 設置 Web 服務器端口 | ||||||
|  |         manager.set_web_server_port(web_manager.port) | ||||||
|  |         debug_log(f"桌面應用將連接到: http://localhost:{web_manager.port}") | ||||||
|  | 
 | ||||||
|  |         # 啟動桌面應用 | ||||||
|  |         desktop_success = await manager.launch_desktop_app(summary, project_dir) | ||||||
|  | 
 | ||||||
|  |         if desktop_success: | ||||||
|  |             debug_log("桌面應用啟動成功,等待用戶回饋...") | ||||||
|  |             # 等待用戶回饋 | ||||||
|  |             result = await session.wait_for_feedback(timeout) | ||||||
|  |             debug_log("收到桌面應用用戶回饋") | ||||||
|  |             return result | ||||||
|  |         else: | ||||||
|  |             debug_log("桌面應用啟動失敗,回退到 Web 模式") | ||||||
|  |             # 回退到 Web 模式 | ||||||
|  |             from ..web import launch_web_feedback_ui | ||||||
|  |             return await launch_web_feedback_ui(project_dir, summary, timeout) | ||||||
|  | 
 | ||||||
|  |     except Exception as e: | ||||||
|  |         debug_log(f"桌面應用啟動過程中出錯: {e}") | ||||||
|  |         debug_log("回退到 Web 模式") | ||||||
|  |         # 回退到 Web 模式 | ||||||
|  |         from ..web import launch_web_feedback_ui | ||||||
|  |         return await launch_web_feedback_ui(project_dir, summary, timeout) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class ElectronManager: | ||||||
|  |     """Electron 管理器 - 預留接口""" | ||||||
|  |      | ||||||
|  |     def __init__(self): | ||||||
|  |         """初始化 Electron 管理器""" | ||||||
|  |         self.electron_process = None | ||||||
|  |         self.web_server_port = None | ||||||
|  |         debug_log("ElectronManager 初始化(預留實現)") | ||||||
|  |      | ||||||
|  |     async def launch_desktop_app(self, summary: str, project_dir: str) -> bool: | ||||||
|  |         """ | ||||||
|  |         啟動桌面應用 | ||||||
|  |          | ||||||
|  |         Args: | ||||||
|  |             summary: AI 工作摘要 | ||||||
|  |             project_dir: 專案目錄 | ||||||
|  |              | ||||||
|  |         Returns: | ||||||
|  |             bool: 啟動是否成功 | ||||||
|  |         """ | ||||||
|  |         debug_log("桌面應用啟動功能尚未實現") | ||||||
|  |         debug_log("此功能將在階段 2 中實現") | ||||||
|  |         return False | ||||||
|  |      | ||||||
|  |     def is_available(self) -> bool: | ||||||
|  |         """檢查桌面管理器是否可用""" | ||||||
|  |         return is_desktop_available() | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | # 主要導出介面 | ||||||
|  | __all__ = [ | ||||||
|  |     "is_desktop_available", | ||||||
|  |     "launch_desktop_app",  | ||||||
|  |     "ElectronManager" | ||||||
|  | ] | ||||||
							
								
								
									
										39
									
								
								src/mcp_feedback_enhanced/desktop/assets/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								src/mcp_feedback_enhanced/desktop/assets/README.md
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,39 @@ | |||||||
|  | # 應用圖標資源 | ||||||
|  | 
 | ||||||
|  | 此目錄包含桌面應用的圖標資源文件。 | ||||||
|  | 
 | ||||||
|  | ## 需要的圖標文件 | ||||||
|  | 
 | ||||||
|  | ### Windows | ||||||
|  | - `icon.ico` - Windows 圖標文件(多尺寸:16x16, 32x32, 48x48, 256x256) | ||||||
|  | 
 | ||||||
|  | ### macOS   | ||||||
|  | - `icon.icns` - macOS 圖標文件(多尺寸:16x16 到 1024x1024) | ||||||
|  | - `entitlements.mac.plist` - macOS 權限配置文件 | ||||||
|  | 
 | ||||||
|  | ### Linux | ||||||
|  | - `icon.png` - Linux 圖標文件(建議 512x512) | ||||||
|  | 
 | ||||||
|  | ### 打包資源 | ||||||
|  | - `dmg-background.png` - macOS DMG 背景圖片(540x380) | ||||||
|  | 
 | ||||||
|  | ## 圖標設計建議 | ||||||
|  | 
 | ||||||
|  | - 使用 MCP Feedback Enhanced 的品牌色彩 | ||||||
|  | - 簡潔明了的設計,在小尺寸下仍然清晰 | ||||||
|  | - 符合各平台的設計規範 | ||||||
|  | - 建議使用矢量圖形作為源文件 | ||||||
|  | 
 | ||||||
|  | ## 臨時解決方案 | ||||||
|  | 
 | ||||||
|  | 在開發階段,可以使用以下方式創建基本圖標: | ||||||
|  | 
 | ||||||
|  | 1. 使用線上圖標生成器 | ||||||
|  | 2. 從現有的 Web UI favicon 轉換 | ||||||
|  | 3. 使用系統預設圖標 | ||||||
|  | 
 | ||||||
|  | ## 注意事項 | ||||||
|  | 
 | ||||||
|  | - 圖標文件應該放在此目錄中 | ||||||
|  | - electron-builder 會自動處理圖標的打包 | ||||||
|  | - 確保圖標文件的版權合規性 | ||||||
| @ -0,0 +1,28 @@ | |||||||
|  | # 圖標佔位符 | ||||||
|  | 
 | ||||||
|  | 此文件標記圖標資源的位置。在實際部署中,需要替換為真實的圖標文件: | ||||||
|  | 
 | ||||||
|  | ## 需要的圖標文件: | ||||||
|  | 
 | ||||||
|  | ### Windows | ||||||
|  | - icon.ico (多尺寸:16x16, 32x32, 48x48, 256x256) | ||||||
|  | 
 | ||||||
|  | ### macOS | ||||||
|  | - icon.icns (多尺寸:16x16 到 1024x1024) | ||||||
|  | 
 | ||||||
|  | ### Linux | ||||||
|  | - icon.png (建議 512x512) | ||||||
|  | 
 | ||||||
|  | ## 臨時解決方案 | ||||||
|  | 
 | ||||||
|  | 在開發階段,可以: | ||||||
|  | 1. 使用 Electron 預設圖標 | ||||||
|  | 2. 從現有 Web UI 的 favicon 轉換 | ||||||
|  | 3. 使用線上圖標生成器創建基本圖標 | ||||||
|  | 
 | ||||||
|  | ## 圖標設計建議 | ||||||
|  | 
 | ||||||
|  | - 簡潔明了,體現 MCP 和回饋收集的概念 | ||||||
|  | - 在小尺寸下仍然清晰可辨 | ||||||
|  | - 符合各平台的設計規範 | ||||||
|  | - 使用一致的色彩方案 | ||||||
							
								
								
									
										309
									
								
								src/mcp_feedback_enhanced/desktop/electron_manager.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										309
									
								
								src/mcp_feedback_enhanced/desktop/electron_manager.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,309 @@ | |||||||
|  | #!/usr/bin/env python3 | ||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  | """ | ||||||
|  | Electron 管理器 | ||||||
|  | ============== | ||||||
|  | 
 | ||||||
|  | 此模組負責管理 Electron 進程的生命週期,包括: | ||||||
|  | - Electron 應用啟動和停止 | ||||||
|  | - 與 Web 服務器的整合 | ||||||
|  | - 依賴檢測和自動安裝 | ||||||
|  | - 進程管理和錯誤處理 | ||||||
|  | 
 | ||||||
|  | 此文件為階段 1 的預留實現,完整功能將在階段 2 中實現。 | ||||||
|  | 
 | ||||||
|  | 作者: Augment Agent | ||||||
|  | 版本: 2.3.0 | ||||||
|  | """ | ||||||
|  | 
 | ||||||
|  | import subprocess | ||||||
|  | import asyncio | ||||||
|  | import os | ||||||
|  | from pathlib import Path | ||||||
|  | from typing import Optional | ||||||
|  | 
 | ||||||
|  | from ..debug import web_debug_log as debug_log | ||||||
|  | from ..utils.error_handler import ErrorHandler, ErrorType | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class ElectronManager: | ||||||
|  |     """Electron 進程管理器""" | ||||||
|  |      | ||||||
|  |     def __init__(self): | ||||||
|  |         """初始化 Electron 管理器""" | ||||||
|  |         self.electron_process: Optional[subprocess.Popen] = None | ||||||
|  |         self.desktop_dir = Path(__file__).parent | ||||||
|  |         self.web_server_port: Optional[int] = None | ||||||
|  |          | ||||||
|  |         debug_log("ElectronManager 初始化完成") | ||||||
|  |         debug_log(f"桌面模組目錄: {self.desktop_dir}") | ||||||
|  |      | ||||||
|  |     async def launch_desktop_app(self, summary: str, project_dir: str) -> bool: | ||||||
|  |         """ | ||||||
|  |         啟動 Electron 桌面應用 | ||||||
|  | 
 | ||||||
|  |         Args: | ||||||
|  |             summary: AI 工作摘要 | ||||||
|  |             project_dir: 專案目錄 | ||||||
|  | 
 | ||||||
|  |         Returns: | ||||||
|  |             bool: 啟動是否成功 | ||||||
|  |         """ | ||||||
|  |         debug_log("=== 桌面應用啟動 ===") | ||||||
|  |         debug_log(f"摘要: {summary}") | ||||||
|  |         debug_log(f"專案目錄: {project_dir}") | ||||||
|  | 
 | ||||||
|  |         try: | ||||||
|  |             # 確保依賴已安裝 | ||||||
|  |             if not await self.ensure_dependencies(): | ||||||
|  |                 debug_log("依賴檢查失敗,無法啟動桌面應用") | ||||||
|  |                 return False | ||||||
|  | 
 | ||||||
|  |             # 啟動 Electron 應用 | ||||||
|  |             success = await self._start_electron_process() | ||||||
|  |             if success: | ||||||
|  |                 debug_log("Electron 桌面應用啟動成功") | ||||||
|  |                 return True | ||||||
|  |             else: | ||||||
|  |                 debug_log("Electron 桌面應用啟動失敗") | ||||||
|  |                 return False | ||||||
|  | 
 | ||||||
|  |         except Exception as e: | ||||||
|  |             error_id = ErrorHandler.log_error_with_context( | ||||||
|  |                 e, | ||||||
|  |                 context={"operation": "桌面應用啟動", "project_dir": project_dir}, | ||||||
|  |                 error_type=ErrorType.SYSTEM | ||||||
|  |             ) | ||||||
|  |             debug_log(f"桌面應用啟動異常 [錯誤ID: {error_id}]: {e}") | ||||||
|  |             return False | ||||||
|  |      | ||||||
|  |     def set_web_server_port(self, port: int): | ||||||
|  |         """設置 Web 服務器端口""" | ||||||
|  |         self.web_server_port = port | ||||||
|  |         debug_log(f"設置 Web 服務器端口: {port}") | ||||||
|  |      | ||||||
|  |     def is_electron_available(self) -> bool: | ||||||
|  |         """檢查 Electron 是否可用""" | ||||||
|  |         try: | ||||||
|  |             # 檢查 Node.js | ||||||
|  |             result = subprocess.run(['node', '--version'],  | ||||||
|  |                                   capture_output=True,  | ||||||
|  |                                   text=True,  | ||||||
|  |                                   timeout=5) | ||||||
|  |             if result.returncode != 0: | ||||||
|  |                 debug_log("Node.js 不可用") | ||||||
|  |                 return False | ||||||
|  |              | ||||||
|  |             debug_log(f"Node.js 版本: {result.stdout.strip()}") | ||||||
|  |              | ||||||
|  |             # 檢查 package.json 是否存在 | ||||||
|  |             package_json = self.desktop_dir / "package.json" | ||||||
|  |             if not package_json.exists(): | ||||||
|  |                 debug_log("package.json 不存在,需要在階段 2 中創建") | ||||||
|  |                 return False | ||||||
|  |              | ||||||
|  |             return True | ||||||
|  |              | ||||||
|  |         except Exception as e: | ||||||
|  |             debug_log(f"Electron 可用性檢查失敗: {e}") | ||||||
|  |             return False | ||||||
|  |      | ||||||
|  |     async def ensure_dependencies(self) -> bool: | ||||||
|  |         """確保依賴已安裝""" | ||||||
|  |         debug_log("檢查 Electron 依賴...") | ||||||
|  | 
 | ||||||
|  |         try: | ||||||
|  |             # 檢查 package.json 是否存在 | ||||||
|  |             package_json = self.desktop_dir / "package.json" | ||||||
|  |             if not package_json.exists(): | ||||||
|  |                 debug_log("package.json 不存在,創建中...") | ||||||
|  |                 await self._create_package_json() | ||||||
|  | 
 | ||||||
|  |             # 檢查 node_modules 是否存在 | ||||||
|  |             node_modules = self.desktop_dir / "node_modules" | ||||||
|  |             if not node_modules.exists(): | ||||||
|  |                 debug_log("node_modules 不存在,安裝依賴中...") | ||||||
|  |                 if not await self._install_dependencies(): | ||||||
|  |                     return False | ||||||
|  | 
 | ||||||
|  |             # 檢查 main.js 是否存在 | ||||||
|  |             main_js = self.desktop_dir / "main.js" | ||||||
|  |             if not main_js.exists(): | ||||||
|  |                 debug_log("main.js 不存在,將在後續步驟中創建") | ||||||
|  |                 return False | ||||||
|  | 
 | ||||||
|  |             debug_log("所有依賴檢查通過") | ||||||
|  |             return True | ||||||
|  | 
 | ||||||
|  |         except Exception as e: | ||||||
|  |             error_id = ErrorHandler.log_error_with_context( | ||||||
|  |                 e, | ||||||
|  |                 context={"operation": "依賴檢查"}, | ||||||
|  |                 error_type=ErrorType.DEPENDENCY | ||||||
|  |             ) | ||||||
|  |             debug_log(f"依賴檢查失敗 [錯誤ID: {error_id}]: {e}") | ||||||
|  |             return False | ||||||
|  |      | ||||||
|  |     def cleanup(self): | ||||||
|  |         """清理資源""" | ||||||
|  |         if self.electron_process: | ||||||
|  |             try: | ||||||
|  |                 # 檢查進程是否還在運行 | ||||||
|  |                 if self.electron_process.returncode is None: | ||||||
|  |                     self.electron_process.terminate() | ||||||
|  |                     debug_log("Electron 進程已終止") | ||||||
|  |                 else: | ||||||
|  |                     debug_log("Electron 進程已自然結束") | ||||||
|  |             except Exception as e: | ||||||
|  |                 debug_log(f"終止 Electron 進程時出錯: {e}") | ||||||
|  |                 try: | ||||||
|  |                     if self.electron_process.returncode is None: | ||||||
|  |                         self.electron_process.kill() | ||||||
|  |                         debug_log("強制終止 Electron 進程") | ||||||
|  |                 except Exception as kill_error: | ||||||
|  |                     debug_log(f"強制終止 Electron 進程失敗: {kill_error}") | ||||||
|  |             finally: | ||||||
|  |                 # 關閉管道以避免 ResourceWarning | ||||||
|  |                 try: | ||||||
|  |                     # 對於 asyncio 子進程,需要特殊處理 | ||||||
|  |                     if hasattr(self.electron_process, 'stdout') and self.electron_process.stdout: | ||||||
|  |                         if hasattr(self.electron_process.stdout, 'close'): | ||||||
|  |                             self.electron_process.stdout.close() | ||||||
|  |                     if hasattr(self.electron_process, 'stderr') and self.electron_process.stderr: | ||||||
|  |                         if hasattr(self.electron_process.stderr, 'close'): | ||||||
|  |                             self.electron_process.stderr.close() | ||||||
|  |                     if hasattr(self.electron_process, 'stdin') and self.electron_process.stdin: | ||||||
|  |                         if hasattr(self.electron_process.stdin, 'close'): | ||||||
|  |                             self.electron_process.stdin.close() | ||||||
|  |                 except Exception: | ||||||
|  |                     # 忽略管道關閉錯誤,這些通常是無害的 | ||||||
|  |                     pass | ||||||
|  | 
 | ||||||
|  |                 self.electron_process = None | ||||||
|  |      | ||||||
|  |     async def _create_package_json(self): | ||||||
|  |         """創建 package.json 文件""" | ||||||
|  |         package_config = { | ||||||
|  |             "name": "mcp-feedback-enhanced-desktop", | ||||||
|  |             "version": "2.3.0", | ||||||
|  |             "description": "MCP Feedback Enhanced Desktop Application", | ||||||
|  |             "main": "main.js", | ||||||
|  |             "scripts": { | ||||||
|  |                 "start": "electron .", | ||||||
|  |                 "dev": "electron . --dev" | ||||||
|  |             }, | ||||||
|  |             "dependencies": { | ||||||
|  |                 "electron": "^28.0.0" | ||||||
|  |             }, | ||||||
|  |             "devDependencies": { | ||||||
|  |                 "electron-builder": "^24.0.0" | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         package_json_path = self.desktop_dir / "package.json" | ||||||
|  |         with open(package_json_path, 'w', encoding='utf-8') as f: | ||||||
|  |             import json | ||||||
|  |             json.dump(package_config, f, indent=2, ensure_ascii=False) | ||||||
|  | 
 | ||||||
|  |         debug_log(f"已創建 package.json: {package_json_path}") | ||||||
|  | 
 | ||||||
|  |     async def _install_dependencies(self) -> bool: | ||||||
|  |         """安裝 Node.js 依賴""" | ||||||
|  |         debug_log("開始安裝 Node.js 依賴...") | ||||||
|  | 
 | ||||||
|  |         try: | ||||||
|  |             # 使用 npm install | ||||||
|  |             install_cmd = ['npm', 'install'] | ||||||
|  |             process = await asyncio.create_subprocess_exec( | ||||||
|  |                 *install_cmd, | ||||||
|  |                 cwd=self.desktop_dir, | ||||||
|  |                 stdout=asyncio.subprocess.PIPE, | ||||||
|  |                 stderr=asyncio.subprocess.PIPE | ||||||
|  |             ) | ||||||
|  | 
 | ||||||
|  |             _, stderr = await process.communicate() | ||||||
|  | 
 | ||||||
|  |             if process.returncode == 0: | ||||||
|  |                 debug_log("Node.js 依賴安裝成功") | ||||||
|  |                 return True | ||||||
|  |             else: | ||||||
|  |                 debug_log(f"依賴安裝失敗: {stderr.decode()}") | ||||||
|  |                 return False | ||||||
|  | 
 | ||||||
|  |         except Exception as e: | ||||||
|  |             debug_log(f"依賴安裝過程中出錯: {e}") | ||||||
|  |             return False | ||||||
|  | 
 | ||||||
|  |     async def _start_electron_process(self) -> bool: | ||||||
|  |         """啟動 Electron 進程""" | ||||||
|  |         debug_log("啟動 Electron 進程...") | ||||||
|  | 
 | ||||||
|  |         try: | ||||||
|  |             # 構建 Electron 命令 - 使用本地安裝的 electron | ||||||
|  |             import platform | ||||||
|  |             if platform.system() == "Windows": | ||||||
|  |                 electron_path = self.desktop_dir / "node_modules" / ".bin" / "electron.cmd" | ||||||
|  |             else: | ||||||
|  |                 electron_path = self.desktop_dir / "node_modules" / ".bin" / "electron" | ||||||
|  | 
 | ||||||
|  |             if electron_path.exists(): | ||||||
|  |                 electron_cmd = [ | ||||||
|  |                     str(electron_path), '.', | ||||||
|  |                     '--port', str(self.web_server_port or 8765) | ||||||
|  |                 ] | ||||||
|  |             else: | ||||||
|  |                 # 回退到 npx | ||||||
|  |                 electron_cmd = [ | ||||||
|  |                     'npx', 'electron', '.', | ||||||
|  |                     '--port', str(self.web_server_port or 8765) | ||||||
|  |                 ] | ||||||
|  | 
 | ||||||
|  |             debug_log(f"使用 Electron 命令: {' '.join(electron_cmd)}") | ||||||
|  | 
 | ||||||
|  |             # 啟動 Electron 進程 | ||||||
|  |             self.electron_process = await asyncio.create_subprocess_exec( | ||||||
|  |                 *electron_cmd, | ||||||
|  |                 cwd=self.desktop_dir, | ||||||
|  |                 stdout=asyncio.subprocess.PIPE, | ||||||
|  |                 stderr=asyncio.subprocess.PIPE | ||||||
|  |             ) | ||||||
|  | 
 | ||||||
|  |             debug_log(f"Electron 進程已啟動,PID: {self.electron_process.pid}") | ||||||
|  | 
 | ||||||
|  |             # 等待一小段時間確保進程正常啟動 | ||||||
|  |             await asyncio.sleep(2) | ||||||
|  | 
 | ||||||
|  |             # 檢查進程是否仍在運行 | ||||||
|  |             if self.electron_process.returncode is None: | ||||||
|  |                 debug_log("Electron 進程運行正常") | ||||||
|  |                 return True | ||||||
|  |             else: | ||||||
|  |                 debug_log(f"Electron 進程異常退出,返回碼: {self.electron_process.returncode}") | ||||||
|  |                 # 讀取錯誤輸出 | ||||||
|  |                 try: | ||||||
|  |                     _, stderr = await self.electron_process.communicate() | ||||||
|  |                     if stderr: | ||||||
|  |                         debug_log(f"Electron 錯誤輸出: {stderr.decode()}") | ||||||
|  |                 except Exception as e: | ||||||
|  |                     debug_log(f"讀取 Electron 錯誤輸出失敗: {e}") | ||||||
|  |                 return False | ||||||
|  | 
 | ||||||
|  |         except Exception as e: | ||||||
|  |             debug_log(f"啟動 Electron 進程失敗: {e}") | ||||||
|  |             return False | ||||||
|  | 
 | ||||||
|  |     def __del__(self): | ||||||
|  |         """析構函數""" | ||||||
|  |         self.cleanup() | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | # 便利函數 | ||||||
|  | async def create_electron_manager() -> ElectronManager: | ||||||
|  |     """創建 Electron 管理器實例""" | ||||||
|  |     manager = ElectronManager() | ||||||
|  |      | ||||||
|  |     # 檢查可用性 | ||||||
|  |     if not manager.is_electron_available(): | ||||||
|  |         debug_log("Electron 環境不可用,建議使用 Web 模式") | ||||||
|  |      | ||||||
|  |     return manager | ||||||
							
								
								
									
										306
									
								
								src/mcp_feedback_enhanced/desktop/main.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										306
									
								
								src/mcp_feedback_enhanced/desktop/main.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,306 @@ | |||||||
|  | #!/usr/bin/env node
 | ||||||
|  | /** | ||||||
|  |  * Electron 主進程 | ||||||
|  |  * =============== | ||||||
|  |  *  | ||||||
|  |  * 此文件是 MCP Feedback Enhanced 桌面應用的主進程入口點。 | ||||||
|  |  * 負責創建和管理 BrowserWindow,以及與現有 Web UI 的整合。 | ||||||
|  |  *  | ||||||
|  |  * 主要功能: | ||||||
|  |  * - 創建和管理應用視窗 | ||||||
|  |  * - 載入本地 Web 服務器內容 | ||||||
|  |  * - 處理應用生命週期事件 | ||||||
|  |  * - 提供桌面應用特有的功能 | ||||||
|  |  *  | ||||||
|  |  * 作者: Augment Agent | ||||||
|  |  * 版本: 2.3.0 | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | const { app, BrowserWindow, ipcMain, shell } = require('electron'); | ||||||
|  | const path = require('path'); | ||||||
|  | const os = require('os'); | ||||||
|  | 
 | ||||||
|  | // 應用配置
 | ||||||
|  | const APP_CONFIG = { | ||||||
|  |     name: 'MCP Feedback Enhanced', | ||||||
|  |     width: 1200, | ||||||
|  |     height: 800, | ||||||
|  |     minWidth: 800, | ||||||
|  |     minHeight: 600, | ||||||
|  |     defaultPort: 8765 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | // 全局變數
 | ||||||
|  | let mainWindow = null; | ||||||
|  | let webServerPort = APP_CONFIG.defaultPort; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Electron 應用類 | ||||||
|  |  */ | ||||||
|  | class ElectronApp { | ||||||
|  |     constructor() { | ||||||
|  |         this.mainWindow = null; | ||||||
|  |         this.webServerPort = APP_CONFIG.defaultPort; | ||||||
|  |         this.isDevMode = process.argv.includes('--dev'); | ||||||
|  |          | ||||||
|  |         this.setupEventHandlers(); | ||||||
|  |         this.parseCommandLineArgs(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 解析命令行參數 | ||||||
|  |      */ | ||||||
|  |     parseCommandLineArgs() { | ||||||
|  |         const args = process.argv; | ||||||
|  |         const portIndex = args.indexOf('--port'); | ||||||
|  |          | ||||||
|  |         if (portIndex !== -1 && portIndex + 1 < args.length) { | ||||||
|  |             const port = parseInt(args[portIndex + 1]); | ||||||
|  |             if (!isNaN(port) && port > 0 && port < 65536) { | ||||||
|  |                 this.webServerPort = port; | ||||||
|  |                 console.log(`使用指定端口: ${port}`); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 設置事件處理器 | ||||||
|  |      */ | ||||||
|  |     setupEventHandlers() { | ||||||
|  |         // 應用準備就緒
 | ||||||
|  |         app.whenReady().then(() => { | ||||||
|  |             this.createWindow(); | ||||||
|  |             this.setupIpcHandlers(); | ||||||
|  |         }); | ||||||
|  | 
 | ||||||
|  |         // 所有視窗關閉
 | ||||||
|  |         app.on('window-all-closed', () => { | ||||||
|  |             if (process.platform !== 'darwin') { | ||||||
|  |                 app.quit(); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  | 
 | ||||||
|  |         // 應用激活(macOS)
 | ||||||
|  |         app.on('activate', () => { | ||||||
|  |             if (BrowserWindow.getAllWindows().length === 0) { | ||||||
|  |                 this.createWindow(); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  | 
 | ||||||
|  |         // 處理證書錯誤(開發模式)
 | ||||||
|  |         app.on('certificate-error', (event, webContents, url, error, certificate, callback) => { | ||||||
|  |             if (this.isDevMode && url.startsWith('https://localhost')) { | ||||||
|  |                 event.preventDefault(); | ||||||
|  |                 callback(true); | ||||||
|  |             } else { | ||||||
|  |                 callback(false); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 創建主視窗 | ||||||
|  |      */ | ||||||
|  |     async createWindow() { | ||||||
|  |         console.log('創建主視窗...'); | ||||||
|  |         console.log(`視窗配置: ${APP_CONFIG.width}x${APP_CONFIG.height}`); | ||||||
|  | 
 | ||||||
|  |         this.mainWindow = new BrowserWindow({ | ||||||
|  |             width: APP_CONFIG.width, | ||||||
|  |             height: APP_CONFIG.height, | ||||||
|  |             minWidth: APP_CONFIG.minWidth, | ||||||
|  |             minHeight: APP_CONFIG.minHeight, | ||||||
|  |             webPreferences: { | ||||||
|  |                 nodeIntegration: false, | ||||||
|  |                 contextIsolation: true, | ||||||
|  |                 enableRemoteModule: false, | ||||||
|  |                 preload: path.join(__dirname, 'preload.js'), | ||||||
|  |                 webSecurity: !this.isDevMode | ||||||
|  |             }, | ||||||
|  |             icon: this.getAppIcon(), | ||||||
|  |             title: APP_CONFIG.name, | ||||||
|  |             show: false, // 先隱藏,等載入完成後顯示
 | ||||||
|  |             titleBarStyle: process.platform === 'darwin' ? 'hiddenInset' : 'default', | ||||||
|  |             alwaysOnTop: false, // 不總是置頂,但確保可見
 | ||||||
|  |             center: true, // 居中顯示
 | ||||||
|  |             resizable: true | ||||||
|  |         }); | ||||||
|  | 
 | ||||||
|  |         // 載入 Web UI
 | ||||||
|  |         await this.loadWebUI(); | ||||||
|  | 
 | ||||||
|  |         // 視窗準備顯示
 | ||||||
|  |         this.mainWindow.once('ready-to-show', () => { | ||||||
|  |             console.log('視窗準備顯示,正在顯示視窗...'); | ||||||
|  |             this.mainWindow.show(); | ||||||
|  |             this.mainWindow.focus(); // 確保視窗獲得焦點
 | ||||||
|  | 
 | ||||||
|  |             if (this.isDevMode) { | ||||||
|  |                 this.mainWindow.webContents.openDevTools(); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  | 
 | ||||||
|  |         // 備用顯示機制:如果 ready-to-show 沒有觸發,強制顯示
 | ||||||
|  |         setTimeout(() => { | ||||||
|  |             if (this.mainWindow && !this.mainWindow.isVisible()) { | ||||||
|  |                 console.log('備用顯示機制:強制顯示視窗'); | ||||||
|  |                 this.mainWindow.show(); | ||||||
|  |                 this.mainWindow.focus(); | ||||||
|  |             } | ||||||
|  |         }, 3000); // 3秒後強制顯示
 | ||||||
|  | 
 | ||||||
|  |         // 處理視窗關閉
 | ||||||
|  |         this.mainWindow.on('closed', () => { | ||||||
|  |             this.mainWindow = null; | ||||||
|  |         }); | ||||||
|  | 
 | ||||||
|  |         // 處理外部連結
 | ||||||
|  |         this.mainWindow.webContents.setWindowOpenHandler(({ url }) => { | ||||||
|  |             shell.openExternal(url); | ||||||
|  |             return { action: 'deny' }; | ||||||
|  |         }); | ||||||
|  | 
 | ||||||
|  |         console.log('主視窗創建完成'); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 載入 Web UI | ||||||
|  |      */ | ||||||
|  |     async loadWebUI() { | ||||||
|  |         const webUrl = `http://localhost:${this.webServerPort}`; | ||||||
|  |         console.log(`載入 Web UI: ${webUrl}`); | ||||||
|  | 
 | ||||||
|  |         try { | ||||||
|  |             await this.mainWindow.loadURL(webUrl); | ||||||
|  |             console.log('Web UI 載入成功'); | ||||||
|  |         } catch (error) { | ||||||
|  |             console.error('載入 Web UI 失敗:', error); | ||||||
|  | 
 | ||||||
|  |             // 載入錯誤頁面
 | ||||||
|  |             const errorHtml = this.createErrorPage(error); | ||||||
|  |             await this.mainWindow.loadURL(`data:text/html;charset=utf-8,${encodeURIComponent(errorHtml)}`); | ||||||
|  | 
 | ||||||
|  |             // 確保錯誤頁面也能顯示視窗
 | ||||||
|  |             console.log('載入錯誤頁面,強制顯示視窗'); | ||||||
|  |             this.mainWindow.show(); | ||||||
|  |             this.mainWindow.focus(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 創建錯誤頁面 | ||||||
|  |      */ | ||||||
|  |     createErrorPage(error) { | ||||||
|  |         return ` | ||||||
|  |         <!DOCTYPE html> | ||||||
|  |         <html> | ||||||
|  |         <head> | ||||||
|  |             <title>連接錯誤 - ${APP_CONFIG.name}</title> | ||||||
|  |             <style> | ||||||
|  |                 body {  | ||||||
|  |                     font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif; | ||||||
|  |                     background: #f5f5f5; | ||||||
|  |                     margin: 0; | ||||||
|  |                     padding: 40px; | ||||||
|  |                     display: flex; | ||||||
|  |                     align-items: center; | ||||||
|  |                     justify-content: center; | ||||||
|  |                     min-height: 100vh; | ||||||
|  |                 } | ||||||
|  |                 .error-container { | ||||||
|  |                     background: white; | ||||||
|  |                     padding: 40px; | ||||||
|  |                     border-radius: 8px; | ||||||
|  |                     box-shadow: 0 2px 10px rgba(0,0,0,0.1); | ||||||
|  |                     text-align: center; | ||||||
|  |                     max-width: 500px; | ||||||
|  |                 } | ||||||
|  |                 h1 { color: #e74c3c; margin-bottom: 20px; } | ||||||
|  |                 p { color: #666; line-height: 1.6; } | ||||||
|  |                 .retry-btn { | ||||||
|  |                     background: #3498db; | ||||||
|  |                     color: white; | ||||||
|  |                     border: none; | ||||||
|  |                     padding: 12px 24px; | ||||||
|  |                     border-radius: 4px; | ||||||
|  |                     cursor: pointer; | ||||||
|  |                     margin-top: 20px; | ||||||
|  |                 } | ||||||
|  |                 .retry-btn:hover { background: #2980b9; } | ||||||
|  |             </style> | ||||||
|  |         </head> | ||||||
|  |         <body> | ||||||
|  |             <div class="error-container"> | ||||||
|  |                 <h1>無法連接到 Web 服務器</h1> | ||||||
|  |                 <p>桌面應用無法連接到本地 Web 服務器(端口 ${this.webServerPort})。</p> | ||||||
|  |                 <p>請確保 MCP 服務器正在運行。</p> | ||||||
|  |                 <p><strong>錯誤詳情:</strong> ${error.message}</p> | ||||||
|  |                 <button class="retry-btn" onclick="location.reload()">重試</button> | ||||||
|  |             </div> | ||||||
|  |         </body> | ||||||
|  |         </html> | ||||||
|  |         `;
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 獲取應用圖標 | ||||||
|  |      */ | ||||||
|  |     getAppIcon() { | ||||||
|  |         const iconPath = path.join(__dirname, 'assets'); | ||||||
|  |          | ||||||
|  |         if (process.platform === 'win32') { | ||||||
|  |             return path.join(iconPath, 'icon.ico'); | ||||||
|  |         } else if (process.platform === 'darwin') { | ||||||
|  |             return path.join(iconPath, 'icon.icns'); | ||||||
|  |         } else { | ||||||
|  |             return path.join(iconPath, 'icon.png'); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 設置 IPC 處理器 | ||||||
|  |      */ | ||||||
|  |     setupIpcHandlers() { | ||||||
|  |         // 視窗控制
 | ||||||
|  |         ipcMain.handle('window-minimize', () => { | ||||||
|  |             if (this.mainWindow) { | ||||||
|  |                 this.mainWindow.minimize(); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  | 
 | ||||||
|  |         ipcMain.handle('window-maximize', () => { | ||||||
|  |             if (this.mainWindow) { | ||||||
|  |                 if (this.mainWindow.isMaximized()) { | ||||||
|  |                     this.mainWindow.unmaximize(); | ||||||
|  |                 } else { | ||||||
|  |                     this.mainWindow.maximize(); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  | 
 | ||||||
|  |         ipcMain.handle('window-close', () => { | ||||||
|  |             if (this.mainWindow) { | ||||||
|  |                 this.mainWindow.close(); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  | 
 | ||||||
|  |         // 獲取系統資訊
 | ||||||
|  |         ipcMain.handle('get-system-info', () => { | ||||||
|  |             return { | ||||||
|  |                 platform: process.platform, | ||||||
|  |                 arch: process.arch, | ||||||
|  |                 nodeVersion: process.version, | ||||||
|  |                 electronVersion: process.versions.electron, | ||||||
|  |                 chromeVersion: process.versions.chrome | ||||||
|  |             }; | ||||||
|  |         }); | ||||||
|  | 
 | ||||||
|  |         console.log('IPC 處理器設置完成'); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // 創建應用實例
 | ||||||
|  | const electronApp = new ElectronApp(); | ||||||
|  | 
 | ||||||
|  | // 導出供外部使用
 | ||||||
|  | module.exports = electronApp; | ||||||
							
								
								
									
										169
									
								
								src/mcp_feedback_enhanced/desktop/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										169
									
								
								src/mcp_feedback_enhanced/desktop/package.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,169 @@ | |||||||
|  | { | ||||||
|  |   "name": "mcp-feedback-enhanced-desktop", | ||||||
|  |   "version": "2.3.0", | ||||||
|  |   "description": "MCP Feedback Enhanced Desktop Application - Electron-based desktop interface for AI feedback collection", | ||||||
|  |   "main": "main.js", | ||||||
|  |   "author": { | ||||||
|  |     "name": "Augment Agent", | ||||||
|  |     "email": "minidora0702@gmail.com" | ||||||
|  |   }, | ||||||
|  |   "license": "MIT", | ||||||
|  |   "homepage": "https://github.com/minidoracat/mcp-feedback-enhanced", | ||||||
|  |   "repository": { | ||||||
|  |     "type": "git", | ||||||
|  |     "url": "https://github.com/minidoracat/mcp-feedback-enhanced.git" | ||||||
|  |   }, | ||||||
|  |   "keywords": [ | ||||||
|  |     "electron", | ||||||
|  |     "mcp", | ||||||
|  |     "feedback", | ||||||
|  |     "ai", | ||||||
|  |     "desktop", | ||||||
|  |     "cross-platform" | ||||||
|  |   ], | ||||||
|  |   "engines": { | ||||||
|  |     "node": ">=16.0.0", | ||||||
|  |     "npm": ">=8.0.0" | ||||||
|  |   }, | ||||||
|  |   "scripts": { | ||||||
|  |     "start": "electron .", | ||||||
|  |     "dev": "electron . --dev", | ||||||
|  |     "build": "electron-builder", | ||||||
|  |     "build:win": "electron-builder --windows", | ||||||
|  |     "build:mac": "electron-builder --mac", | ||||||
|  |     "build:linux": "electron-builder --linux", | ||||||
|  |     "dist": "npm run build", | ||||||
|  |     "pack": "electron-builder --dir", | ||||||
|  |     "postinstall": "electron-builder install-app-deps", | ||||||
|  |     "clean": "rimraf dist node_modules", | ||||||
|  |     "lint": "eslint *.js", | ||||||
|  |     "test": "echo \"桌面應用測試功能待實現\" && exit 0" | ||||||
|  |   }, | ||||||
|  |   "dependencies": { | ||||||
|  |     "electron": "^28.0.0" | ||||||
|  |   }, | ||||||
|  |   "devDependencies": { | ||||||
|  |     "electron-builder": "^24.0.0", | ||||||
|  |     "rimraf": "^5.0.0", | ||||||
|  |     "eslint": "^8.0.0" | ||||||
|  |   }, | ||||||
|  |   "build": { | ||||||
|  |     "appId": "com.minidoracat.mcp-feedback-enhanced", | ||||||
|  |     "productName": "MCP Feedback Enhanced", | ||||||
|  |     "directories": { | ||||||
|  |       "output": "dist", | ||||||
|  |       "buildResources": "assets" | ||||||
|  |     }, | ||||||
|  |     "files": [ | ||||||
|  |       "main.js", | ||||||
|  |       "preload.js", | ||||||
|  |       "assets/**/*", | ||||||
|  |       "node_modules/**/*" | ||||||
|  |     ], | ||||||
|  |     "extraResources": [ | ||||||
|  |       { | ||||||
|  |         "from": "../", | ||||||
|  |         "to": "app", | ||||||
|  |         "filter": [ | ||||||
|  |           "**/*", | ||||||
|  |           "!desktop/node_modules", | ||||||
|  |           "!desktop/dist", | ||||||
|  |           "!**/*.pyc", | ||||||
|  |           "!**/__pycache__" | ||||||
|  |         ] | ||||||
|  |       } | ||||||
|  |     ], | ||||||
|  |     "win": { | ||||||
|  |       "target": [ | ||||||
|  |         { | ||||||
|  |           "target": "nsis", | ||||||
|  |           "arch": ["x64", "ia32"] | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |           "target": "portable", | ||||||
|  |           "arch": ["x64"] | ||||||
|  |         } | ||||||
|  |       ], | ||||||
|  |       "icon": "assets/icon.ico", | ||||||
|  |       "requestedExecutionLevel": "asInvoker" | ||||||
|  |     }, | ||||||
|  |     "mac": { | ||||||
|  |       "target": [ | ||||||
|  |         { | ||||||
|  |           "target": "dmg", | ||||||
|  |           "arch": ["x64", "arm64"] | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |           "target": "zip", | ||||||
|  |           "arch": ["x64", "arm64"] | ||||||
|  |         } | ||||||
|  |       ], | ||||||
|  |       "icon": "assets/icon.icns", | ||||||
|  |       "category": "public.app-category.developer-tools", | ||||||
|  |       "hardenedRuntime": true, | ||||||
|  |       "gatekeeperAssess": false, | ||||||
|  |       "entitlements": "assets/entitlements.mac.plist", | ||||||
|  |       "entitlementsInherit": "assets/entitlements.mac.plist" | ||||||
|  |     }, | ||||||
|  |     "linux": { | ||||||
|  |       "target": [ | ||||||
|  |         { | ||||||
|  |           "target": "AppImage", | ||||||
|  |           "arch": ["x64"] | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |           "target": "deb", | ||||||
|  |           "arch": ["x64"] | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |           "target": "rpm", | ||||||
|  |           "arch": ["x64"] | ||||||
|  |         } | ||||||
|  |       ], | ||||||
|  |       "icon": "assets/icon.png", | ||||||
|  |       "category": "Development", | ||||||
|  |       "desktop": { | ||||||
|  |         "StartupNotify": "true", | ||||||
|  |         "Encoding": "UTF-8", | ||||||
|  |         "MimeType": "x-scheme-handler/mcp-feedback" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "nsis": { | ||||||
|  |       "oneClick": false, | ||||||
|  |       "allowToChangeInstallationDirectory": true, | ||||||
|  |       "createDesktopShortcut": true, | ||||||
|  |       "createStartMenuShortcut": true, | ||||||
|  |       "shortcutName": "MCP Feedback Enhanced" | ||||||
|  |     }, | ||||||
|  |     "dmg": { | ||||||
|  |       "title": "MCP Feedback Enhanced", | ||||||
|  |       "icon": "assets/icon.icns", | ||||||
|  |       "background": "assets/dmg-background.png", | ||||||
|  |       "contents": [ | ||||||
|  |         { | ||||||
|  |           "x": 130, | ||||||
|  |           "y": 220, | ||||||
|  |           "type": "file" | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |           "x": 410, | ||||||
|  |           "y": 220, | ||||||
|  |           "type": "link", | ||||||
|  |           "path": "/Applications" | ||||||
|  |         } | ||||||
|  |       ], | ||||||
|  |       "window": { | ||||||
|  |         "width": 540, | ||||||
|  |         "height": 380 | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "publish": { | ||||||
|  |       "provider": "github", | ||||||
|  |       "owner": "minidoracat", | ||||||
|  |       "repo": "mcp-feedback-enhanced" | ||||||
|  |     } | ||||||
|  |   }, | ||||||
|  |   "electronDownload": { | ||||||
|  |     "mirror": "https://npmmirror.com/mirrors/electron/" | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										204
									
								
								src/mcp_feedback_enhanced/desktop/preload.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										204
									
								
								src/mcp_feedback_enhanced/desktop/preload.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,204 @@ | |||||||
|  | /** | ||||||
|  |  * Electron 預載腳本 | ||||||
|  |  * ================== | ||||||
|  |  *  | ||||||
|  |  * 此腳本在渲染進程中運行,但在網頁內容載入之前執行。 | ||||||
|  |  * 它提供了安全的方式讓網頁與主進程通信,同時保持安全性。 | ||||||
|  |  *  | ||||||
|  |  * 主要功能: | ||||||
|  |  * - 提供安全的 IPC 通信接口 | ||||||
|  |  * - 擴展現有的 WebSocket 管理器 | ||||||
|  |  * - 添加桌面應用特有的 API | ||||||
|  |  * - 標記桌面環境 | ||||||
|  |  *  | ||||||
|  |  * 作者: Augment Agent | ||||||
|  |  * 版本: 2.3.0 | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | const { contextBridge, ipcRenderer } = require('electron'); | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * 桌面 API 接口 | ||||||
|  |  * 通過 contextBridge 安全地暴露給渲染進程 | ||||||
|  |  */ | ||||||
|  | const desktopAPI = { | ||||||
|  |     // 環境標識
 | ||||||
|  |     isDesktop: true, | ||||||
|  |     platform: process.platform, | ||||||
|  | 
 | ||||||
|  |     // 視窗控制
 | ||||||
|  |     window: { | ||||||
|  |         minimize: () => ipcRenderer.invoke('window-minimize'), | ||||||
|  |         maximize: () => ipcRenderer.invoke('window-maximize'), | ||||||
|  |         close: () => ipcRenderer.invoke('window-close') | ||||||
|  |     }, | ||||||
|  | 
 | ||||||
|  |     // 系統資訊
 | ||||||
|  |     system: { | ||||||
|  |         getInfo: () => ipcRenderer.invoke('get-system-info'), | ||||||
|  |         platform: process.platform, | ||||||
|  |         arch: process.arch | ||||||
|  |     }, | ||||||
|  | 
 | ||||||
|  |     // 事件監聽
 | ||||||
|  |     events: { | ||||||
|  |         onSessionUpdate: (callback) => { | ||||||
|  |             const wrappedCallback = (event, ...args) => callback(...args); | ||||||
|  |             ipcRenderer.on('session-updated', wrappedCallback); | ||||||
|  |              | ||||||
|  |             // 返回清理函數
 | ||||||
|  |             return () => { | ||||||
|  |                 ipcRenderer.removeListener('session-updated', wrappedCallback); | ||||||
|  |             }; | ||||||
|  |         }, | ||||||
|  | 
 | ||||||
|  |         onFeedbackRequest: (callback) => { | ||||||
|  |             const wrappedCallback = (event, ...args) => callback(...args); | ||||||
|  |             ipcRenderer.on('feedback-request', wrappedCallback); | ||||||
|  |              | ||||||
|  |             return () => { | ||||||
|  |                 ipcRenderer.removeListener('feedback-request', wrappedCallback); | ||||||
|  |             }; | ||||||
|  |         } | ||||||
|  |     }, | ||||||
|  | 
 | ||||||
|  |     // 回饋處理
 | ||||||
|  |     feedback: { | ||||||
|  |         send: (data) => ipcRenderer.invoke('send-feedback', data), | ||||||
|  |         cancel: () => ipcRenderer.invoke('cancel-feedback') | ||||||
|  |     }, | ||||||
|  | 
 | ||||||
|  |     // 開發者工具
 | ||||||
|  |     dev: { | ||||||
|  |         openDevTools: () => ipcRenderer.invoke('open-dev-tools'), | ||||||
|  |         reload: () => ipcRenderer.invoke('reload-window') | ||||||
|  |     } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * 擴展現有的 Web UI 功能 | ||||||
|  |  */ | ||||||
|  | const webUIExtensions = { | ||||||
|  |     // 檢測桌面環境
 | ||||||
|  |     isDesktopMode: () => true, | ||||||
|  | 
 | ||||||
|  |     // 獲取桌面特有的配置
 | ||||||
|  |     getDesktopConfig: () => ({ | ||||||
|  |         windowControls: true, | ||||||
|  |         nativeMenus: process.platform === 'darwin', | ||||||
|  |         customTitleBar: process.platform !== 'darwin' | ||||||
|  |     }), | ||||||
|  | 
 | ||||||
|  |     // 桌面通知
 | ||||||
|  |     showNotification: (title, body, options = {}) => { | ||||||
|  |         if ('Notification' in window && Notification.permission === 'granted') { | ||||||
|  |             return new Notification(title, { body, ...options }); | ||||||
|  |         } | ||||||
|  |         return null; | ||||||
|  |     }, | ||||||
|  | 
 | ||||||
|  |     // 請求通知權限
 | ||||||
|  |     requestNotificationPermission: async () => { | ||||||
|  |         if ('Notification' in window) { | ||||||
|  |             return await Notification.requestPermission(); | ||||||
|  |         } | ||||||
|  |         return 'denied'; | ||||||
|  |     } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * 日誌工具 | ||||||
|  |  */ | ||||||
|  | const logger = { | ||||||
|  |     debug: (...args) => { | ||||||
|  |         if (process.env.NODE_ENV === 'development') { | ||||||
|  |             console.log('[Desktop Debug]', ...args); | ||||||
|  |         } | ||||||
|  |     }, | ||||||
|  |     info: (...args) => console.log('[Desktop Info]', ...args), | ||||||
|  |     warn: (...args) => console.warn('[Desktop Warn]', ...args), | ||||||
|  |     error: (...args) => console.error('[Desktop Error]', ...args) | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | // 暴露 API 到渲染進程
 | ||||||
|  | try { | ||||||
|  |     // 主要的桌面 API
 | ||||||
|  |     contextBridge.exposeInMainWorld('electronAPI', desktopAPI); | ||||||
|  |      | ||||||
|  |     // Web UI 擴展
 | ||||||
|  |     contextBridge.exposeInMainWorld('desktopExtensions', webUIExtensions); | ||||||
|  |      | ||||||
|  |     // 日誌工具
 | ||||||
|  |     contextBridge.exposeInMainWorld('desktopLogger', logger); | ||||||
|  |      | ||||||
|  |     // 標記桌面環境(向後兼容)
 | ||||||
|  |     contextBridge.exposeInMainWorld('MCP_DESKTOP_MODE', true); | ||||||
|  |      | ||||||
|  |     logger.info('桌面 API 已成功暴露到渲染進程'); | ||||||
|  |      | ||||||
|  | } catch (error) { | ||||||
|  |     console.error('暴露桌面 API 失敗:', error); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * DOM 載入完成後的初始化 | ||||||
|  |  */ | ||||||
|  | window.addEventListener('DOMContentLoaded', () => { | ||||||
|  |     logger.debug('DOM 載入完成,開始桌面環境初始化'); | ||||||
|  |      | ||||||
|  |     // 添加桌面環境樣式類
 | ||||||
|  |     document.body.classList.add('desktop-mode'); | ||||||
|  |     document.body.classList.add(`platform-${process.platform}`); | ||||||
|  |      | ||||||
|  |     // 設置桌面環境變數
 | ||||||
|  |     document.documentElement.style.setProperty('--is-desktop', '1'); | ||||||
|  |      | ||||||
|  |     // 如果是 macOS,添加特殊樣式
 | ||||||
|  |     if (process.platform === 'darwin') { | ||||||
|  |         document.body.classList.add('macos-titlebar'); | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     // 監聽鍵盤快捷鍵
 | ||||||
|  |     document.addEventListener('keydown', (event) => { | ||||||
|  |         // Ctrl/Cmd + R: 重新載入
 | ||||||
|  |         if ((event.ctrlKey || event.metaKey) && event.key === 'r') { | ||||||
|  |             if (process.env.NODE_ENV === 'development') { | ||||||
|  |                 event.preventDefault(); | ||||||
|  |                 location.reload(); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         // F12: 開發者工具
 | ||||||
|  |         if (event.key === 'F12' && process.env.NODE_ENV === 'development') { | ||||||
|  |             event.preventDefault(); | ||||||
|  |             desktopAPI.dev.openDevTools(); | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         // Escape: 最小化視窗
 | ||||||
|  |         if (event.key === 'Escape' && event.ctrlKey) { | ||||||
|  |             event.preventDefault(); | ||||||
|  |             desktopAPI.window.minimize(); | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|  |      | ||||||
|  |     logger.debug('桌面環境初始化完成'); | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * 錯誤處理 | ||||||
|  |  */ | ||||||
|  | window.addEventListener('error', (event) => { | ||||||
|  |     logger.error('渲染進程錯誤:', { | ||||||
|  |         message: event.message, | ||||||
|  |         filename: event.filename, | ||||||
|  |         lineno: event.lineno, | ||||||
|  |         colno: event.colno, | ||||||
|  |         error: event.error | ||||||
|  |     }); | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | window.addEventListener('unhandledrejection', (event) => { | ||||||
|  |     logger.error('未處理的 Promise 拒絕:', event.reason); | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | logger.info('預載腳本載入完成'); | ||||||
| @ -31,6 +31,7 @@ import tempfile | |||||||
| import asyncio | import asyncio | ||||||
| import base64 | import base64 | ||||||
| from typing import Annotated, List | from typing import Annotated, List | ||||||
|  | from enum import Enum | ||||||
| import io | import io | ||||||
| 
 | 
 | ||||||
| from fastmcp import FastMCP | from fastmcp import FastMCP | ||||||
| @ -108,6 +109,34 @@ SERVER_NAME = "互動式回饋收集 MCP" | |||||||
| SSH_ENV_VARS = ['SSH_CONNECTION', 'SSH_CLIENT', 'SSH_TTY'] | SSH_ENV_VARS = ['SSH_CONNECTION', 'SSH_CLIENT', 'SSH_TTY'] | ||||||
| REMOTE_ENV_VARS = ['REMOTE_CONTAINERS', 'CODESPACES'] | REMOTE_ENV_VARS = ['REMOTE_CONTAINERS', 'CODESPACES'] | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  | # ===== 回饋模式枚舉 ===== | ||||||
|  | class FeedbackMode(Enum): | ||||||
|  |     """回饋模式枚舉""" | ||||||
|  |     WEB = "web" | ||||||
|  |     DESKTOP = "desktop" | ||||||
|  |     AUTO = "auto" | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def get_feedback_mode() -> FeedbackMode: | ||||||
|  |     """ | ||||||
|  |     從環境變數獲取回饋模式 | ||||||
|  | 
 | ||||||
|  |     環境變數 MCP_FEEDBACK_MODE 可設置為: | ||||||
|  |     - 'web': 強制使用 Web 模式 | ||||||
|  |     - 'desktop': 強制使用桌面模式 | ||||||
|  |     - 'auto': 自動檢測(預設) | ||||||
|  | 
 | ||||||
|  |     Returns: | ||||||
|  |         FeedbackMode: 回饋模式 | ||||||
|  |     """ | ||||||
|  |     mode = os.environ.get('MCP_FEEDBACK_MODE', 'auto').lower() | ||||||
|  |     try: | ||||||
|  |         return FeedbackMode(mode) | ||||||
|  |     except ValueError: | ||||||
|  |         debug_log(f"無效的 MCP_FEEDBACK_MODE 值: {mode},使用預設值 'auto'") | ||||||
|  |         return FeedbackMode.AUTO | ||||||
|  | 
 | ||||||
| # 初始化 MCP 服務器 | # 初始化 MCP 服務器 | ||||||
| from . import __version__ | from . import __version__ | ||||||
| 
 | 
 | ||||||
| @ -452,8 +481,16 @@ async def interactive_feedback( | |||||||
|             project_directory = os.getcwd() |             project_directory = os.getcwd() | ||||||
|         project_directory = os.path.abspath(project_directory) |         project_directory = os.path.abspath(project_directory) | ||||||
|          |          | ||||||
|         # 使用 Web UI |         # 根據模式選擇啟動方式 | ||||||
|         result = await launch_web_ui_with_timeout(project_directory, summary, timeout) |         mode = get_feedback_mode() | ||||||
|  |         debug_log(f"回饋模式: {mode.value}") | ||||||
|  | 
 | ||||||
|  |         if mode == FeedbackMode.DESKTOP: | ||||||
|  |             result = await launch_desktop_feedback_ui(project_directory, summary, timeout) | ||||||
|  |         elif mode == FeedbackMode.WEB: | ||||||
|  |             result = await launch_web_feedback_ui(project_directory, summary, timeout) | ||||||
|  |         else:  # AUTO | ||||||
|  |             result = await launch_auto_feedback_ui(project_directory, summary, timeout) | ||||||
|          |          | ||||||
|         # 處理取消情況 |         # 處理取消情況 | ||||||
|         if not result: |         if not result: | ||||||
| @ -499,7 +536,7 @@ async def interactive_feedback( | |||||||
|         return [TextContent(type="text", text=user_error_msg)] |         return [TextContent(type="text", text=user_error_msg)] | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| async def launch_web_ui_with_timeout(project_dir: str, summary: str, timeout: int) -> dict: | async def launch_web_feedback_ui(project_dir: str, summary: str, timeout: int) -> dict: | ||||||
|     """ |     """ | ||||||
|     啟動 Web UI 收集回饋,支援自訂超時時間 |     啟動 Web UI 收集回饋,支援自訂超時時間 | ||||||
| 
 | 
 | ||||||
| @ -515,10 +552,10 @@ async def launch_web_ui_with_timeout(project_dir: str, summary: str, timeout: in | |||||||
| 
 | 
 | ||||||
|     try: |     try: | ||||||
|         # 使用新的 web 模組 |         # 使用新的 web 模組 | ||||||
|         from .web import launch_web_feedback_ui, stop_web_ui |         from .web import launch_web_feedback_ui as web_launch, stop_web_ui | ||||||
| 
 | 
 | ||||||
|         # 傳遞 timeout 參數給 Web UI |         # 傳遞 timeout 參數給 Web UI | ||||||
|         return await launch_web_feedback_ui(project_dir, summary, timeout) |         return await web_launch(project_dir, summary, timeout) | ||||||
|     except ImportError as e: |     except ImportError as e: | ||||||
|         # 使用統一錯誤處理 |         # 使用統一錯誤處理 | ||||||
|         error_id = ErrorHandler.log_error_with_context( |         error_id = ErrorHandler.log_error_with_context( | ||||||
| @ -534,49 +571,66 @@ async def launch_web_ui_with_timeout(project_dir: str, summary: str, timeout: in | |||||||
|             "interactive_feedback": user_error_msg, |             "interactive_feedback": user_error_msg, | ||||||
|             "images": [] |             "images": [] | ||||||
|         } |         } | ||||||
|     except TimeoutError as e: |  | ||||||
|         debug_log(f"Web UI 超時: {e}") |  | ||||||
|         # 超時時確保停止 Web 服務器 |  | ||||||
|         try: |  | ||||||
|             from .web import stop_web_ui |  | ||||||
|             stop_web_ui() |  | ||||||
|             debug_log("Web UI 服務器已因超時而停止") |  | ||||||
|         except Exception as stop_error: |  | ||||||
|             debug_log(f"停止 Web UI 服務器時發生錯誤: {stop_error}") |  | ||||||
| 
 | 
 | ||||||
|         return { | 
 | ||||||
|             "command_logs": "", | async def launch_desktop_feedback_ui(project_dir: str, summary: str, timeout: int) -> dict: | ||||||
|             "interactive_feedback": f"回饋收集超時({timeout}秒),介面已自動關閉。", |     """ | ||||||
|             "images": [] |     啟動桌面應用收集回饋 | ||||||
|         } | 
 | ||||||
|  |     Args: | ||||||
|  |         project_dir: 專案目錄路徑 | ||||||
|  |         summary: AI 工作摘要 | ||||||
|  |         timeout: 超時時間(秒) | ||||||
|  | 
 | ||||||
|  |     Returns: | ||||||
|  |         dict: 收集到的回饋資料 | ||||||
|  |     """ | ||||||
|  |     debug_log(f"啟動桌面應用介面,超時時間: {timeout} 秒") | ||||||
|  | 
 | ||||||
|  |     try: | ||||||
|  |         # 嘗試導入桌面模組 | ||||||
|  |         from .desktop import launch_desktop_app | ||||||
|  |         return await launch_desktop_app(project_dir, summary, timeout) | ||||||
|  |     except ImportError as e: | ||||||
|  |         debug_log(f"桌面模組未安裝或不可用,回退到 Web 模式: {e}") | ||||||
|  |         # 回退到 Web 模式 | ||||||
|  |         return await launch_web_feedback_ui(project_dir, summary, timeout) | ||||||
|     except Exception as e: |     except Exception as e: | ||||||
|         # 使用統一錯誤處理 |         debug_log(f"桌面應用啟動失敗,回退到 Web 模式: {e}") | ||||||
|         error_id = ErrorHandler.log_error_with_context( |         # 回退到 Web 模式 | ||||||
|             e, |         return await launch_web_feedback_ui(project_dir, summary, timeout) | ||||||
|             context={"operation": "Web UI 啟動", "timeout": timeout}, |  | ||||||
|             error_type=ErrorType.SYSTEM |  | ||||||
|         ) |  | ||||||
|         user_error_msg = ErrorHandler.format_user_error(e, include_technical=False) |  | ||||||
|         debug_log(f"❌ Web UI 錯誤 [錯誤ID: {error_id}]: {e}") |  | ||||||
| 
 | 
 | ||||||
|         # 發生錯誤時也要停止 Web 服務器 |  | ||||||
|         try: |  | ||||||
|             from .web import stop_web_ui |  | ||||||
|             stop_web_ui() |  | ||||||
|             debug_log("Web UI 服務器已因錯誤而停止") |  | ||||||
|         except Exception as stop_error: |  | ||||||
|             ErrorHandler.log_error_with_context( |  | ||||||
|                 stop_error, |  | ||||||
|                 context={"operation": "Web UI 服務器停止"}, |  | ||||||
|                 error_type=ErrorType.SYSTEM |  | ||||||
|             ) |  | ||||||
|             debug_log(f"停止 Web UI 服務器時發生錯誤: {stop_error}") |  | ||||||
| 
 | 
 | ||||||
|         return { | async def launch_auto_feedback_ui(project_dir: str, summary: str, timeout: int) -> dict: | ||||||
|             "command_logs": "", |     """ | ||||||
|             "interactive_feedback": user_error_msg, |     自動檢測環境並選擇合適的回饋介面 | ||||||
|             "images": [] | 
 | ||||||
|         } |     Args: | ||||||
|  |         project_dir: 專案目錄路徑 | ||||||
|  |         summary: AI 工作摘要 | ||||||
|  |         timeout: 超時時間(秒) | ||||||
|  | 
 | ||||||
|  |     Returns: | ||||||
|  |         dict: 收集到的回饋資料 | ||||||
|  |     """ | ||||||
|  |     debug_log("自動檢測環境以選擇回饋介面") | ||||||
|  | 
 | ||||||
|  |     # 檢測是否為遠程環境 | ||||||
|  |     if is_remote_environment(): | ||||||
|  |         debug_log("檢測到遠程環境,使用 Web 模式") | ||||||
|  |         return await launch_web_feedback_ui(project_dir, summary, timeout) | ||||||
|  | 
 | ||||||
|  |     # 本地環境:嘗試桌面模式,失敗則回退到 Web 模式 | ||||||
|  |     try: | ||||||
|  |         from .desktop import is_desktop_available | ||||||
|  |         if is_desktop_available(): | ||||||
|  |             debug_log("檢測到桌面環境可用,使用桌面模式") | ||||||
|  |             return await launch_desktop_feedback_ui(project_dir, summary, timeout) | ||||||
|  |     except ImportError: | ||||||
|  |         debug_log("桌面模組不可用") | ||||||
|  | 
 | ||||||
|  |     debug_log("使用 Web 模式作為預設選擇") | ||||||
|  |     return await launch_web_feedback_ui(project_dir, summary, timeout) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @mcp.tool() | @mcp.tool() | ||||||
|  | |||||||
| @ -100,6 +100,14 @@ class WebUIManager: | |||||||
|         self.server_process = None |         self.server_process = None | ||||||
|         self.i18n = get_i18n_manager() |         self.i18n = get_i18n_manager() | ||||||
| 
 | 
 | ||||||
|  |         # 添加模式檢測支援 | ||||||
|  |         self.mode = self._detect_feedback_mode() | ||||||
|  |         self.desktop_manager = None | ||||||
|  | 
 | ||||||
|  |         # 如果是桌面模式,嘗試初始化桌面管理器 | ||||||
|  |         if self.mode == "desktop": | ||||||
|  |             self._init_desktop_manager() | ||||||
|  | 
 | ||||||
|         # 設置靜態文件和模板 |         # 設置靜態文件和模板 | ||||||
|         self._setup_static_files() |         self._setup_static_files() | ||||||
|         self._setup_templates() |         self._setup_templates() | ||||||
| @ -108,6 +116,43 @@ class WebUIManager: | |||||||
|         setup_routes(self) |         setup_routes(self) | ||||||
| 
 | 
 | ||||||
|         debug_log(f"WebUIManager 初始化完成,將在 {self.host}:{self.port} 啟動") |         debug_log(f"WebUIManager 初始化完成,將在 {self.host}:{self.port} 啟動") | ||||||
|  |         debug_log(f"回饋模式: {self.mode}") | ||||||
|  | 
 | ||||||
|  |     def _detect_feedback_mode(self) -> str: | ||||||
|  |         """檢測回饋模式""" | ||||||
|  |         mode = os.environ.get('MCP_FEEDBACK_MODE', 'auto').lower() | ||||||
|  |         if mode in ['web', 'desktop', 'auto']: | ||||||
|  |             return mode | ||||||
|  |         else: | ||||||
|  |             debug_log(f"無效的 MCP_FEEDBACK_MODE 值: {mode},使用預設值 'auto'") | ||||||
|  |             return 'auto' | ||||||
|  | 
 | ||||||
|  |     def _init_desktop_manager(self): | ||||||
|  |         """初始化桌面管理器(如果可用)""" | ||||||
|  |         try: | ||||||
|  |             # 嘗試導入桌面模組 | ||||||
|  |             from ..desktop import ElectronManager | ||||||
|  |             self.desktop_manager = ElectronManager() | ||||||
|  |             debug_log("桌面管理器初始化成功") | ||||||
|  |         except ImportError: | ||||||
|  |             debug_log("桌面模組不可用,將在需要時回退到 Web 模式") | ||||||
|  |             self.desktop_manager = None | ||||||
|  |         except Exception as e: | ||||||
|  |             debug_log(f"桌面管理器初始化失敗: {e}") | ||||||
|  |             self.desktop_manager = None | ||||||
|  | 
 | ||||||
|  |     def should_use_desktop_mode(self) -> bool: | ||||||
|  |         """判斷是否應該使用桌面模式""" | ||||||
|  |         if self.mode == "web": | ||||||
|  |             return False | ||||||
|  |         elif self.mode == "desktop": | ||||||
|  |             return self.desktop_manager is not None | ||||||
|  |         else:  # auto | ||||||
|  |             # 自動模式:檢測環境 | ||||||
|  |             from ..server import is_remote_environment | ||||||
|  |             if is_remote_environment(): | ||||||
|  |                 return False | ||||||
|  |             return self.desktop_manager is not None | ||||||
| 
 | 
 | ||||||
|     def _setup_compression_middleware(self): |     def _setup_compression_middleware(self): | ||||||
|         """設置壓縮和緩存中間件""" |         """設置壓縮和緩存中間件""" | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user
	 Minidoracat
						Minidoracat