ha-mcp-dev 7.5.0.dev527__tar.gz → 7.5.0.dev529__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (111) hide show
  1. {ha_mcp_dev-7.5.0.dev527/src/ha_mcp_dev.egg-info → ha_mcp_dev-7.5.0.dev529}/PKG-INFO +1 -1
  2. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/pyproject.toml +1 -1
  3. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_addons.py +25 -8
  4. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_config_scenes.py +42 -41
  5. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529/src/ha_mcp_dev.egg-info}/PKG-INFO +1 -1
  6. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/LICENSE +0 -0
  7. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/MANIFEST.in +0 -0
  8. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/README.md +0 -0
  9. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/setup.cfg +0 -0
  10. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/__init__.py +0 -0
  11. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/__main__.py +0 -0
  12. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/_pypi_marker +0 -0
  13. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/_version.py +0 -0
  14. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/auth/__init__.py +0 -0
  15. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/auth/consent_form.py +0 -0
  16. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/auth/provider.py +0 -0
  17. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/client/__init__.py +0 -0
  18. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/client/rest_client.py +0 -0
  19. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/client/supervisor_client.py +0 -0
  20. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/client/websocket_client.py +0 -0
  21. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/client/websocket_listener.py +0 -0
  22. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/config.py +0 -0
  23. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/errors.py +0 -0
  24. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/py.typed +0 -0
  25. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/resources/skills-vendor/.claude/settings.json +0 -0
  26. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/resources/skills-vendor/.claude-plugin/marketplace.json +0 -0
  27. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/resources/skills-vendor/.claude-plugin/plugin.json +0 -0
  28. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/resources/skills-vendor/.github/ISSUE_TEMPLATE/skill-rca.md +0 -0
  29. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/resources/skills-vendor/AGENTS.md +0 -0
  30. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/resources/skills-vendor/CLAUDE.md +0 -0
  31. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/resources/skills-vendor/CONTRIBUTING.md +0 -0
  32. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/resources/skills-vendor/LICENSE +0 -0
  33. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/resources/skills-vendor/README.md +0 -0
  34. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/resources/skills-vendor/skills/home-assistant-best-practices/SKILL.md +0 -0
  35. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/resources/skills-vendor/skills/home-assistant-best-practices/evals/evals.json +0 -0
  36. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/resources/skills-vendor/skills/home-assistant-best-practices/references/automation-patterns.md +0 -0
  37. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/resources/skills-vendor/skills/home-assistant-best-practices/references/dashboard-cards.md +0 -0
  38. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/resources/skills-vendor/skills/home-assistant-best-practices/references/dashboard-guide.md +0 -0
  39. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/resources/skills-vendor/skills/home-assistant-best-practices/references/device-control.md +0 -0
  40. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/resources/skills-vendor/skills/home-assistant-best-practices/references/domain-docs.md +0 -0
  41. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/resources/skills-vendor/skills/home-assistant-best-practices/references/examples.yaml +0 -0
  42. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/resources/skills-vendor/skills/home-assistant-best-practices/references/helper-selection.md +0 -0
  43. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/resources/skills-vendor/skills/home-assistant-best-practices/references/safe-refactoring.md +0 -0
  44. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/resources/skills-vendor/skills/home-assistant-best-practices/references/template-guidelines.md +0 -0
  45. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/server.py +0 -0
  46. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/settings_ui.py +0 -0
  47. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/smoke_test.py +0 -0
  48. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/__init__.py +0 -0
  49. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/backup.py +0 -0
  50. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/best_practice_checker.py +0 -0
  51. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/device_control.py +0 -0
  52. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/enhanced.py +0 -0
  53. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/helpers.py +0 -0
  54. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/reference_validator.py +0 -0
  55. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/registry.py +0 -0
  56. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/smart_search.py +0 -0
  57. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_areas.py +0 -0
  58. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_blueprints.py +0 -0
  59. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_bug_report.py +0 -0
  60. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_calendar.py +0 -0
  61. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_camera.py +0 -0
  62. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_categories.py +0 -0
  63. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_code.py +0 -0
  64. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_config_automations.py +0 -0
  65. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_config_dashboards.py +0 -0
  66. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_config_entry_flow.py +0 -0
  67. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_config_helpers.py +0 -0
  68. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_config_scripts.py +0 -0
  69. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_energy.py +0 -0
  70. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_entities.py +0 -0
  71. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_filesystem.py +0 -0
  72. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_groups.py +0 -0
  73. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_hacs.py +0 -0
  74. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_history.py +0 -0
  75. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_integrations.py +0 -0
  76. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_labels.py +0 -0
  77. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_mcp_component.py +0 -0
  78. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_registry.py +0 -0
  79. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_resources.py +0 -0
  80. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_search.py +0 -0
  81. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_service.py +0 -0
  82. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_services.py +0 -0
  83. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_system.py +0 -0
  84. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_todo.py +0 -0
  85. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_traces.py +0 -0
  86. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_updates.py +0 -0
  87. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_utility.py +0 -0
  88. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_voice_assistant.py +0 -0
  89. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_yaml_config.py +0 -0
  90. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/tools_zones.py +0 -0
  91. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/tools/util_helpers.py +0 -0
  92. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/transforms/__init__.py +0 -0
  93. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/transforms/categorized_search.py +0 -0
  94. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/transforms/lite_docstrings.py +0 -0
  95. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/utils/__init__.py +0 -0
  96. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/utils/config_hash.py +0 -0
  97. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/utils/data_paths.py +0 -0
  98. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/utils/domain_handlers.py +0 -0
  99. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/utils/fuzzy_search.py +0 -0
  100. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/utils/kill_signal_diagnostics.py +0 -0
  101. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/utils/operation_manager.py +0 -0
  102. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/utils/python_sandbox.py +0 -0
  103. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp/utils/usage_logger.py +0 -0
  104. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp_dev.egg-info/SOURCES.txt +0 -0
  105. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp_dev.egg-info/dependency_links.txt +0 -0
  106. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp_dev.egg-info/entry_points.txt +0 -0
  107. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp_dev.egg-info/requires.txt +0 -0
  108. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/src/ha_mcp_dev.egg-info/top_level.txt +0 -0
  109. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/tests/__init__.py +0 -0
  110. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/tests/test_constants.py +0 -0
  111. {ha_mcp_dev-7.5.0.dev527 → ha_mcp_dev-7.5.0.dev529}/tests/test_env_manager.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: ha-mcp-dev
3
- Version: 7.5.0.dev527
3
+ Version: 7.5.0.dev529
4
4
  Summary: Home Assistant MCP Server - Complete control of Home Assistant through MCP
5
5
  Author-email: Julien <github@qc-h.net>
6
6
  License: MIT
@@ -4,7 +4,7 @@ build-backend = "setuptools.build_meta"
4
4
 
5
5
  [project]
6
6
  name = "ha-mcp-dev"
7
- version = "7.5.0.dev527"
7
+ version = "7.5.0.dev529"
8
8
  description = "Home Assistant MCP Server - Complete control of Home Assistant through MCP"
9
9
  readme = "README.md"
10
10
  requires-python = ">=3.13,<3.14"
@@ -1554,19 +1554,36 @@ async def _call_addon_api(
1554
1554
  "token has admin rights and try again."
1555
1555
  )
1556
1556
  elif response.status_code == 403:
1557
+ ports_dict = addon.get("network") or addon.get("ports") or {}
1558
+ # Ingress nginx in peer add-ons typically rejects cross-container
1559
+ # traffic; the direct-port fallback only works if the user mapped
1560
+ # a host port. Detect and call out the unmapped case explicitly.
1561
+ unmapped = sorted(k for k, v in ports_dict.items() if v is None)
1557
1562
  result["addon_config"] = {
1558
1563
  "options": addon.get("options"),
1559
- "ports": addon.get("network") or addon.get("ports"),
1564
+ "ports": ports_dict or None,
1560
1565
  "host_network": addon.get("host_network"),
1561
1566
  "ingress_port": addon.get("ingress_port"),
1562
1567
  }
1563
- result["suggestion"] = (
1564
- "This add-on is blocking direct connections (likely Nginx IP restriction). "
1565
- "Try using the 'port' parameter to connect to the add-on's direct access port "
1566
- "(see addon_config.ports above) with 'leave_front_door_open' enabled. "
1567
- "Example: ha_manage_addon(slug='...', path='...', port=<direct_port>). "
1568
- "The user may need to change add-on settings in the HA UI and restart the add-on."
1569
- )
1568
+ slug = addon.get("slug") or "<slug>"
1569
+ example_proto = unmapped[0] if unmapped else ""
1570
+ example_port = example_proto.split("/", 1)[0] if example_proto else ""
1571
+ if unmapped and example_port.isdigit():
1572
+ addon_label = addon.get("name") or slug
1573
+ result["suggestion"] = (
1574
+ f"Map {example_proto} to a host port in the HA UI "
1575
+ f"('{addon_label}' → Configuration → Network), restart the "
1576
+ f"add-on, then retry with ha_manage_addon(slug='{slug}', "
1577
+ f"path='...', port={example_port})."
1578
+ )
1579
+ else:
1580
+ result["suggestion"] = (
1581
+ "This add-on is blocking direct connections (likely Nginx IP restriction). "
1582
+ "Try using the 'port' parameter to connect to the add-on's direct access port "
1583
+ "(see addon_config.ports above) with 'leave_front_door_open' enabled. "
1584
+ "Example: ha_manage_addon(slug='...', path='...', port=<direct_port>). "
1585
+ "The user may need to change add-on settings in the HA UI and restart the add-on."
1586
+ )
1570
1587
 
1571
1588
  return result
1572
1589
 
@@ -32,6 +32,7 @@ from .helpers import (
32
32
  log_tool_usage,
33
33
  raise_tool_error,
34
34
  register_tool_methods,
35
+ validate_identifier_not_empty,
35
36
  )
36
37
  from .reference_validator import validate_config_references
37
38
  from .util_helpers import (
@@ -221,20 +222,20 @@ class ConfigSceneTools:
221
222
  # Issue #1168 R6 blocker 16: empty ``scene_id`` previously
222
223
  # surfaced as ``RESOURCE_NOT_FOUND`` with a misleading
223
224
  # `entities`-related suggestion. Pre-flight here so the caller
224
- # gets the actual problem.
225
- if not scene_id or not scene_id.strip():
226
- raise_tool_error(
227
- create_error_response(
228
- ErrorCode.VALIDATION_INVALID_PARAMETER,
229
- "scene_id must not be empty",
230
- suggestions=[
231
- "Pass a non-empty scene identifier (e.g. 'movie_night')",
232
- "Use ha_search_entities(domain_filter='scene') "
233
- "to find existing scene_ids",
234
- ],
235
- context={"scene_id": scene_id},
236
- )
237
- )
225
+ # gets the actual problem. Migrated to the shared
226
+ # ``validate_identifier_not_empty`` helper (#1314) message
227
+ # and ``context["scene_id"]`` key preserved for callers.
228
+ validate_identifier_not_empty(
229
+ scene_id,
230
+ "scene_id",
231
+ message="scene_id must not be empty",
232
+ suggestions=[
233
+ "Pass a non-empty scene identifier (e.g. 'movie_night')",
234
+ "Use ha_search_entities(domain_filter='scene') "
235
+ "to find existing scene_ids",
236
+ ],
237
+ context={"scene_id": scene_id},
238
+ )
238
239
  # Issue #1168 R3 blockers 3 + 6: unwrap the rest-client envelope
239
240
  # so the response carries the scene body directly (no nested
240
241
  # `success`/`scene_id`/`config` chain), and use the storage key
@@ -520,19 +521,19 @@ class ConfigSceneTools:
520
521
  # Issue #1168 R6 blocker 16: empty ``scene_id`` pre-flight before
521
522
  # any config dispatch — keeps the error code/message aligned with
522
523
  # the actual problem rather than the misleading
523
- # ``RESOURCE_NOT_FOUND`` from a downstream lookup.
524
- if not scene_id or not scene_id.strip():
525
- raise_tool_error(
526
- create_error_response(
527
- ErrorCode.VALIDATION_INVALID_PARAMETER,
528
- "scene_id must not be empty",
529
- suggestions=[
530
- "Pass a non-empty scene identifier (e.g. 'movie_night')",
531
- "For a fresh create, use a name-derived slug",
532
- ],
533
- context={"scene_id": scene_id},
534
- )
535
- )
524
+ # ``RESOURCE_NOT_FOUND`` from a downstream lookup. Migrated to
525
+ # the shared ``validate_identifier_not_empty`` helper (#1314)
526
+ # message and ``context["scene_id"]`` key preserved for callers.
527
+ validate_identifier_not_empty(
528
+ scene_id,
529
+ "scene_id",
530
+ message="scene_id must not be empty",
531
+ suggestions=[
532
+ "Pass a non-empty scene identifier (e.g. 'movie_night')",
533
+ "For a fresh create, use a name-derived slug",
534
+ ],
535
+ context={"scene_id": scene_id},
536
+ )
536
537
  # Validate mutual exclusivity of config and python_transform
537
538
  if config is not None and python_transform is not None:
538
539
  raise_tool_error(
@@ -942,20 +943,20 @@ class ConfigSceneTools:
942
943
  try:
943
944
  # Issue #1168 R6 blocker 16: empty ``scene_id`` pre-flight before
944
945
  # the resolver — keeps the error code/message aligned with the
945
- # actual problem.
946
- if not scene_id or not scene_id.strip():
947
- raise_tool_error(
948
- create_error_response(
949
- ErrorCode.VALIDATION_INVALID_PARAMETER,
950
- "scene_id must not be empty",
951
- suggestions=[
952
- "Pass a non-empty scene identifier (e.g. 'old_scene')",
953
- "Use ha_search_entities(domain_filter='scene') "
954
- "to find existing scene_ids",
955
- ],
956
- context={"scene_id": scene_id},
957
- )
958
- )
946
+ # actual problem. Migrated to the shared
947
+ # ``validate_identifier_not_empty`` helper (#1314) message
948
+ # and ``context["scene_id"]`` key preserved for callers.
949
+ validate_identifier_not_empty(
950
+ scene_id,
951
+ "scene_id",
952
+ message="scene_id must not be empty",
953
+ suggestions=[
954
+ "Pass a non-empty scene identifier (e.g. 'old_scene')",
955
+ "Use ha_search_entities(domain_filter='scene') "
956
+ "to find existing scene_ids",
957
+ ],
958
+ context={"scene_id": scene_id},
959
+ )
959
960
  # Issue #1168 R3 blocker 6: resolve once up-front so every later
960
961
  # callsite (entity_id resolver, delete call, response) uses the
961
962
  # storage key consistently — outer ``scene_id`` matches the
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: ha-mcp-dev
3
- Version: 7.5.0.dev527
3
+ Version: 7.5.0.dev529
4
4
  Summary: Home Assistant MCP Server - Complete control of Home Assistant through MCP
5
5
  Author-email: Julien <github@qc-h.net>
6
6
  License: MIT