spool_assignment_notifications.py 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. import logging
  2. from backend.app.core.database import async_session
  3. from backend.app.core.websocket import ws_manager
  4. from backend.app.models.printer import Printer
  5. from backend.app.models.spool_assignment import SpoolAssignment
  6. from backend.app.services.bambu_mqtt import PrinterState
  7. from backend.app.services.notification_service import notification_service
  8. from backend.app.services.printer_manager import printer_manager
  9. def _global_tray_from_assignment(ams_id: int, tray_id: int) -> int:
  10. """Convert an assignment tuple to Bambuddy global tray ID."""
  11. if ams_id in (254, 255):
  12. return 254 + tray_id
  13. if ams_id >= 128:
  14. return ams_id
  15. return ams_id * 4 + tray_id
  16. def _slot_label_from_global_tray(global_tray_id: int) -> str:
  17. """Return a human-readable slot label from a global tray ID."""
  18. if global_tray_id == 254:
  19. return "Ext-L"
  20. if global_tray_id == 255:
  21. return "Ext-R"
  22. if global_tray_id >= 128:
  23. return f"HT-{chr(65 + (global_tray_id - 128))}"
  24. ams_id = global_tray_id // 4
  25. tray_id = global_tray_id % 4
  26. return f"{chr(65 + ams_id)}{tray_id + 1}"
  27. def _tray_profile_and_color_for_global_id(state: PrinterState | None, global_tray_id: int) -> tuple[str, str]:
  28. """Resolve expected tray material/profile and color for a global tray ID from current printer state."""
  29. if not state or not state.raw_data:
  30. return ("Unknown", "Unknown")
  31. ams_raw = state.raw_data.get("ams", {})
  32. ams_units = ams_raw.get("ams", []) if isinstance(ams_raw, dict) else ams_raw if isinstance(ams_raw, list) else []
  33. vt_trays = state.raw_data.get("vt_tray", [])
  34. if not isinstance(vt_trays, list):
  35. vt_trays = []
  36. for tray in vt_trays:
  37. if not isinstance(tray, dict):
  38. continue
  39. if int(tray.get("id", -1)) == global_tray_id:
  40. profile = tray.get("tray_sub_brands") or tray.get("tray_type") or "Unknown"
  41. color = tray.get("tray_color") or "Unknown"
  42. return (profile, color)
  43. for ams in ams_units:
  44. if not isinstance(ams, dict):
  45. continue
  46. ams_id = int(ams.get("id", -1))
  47. trays = ams.get("tray", [])
  48. if not isinstance(trays, list):
  49. continue
  50. for tray in trays:
  51. if not isinstance(tray, dict):
  52. continue
  53. tray_id = int(tray.get("id", -1))
  54. candidate = ams_id if ams_id >= 128 else (ams_id * 4 + tray_id)
  55. if candidate == global_tray_id:
  56. profile = tray.get("tray_sub_brands") or tray.get("tray_type") or "Unknown"
  57. color = tray.get("tray_color") or "Unknown"
  58. return (profile, color)
  59. return ("Unknown", "Unknown")
  60. def _decode_mqtt_mapping_to_global_trays(mapping_raw: object) -> list[int]:
  61. """Decode printer MQTT mapping values into Bambuddy global tray IDs."""
  62. if not isinstance(mapping_raw, list) or not mapping_raw:
  63. return []
  64. decoded: list[int] = []
  65. for value in mapping_raw:
  66. try:
  67. if isinstance(value, int):
  68. encoded = value
  69. elif isinstance(value, str):
  70. encoded = int(value, 10)
  71. else:
  72. continue
  73. except ValueError:
  74. continue
  75. if encoded >= 65535:
  76. continue
  77. ams_hw_id = (encoded >> 8) & 0xFF
  78. slot = encoded & 0xFF
  79. if 0 <= ams_hw_id <= 3:
  80. decoded.append(ams_hw_id * 4 + (slot & 0x03))
  81. elif 128 <= ams_hw_id <= 135:
  82. decoded.append(ams_hw_id)
  83. elif ams_hw_id in (254, 255):
  84. decoded.append(255 if slot == 255 else 254)
  85. return decoded
  86. async def notify_missing_spool_assignments_on_print_start(
  87. printer_id: int,
  88. data: dict,
  89. logger: logging.Logger,
  90. ) -> None:
  91. """Send notification when print-start mapping references unassigned trays."""
  92. explicit_mapping = data.get("ams_mapping")
  93. explicit_values = (
  94. [value for value in explicit_mapping if isinstance(value, int)] if isinstance(explicit_mapping, list) else []
  95. )
  96. raw_mapping = data.get("raw_data", {}).get("mapping") if isinstance(data.get("raw_data"), dict) else None
  97. decoded_values = _decode_mqtt_mapping_to_global_trays(raw_mapping)
  98. mapping_values = explicit_values if explicit_values else decoded_values
  99. used_global_trays = {value for value in mapping_values if value >= 0}
  100. if not used_global_trays:
  101. return
  102. try:
  103. async with async_session() as db:
  104. printer = await db.get(Printer, printer_id)
  105. printer_name = printer.name if printer else f"Printer {printer_id}"
  106. assignments_result = await db.execute(
  107. SpoolAssignment.__table__.select().where(SpoolAssignment.printer_id == printer_id)
  108. )
  109. assignments = assignments_result.fetchall()
  110. assigned_global_trays = {
  111. _global_tray_from_assignment(assignment.ams_id, assignment.tray_id) for assignment in assignments
  112. }
  113. missing_global = sorted(used_global_trays - assigned_global_trays)
  114. if not missing_global:
  115. return
  116. state = printer_manager.get_status(printer_id)
  117. missing_slots = []
  118. for global_id in missing_global:
  119. profile, color = _tray_profile_and_color_for_global_id(state, global_id)
  120. missing_slots.append(
  121. {
  122. "slot": _slot_label_from_global_tray(global_id),
  123. "profile": profile,
  124. "color": color,
  125. }
  126. )
  127. await ws_manager.send_missing_spool_assignment(
  128. printer_id=printer_id,
  129. printer_name=printer_name,
  130. missing_slots=missing_slots,
  131. )
  132. await notification_service.on_print_missing_spool_assignment(
  133. printer_id=printer_id,
  134. printer_name=printer_name,
  135. missing_slots=missing_slots,
  136. db=db,
  137. )
  138. except Exception as e:
  139. logger.warning("Missing spool-assignment notification failed: %s", e)