spool_assignment_notifications.py 6.8 KB

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