177 Commits (73b5104d0f4cb91065e31174934bbb65c824e1e8)
| Author | SHA1 | Message | Date |
|---|---|---|---|
|
|
77d262b014 | chore: bump version to 9.3.1 | |
|
|
111c9d5595
|
Merge pull request #663 from CoplayDev/main
chore: sync main (v9.3.0) into beta |
|
|
|
44968fd763 | chore: bump version to 9.3.0 | |
|
|
424df879d4
|
Update warning message for Camera Capture (#661) | |
|
|
140a7e5c55
|
Asset store updates (#660)
* Add resource discovery service and UI for managing MCP resources * Consolidate duplicate IsBuiltIn logic into StringCaseUtility.IsBuiltInMcpType * Add resource enable/disable enforcement and improve error response handling - Block execution of disabled resources in TransportCommandDispatcher with clear error message - Add parse_resource_response() utility to handle error responses without Pydantic validation failures - Replace inline response parsing with parse_resource_response() across all resource handlers - Export parse_resource_response from models/__init__.py for consistent usage * Block execution of disabled built-in tools in TransportCommandDispatcher with clear error message Add tool enable/disable enforcement before command execution. Check tool metadata and enabled state, returning error response if tool is disabled. Prevents execution of disabled tools with user-friendly error message. * Fire warning in the rare chance there are duplicate names * Add Asset Store version checking with separate cache from Git installations To make this work I've added a publicly available JSON that's updated after every release. We can get the info from the asset store page that's against Unity's terms of service, so we want to avoid trouble. The release approval is manual, so this method suffices * Change LastUpdateCheck from Int to String type and add Asset Store version check EditorPrefs * Add EditorPrefs keys for local HTTP server state tracking * Add remote URL configuration parameter for Asset Store release preparation Needed to update this to set the default scope to "remote" because now it's a separate transport mode |
|
|
|
4d969419d4
|
Fix WebSocket connection reliability and domain reload recovery (#656)
* Add server-side ping/pong heartbeat to detect dead WebSocket connections On Windows, WebSocket connections can die silently (OSError 64) without either side being notified. This causes commands to fail with "Unity session not available" until Unity eventually detects the dead connection. Changes: - Add PingMessage model for server->client pings - Add ping loop in PluginHub that sends pings every 10 seconds - Track last pong time per session; close connection if no pong within 20s - Include session_id in pong messages from Unity for server-side tracking - Clean up debug/timing logs from Issue #654 investigation The server will now proactively detect dead connections within 20 seconds instead of waiting indefinitely for the next command to fail. Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * Fix connection recovery after Unity domain reloads (#654) When Unity performs a domain reload (after script changes, test runs, or large payload transfers), the MCP connection drops and needs to reconnect. The previous reconnection timeout (2s) was too short for domain reloads which can take 10-30s. Changes: - Increase UNITY_MCP_RELOAD_MAX_WAIT_S default from 2s to 30s - Increase backoff cap when reloading from 0.8s to 5.0s - Skip PluginHub session resolution for stdio transport (was causing unnecessary waits on every command) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * Fix ping/pong heartbeat, reduce timeout to 20s, fix test flakiness - Add server-side ping loop to detect dead WebSocket connections - Include session_id in pong messages for tracking - Reduce domain reload timeout from 30s to 20s - Add ClassVar annotations for mutable class attributes - Add lock protection for _last_pong access - Change debug stack trace log from Warn to Debug level - Remove unused TIMING-STDIO variable - Fix flaky async duration test (allow 20% timer variance) - Fix Python test that cleared HOME env var on Windows - Skip Unix-path test on Windows (path separator difference) - Add LogAssert.Expect to PropertyConversion tests Fixes #654, #643 Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> --------- Co-authored-by: Claude Opus 4.5 <noreply@anthropic.com> |
|
|
|
61284cc172
|
Display resources (#658)
* Add resource discovery service and UI for managing MCP resources * Consolidate duplicate IsBuiltIn logic into StringCaseUtility.IsBuiltInMcpType * Add resource enable/disable enforcement and improve error response handling - Block execution of disabled resources in TransportCommandDispatcher with clear error message - Add parse_resource_response() utility to handle error responses without Pydantic validation failures - Replace inline response parsing with parse_resource_response() across all resource handlers - Export parse_resource_response from models/__init__.py for consistent usage * Block execution of disabled built-in tools in TransportCommandDispatcher with clear error message Add tool enable/disable enforcement before command execution. Check tool metadata and enabled state, returning error response if tool is disabled. Prevents execution of disabled tools with user-friendly error message. * Fire warning in the rare chance there are duplicate names * Handle rare case a resource name is null Co-authored-by: coderabbitai[bot] <136622811+coderabbitai[bot]@users.noreply.github.com> --------- Co-authored-by: coderabbitai[bot] <136622811+coderabbitai[bot]@users.noreply.github.com> |
|
|
|
664a43b76c
|
Remote server auth (#644)
* Disable the gloabl default to first session when hosting remotely
* Remove calls to /plugin/sessions
The newer /api/instances covers that data, and we want to remove these "expose all" endpoints
* Disable CLI routes when running in remote hosted mode
* Update server README
* feat: add API key authentication support for remote-hosted HTTP transport
- Add API key field to connection UI (visible only in HTTP Remote mode)
- Add "Get API Key" and "Clear" buttons with login URL retrieval
- Include X-API-Key header in WebSocket connections when configured
- Add API key to CLI commands (mcp add, claude mcp add) when set
- Update config.json generation to include headers with API key
- Add API key validation service with caching and configurable endpoints
- Add /api/auth/login-url endpoint
* feat: add environment variable support for HTTP remote hosted mode
- Add UNITY_MCP_HTTP_REMOTE_HOSTED environment variable as alternative to --http-remote-hosted flag
- Accept "true", "1", or "yes" values (case-insensitive)
- Update CLI help text to document environment variable option
* feat: add user isolation enforcement for remote-hosted mode session listing
- Raise ValueError when list_sessions() called without user_id in remote-hosted mode
- Add comprehensive integration tests for multi-user session isolation
- Add unit tests for PluginRegistry user-scoped session filtering
- Verify cross-user isolation with same project hash
- Test unity_instances resource and set_active_instance user filtering
* feat: add comprehensive integration tests for API key authentication
- Add ApiKeyService tests covering validation, caching, retries, and singleton lifecycle
- Add startup config validation tests for remote-hosted mode requirements
- Test cache hit/miss scenarios, TTL expiration, and manual invalidation
- Test transient failure handling (5xx, timeouts, connection errors) with retry logic
- Test service token header injection and empty key fast-path validation
- Test startup validation requiring
* test: add autouse fixture to restore config state after startup validation tests
Ensures test isolation for config-dependent integration tests
* feat: skip user_id resolution in non-remote-hosted mode
Prevents unnecessary API key validation when not in remote-hosted mode
* test: add missing mock attributes to instance routing tests
- Add client_id to test context mock in set_active_instance test
- Add get_state mock to context in global instance routing test
* Fix broken telemetry test
* Add comprehensive API key authentication documentation
- Add user guide covering configuration, setup, and troubleshooting
- Add architecture reference documenting internal design and request flows
* Add remote-hosted mode and API key authentication documentation to server README
* Update reference doc for Docker Hub
* Specify exception being caught
* Ensure caplog handler cleanup in telemetry queue worker test
* Use NoUnitySessionError instead of RuntimeError in session isolation test
* Remove unusued monkeypatch arg
* Use more obviously fake API keys
* Reject connections when ApiKeyService is not initialized in remote-hosted mode
- Validate that user_id is present after successful key validation
- Expand transient error detection to include timeout and service errors
- Use consistent 1013 status code for retryable auth failures
* Accept "on" for UNITY_MCP_HTTP_REMOTE_HOSTED env var
Consistent with repo
* Invalidate cached login URL when HTTP base URL changes
* Pass API key as parameter instead of reading from EditorPrefs in RegisterWithCapturedValues
* Cache API key in field instead of reading from EditorPrefs on each reconnection
* Align markdown table formatting in remote server auth documentation
* Minor fixes
* security: Sanitize API key values in shell commands and fix minor issues
Add SanitizeShellHeaderValue() method to escape special shell characters (", \, `, $, !) in API keys before including them in shell command arguments. Apply sanitization to all three locations where API keys are embedded in shell commands (two in RegisterWithCapturedValues, one in GetManualInstructions).
Also fix deprecated passwordCharacter property (now maskChar) and improve exception logging in _resolve_user_id_from_request
* Consolidate duplicate instance selection error messages into InstanceSelectionRequiredError class
Add InstanceSelectionRequiredError exception class with centralized error messages (_SELECTION_REQUIRED and _MULTIPLE_INSTANCES). Replace 4 duplicate RuntimeError raises with new exception type. Update tests to catch InstanceSelectionRequiredError instead of RuntimeError.
* Replace hardcoded "X-API-Key" strings with AuthConstants.ApiKeyHeader constant across C# and Python codebases
Add AuthConstants class in C# and API_KEY_HEADER constant in Python to centralize the API key header name definition. Update all 8 locations where "X-API-Key" was hardcoded (4 in C#, 4 in Python) to use the new constants instead.
* Fix imports
* Filter session listing by user_id in all code paths to prevent cross-user session access
Remove conditional logic that only filtered sessions by user_id in remote-hosted mode. Now all session listings are filtered by user_id regardless of hosting mode, ensuring users can only see and interact with their own sessions.
* Consolidate get_session_id_by_hash methods into single method with optional user_id parameter
Merge get_session_id_by_hash and get_session_id_by_user_hash into a single method that accepts an optional user_id parameter. Update all call sites to use the unified method signature with user_id as the second parameter. Update tests and documentation to reflect the simplified API.
* Add environment variable support for project-scoped-tools flag [skip ci]
Support UNITY_MCP_PROJECT_SCOPED_TOOLS environment variable as alternative to --project-scoped-tools command line flag. Accept "true", "1", "yes", or "on" as truthy values (case-insensitive). Update help text to document the environment variable option.
* Fix Python tests
* Update validation logic to only require API key validation URL when both http_remote_hosted is enabled AND transport mode is "http", preventing false validation errors in stdio mode.
* Update Server/src/main.py
Co-authored-by: coderabbitai[bot] <136622811+coderabbitai[bot]@users.noreply.github.com>
* Refactor HTTP transport configuration to support separate local and remote URLs
Split HTTP transport into HttpLocal and HttpRemote modes with separate EditorPrefs storage (HttpBaseUrl and HttpRemoteBaseUrl).
Add HttpEndpointUtility methods to get/save local and remote URLs independently, and introduce IsRemoteScope() and GetCurrentServerTransport() helpers to centralize 3-way transport determination (Stdio/Http/HttpRemote). Update all client configuration code to distinguish between local and remote HTTP
* Only include API key headers in HTTP/WebSocket configuration when in remote-hosted mode
Update all locations where API key headers are added to HTTP/WebSocket configurations to check HttpEndpointUtility.IsRemoteScope() or serverTransport == HttpRemote before including the API key. This prevents local HTTP mode from unnecessarily including API key headers in shell commands, config JSON, and WebSocket connections.
* Hide Manual Server Launch foldout when not in HTTP Local mode
* Fix failing test
* Improve error messaging and API key validation for HTTP Remote transport
Add detailed error messages to WebSocket connection failures that guide users to check server URL, server status, and API key validity. Store error state in TransportState for propagation to UI. Disable "Start Session" button when HTTP Remote mode is selected without an API key, with tooltip explaining requirement. Display error dialog on connection failure with specific error message from transport state. Update connection
* Add missing .meta file
* Store transport mode in ServerConfig instead of environment variable
* Add autouse fixture to restore global config state between tests
Add restore_global_config fixture in conftest.py that automatically saves and restores global config attributes and UNITY_MCP_TRANSPORT environment variable between tests. Update integration tests to use monkeypatch.setattr on config.transport_mode instead of monkeypatch.setenv to prevent test pollution and ensure clean state isolation.
* Fix startup
* Replace _current_transport() calls with direct config.transport_mode access
* Minor cleanup
* Add integration tests for HTTP transport authentication behavior
Verify that HTTP local mode allows requests without user_id while HTTP remote-hosted mode rejects them with auth_required error.
* Add smoke tests for transport routing paths across HTTP local, HTTP remote, and stdio modes
Verify that HTTP local routes through PluginHub without user_id, HTTP remote routes through PluginHub with user_id, and stdio calls legacy send function with instance_id. Each test uses monkeypatch to configure transport mode and mock appropriate transport layer functions.
---------
Co-authored-by: coderabbitai[bot] <136622811+coderabbitai[bot]@users.noreply.github.com>
|
|
|
|
8ee9700327
|
Add GitHub Copilot CLI support to MCP client list and configurator (#641) | |
|
|
13aca5fb1c
|
fix: Windows Claude Desktop stdio + Codex beta/timeout config (#650)
* fix: Remove cmd.exe wrapper for Claude Desktop on Windows The cmd.exe /c wrapper was causing stdio pipe issues on Windows, resulting in "Server disconnected" errors in Claude Desktop. Testing confirmed that calling uvx.exe directly works reliably. Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * fix: Codex config uses centralized beta args and adds startup timeout - Use GetBetaServerFromArgsList() for consistent --prerelease handling - Add startup_timeout_sec = 60 to allow time for uvx package downloads Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> --------- Co-authored-by: Claude Opus 4.5 <noreply@anthropic.com> |
|
|
|
62c015d873
|
Add create_child parameter to manage_prefabs modify_contents (#646)
* Add create_child parameter to manage_prefabs modify_contents Enables adding child GameObjects to existing prefabs via headless editing. Supports single object or array for batch creation in one save operation. Features: - Create children with primitive types (Cube, Sphere, etc.) - Set position, rotation, scale on new children - Add components to children - Specify parent within prefab hierarchy for nested children - Set tag, layer, and active state Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * Address code review feedback for create_child validation - Fix type hint to `tuple[dict | None, str | None]` to match actual returns - Add explicit dict validation with clear error message including actual type - Error on invalid component entries instead of silently ignoring them - Return ErrorResponse for invalid tag/layer instead of just logging warnings Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * Add unit tests for create_child prefab functionality Tests cover: - Single child with primitive type - Empty GameObject (no primitive_type) - Multiple children from array (batch creation) - Nested parenting within prefab - Error handling for invalid inputs Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> --------- Co-authored-by: Claude Opus 4.5 <noreply@anthropic.com> |
|
|
|
6ec31cb88d
|
Large Cleanup and Refactor + Many new Tests added (#642)
* docs: Add codebase overview and comprehensive refactor plan
- Add .claude/OVERVIEW.md with repository structure snapshot for future agents
* Documents 10 major components/domains
* Maps architecture layers and file organization
* Lists 94 Python files, 163 C# files, 27 MCP tools
* Identifies known improvement areas and patterns
- Add results/REFACTOR_PLAN.md with comprehensive refactoring strategy
* Synthesis of findings from 10 parallel domain analyses
* P0-P3 prioritized refactor items targeting 25-40% code reduction
* 23 specific refactoring tasks with effort estimates
* Regression-safe refactoring methodology:
- Characterization tests for current behavior
- One-commit-one-change discipline
- Parallel implementation patterns for verification
- Feature flags for instant rollback (EditorPrefs + environment)
* 4-phase parallel subagent execution workflow:
- Phase 1: Write characterization tests (10 agents in parallel)
- Phase 2: Execute refactorings (10 agents in parallel)
- Phase 3: Fix failing tests (10 agents in parallel)
- Phase 4: Cleanup legacy code (parallel)
* Domain-to-agent mapping and detailed prompt templates
* Safety guarantees and regression detection strategy
This plan enables structured, low-risk refactoring of the unity-mcp codebase
while maintaining full backward compatibility and reducing code duplication.
Co-Authored-By: Claude Haiku 4.5 <noreply@anthropic.com>
* More stuff for cleanup
* docs: Document null parameter handling inconsistency and test validation blocker
Characterization test fixes:
- Fix ManageEditor test to expect NullReferenceException (actual behavior)
- Fix FindGameObjects test to expect ErrorResponse (actual behavior)
Discovered issues:
- Inconsistent null handling: ManageEditor throws, FindGameObjects handles gracefully
- Running all EditMode tests triggers domain reloads that break MCP connection
Documentation updates:
- Add null handling inconsistency to REFACTOR_PLAN.md P1-1 section
- Create REFACTOR_PROGRESS.md to track refactoring work
- Document blocker: domain reload tests break MCP during test runs
Files:
- TestProjects/UnityMCPTests/Assets/Tests/EditMode/Tools/Characterization/EditorTools_Characterization.cs:32-47
- results/REFACTOR_PLAN.md (P1-1 section)
- REFACTOR_PROGRESS.md (new file)
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
* fix: Prevent characterization tests from mutating editor state
Root causes identified:
1. Tests calling ManageEditor.HandleCommand with "play" action entered play mode
2. Test executing "Window/General/Console" menu item opened Console window
Both actions caused Unity to steal focus from terminal
Fixes:
- Replaced "play" actions with "telemetry_status" (read-only) in 5 tests
- Fixed FindGameObjects tests to use "searchTerm" instead of "query" parameter
- Marked ExecuteMenuItem Console window test as [Explicit]
Result: 37/38 characterization tests pass without entering play mode or stealing focus
Tests fixed:
- HandleCommand_ActionNormalization_CaseInsensitive
- HandleCommand_ManageEditor_DifferentActionsDispatchToDifferentHandlers
- HandleCommand_ManageEditor_ReturnsResponseObject
- HandleCommand_ManageEditor_ReadOnlyActionsDoNotMutateState
- HandleCommand_ManageEditor_ActionsRecognized
- HandleCommand_ExecuteMenuItem_ExecutesNonBlacklistedItems (marked Explicit)
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
* docs: Mark characterization test validation complete
Updated REFACTOR_PROGRESS.md:
- Status: Ready for refactoring
- Completed characterization test validation (37/38 passing)
- Documented fixes for play mode and focus stealing issues
- Next steps: Begin Phase 1 Quick Wins
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
* fix: Mark StopLocalHttpServer test as Explicit - kills MCP connection
Root cause: ServerManagementService_StopLocalHttpServer_PrefersPidfileBasedApproach
calls service.StopLocalHttpServer() which actually stops the running MCP server,
causing the MCP connection to drop and test framework to crash.
Fix: Marked test as [Explicit("Stops the MCP server - kills connection")]
Result: 25/26 ServicesCharacterizationTests pass without killing MCP server
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
* docs: Update progress with complete characterization test validation
Validated both characterization test suites:
- EditorToolsCharacterizationTests: 37 passing, 1 explicit
- ServicesCharacterizationTests: 25 passing, 1 explicit
Total characterization tests: 62 passing, 2 explicit (64 total)
Combined with 280 existing regression tests: 342 C# tests
Total project coverage: ~545 tests (342 C# + 203 Python)
All tests run without:
- Play mode entry
- Focus stealing
- MCP server crashes
- Assembly reload issues
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
* test: Add 29 Windows/UI domain characterization tests
Add comprehensive characterization tests documenting UI patterns:
- EditorPrefs binding patterns (3 tests)
- UI lifecycle patterns (6 tests)
- Callback registration patterns (4 tests)
- Cross-component communication (5 tests)
- Visibility/refresh logic (2 tests)
All 29 tests pass (validated in EditMode).
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
* docs: Update progress with Windows characterization tests complete
- Added 29 Windows/UI characterization tests (all passing)
- Updated total C# tests: 371 passing, 2 explicit
- Updated total coverage: ~574 tests (371 C# + 203 Python)
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
* test: Add 53 Models domain characterization tests
Add comprehensive characterization tests documenting model patterns:
- McpStatus enum (3 tests)
- ConfiguredTransport enum (2 tests)
- McpClient class (20 tests) - documents 6 capability flags
- McpConfigServer class (10 tests) - JSON.NET NullValueHandling
- McpConfigServers class (4 tests) - JsonProperty("unityMCP")
- McpConfig class (5 tests) - three-level hierarchy
- Command class (8 tests) - JObject params handling
- Round-trip serialization (1 test)
All 53 tests pass (validated in EditMode).
Captures P2-3 target: McpClient over-configuration issue.
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
* docs: Update progress with Models tests complete and bug documentation
- Added 53 Models characterization tests (all passing)
- Updated total C# tests: 424 passing, 2 explicit
- Updated total coverage: ~627 tests (424 C# + 203 Python)
- All characterization test domains now complete
- Documented McpClient.SetStatus() NullReferenceException bug
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
* feat: Add pagination and filtering to tests resource
Reduces token usage from 13K+ to ~500 tokens for typical queries.
C# (Unity) Changes:
- Add pagination support (page_size, cursor, page_number)
- Add name filter parameter (case-insensitive contains)
- Default page_size: 50, max: 200
- Returns PaginationResponse with items, cursor, nextCursor, totalCount
- Both get_tests and get_tests_for_mode now support pagination
Python (MCP Server) Changes:
- Update resource signatures to accept pagination parameters
- Add PaginatedTestsData model for new response format
- Support both new paginated format and legacy list format
- Forward all parameters (mode, filter, page_size, cursor) to Unity
- Mark get_tests_for_mode as DEPRECATED (use get_tests with mode param)
Usage Examples:
- mcpforunity://tests?page_size=10
- mcpforunity://tests?mode=EditMode&filter=Characterization
- mcpforunity://tests?page_size=50&cursor=50
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
* fix: Simplify tests resource to work with fastmcp URI constraints
FastMCP resources require URI path parameters, not function parameters.
Simplified Python resource handlers to pass empty params to Unity.
Tested and verified:
- mcpforunity://tests - Returns first 50 of 426 tests (paginated)
- mcpforunity://tests/EditMode - Returns first 50 of 421 EditMode tests
Token savings: ~85% reduction (~6,150 → ~725 tokens per query)
C# handler (already committed) supports:
- mode, filter, page_size, cursor, page_number parameters
- Default page_size: 50, max: 200
- Returns PaginatedTestsData with nextCursor for pagination
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
* docs: Complete pre-refactor utility audit
Audited existing utilities to avoid duplication and identify opportunities to patch in existing helpers rather than creating new ones.
Key findings:
- AssetPathUtility.cs already exists (QW-3: patch in, don't create)
- ParamCoercion.cs already exists (foundation for P1-1)
- JSON parser pattern exists but not extracted (QW-2: create)
- Search method constants duplicated 14 times in vfx.py alone (QW-4: create)
- Confirmation dialog duplicated in 5 files (QW-5: create)
Updated REFACTOR_PLAN.md to reflect Create vs Patch In actions.
Created UTILITY_AUDIT.md with full analysis.
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
* refactor: QW-1 Delete dead code
Removed confirmed dead code:
- Server/src/utils/reload_sentinel.py (entire deprecated file)
- Server/src/transport/unity_transport.py:28-76 (with_unity_instance decorator - never used)
- Server/src/core/config.py:49-51 (configure_logging method - never called)
- MCPForUnity/Editor/Services/Transport/TransportManager.cs:26-27 (ActiveTransport, ActiveMode deprecated accessors)
- MCPForUnity/Editor/Windows/McpSetupWindow.cs:37 (commented maxSize line)
- MCPForUnity/Editor/Windows/Components/Connection/McpConnectionSection.cs (stopHttpServerButton backward-compat code and references)
Updated characterization tests to document removal of configure_logging.
NOT removed (refactor plan was incorrect - these are actively used):
- port_registry_ttl (used in stdio_port_registry.py)
- reload_retry_ms (used in plugin_hub.py, unity_connection.py)
- STDIO framing config (used in unity_connection.py)
All 59 config/transport tests passing.
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
* docs: Update progress with QW-1 complete
QW-1 (Delete Dead Code) completed - 86 lines removed.
Updated refactor plan to document:
- What was actually deleted (6 items, 86 lines)
- What was NOT dead code (port_registry_ttl, reload_retry_ms, STDIO framing config - all actively used)
- Test verification (59 config/transport tests passing)
Updated progress tracking with QW-1 completion details.
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
* refactor: QW-2 Create JSON parser utility
Created Server/src/cli/utils/parsers.py with comprehensive JSON parsing utilities:
- parse_value_safe(): JSON → float → string fallback (no exit)
- parse_json_or_exit(): JSON with quote/bool fixes, exits on error
- parse_json_dict_or_exit(): Ensures result is dict
- parse_json_list_or_exit(): Ensures result is list
Updated 8 CLI command modules to use new utilities:
- material.py: 2 patterns replaced (JSON → float → string, dict parsing)
- component.py: 3 patterns replaced (value parsing, 2x dict parsing)
- texture.py: Removed local try_parse_json (14 lines), now uses utility
- vfx.py: 2 patterns replaced (list and dict parsing)
- asset.py: 1 pattern replaced (dict parsing)
- editor.py: 1 pattern replaced (dict parsing)
- script.py: 1 pattern replaced (list parsing)
- batch.py: 1 pattern replaced (list parsing)
Eliminated ~60 lines of duplicated JSON parsing code.
All 23 material/component CLI tests passing.
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
* docs: Update progress with QW-2 complete
QW-2 (Create JSON Parser Utility) completed - ~60 lines eliminated.
Created comprehensive parser utility with 4 functions:
- parse_value_safe(): JSON → float → string (no exit)
- parse_json_or_exit(): JSON with fixes, exits on error
- parse_json_dict_or_exit(): Ensures dict result
- parse_json_list_or_exit(): Ensures list result
Updated 8 CLI modules, eliminated ~60 lines of duplication.
All 23 CLI tests passing.
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
* refactor: QW-3 Patch in AssetPathUtility for path normalization
Replaced duplicated path normalization patterns with AssetPathUtility.NormalizeSeparators():
Files updated:
- ManageScene.cs: 2 occurrences (lines 104, 131)
- ManageShader.cs: 2 occurrences (lines 69, 85)
- ManageScript.cs: 4 occurrences (lines 63, 66, 81, 82, 185, 2639)
- GameObjectModify.cs: 1 occurrence (line 50)
- ManageScriptableObject.cs: 1 occurrence (line 1444)
Total: 10+ path.Replace('\\', '/') patterns replaced with utility calls.
AssetPathUtility.NormalizeSeparators() provides centralized, tested path normalization that:
- Converts backslashes to forward slashes
- Handles null/empty paths safely
- Is already used throughout the codebase
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
* docs: Update progress with QW-3 complete
QW-3 (Patch in AssetPathUtility) completed - 10+ patterns replaced.
Patched existing AssetPathUtility.NormalizeSeparators() into 5 Editor tool files:
- ManageScene.cs: 2 patterns
- ManageShader.cs: 2 patterns
- ManageScript.cs: 4 patterns
- GameObjectModify.cs: 1 pattern
- ManageScriptableObject.cs: 1 pattern
Replaced duplicated path.Replace('\\', '/') patterns with centralized utility.
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
* refactor: QW-4 Create search method constants for CLI commands
Created centralized constants module to eliminate duplicated search method
choices across CLI commands. This establishes a single source of truth for
GameObject/component search patterns.
Changes:
- Created Server/src/cli/utils/constants.py with 4 search method sets:
* SEARCH_METHODS_FULL (6 methods) - for gameobject commands
* SEARCH_METHODS_BASIC (3 methods) - for component/animation/audio
* SEARCH_METHODS_RENDERER (5 methods) - for material commands
* SEARCH_METHODS_TAGGED (4 methods) - for VFX commands
- Updated 6 CLI command modules to use new constants:
* vfx.py: 14 occurrences replaced with SEARCH_METHOD_CHOICE_TAGGED
* gameobject.py: Multiple occurrences with FULL and TAGGED
* component.py: All occurrences with BASIC
* material.py: All occurrences with RENDERER
* animation.py: All occurrences with BASIC
* audio.py: All occurrences with BASIC
Impact:
- Eliminates ~30+ lines of duplicated Click.Choice declarations
- Makes search method changes easier (single source of truth)
- Prevents inconsistencies across commands
Testing: All 49 CLI characterization tests passing
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
* docs: Update REFACTOR_PLAN with QW-4 completion status
* refactor: QW-5 Create confirmation dialog utility for CLI commands
Created centralized confirmation utility to eliminate duplicated confirmation
dialog patterns across CLI commands. Provides consistent UX for destructive
operations.
Changes:
- Created Server/src/cli/utils/confirmation.py with confirm_destructive_action()
* Flexible message formatting for different contexts
* Respects --force flag to skip prompts
* Raises click.Abort if user declines
- Updated 5 CLI command modules to use new utility:
* component.py: Remove component confirmation
* gameobject.py: Delete GameObject confirmation
* script.py: Delete script confirmation
* shader.py: Delete shader confirmation
* asset.py: Delete asset confirmation
Impact:
- Eliminates 5+ duplicate "if not force: click.confirm(...)" patterns
- Consistent confirmation message formatting
- Single location to enhance confirmation behavior
Testing: All 49 CLI characterization tests passing
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
* docs: Add QW-5 completion and comprehensive verification summary
All Quick Wins (QW-1 through QW-5) now complete and fully verified with:
- 108/108 Python tests passing
- 322/327 C# Unity tests passing (5 explicit skipped)
- Live integration tests successful
Total impact: ~180+ lines removed, 3 new utilities created, 16 files refactored
* docs: Add URI to all 21 MCP resource descriptions for better discoverability
Added explicit URI documentation to every MCP resource description to prevent
confusion between resource names (snake_case) and URIs (slash/hyphen separated).
Changes:
- Updated 21 MCP resources across 14 Python files
- Format: description + newline + URI: mcpforunity://...
- Added MCP Resources section to README.md explaining URI format
- Emphasized that resource names != URIs (editor_state vs editor/state)
Impact:
- Future AI agents will not fumble with URI format
- Self-documenting resource catalog
- Clear distinction between name and URI fields
Files updated (14 Python files, 21 resources total):
- tags.py, editor_state.py, unity_instances.py, project_info.py
- prefab_stage.py, custom_tools.py, windows.py, selection.py
- menu_items.py, layers.py, active_tool.py
- prefab.py (3 resources), gameobject.py (4 resources), tests.py (2 resources)
- README.md (added MCP Resources documentation section)
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
* refactor: P1-1 Create ToolParams validation wrapper
- Add ToolParams helper class for unified parameter validation
- Add Result<T> type for operation results
- Implements snake_case/camelCase fallback automatically
- Add comprehensive unit tests for ToolParams
- Refactor ManageEditor.cs to use ToolParams (fixes null params issue)
- Refactor FindGameObjects.cs to use ToolParams
This eliminates repetitive IsNullOrEmpty checks and provides consistent
error messages across all tools. First step towards removing 997+ lines
of duplicated validation code.
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
* refactor: P1-1 Apply ToolParams to ManageScript and ReadConsole
- Refactor ManageScript.cs to use ToolParams wrapper
- Refactor ReadConsole.cs to use ToolParams wrapper
- Simplifies parameter extraction and validation
- Maintains backwards compatibility with snake_case/camelCase
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
* fix: Resolve compilation errors in ToolParams implementation
- Rename Result<T>.Error property to ErrorMessage to avoid conflict with Error() static method
- Update all references to use ErrorMessage instead of Error
- Fix SearchMethods constant reference in FindGameObjects
- Rename options variable to optionsToken in ManageScript to avoid scope conflict
- Verify compilation succeeds with no errors
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
* test: Update ManageEditor null params test to reflect P1-1 fix
The P1-1 ToolParams refactoring fixed ManageEditor to handle null params
gracefully by returning an ErrorResponse instead of throwing NullReferenceException.
Update the characterization test to validate this new, correct behavior.
* docs: Add P1-1.5 Python MCP Parameter Aliasing plan
Identified gap: C# ToolParams provides snake_case/camelCase flexibility,
but Python MCP layer (FastMCP/pydantic) rejects non-matching parameter names.
This creates user friction when they guess wrong on naming convention.
Plan adds parameter normalization decorator to Python tool registration,
making the entire stack forgiving of naming conventions.
Scope: ~20 tools, ~50+ parameters
Estimated effort: 2 hours
Risk: Low (additive, does not modify existing behavior)
Impact: High (eliminates entire class of user errors)
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* fix: Address PR #642 CodeRabbit review feedback
- ToolParams: Add GetToken helper for consistent snake/camel fallback
in GetBool, Has, and GetRaw methods (not just string getters)
- ManageScript: Guard options token type with `as JObject` before indexing
- constants.py: Add `by_id` to SEARCH_METHODS_RENDERER for consistency
- McpClient: Add null-safe check for configStatus in GetStatusDisplayString
Added 6 new tests for snake/camel fallback in GetBool, Has, GetRaw.
All 458 EditMode tests passing (452 pass, 6 expected skips).
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* fix: Address remaining PR #642 CodeRabbit feedback
- texture.py: Remove unused `json` import (now using centralized parser)
- GetTests.cs: Clamp pageSize before computing cursor to fix inconsistency
when page_number is used with large page_size values
- mcp.json: Use ${workspaceFolder} instead of hardcoded absolute path
- settings.local.json: Remove duplicate unity-mcp permission entry,
rename server to UnityMCP for consistency
All 458 EditMode tests passing. 22 Python texture tests passing.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* fix: Address final PR #642 CodeRabbit feedback for tests
- Rename HandleCommand_AllTools_SafelyHandleNullTokens to
HandleCommand_ManageEditor_SafelyHandlesNullTokens (scope accuracy)
- Strengthen assertion from ContainsKey("success") to (bool)jo["success"]
- Fix incorrect parameter name from "query" to "searchTerm" in
HandleCommand_FindGameObjects_SearchMethodOptions test
All 458 EditMode tests passing.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* docs: Integrate CodeRabbit feedback into P1-1.5 plan
Updated the Python MCP Parameter Aliasing plan based on PR review:
- Add preliminary audit step to check sync vs async tool functions
- Update decorator to handle both sync and async functions
- Improve camel_to_snake regex for consecutive capitals (HTMLParser)
- Add conflict detection when both naming conventions are provided
- Add edge cases table with expected behavior
- Expand unit test requirements for new scenarios
- Adjust time estimate from 2h to 2.5h
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* feat: P1-1.5 Add parameter normalization middleware for camelCase support
Implements Python MCP parameter aliasing via FastMCP middleware.
This allows MCP clients to use either camelCase or snake_case for
parameter names (e.g., searchMethod or search_method).
Implementation:
- ParamNormalizerMiddleware intercepts tool calls before FastMCP validation
- Normalizes camelCase params to snake_case in the request message
- When both conventions are provided, explicit snake_case takes precedence
Files added:
- transport/param_normalizer_middleware.py - Middleware implementation
- services/tools/param_normalizer.py - Decorator version (backup approach)
- tests/test_param_normalizer.py - 23 comprehensive tests
Changes:
- main.py: Register ParamNormalizerMiddleware before UnityInstanceMiddleware
- services/tools/__init__.py: Remove decorator approach (middleware handles it)
All 23 param normalizer tests passing.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* refactor: P1-1.5 Use Pydantic AliasChoices instead of middleware
The middleware approach didn't work because FastMCP validates parameters
during JSON-RPC parsing, before middleware runs. Pydantic's AliasChoices
with Field(validation_alias=...) works correctly at the validation layer.
Changes:
- Update find_gameobjects.py to use AliasChoices pattern
- Remove ParamNormalizerMiddleware (validation happens before middleware)
- Delete param_normalizer.py decorator (same issue - runs after validation)
- Rewrite tests to verify AliasChoices pattern only
This allows tools to accept both snake_case and camelCase parameter names
(e.g., search_term and searchTerm both work).
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* docs: Update P1-1.5 status - pattern established, expansion bookmarked
The AliasChoices pattern works but adds verbosity. Decision: keep
find_gameobjects as proof-of-concept, expand to other tools only if
models frequently struggle with snake_case parameter names.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* refactor: P1-6 Consolidate duplicate test fixtures
Remove duplicate DummyMCP definitions from 4 test files - now import
from test_helpers.py instead. Also consolidate duplicate setup_*_tools
functions where identical to test_helpers.setup_script_tools.
- test_validate_script_summary.py: -27 lines
- test_manage_script_uri.py: -22 lines
- test_script_tools.py: -35 lines
- test_read_console_truncate.py: -11 lines
Total: ~95 lines removed, 18 tests still passing.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* docs: Update progress - P1-6 done, P1-2 and P2-3 skipped
- P1-6 (test fixtures): Complete, 95 lines removed
- P1-2 (EditorPrefs binding): Skipped - low impact, keys already centralized
- P2-3 (Configurator builder): Skipped - configurators already well-factored
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* refactor: P2-1 Add handle_unity_errors decorator for CLI commands
Create a reusable decorator that handles the repeated try/except
UnityConnectionError pattern found 99 times across 19 CLI files.
- Add handle_unity_errors() decorator to connection.py
- Refactor scene.py (7 commands) as proof-of-concept: -24 lines
- Pattern ready to apply to remaining 18 CLI command files
Each application eliminates ~3 lines per command (try/except/sys.exit).
Estimated total reduction when fully applied: ~200 lines.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* docs: Update progress - P2-1 in progress
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* refactor: P2-1 Complete - Apply handle_unity_errors decorator to all CLI commands
Applied the @handle_unity_errors decorator to 83 CLI commands across 18 files,
eliminating ~296 lines of repetitive try/except UnityConnectionError boilerplate.
Files updated:
- animation.py, asset.py, audio.py, batch.py, code.py, component.py
- editor.py, gameobject.py, instance.py, lighting.py, material.py
- prefab.py, script.py, shader.py, texture.py, tool.py, ui.py, vfx.py
Remaining intentional exceptions:
- editor.py:446 - Silent catch for suggestion lookup
- gameobject.py:191 - Track component failures in loop
- main.py - Special handling for status/ping/interactive commands
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* docs: Update progress - P2-1 complete
P2-1 (CLI Command Wrapper) is now complete:
- Created @handle_unity_errors decorator
- Applied to 83 commands across 18 files
- Eliminated ~296 lines of boilerplate
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* docs: Add P2-8 CLI Consistency Pass to refactor plan
Identified during live CLI testing - inconsistent patterns cause user errors:
- Missing --force flags on some destructive commands (texture, shader)
- Subcommand structure confusion (vfx particle info vs vfx particle-info)
- Inconsistent positional vs named arguments
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* refactor: P1-3 Add nullable coercion methods and consolidate TryParse patterns
Added nullable coercion overloads to ParamCoercion:
- CoerceIntNullable(JToken) - returns int? for optional params
- CoerceBoolNullable(JToken) - returns bool? for optional params
- CoerceFloatNullable(JToken) - returns float? for optional params
Refactored tools to use ParamCoercion instead of duplicated patterns:
- ManageScene.cs: Removed local BI()/BB() functions (~27 lines)
- RunTests.cs: Simplified bool parsing (~15 lines)
- GetTestJob.cs: Simplified bool parsing (~17 lines)
- RefreshUnity.cs: Simplified bool parsing (~10 lines)
Total: 87 lines of duplicated code eliminated, replaced with reusable utility calls.
All 458 Unity tests passing.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* docs: Update progress - P1-3 complete
Added nullable coercion methods and consolidated TryParse patterns.
~87 lines eliminated from 4 tool files.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* docs: Add P2-9 focus nudge improvements task to refactor plan
Problem identified during testing: Unity gets re-throttled by macOS
before enough test progress is made. 0.5s focus duration + 5s rate
limit creates cycle where Unity is throttled most of the time.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* refactor(P2-8): Add --force flag to texture delete command
texture delete was the only destructive CLI command missing the
confirmation prompt and --force flag. Now consistent with:
- script delete
- shader delete
- asset delete
- gameobject delete
- component remove
All 173 CLI tests passing.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* docs: Update P2-8 CLI Consistency Pass status
Core consistency issues addressed:
- texture delete now has --force/-f flag
- All --force flags verified to have -f short option
VFX clear commands intentionally left without confirmation (ephemeral).
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* fix: Address CodeRabbit PR feedback
REFACTOR_PROGRESS.md:
- Add blank line after "### Python Tests" heading before table (MD058)
- Convert bold table header to proper heading (MD036)
- Add blank lines around scope analysis table
Server/src/cli/commands/ui.py:
- Add error handling for Canvas component creation loop
- Track and report failed components instead of silently ignoring
EditorTools_Characterization.cs:
- Fix "query" to "searchTerm" in FindGameObjects tests
- HandleCommand_FindGameObjects_ReturnsPaginationMetadata
- HandleCommand_FindGameObjects_PageSizeRange
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* test(P3-1): Add ServerManagementService characterization tests
Add focused behavioral tests for ServerManagementService public methods
before decomposition refactoring:
- IsLocalUrl tests (localhost, 127.0.0.1, remote, empty)
- CanStartLocalServer tests (HTTP disabled, enabled with local/remote URL)
- TryGetLocalHttpServerCommand tests (HTTP disabled, remote URL, local URL)
- IsLocalHttpServerReachable tests (no server, remote URL)
- IsLocalHttpServerRunning tests (remote URL, error handling)
- ClearUvxCache error handling test
- Private method characterization via reflection
These tests establish a regression baseline before extracting:
ProcessDetector, PidFileManager, ProcessTerminator, ServerCommandBuilder,
and TerminalLauncher components.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* refactor(P3-1): Add Server component interfaces
Add interface definitions for ServerManagementService decomposition:
- IProcessDetector: Platform-specific process inspection
- LooksLikeMcpServerProcess, TryGetProcessCommandLine
- GetListeningProcessIdsForPort, GetCurrentProcessId, ProcessExists
- IPidFileManager: PID file and handshake state management
- GetPidFilePath, TryReadPid, DeletePidFile
- StoreHandshake, TryGetHandshake, StoreTracking, TryGetStoredPid
- IProcessTerminator: Platform-specific process termination
- Terminate (graceful-then-forced approach)
- IServerCommandBuilder: uvx/server command construction
- TryBuildCommand, BuildUvPathFromUvx, GetPlatformSpecificPathPrepend
- ITerminalLauncher: Platform-specific terminal launching
- CreateTerminalProcessStartInfo (macOS, Windows, Linux)
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* refactor(P3-1): Extract ProcessDetector from ServerManagementService
Create ProcessDetector implementing IProcessDetector:
- LooksLikeMcpServerProcess: Multi-strategy process identification
- TryGetProcessCommandLine: Platform-specific command line retrieval
- GetListeningProcessIdsForPort: Port-to-PID mapping via netstat/lsof
- GetCurrentProcessId: Safe Unity process ID retrieval
- ProcessExists: Cross-platform process existence check
- NormalizeForMatch: String normalization for matching
Update ServerManagementService:
- Add IProcessDetector dependency via constructor injection
- Delegate process inspection calls to injected detector
- Maintain backward compatibility with parameterless constructor
Add ProcessDetectorTests (25 tests):
- NormalizeForMatch edge cases and string handling
- GetCurrentProcessId consistency and validity
- ProcessExists for current process and invalid PIDs
- GetListeningProcessIdsForPort validation
- LooksLikeMcpServerProcess safety checks
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* refactor(P3-1): Extract PidFileManager from ServerManagementService
Create PidFileManager implementing IPidFileManager:
- GetPidDirectory/GetPidFilePath: PID file path construction
- TryReadPid: Parse PID from file with whitespace tolerance
- TryGetPortFromPidFilePath: Extract port from PID file name
- DeletePidFile: Safe PID file deletion
- StoreHandshake/TryGetHandshake: EditorPrefs handshake management
- StoreTracking/TryGetStoredPid: EditorPrefs PID tracking
- GetStoredArgsHash: Retrieve stored args fingerprint
- ClearTracking: Clear all EditorPrefs tracking keys
- ComputeShortHash: SHA256-based fingerprint generation
Update ServerManagementService:
- Add IPidFileManager dependency via constructor injection
- Delegate all PID file operations to injected manager
- Remove redundant static methods
Add PidFileManagerTests (33 tests):
- GetPidFilePath and GetPidDirectory validation
- TryReadPid with valid/invalid files, whitespace, edge cases
- TryGetPortFromPidFilePath parsing
- Handshake store/retrieve
- Tracking store/retrieve/clear
- ComputeShortHash determinism and edge cases
- DeletePidFile safety
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* refactor(P3-1): Extract ProcessTerminator from ServerManagementService
Create ProcessTerminator implementing IProcessTerminator:
- Terminate: Platform-specific process termination
- Windows: taskkill with /T (tree kill), escalates to /F if needed
- Unix: SIGTERM (kill -15) with 8s grace period, escalates to SIGKILL (kill -9)
- Verifies process termination via ProcessDetector.ProcessExists()
Update ServerManagementService:
- Add IProcessTerminator dependency via constructor injection
- Delegate TerminateProcess calls to injected terminator
- Remove ProcessExistsUnix helper (used via ProcessDetector)
Add ProcessTerminatorTests (10 tests):
- Constructor validation (null detector throws)
- Terminate with invalid/zero/non-existent PIDs
- Interface implementation verification
- Integration test with real detector
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* refactor(P3-1): Extract ServerCommandBuilder from ServerManagementService
Create ServerCommandBuilder implementing IServerCommandBuilder:
- TryBuildCommand: Constructs uvx command for HTTP server launch
- Validates HTTP transport enabled
- Validates local URL (localhost, 127.0.0.1, 0.0.0.0, ::1)
- Integrates with AssetPathUtility for uvx path discovery
- Handles dev mode refresh flags and project-scoped tools
- BuildUvPathFromUvx: Converts uvx path to uv path
- GetPlatformSpecificPathPrepend: Platform-specific PATH prefixes
- QuoteIfNeeded: Quote paths containing spaces
Update ServerManagementService:
- Add IServerCommandBuilder dependency via constructor injection
- Delegate command building to injected builder
- Remove redundant static methods (BuildUvPathFromUvx, GetPlatformSpecificPathPrepend)
Add ServerCommandBuilderTests (19 tests):
- QuoteIfNeeded edge cases (spaces, null, empty, already quoted)
- BuildUvPathFromUvx path conversion (Unix, Windows, null, filename-only)
- GetPlatformSpecificPathPrepend platform handling
- TryBuildCommand validation (HTTP disabled, remote URL, local URL)
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* refactor(P3-1): Extract TerminalLauncher from ServerManagementService
Create TerminalLauncher implementing ITerminalLauncher:
- CreateTerminalProcessStartInfo: Platform-specific terminal launch
- macOS: Uses .command script + /usr/bin/open -a Terminal
- Windows: Uses .cmd script + cmd.exe /c start
- Linux: Auto-detects gnome-terminal, xterm, konsole, xfce4-terminal
- GetProjectRootPath: Unity project root discovery
Update ServerManagementService:
- Add ITerminalLauncher dependency via constructor injection
- Delegate terminal operations to injected launcher
- Remove 110+ lines of platform-specific terminal code
Add TerminalLauncherTests (15 tests):
- GetProjectRootPath validation (non-empty, exists, not Assets)
- CreateTerminalProcessStartInfo error handling (empty, null, whitespace)
- ProcessStartInfo configuration validation
- Platform-specific behavior verification
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* refactor(P3-1): Complete ServerManagementService decomposition
Final cleanup of ServerManagementService after extracting 5 focused components:
- Remove unused imports (System.Globalization, System.Security.Cryptography, System.Text)
- Remove unused static field (LoggedStopDiagnosticsPids)
- Remove unused methods (GetProjectRootPath, StoreLocalServerPidTracking, LogStopDiagnosticsOnce, TrimForLog)
ServerManagementService is now a clean orchestrator at 876 lines (down from 1489),
delegating to: ProcessDetector, PidFileManager, ProcessTerminator, ServerCommandBuilder, TerminalLauncher
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* fix(critical): Prevent ProcessTerminator from killing all processes
Add PID validation before any kill operation:
- Reject PID <= 1 (prevents kill -1 catastrophe and init termination)
- Reject current Unity process PID
On Unix, kill(-1) sends signal to ALL processes the user can signal.
This caused all Mac applications to exit when tests ran Terminate(-1).
Added tests for PID 1 and current process protection.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* fix(tests): Correct characterization tests to document actual behavior
- IsLocalUrl_IPv6Loopback: Changed to assert false (known limitation)
- IsLocalUrl_Static reflection test: Same IPv6 fix
- BuildUvPathFromUvx_WindowsPath: Skip on non-Windows platforms
Characterization tests should document actual behavior, not desired behavior.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* refactor(P1-5): Add EditorConfigurationCache to eliminate scattered EditorPrefs reads
- Create EditorConfigurationCache singleton to centralize frequently-read settings
- Replace 25 direct EditorPrefs.GetBool(UseHttpTransport) calls with cached access
- Add change notification event for reactive UI updates
- Add Refresh() method for explicit cache invalidation
- Add 13 unit tests for cache behavior (singleton, read, write, invalidation)
- Update test files to refresh cache when modifying EditorPrefs directly
Files using cache: ServerManagementService, BridgeControlService, ConfigJsonBuilder,
McpClientConfiguratorBase, McpConnectionSection, McpClientConfigSection,
StdioBridgeHost, StdioBridgeReloadHandler, HttpBridgeReloadHandler,
McpEditorShutdownCleanup, ServerCommandBuilder, ClaudeDesktopConfigurator,
CherryStudioConfigurator
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* docs: Mark P1-5 Configuration Cache as complete
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* docs: Fix misleading parameter documentation in tests.py resources
The get_tests and get_tests_for_mode MCP resources claimed to support
optional parameters (filter, page_size, cursor) that were not actually
being forwarded to Unity. Updated docstrings to accurately describe
current behavior (returns first page with defaults) and direct users
to run_tests tool for advanced filtering/pagination.
Addresses CodeRabbit review comment about documentation/implementation
consistency.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* docs: Update REFACTOR_PROGRESS.md with P3-1 and P1-5 completions
- Added P3-1: ServerManagementService decomposition (1489→300 lines, 5 new services)
- Added P1-5: EditorConfigurationCache (25 EditorPrefs reads centralized)
- Updated test counts: 594 passing, 6 explicit (600 total)
- Updated current status header
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* docs: Update P2-6 plan with detailed VFX split + utility consolidation
Revised P2-6 to include:
- Part 1: Extract VFX Graph code into VfxGraphAssets/Read/Write/Control.cs
- Part 2: Consolidate ToCamelCase/ToSnakeCase into StringCaseUtility.cs
- Eliminates 6x duplication of string case conversion code
- Reduces ManageVFX.cs from 1023 to ~350 lines
Also marked P1-4 (Session Model Consolidation) as skipped - low impact
after evaluation showed only 1 conversion site with 4 lines of code.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* refactor(P2-6): Consolidate string case utilities
Create StringCaseUtility.cs with ToSnakeCase and ToCamelCase methods.
Update 5 files to use the shared utility, removing 6 duplicate implementations.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* refactor(P2-6): Extract VFX Graph code from ManageVFX
Extract ~590 lines of VFX Graph code into 5 dedicated files:
- VfxGraphAssets.cs: Asset management (create, assign, list)
- VfxGraphRead.cs: Read operations (get_info)
- VfxGraphWrite.cs: Parameter setters
- VfxGraphControl.cs: Playback control
- VfxGraphCommon.cs: Shared utilities
ManageVFX.cs reduced from 1006 to 411 lines (59% reduction).
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* docs: Update REFACTOR_PROGRESS.md with P2-6 completion
- ManageVFX.cs reduced from 1006 to 411 lines (59% reduction)
- 5 new VFX Graph files created
- StringCaseUtility consolidates 6 duplicate implementations
- P1-4 marked as skipped (low impact)
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* fix(P1-5): Add cache refresh when toggling HTTP/STDIO transport
McpConnectionSection was updating EditorPrefs but not refreshing
EditorConfigurationCache when user switched transports. Cache would
return stale value until manual refresh.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* refactor(P2-9): Improve focus nudge timing for better test reliability
- Increase default focus duration from 0.5s to 2.0s
- Reduce minimum nudge interval from 5.0s to 2.0s
- Add environment variable configuration:
- UNITY_MCP_NUDGE_DURATION_S: focus duration
- UNITY_MCP_NUDGE_INTERVAL_S: min interval between nudges
- Fix test_texture_delete to include --force flag (from P2-8)
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* docs: Mark refactor plan complete - all items evaluated
P2-9 (Focus Nudge) completed. Remaining items evaluated and skipped:
- P2-2, P2-4, P2-5, P2-7: Low impact or already addressed
- P3-2, P3-3, P3-4, P3-5: High effort/risk, diminishing returns
15 items completed, 12 items skipped. 600+ tests passing.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* fix: Add conftest.py to fix Python path for pytest
Add conftest.py that adds src/ to sys.path so pytest can properly import
cli, transport, and other modules. This fixes test failures where CLI
commands weren't being found.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* test: Enable domain reload resilience tests
Remove [Explicit] attribute from DomainReloadResilienceTests to include
them in regular test runs. These tests verify MCP remains functional
during Unity domain reloads (e.g., when scripts are created/compiled).
Tests now run automatically with improved focus nudge timing from P2-9.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* refactor(P2-9): Implement exponential backoff for focus nudges
Replace fixed interval with exponential backoff to handle different scenarios:
- Start aggressive: 1s base interval for quick stall detection
- Back off gracefully: Double interval after each nudge (1s→2s→4s→8s→10s max)
- Reset on progress: Return to base interval when tests make progress
- Longer focus duration: 3s default (up from 0.5s) for compilation/domain reloads
Also reduced stall threshold from 10s to 3s for faster stall detection.
This should handle domain reload tests that require sustained focus during
compilation while preventing excessive focus thrashing.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* fix(P2-9): Wait for window switch and use exponential focus duration
Two critical fixes for focus nudging:
1. **Wait for window switch to complete**: Added 0.5s delay after activate
command to let macOS window switching animation finish before starting
the focus timer. The activate command is asynchronous - it starts the
switch but returns immediately. This caused Unity to barely be visible
(or not visible at all) before switching back.
2. **Exponential focus duration**: Now increases focus time with consecutive
nudges (3s → 5s → 8s → 12s). Previous version only increased interval
between nudges, but kept duration fixed at 3s. Domain reloads need
longer sustained focus (12s) to complete compilation.
This should make focus swaps visibly perceptible and give Unity enough
time to complete compilation during domain reload tests.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* feat(P2-9): Add PID-based focus nudging for multi-instance support
- Add project_path to Unity registration message and PluginSession
- Unity sends project root path (dataPath without /Assets) during registration
- Focus nudge finds specific Unity instance by matching -projectpath in ps output
- Use AppleScript with Unix PID for precise window activation on macOS
- Handles multiple Unity instances correctly (even with same project name)
- Falls back to project_name matching if full path unavailable
* fix(P2-9): Use bundle ID activation to fully wake Unity on macOS
Two-step activation process:
1. Set frontmost to bring window to front
2. Activate via bundle identifier to trigger full app activation
This ensures Unity receives focus events and starts processing,
matching the behavior of cmd+tab or clicking the window.
Without step 2, Unity comes to foreground visually but doesn't
actually wake up until user interacts with it.
* fix(tests): Fix asyncio event loop issues in transport tests
- Change configured_plugin_hub to async fixture using @pytest_asyncio.fixture
- Use asyncio.get_running_loop() instead of deprecated get_event_loop()
- Import pytest_asyncio module
- Fixes 'RuntimeError: There is no current event loop' error
Also:
- Update telemetry test patches to use correct module (core.telemetry)
- Mark one telemetry test as skipped pending proper mock fix
Test results: 476/502 passing (25 telemetry mock tests need fixing)
* fix(tests): Fix telemetry mock patches to use correct import location
Changed all telemetry mock patches from:
- core.telemetry.record_tool_usage -> core.telemetry_decorator.record_tool_usage
- core.telemetry.record_resource_usage -> core.telemetry_decorator.record_resource_usage
- core.telemetry.record_milestone -> core.telemetry_decorator.record_milestone
The decorator imports these functions at module level, so mocks must patch
where they're used (telemetry_decorator) not where they're defined (telemetry).
All 51 telemetry tests now pass when run in isolation.
Note: Full test suite has interaction issues causing some telemetry tests
to fail and Python to crash. Investigating separately.
* fix(tests): Add telemetry singleton cleanup to prevent Python crashes
Added shutdown mechanism to TelemetryCollector:
- Added _shutdown flag to gracefully stop worker thread
- Modified _worker_loop to check shutdown flag and use timeout on queue.get()
- Added shutdown() method to stop worker thread
- Added reset_telemetry() function to reset global singleton
Added pytest fixtures for telemetry cleanup:
- Module-scoped cleanup_telemetry fixture (autouse) prevents crashes
- Class-scoped fresh_telemetry fixture for tests needing clean state
- Added fresh_telemetry to telemetry test classes
Results:
- ✅ No more Python crashes when running full test suite
- ✅ All tests pass when run without integration tests (292/292)
- ✅ All integration tests pass (124/124)
- ⚠️ 26 telemetry tests fail when run after integration tests (test order dependency)
The 26 failures are due to integration tests initializing telemetry before
characterization tests can mock it. Tests pass individually and in subsets.
Next: Investigate test ordering or mark flaky tests.
* fix(tests): Reorder test collection to run characterization tests before integration
Added pytest_collection_modifyitems hook in conftest.py to reorder tests:
- Characterization/unit tests run first
- Integration tests run last
This prevents integration tests from initializing the telemetry singleton
before characterization tests can mock it.
Result: ✅ ALL 502 PYTHON TESTS PASSING!
Test Results:
- Unity C# Tests: 605/605 ✓
- Python Tests: 502/502 ✓ (was 476/502)
Fixed the 26 telemetry test failures that were caused by test order dependency.
* docs: Clean up refactor artifacts and rewrite developer guide
- Delete 19 refactor/characterization markdown files
- Rewrite README-DEV.md with essentials: branching, local dev setup, running tests
- Align README-DEV-zh.md with English version
- Add CLAUDE.md with repo overview and code philosophy for AI assistants
- Update mcp_source.py to add upstream beta option (4 choices now)
- Remove CLAUDE.md from .gitignore so it can be shared
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* fix: Remove absolute path from docstring example
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* fix: Remove orphaned .meta files for deleted markdown docs
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* fix: Gate MCP startup logs behind debug mode toggle
Changed McpLog.Info calls to pass always=false so they only
appear when debug logging is enabled in Advanced Settings.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* fix: Use relative path for MCP package in test project manifest
Fixes CI failure - was using absolute local path that doesn't exist on runners.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* fix: Remove personal Claude settings and gitignore it
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* fix: Remove orphaned test README files referencing deleted docs
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* fix: Remove test artifact Materials and Prefabs
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* fix: Remove test artifacts (QW3 scene, screenshots, textures, models characterization)
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* fix: Remove file with corrupted filename
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* docs: Remove redundant OVERVIEW.md (covered by CLAUDE.md)
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* fix: Address CodeRabbit review feedback
- VfxGraphControl: Return error for unknown actions instead of success
- focus_nudge.py: Remove pointless f-string, narrow bare except
- test_transport_characterization.py: Fix unused params (_ctx), remove unused vars, track background task
- test_core_infrastructure_characterization.py: Use _ for unused loop variable
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
* fix(coderabbit): Address critical CodeRabbit feedback issues
- VfxGraphCommon: Add null guard in FindVisualEffect before accessing params
- run_tests.py: Parse Name@hash format before session lookup for multi-instance focus nudging
- WebSocketTransportClient: Use Path.GetFileName/GetDirectoryName for robust trailing separator handling
- focus_nudge.py: Safe float parsing for environment variables with fallback + warning logging
- LineWrite: Add debug logging to diagnose LineRenderer position persistence issue
Co-Authored-By: Claude Haiku 4.5 <noreply@anthropic.com>
* fix(coderabbit): Address linting and validation feedback
- CLAUDE.md: Add language identifiers to markdown code blocks, fix "etc" -> "etc."
- StringCaseUtility: Fix ToSnakeCase regex to match digit→Uppercase boundaries (param1Value -> param1_value)
- VfxGraphWrite: Add validation for unsupported vector dimensions (must be 2, 3, or 4)
- conftest.py: Improve telemetry reset error handling with safe parser and logging
Co-Authored-By: Claude Haiku 4.5 <noreply@anthropic.com>
* debug: Use McpLog.Warn for guaranteed LineRenderer debug visibility
* cleanup: Remove debug logging from LineWrite (tool verified working)
* fix(coderabbit): Safe float parsing and unused import cleanup
- VfxGraphWrite.SendEvent: Use safe float? parsing for size/lifetime to avoid ToObject exceptions
- run_tests.py: Remove unused 'os' import, narrow exception types to (AttributeError, KeyError), use else block for clarity
- conftest.py: Add noqa comment for pytest hook args (pytest requires exact parameter names)
Co-Authored-By: Claude Haiku 4.5 <noreply@anthropic.com>
* fix: OpenCode configurator preserves existing config
- TryLoadConfig now returns null on JSON errors (was returning empty object)
- Configure() preserves existing config and other MCP servers
- Only adds schema when creating new file
- Safely updates only unityMCP entry, preserves antigravity + other servers
- Better error logging for debugging config issues
Fixes issue where Configure button wiped entire config for Codex/OpenCode.
Co-Authored-By: Claude Haiku 4.5 <noreply@anthropic.com>
* security: Fix AppleScript injection vulnerability in focus_nudge.py
- Escape double quotes in app_name parameter before interpolation into AppleScript
- Prevents command injection via untrusted app names in focus_nudge.py:251
- Escaping follows AppleScript string literal requirements
Fixes high-severity vulnerability identified in security review.
Co-Authored-By: Claude Haiku 4.5 <noreply@anthropic.com>
* fix: Fix middleware job state cleanup and improve test error handling
## Changes
### TestJobManager: Auto-fail stalled initialization
- Add 15-second initialization timeout for jobs that fail to start tests
- Jobs in "running" state that never call OnRunStarted() are automatically failed
- Prevents "tests_running" deadlock when tests fail to initialize (e.g., unsaved scene)
- GetJob() now checks for initialization timeout on each poll
### OpenCodeConfigurator: Fix misleading comment
- Update TryLoadConfig() comment to accurately describe behavior when JSON is malformed
- Clarify that returning null causes Configure() to create fresh JObject, losing existing sections
- Note that preserving sections would require different recovery strategy
### run_tests.py: Improve exception handling
- Change _get_unity_project_path() to catch general Exception (not just AttributeError/KeyError)
- Re-raise asyncio.CancelledError to preserve task cancellation behavior
- Ensures registry failures are logged/swallowed while maintaining cancellation semantics
- Add lazy project path resolution: re-resolve project_path when nudging if initially None
- Fixes multi-instance support when registry becomes ready after polling starts
### conftest.py: Future-proof pytest compatibility
- Change item.fspath to item.path in pytest_collection_modifyitems hook
- item.path is pytest 7.0.0+ replacement for deprecated fspath
- Prevents future compatibility issues with newer pytest versions
## Testing
- All 502 Python tests pass
- Verified job state transitions with timeout logic
- Confirmed exception handling preserves cancellation semantics
Co-Authored-By: Claude Haiku 4.5 <noreply@anthropic.com>
* fix: Mark slow process inspection tests as [Explicit]
ProcessDetectorTests and ProcessTerminatorTests execute subprocess commands
(ps, lsof, tasklist, wmic) which can be slow on macOS, especially during
full test suite runs. These tests were blocking other tests from progressing
and causing excessive focus nudging attempts.
Marking both test classes as [Explicit] excludes them from normal test runs
and allows them to be run separately when needed for process detection validation.
Fixes: Tests taking 1+ minute and triggering focus nudge spam
Co-Authored-By: Claude Haiku 4.5 <noreply@anthropic.com>
* fix: Only increment consecutive nudges counter after focus attempt
Move _consecutive_nudges increment to after verifying the focus attempt,
rather than before. This ensures the counter only reflects actual nudge
attempts, not potential nudges that were rate-limited or skipped.
Fixes CodeRabbit issue: Counter was incrementing even if _focus_app
failed or activation didn't complete, leading to unnecessarily long
backoff intervals on subsequent failed attempts.
Co-Authored-By: Claude Haiku 4.5 <noreply@anthropic.com>
* fix: Address remaining CodeRabbit feedback
## Changes
### McpConnectionSection.cs
- Updated stale comment about stdio selection to correctly reference EditorConfigurationCache as source of truth
### find_gameobjects.py
- Removed unused AliasChoices import (never effective with FastMCP function signatures)
- Removed validation_alias decorations from Field definitions (FastMCP uses Python parameter names only)
### focus_nudge.py
- Updated _get_current_focus_duration to use configurable _DEFAULT_FOCUS_DURATION_S instead of hardcoded values
- Durations now scale proportionally from environment-configured default (base, base+2s, base+5s, base+9s)
- Ensures UNITY_MCP_NUDGE_DURATION_S environment variable is actually respected
### test_core_infrastructure_characterization.py
- Removed unused monkeypatch parameter from mock_telemetry_config fixture
- Added explicit fixture references in tests using mock_telemetry_config to suppress unused parameter warnings
- Moved CustomError class definition to test method scope for proper exception type checking in pytest.raises
## Testing
- All 502 Python tests pass
- No regressions in existing functionality
Co-Authored-By: Claude Haiku 4.5 <noreply@anthropic.com>
* fix: Final CodeRabbit feedback - VFX and telemetry hardening
## Changes
### VfxGraphAssets.cs
- FindTemplate: Convert asset paths to absolute filesystem paths before returning
(AssetDatabase.GUIDToAssetPath returns "Assets/...", now converts to full paths)
- FindTemplate/SetVfxAsset: Add path traversal validation to reject ".." sequences,
absolute paths, and backslashes; verify normalized paths don't escape Assets folder
using canonical path comparison
### VfxGraphWrite.cs
- SetParameter<T>: Guard valueToken.ToObject<T>() with try/catch for JsonException
and InvalidCastException; return error response instead of crashing
### focus_nudge.py
- Move _last_nudge_time and _consecutive_nudges updates to only occur after
successful _focus_app() call (prevents backoff advancing on failed attempts)
- _get_current_focus_duration: Scale base durations (3,5,8,12) proportionally by
ratio of configured UNITY_MCP_NUDGE_DURATION_S to default 3.0 seconds
(e.g., if env var = 6.0, durations become 6,10,16,24 seconds)
### test_core_infrastructure_characterization.py
- test_telemetry_collector_records_event: Mock threading.Thread to prevent worker
from consuming queued events during test assertion
- reset_telemetry fixture: Call core.telemetry.reset_telemetry() function to
properly shut down worker threads instead of just setting _telemetry_collector = None
## Testing
- All 502 Python tests pass
- Telemetry tests no longer flaky
- No regressions in existing functionality
Co-Authored-By: Claude Haiku 4.5 <noreply@anthropic.com>
* cleanup: Remove orphaned .meta files for deleted empty folders
Removed .meta files for folders that were previously deleted, preventing Unity warnings about missing directories.
* feat: Add dict/hex format support for vectors and colors
Add support for intuitive parameter formats that LLMs commonly use:
- Dict vectors: position={x:0, y:1, z:2}
- Dict colors: color={r:1, g:0, b:0, a:1}
- Hex colors: #RGB, #RRGGBB, #RRGGBBAA
- Tuple strings: (x, y, z) and (r, g, b, a)
Centralized normalization in utils.py with normalize_vector3() and
normalize_color() functions. Removed ~200 lines of duplicate code.
Updated type annotations to accept dict format in Pydantic schema.
* Fix VFX graph asset handling and harden CI GO merge
* Fix VFX graph asset handling and harden CI GO merge
* Deduplicate VFX template listing
* Avoid duplicate GO fragment merges
* Harden test job handling and tool validation
* Relax VFX version checks and harden VFX tools
---------
Co-authored-by: Claude Haiku 4.5 <noreply@anthropic.com>
|
|
|
|
17c6a36c8d
|
feat: Add beta server mode with PyPI pre-release support (#640)
* feat: add TestPyPI toggle for pre-release server package testing - Add UseTestPyPI editor preference key - Add TestPyPI toggle to Advanced settings UI with tooltip - Configure uvx to use test.pypi.org when TestPyPI mode enabled - Skip version pinning in TestPyPI mode to get latest pre-release - Update ConfigJsonBuilder to handle TestPyPI index URL * Update .meta file * fix: Use PyPI pre-release versions instead of TestPyPI for beta server TestPyPI has polluted packages (broken httpx, mcp, fastapi) that cause server startup failures. Switch to publishing beta versions directly to PyPI as pre-releases (e.g., 9.3.0b20260127). Key changes: - beta-release.yml: Publish to PyPI instead of TestPyPI, use beta suffix - Use --prerelease explicit with version specifier (>=0.0.0a0) to only get prereleases of our package, not broken dependency prereleases - Default "Use Beta Server" toggle to true on beta branch - Rename UI label from "Use TestPyPI" to "Use Beta Server" - Add UseTestPyPI to EditorPrefsWindow known prefs - Add search field and refresh button to EditorPrefsWindow Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com> * feat: Add beta mode indicator to UI badge and server version logging - Show "β" suffix on version badge when beta server mode is enabled - Badge updates dynamically when toggle changes - Add server version to startup log: "MCP for Unity Server v9.2.0 starting up" - Add version field to /health endpoint response Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com> * refactor: Rename UseTestPyPI to UseBetaServer and fix EditorPrefs margin - Rename EditorPref key from UseTestPyPI to UseBetaServer for clarity - Rename all related variables and UXML element names - Increase bottom margin on EditorPrefs search bar to prevent clipping first entry Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com> * refactor: Address code review feedback - Centralize beta server uvx args in AssetPathUtility.GetBetaServerFromArgs() to avoid duplication between HTTP and stdio transports - Cache server version at startup instead of calling get_package_version() on every /health request - Add robustness to beta version parsing in workflow: strip existing pre-release suffix and validate X.Y.Z format before parsing Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com> * Prioritize explicit fromUrl override and optimize search filter - GetBetaServerFromArgs/GetBetaServerFromArgsList now check for explicit GitUrlOverride before applying beta server mode, ensuring local dev paths and custom URLs are honored - EditorPrefsWindow search filter uses IndexOf with OrdinalIgnoreCase instead of ToLowerInvariant().Contains() for fewer allocations Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com> --------- Co-authored-by: Marcus Sanatan <msanatan@gmail.com> Co-authored-by: Claude Sonnet 4.5 <noreply@anthropic.com> |
|
|
|
feb9e3b6b5
|
Update the manual commands (#638)
update readme to use the uvx command |
|
|
|
17eb171e31
|
feat: replace prefab stage actions with headless modify_contents (#635)
Removes stage-based prefab editing (open_stage, close_stage, save_open_stage) in favor of headless modify_contents action that uses PrefabUtility.LoadPrefabContents for reliable automated workflows without UI dialogs. Changes: - Remove open_stage, close_stage, save_open_stage actions - Add modify_contents action for headless prefab editing - Support targeting objects by name or path (e.g., "Turret/Barrel") - Support transform, tag, layer, setActive, name, parent, components operations - Skip saving when no modifications made (avoids unnecessary asset writes) - Delete PrefabStage.cs resource (no longer needed) - Update Python tool description to remove "stages" reference - Consolidate tests from 29 to 14 (covers complex prefabs, reparenting, hierarchy loop guard) Co-authored-by: Claude Opus 4.5 <noreply@anthropic.com> |
|
|
|
9299828eb5
|
Add meta file for PrefabUtilityHelper (#630) | |
|
|
ed11e30b47
|
fix: prefab stage dirty flag, root rename, test fix, and prefab resources (#627)
- Mark prefab stage scene as dirty when manage_components adds/removes/
modifies components, ensuring save_open_stage correctly detects changes
- When renaming the root GameObject of an open prefab stage, also rename
the prefab asset file to match, preventing Unity's "file name must
match" dialog from interrupting automated workflows
- Fix ManagePrefabsCrudTests cleanup order: delete NestedContainer.prefab
before ChildPrefab.prefab to avoid missing prefab reference errors
- Remove incorrect LogAssert.Expect that expected an error that doesn't
occur in the test scenario
- Add new prefab MCP resources for inspecting prefabs:
- mcpforunity://prefab-api: Documentation for prefab resources
- mcpforunity://prefab/{path}: Get prefab asset info
- mcpforunity://prefab/{path}/hierarchy: Get full prefab hierarchy
Addresses #97 (Prefab Editor Inspection & Modification Support)
Co-authored-by: Claude Opus 4.5 <noreply@anthropic.com>
|
|
|
|
300a745bf2
|
feat: Prefab Feature Updates (#611)
* feat: Add prefab read operations (get_info, get_hierarchy, list_prefabs) - Add get_info: retrieve prefab metadata (GUID, type, components, child count, variant info) - Add get_hierarchy: get prefab internal structure with pagination support - Add list_prefabs: search prefabs in project with optional name filtering - Extract PrefabUtilityHelper class for reusable prefab utility methods - Update Python tool descriptions and parameter documentation Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * fix: Use correct API to save prefab stage changes Replace PrefabUtility.SaveAsPrefabAsset (for creating new prefabs) with EditorSceneManager.SaveScene to properly save stage modifications. This fixes the issue where component additions were lost after closing the prefab stage. Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * refactor: improve code quality and error handling - Add pagination constants (DefaultPageSize, MaxPageSize) - Extract SaveAndRefreshStage helper to reduce duplication - Change all user-facing messages to English - Add REQUIRED_PARAMS validation in Python - Split path parameter into prefab_path and folder_path for clarity - Improve error handling with specific exception types Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * feat: Remove list_prefabs action and update related documentation * feat: Enhance prefab management with detailed parameter descriptions and new unlinking option * feat: Simplify prefab creation logic and unify logging for asset replacement * feat: Update SaveStagePrefab method to use SetDirty and SaveAssets for prefab stage saving * feat: Add PrefabUtilityHelper class with utility methods for prefab asset management * feat: Refactor action constants and enhance parameter validation in prefab management * feat: Update ValidateSourceObjectForPrefab method to remove replaceExisting parameter and simplify validation logic * fix: Fix searchInactive parameter and improve prefab management - Fix searchInactive not working correctly for child objects - Improve error message accuracy for object not found - Use Application.dataPath for reliable directory path resolution * feat: Add path validation and security checks for prefab operations * feat: Remove pagination from GetHierarchy method and simplify prefab retrieval * feat: Remove mode parameter from prefab management functions to simplify usage * fix: Improve path validation and replace logic in prefab management * feat: Enhance prefab management by adding nesting depth and parent prefab path retrieval * fix: resolve Unknown pseudo class last-child USS warnings Unity UI Toolkit does not support the :last-child pseudo-class. Replace it with a .section-last class that is applied programmatically to the last section in each .section-stack container. Also moves the Configure All Detected Clients button to the bottom of the Client Configuration section and makes it auto-width. Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * fix: improve prefab stage save for automated workflows - Add force parameter to save_open_stage for automated workflows where isDirty may not be correctly set - Use PrefabUtility.SaveAsPrefabAsset for dialog-free saving - Mark prefab stage scene dirty when modifying GameObjects in prefab mode - Skip save when no changes and force=false (prevents false dirty flag) The force parameter ensures reliable saving in CI/automation scenarios where Unity dirty tracking may be inconsistent with programmatic changes. Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * Update prefab.py * refactor: remove unnecessary blank line before create function * feat: add info and hierarchy commands to prefab CLI for enhanced prefab management * feat: enhance prefab management with comprehensive CRUD tests and ensure dirty state tracking --------- Co-authored-by: Claude Opus 4.5 <noreply@anthropic.com> Co-authored-by: David Sarno <david@lighthaus.us> |
|
|
|
bb56f78ad3
|
Token Optimization for VFX (#626)
* Token Optimization for VFX * Update VFXs * Small fix based on AI feedback * Update ManageVFX.cs |
|
|
|
2eb26b85cf
|
fix: resolve Unknown pseudo class last-child USS warnings (#624)
Unity UI Toolkit does not support the :last-child pseudo-class. Replace it with a .section-last class that is applied programmatically to the last section in each .section-stack container. Also moves the Configure All Detected Clients button to the bottom of the Client Configuration section and makes it auto-width. Co-authored-by: Claude Opus 4.5 <noreply@anthropic.com> |
|
|
|
e00b2aed4f
|
[FEATURE] Procedural Texture2D/Sprite Generation (#621)
* Update for Texture2D/Sprite Generation Given the choice to generate Texture2D based on patterns and color, also introduce pipeline to turn Texture2D direct to Sprite. Update CLI command to include this too. * Texture Size Set Set texture size to 1024X1024 to avoid too large texture set * Add image input * Update to release direct error with large tex2d * Fix for AI advice * Update on action fetch line |
|
|
|
67dda7f9cc
|
fix: improve manage_scene screenshot capture (#600)
* fix: improve manage_scene screenshot capture * fix: address PR review feedback for screenshot capture - Gate pre-2022 ScreenCapture fallback warning to log only once - Downgrade warning to Debug.Log to reduce log noise - Refactor path-building into shared PrepareCaptureResult() helper - Add conditional logging to catch blocks in BestEffortPrepareGameViewForScreenshot - Add timeout/failure logging to ScheduleAssetImportWhenFileExists - Fix grammar in README-DEV.md * fix(unity): resolve screenshot import callback type + FindObjectsOfType deprecation * chore: bump version to 9.2.0 * Update ManageScene.cs * Update ScreenshotUtility.cs * Update error logging in ManageScene.cs --------- Co-authored-by: Marcus Sanatan <msanatan@gmail.com> Co-authored-by: GitHub Actions <actions@github.com> Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com> Co-authored-by: Shutong Wu <51266340+Scriptwonder@users.noreply.github.com> |
|
|
|
cb1a7dd2a1
|
feat: improve editor window UI + add transport mismatch warning (#613) | |
|
|
2c20ddcd10 | chore: bump version to 9.2.0 | |
|
|
30d5bc254e
|
feat: Add OpenCode (opencode.ai) client configurator (#608)
* feat: Add OpenCode (opencode.ai) client configurator Add support for the OpenCode CLI client with automatic configuration. - Create OpenCodeConfigurator implementing IClientConfigurator - Configure via ~/.config/opencode/opencode.json (XDG standard path) - Use McpConfigurationHelper for atomic file writes and directory creation - Support both new config creation and merging with existing config Co-Authored-By: akshay-kiddopia <akshay@kiddopia.com> Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * fix: Address code review feedback for OpenCodeConfigurator - Add TryLoadConfig() helper to consolidate file read/parse logic - Handle JsonException separately (log warning, return empty object to overwrite) - Wrap Configure() in try/catch to prevent crashes, set McpStatus.Error on failure - Respect XDG_CONFIG_HOME environment variable per XDG Base Directory spec Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> --------- Co-authored-by: akshay-kiddopia <akshay@kiddopia.com> Co-authored-by: Claude Opus 4.5 <noreply@anthropic.com> |
|
|
|
3ef2d37790 | chore: bump version to 9.1.0 | |
|
|
7f44e4b53e
|
Add CLI (#606)
* feat: Add CLI for Unity MCP server - Add click-based CLI with 15+ command groups - Commands: gameobject, component, scene, asset, script, editor, prefab, material, lighting, ui, audio, animation, code - HTTP transport to communicate with Unity via MCP server - Output formats: text, json, table - Configuration via environment variables or CLI options - Comprehensive usage guide and unit tests * Update based on AI feedback * Fixes main.py error * Update for further error fix * Update based on AI * Update script.py * Update with better coverage and Tool Readme * Log a message with implicit URI changes Small update for #542 * Minor fixes (#602) * Log a message with implicit URI changes Small update for #542 * Log a message with implicit URI changes Small update for #542 * Add helper scripts to update forks * fix: improve HTTP Local URL validation UX and styling specificity - Rename CSS class from generic "error" to "http-local-url-error" for better specificity - Rename "invalid-url" class to "http-local-invalid-url" for clarity - Disable httpServerCommandField when URL is invalid or transport not HTTP Local - Clear field value and tooltip when showing validation errors - Ensure field is re-enabled when URL becomes valid * Docker mcp gateway (#603) * Log a message with implicit URI changes Small update for #542 * Update docker container to default to stdio Replaces #541 * fix: Rider config path and add MCP registry manifest (#604) - Fix RiderConfigurator to use correct GitHub Copilot config path: - Windows: %LOCALAPPDATA%\github-copilot\intellij\mcp.json - macOS: ~/Library/Application Support/github-copilot/intellij/mcp.json - Linux: ~/.config/github-copilot/intellij/mcp.json - Add mcp.json for GitHub MCP Registry support: - Enables users to install via coplaydev/unity-mcp - Uses uvx with mcpforunityserver from PyPI * Use click.echo instead of print statements * Standardize whitespace * Minor tweak in docs * Use `wait` params * Unrelated but project scoped tools should be off by default * Update lock file * Whitespace cleanup * Update custom_tool_service.py to skip global registration for any tool name that already exists as a built‑in. * Avoid silently falling back to the first Unity session when a specific unity_instance was requested but not found. If a client passes a unity_instance that doesn’t match any session, this code will still route the command to the first available session, which can send commands to the wrong project in multi‑instance environments. Instead, when a unity_instance is provided but no matching session_id is found, return an error (e.g. 400/404 with "Unity instance '' not found") and only default to the first session when no unity_instance was specified. Co-authored-by: sourcery-ai[bot] <58596630+sourcery-ai[bot]@users.noreply.github.com> * Update docs/CLI_USAGE.md Co-authored-by: sourcery-ai[bot] <58596630+sourcery-ai[bot]@users.noreply.github.com> * Updated the CLI command registration to only swallow missing optional modules and to surface real import-time failures, so broken command modules don’t get silently ignored. * Sorted __all__ alphabetically to satisfy RUF022 in __init__.py. * Validate --params is a JSON object before merging. Co-authored-by: coderabbitai[bot] <136622811+coderabbitai[bot]@users.noreply.github.com> --------- Co-authored-by: Shutong Wu <51266340+Scriptwonder@users.noreply.github.com> Co-authored-by: dsarno <david@lighthaus.us> Co-authored-by: sourcery-ai[bot] <58596630+sourcery-ai[bot]@users.noreply.github.com> Co-authored-by: coderabbitai[bot] <136622811+coderabbitai[bot]@users.noreply.github.com> |
|
|
|
f54b1cb552
|
fix: Rider config path and add MCP registry manifest (#604)
- Fix RiderConfigurator to use correct GitHub Copilot config path: - Windows: %LOCALAPPDATA%\github-copilot\intellij\mcp.json - macOS: ~/Library/Application Support/github-copilot/intellij/mcp.json - Linux: ~/.config/github-copilot/intellij/mcp.json - Add mcp.json for GitHub MCP Registry support: - Enables users to install via coplaydev/unity-mcp - Uses uvx with mcpforunityserver from PyPI |
|
|
|
810d756be9
|
Minor fixes (#602)
* Log a message with implicit URI changes Small update for #542 * Log a message with implicit URI changes Small update for #542 * Add helper scripts to update forks * fix: improve HTTP Local URL validation UX and styling specificity - Rename CSS class from generic "error" to "http-local-url-error" for better specificity - Rename "invalid-url" class to "http-local-invalid-url" for clarity - Disable httpServerCommandField when URL is invalid or transport not HTTP Local - Clear field value and tooltip when showing validation errors - Ensure field is re-enabled when URL becomes valid |
|
|
|
1cc582636d
|
Add localhost setup feedback and remove UI issues (#587) | |
|
|
abd596ff4a
|
use localhost to ping server if server binds to 0.0.0.0 (#542)
Co-authored-by: vladimir.ivanov <terakriper@gmail.com> |
|
|
|
b69ee80da9
|
Project scoped tools (#596)
* feat: Add project-scoped tools flag to control custom tool registration behavior Add `--project-scoped-tools` CLI flag and `UNITY_MCP_PROJECT_SCOPED_TOOLS` environment variable to control whether custom tools are registered globally or scoped to specific Unity projects. Closes #416 * Add .meta file * feat: Add project-scoped tools toggle for local HTTP transport Add UI toggle in Connection section to control project-scoped tools flag when using HTTP Local transport. The toggle: - Defaults to enabled (true) - Persists state in EditorPrefs - Only displays when HTTP Local transport is selected - Automatically appends `--project-scoped-tools` flag to uvx server command - Updates manual config display when toggled * Update Server/src/services/custom_tool_service.py Co-authored-by: sourcery-ai[bot] <58596630+sourcery-ai[bot]@users.noreply.github.com> * Pass project_scoped_tools flag directly without environment variable conversion Remove unnecessary environment variable conversion for project_scoped_tools flag. * fix: Improve error handling and logging in global custom tool registration Split exception handling to distinguish between expected RuntimeError (service not initialized) and unexpected errors. --------- Co-authored-by: sourcery-ai[bot] <58596630+sourcery-ai[bot]@users.noreply.github.com> |
|
|
|
8eb684ddc2
|
fix: comprehensive performance optimizations, claude code config, and stability improvements (issue #577) (#595)
* fix: reduce per-frame GC allocations causing editor hitches (issue #577) Eliminate memory allocations that occurred every frame, which triggered garbage collection spikes (~28ms) approximately every second. Changes: - EditorStateCache: Skip BuildSnapshot() entirely when state unchanged (check BEFORE building). Increased poll interval from 0.25s to 1.0s. Cache DeepClone() results to avoid allocations on GetSnapshot(). - TransportCommandDispatcher: Early exit before lock/list allocation when Pending.Count == 0, eliminating per-frame allocations when idle. - StdioBridgeHost: Same early exit pattern for commandQueue. - MCPForUnityEditorWindow: Throttle OnEditorUpdate to 2-second intervals instead of every frame, preventing expensive socket checks 60+/sec. Fixes GitHub issue #577: High performance impact even when MCP server is off * fix: prevent multiple domain reloads when calling refresh_unity (issue #577) Root Cause: - send_command() had a hardcoded retry loop (min 6 attempts) on connection errors - Each retry resent the refresh_unity command, causing Unity to reload 6 times - retry_on_reload=False only controlled reload-state retries, not connection retries The Fix: 1. Unity C# (MCPForUnity/Editor): - Added --reinstall flag to uvx commands in dev mode - Ensures local development changes are picked up by uvx/Claude Code - Applies to all client configurators (Claude Code, Codex, etc.) 2. Python Server (Server/src): - Added max_attempts parameter to send_command() - Pass max_attempts=0 when retry_on_reload=False - Fixed type handling in refresh_unity.py (handle MCPResponse objects) - Added timeout to connection error recovery conditions - Recovery logic now returns success instead of error to prevent client retries Changes: - MCPForUnity/Editor: Added --reinstall to dev mode uvx commands - Server/refresh_unity.py: Fixed type handling, improved error recovery - Server/unity_connection.py: Added max_attempts param, disable retries when retry_on_reload=False Result: refresh_unity with compile=request now triggers only 1 domain reload instead of 6 Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com> * fix: UI and server stability improvements Unity Editor (C#): - Fix "Resuming..." stuck state when manually clicking End Session Clear ResumeStdioAfterReload and ResumeHttpAfterReload flags in OnConnectionToggleClicked and EndOrphanedSessionAsync to prevent UI from getting stuck showing "Resuming..." with disabled button - Remove unsupported --reinstall flag from all uvx command builders uvx does not support --reinstall and shows warning when used Use --no-cache --refresh instead for dev mode cache busting Python Server: - Add "aborted" to connection error patterns in refresh_unity Handle WinError 10053 (connection aborted) gracefully during Unity domain reload, treating it as expected behavior - Add WindowsSafeRotatingFileHandler to suppress log rotation errors Windows file locking prevents log rotation when file is open by another process; catch PermissionError to avoid noisy stack traces - Fix packaging: add py-modules = ["main"] to pyproject.toml setuptools.packages.find only discovers packages (directories with __init__.py), must explicitly list standalone module files Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com> * docs: improve refresh_unity connection loss handling documentation Add detailed comments and logging to clarify why connection loss during compile is treated as success (expected domain reload behavior, not failure). This addresses PR feedback about potentially masking real connection errors. The logic is intentional and correct: - Connection loss only treated as success when compile='request' - Domain reload causing disconnect is expected Unity behavior - Subsequent wait_for_ready loop validates Unity becomes ready - Prevents multiple domain reload loops (issue #577) Added logging for observability: - Info log when expected disconnect detected - Warning log for non-recoverable errors Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com> * fix: add missing logger import in refresh_unity Missing logger import causes NameError at runtime when connection loss handling paths are triggered (lines 82 and 91). Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com> * fix: address code review feedback on thread safety and semantics Addresses four issues raised in code review: 1. EditorStateCache.GetSnapshot() - Remove shared cached clone - Revert to always returning fresh DeepClone() to prevent mutation bugs - Main GC optimization remains: state-change detection prevents unnecessary _cached rebuilds (the expensive operation) - Removed _cachedClone and _cachedCloneSequence fields 2. refresh_unity.py - Fix blocking reason terminology mismatch - Changed "asset_refresh" to "asset_import" to match activityPhase values from EditorStateCache.cs - Ensures asset import is correctly detected as blocking state 3. TransportCommandDispatcher - Fix unsynchronized Count access - Moved Pending.Count check inside PendingLock - Prevents data races and InvalidOperationException from concurrent dictionary access 4. StdioBridgeHost - Fix unsynchronized Count access - Moved commandQueue.Count check inside lockObj - Ensures all collection access is properly serialized All changes maintain the GC allocation optimizations while fixing thread safety violations and semantic contract changes. Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com> * fix: address code review feedback on thread safety and timeout handling - refresh_unity.py: Track readiness explicitly and return failure on timeout instead of silently returning success when wait loop exits without confirming ready_for_tools=true - McpClientConfiguratorBase.cs: Add thread safety guard for Configure() call in CheckStatusWithProjectDir(). Changed default attemptAutoRewrite to false and added runtime check to prevent calling Configure() from background threads. Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> --------- Co-authored-by: Claude Sonnet 4.5 <noreply@anthropic.com> |
|
|
|
aaf6308b33
|
fix: Add special handling for UIDocument serialization to prevent infinite loops (#586)
* fix: search inactive objects when setActive=true in modify When trying to activate an inactive GameObject via manage_gameobject modify with setActive=true, the lookup would fail because inactive objects were not included in the search by default. Now automatically sets searchInactive=true when setActive=true is specified, allowing inactive objects to be found and activated. * fix: Add special handling for UIDocument serialization to prevent infinite loops (#585) UIDocument.rootVisualElement contains circular parent/child references that can cause infinite serialization loops. This adds special handling similar to Transform and Camera components. The fix: - Safely serializes panelSettings, visualTreeAsset, sortingOrder, enabled, parentUI - Explicitly skips rootVisualElement to prevent circular reference issues - Includes a note explaining why rootVisualElement is skipped Tested on Unity 2021.3 and Unity 6.3. * refactor: Extract SerializeAssetReference helper and align UIDocument structure - Add SerializeAssetReference() helper for consistent asset reference serialization - UIDocument now uses same return structure as Camera (typeName, instanceID, properties) - Reduces code duplication in special-case handlers - Enhanced test coverage to verify structure matches Camera pattern * fix: Handle UIDocument subclasses and add negative assertion for rootVisualElement Address code review feedback: - Add IsOrDerivedFrom() helper to detect UIDocument and any subclasses by walking the base-type chain, ensuring derived types also get special-case handling - Add negative assertion verifying rootVisualElement is NOT in serialized output |
|
|
|
f4e441c563
|
fix: Filter EditorApplication.isCompiling false positives in Play mode (#582)
EditorApplication.isCompiling can return true in Play mode even when Unity is not actually compiling, causing MCP tools to incorrectly return "busy/compiling" errors. This adds a GetActualIsCompiling() helper that double-checks with CompilationPipeline.isCompiling via reflection when in Play mode to filter out these false positives. Fixes #549 |
|
|
|
8252e6debc
|
fix: search inactive objects when setActive=true in modify (#581)
When trying to activate an inactive GameObject via manage_gameobject modify with setActive=true, the lookup would fail because inactive objects were not included in the search by default. Now automatically sets searchInactive=true when setActive=true is specified, allowing inactive objects to be found and activated. |
|
|
|
e617753b4e
|
fix: Add Prefab Stage support for GameObject lookup (#573)
* Enhance Prefab Stage support in GameObject lookup and scene management Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * fix: unify path matching and restore fast path lookup --------- Co-authored-by: Claude Opus 4.5 <noreply@anthropic.com> |
|
|
|
2cdc386b55 | chore: bump version to 9.0.8 | |
|
|
322a3d1846
|
fix: resolve Claude Code HTTP Remote UV path override not being detected in System Requirements .#550
* fix: resolve UV path override not being detected in System Requirements Fixes #538 The System Requirements panel showed "UV Package Manager: Not Found" even when a valid UV path override was configured in Advanced Settings. Root cause: PlatformDetectorBase.DetectUv() only searched PATH with bare command names ("uvx", "uv") and never consulted PathResolverService which respects the user's override setting. Changes: - Refactor DetectUv() to use PathResolverService.GetUvxPath() which checks override path first, then system PATH, then falls back to "uvx" - Add TryValidateUvExecutable() to verify executables by running --version instead of just checking File.Exists - Prioritize PATH environment variable in EnumerateUvxCandidates() for better compatibility with official uv install scripts - Fix process output read order (ReadToEnd before WaitForExit) to prevent potential deadlocks Co-Authored-By: ChatGLM 4.7 <noreply@zhipuai.com> * fix: improve uv/uvx detection robustness on macOS and Linux - Read both stdout and stderr when validating uv/uvx executables - Respect WaitForExit timeout return value instead of ignoring it - Fix version parsing to handle extra tokens like "(Homebrew 2025-01-01)" - Resolve bare commands ("uv"/"uvx") to absolute paths after validation - Rename FindExecutableInPath to FindUvxExecutableInPath for clarity Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * refactor: unify process execution with ExecPath.TryRun and add Windows PATH augmentation Replace direct Process.Start calls with ExecPath.TryRun across all platform detectors. This change: - Fixes potential deadlocks by using async output reading - Adds proper timeout handling with process termination - Removes redundant fallback logic and simplifies version parsing - Adds Windows PATH augmentation with common uv, npm, and Python installation paths Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * fix: improve version parsing to handle both spaces and parentheses The version extraction logic now properly handles outputs like: - "uvx 0.9.18" -> "0.9.18" - "uvx 0.9.18 (hash date)" -> "0.9.18" - "uvx 0.9.18 extra info" -> "0.9.18" Uses Math.Min to find the first occurrence of either space or parenthesis. Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * refactor: improve platform detectors with absolute path resolution - Add absolute path resolution in TryValidatePython and TryValidateUvWithPath for better UI display - Fix BuildAugmentedPath to avoid PATH duplication - Add comprehensive comments for version parsing logic - Ensure cross-platform consistency across all three detectors - Fix override path validation logic with clear state handling - Fix platform detector path resolution and Python version detection - Use UserProfile consistently in GetClaudeCliPath instead of Personal - All platforms now use protected BuildAugmentedPath method This change improves user experience by displaying full paths in the UI while maintaining robust fallback behavior if path resolution fails. Co-Authored-By: GLM4.7 <noreply@zhipuai.com> * fix: improve error handling in PathResolverService by logging exceptions * Remove .meta files added after fork and update .gitignore * Update .gitignore * save .meta * refactor: unify uv/uvx naming and path detection across platforms - Rename TryValidateUvExecutable -> TryValidateUvxExecutable for consistency - Add cross-platform FindInPath() helper in ExecPath.cs - Remove platform-specific where/which implementations in favor of unified helper - Add Windows-specific DetectUv() override with enhanced uv/uvx detection - Add WinGet shim path support for Windows uvx installation - Update UI labels: "UV Path" -> "UVX Path" - Only show uvx path status when override is configured Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * fix: improve validation light(uvxPathStatus) logic for UVX path overrides and system paths * refactor: streamline UV version validation and unify path detection methods across platform detectors * fix: add type handling for Claude Code client in config JSON builder * fix: correct command from 'uvx' to 'uv' for Python version listing in WindowsPlatformDetector * feat: add uvx path fallback with warning UI - When override path is invalid, automatically fall back to system path - Add HasUvxPathFallback flag to track fallback state - Show yellow warning indicator when using fallback - Display clear messages for invalid override paths - Updated all platform detectors (Windows, macOS, Linux) to support fallback logic * refactor: remove GetDetails method from PlatformDetectorBase * Update ExecPath.cs update Windows Path lookup --------- Co-authored-by: ChatGLM 4.7 <noreply@zhipuai.com> Co-authored-by: Claude Opus 4.5 <noreply@anthropic.com> Co-authored-by: Shutong Wu <51266340+Scriptwonder@users.noreply.github.com> |
|
|
|
cf177b50af
|
Replace asmdef GUID references (#564) | |
|
|
98796264ef |
Remove unnecessary allowNudge guard from PR #558
The allowNudge parameter was attempting to fix Unity 6 compilation loops by skipping QueuePlayerLoopUpdate, but this was not the root cause. The actual issue (fixed by #559's Unity 6+ preprocessor guard) is that EditorApplication.update callbacks don't survive domain reloads properly in Unity 6+. Since #559 skips WaitForUnityReadyAsync entirely on Unity 6+ when compile is requested, the allowNudge guard is redundant and can be removed. Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com> |
|
|
|
87d0f1d422
|
fix: Prevent infinite compilation loop in Unity 6 when using wait_for_ready (#559)
* fix: Prevent infinite compilation loop in Unity 6 when using wait_for_ready Skip WaitForUnityReadyAsync when compileRequested is true. The EditorApplication.update polling doesn't survive domain reloads properly in Unity 6, causing infinite compilation loops. When compilation is requested, return immediately and let the client poll editor_state resource instead. Fixes #557 Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * fix: Use actuallyWaited for hint message consistency Address code review feedback: when compile was requested and we skip WaitForUnityReadyAsync, the hint should correctly indicate that the client needs to poll editor_state, not claim the editor is ready. Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * refactor: Gate Unity 6 fix with version check and rename variable Address code review feedback: - Use UNITY_6000_0_OR_NEWER preprocessor directive to only apply the compilation wait bypass on Unity 6+, preserving original behavior for earlier versions - Rename actuallyWaited to shouldWaitForReady for clarity, as it represents the decision to wait rather than a post-hoc result Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> --------- Co-authored-by: Claude Opus 4.5 <noreply@anthropic.com> |
|
|
|
b6fa293cd3
|
Guard refresh wait nudge during compile (#558) | |
|
|
3f08ac0e43
|
Fix local HTTP server UI check (#556)
* Fix local HTTP server UI check * Fix connection section cleanup |
|
|
|
b611b268ef | chore: bump version to 9.0.7 | |
|
|
218f5f556d | chore: bump version to 9.0.6 | |
|
|
8df89587ec | chore: bump version to 9.0.5 | |
|
|
787e667c76 | chore: bump version to 9.0.4 | |
|
|
39bff4ae27
|
Fix PlayMode tests stalling when unfocused (python refresh utility), improve domain reload recovery and refresh tool (#554)
* Fix test job state management after domain reload - TestRunnerService.RunFinished: Always clean up job state even when _runCompletionSource is null (happens after PlayMode domain reload) - TestJobManager: Detect and clear stale jobs (5+ min without updates) on startup to recover from stuck state after domain reload - refresh_unity.py: Add "could not connect" to retryable errors when wait_for_ready=True, so connection failures during domain reload trigger waiting instead of immediate failure Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * Add focus nudge to handle OS-level throttling during PlayMode tests When Unity is unfocused, macOS App Nap (and similar OS features) can throttle the process, causing PlayMode tests to stall even with Unity No Throttling mode enabled. Changes: - Add ApplyNoThrottlingPreemptive() to TestRunnerNoThrottle for early throttle prevention before PlayMode Execute() - Add focus_nudge.py utility that temporarily focuses Unity and returns focus to the original app (supports macOS, Windows, Linux) - Integrate focus nudge into get_test_job polling - when tests appear stalled (unfocused + no progress for 10s), automatically nudge Unity Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * Fix code review issues in focus_nudge.py - Remove redundant time import (already imported at module level) - Escape window titles in PowerShell script to prevent injection - Remove unused Callable import Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * Improve focus nudge logging and fix skipped tests - Improve logging in focus_nudge.py: rate limit skip and focus return at INFO level - Improve logging in run_tests.py: show nudge completion status - Fix path resolution in test_logging_stdout.py and test_transport_framing.py - Add PlayMode tests to UnityMCPTests project for testing PlayMode runner Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * Add troubleshooting note about focus permission requests When running PlayMode tests with Unity in the background, the focus nudge feature may trigger OS permission prompts (especially on macOS for accessibility/automation). Document this expected behavior. Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> --------- Co-authored-by: Claude Opus 4.5 <noreply@anthropic.com> |
|
|
|
b874922cb0
|
Fix manage_components set_property for object references (#551) |