test_failure_reason_derivation.py 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899
  1. """Regression tests for derive_failure_reason in backend.app.main.
  2. Ensures user-cancelled prints don't get archived as "Layer shift" — the bug
  3. seen on H2D where the firmware's cancel-sequence module-0x0C HMS was being
  4. matched by the old broad heuristic (`module == 0x0C → Layer shift`).
  5. """
  6. from __future__ import annotations
  7. import pytest
  8. from backend.app.main import derive_failure_reason
  9. # ---------------------------------------------------------------------------
  10. # Status-based reasons (no HMS lookup needed)
  11. # ---------------------------------------------------------------------------
  12. @pytest.mark.parametrize("status", ["aborted", "cancelled"])
  13. def test_user_cancel_status_yields_user_cancelled(status: str) -> None:
  14. assert derive_failure_reason(status, None) == "User cancelled"
  15. assert derive_failure_reason(status, []) == "User cancelled"
  16. def test_completed_status_returns_none() -> None:
  17. assert derive_failure_reason("completed", None) is None
  18. # ---------------------------------------------------------------------------
  19. # H2D regression: cancel-sequence HMS must not be labelled "Layer shift"
  20. # ---------------------------------------------------------------------------
  21. def test_h2d_cancel_module_0x0c_is_not_layer_shift() -> None:
  22. """0C00_001B is the H2D cancel-sequence echo, not a real layer-shift code.
  23. The old `module == 0x0C → Layer shift` heuristic mislabeled every user-cancel
  24. on H2D as a layer-shift failure. This pins that code to None.
  25. """
  26. h2d_cancel_hms = [
  27. {"code": "0x2001b", "attr": 0x0C000C00, "module": 0x0C, "severity": 1},
  28. {"code": "0x400c", "attr": 0x03002C0C, "module": 0x03, "severity": 3},
  29. ]
  30. assert derive_failure_reason("failed", h2d_cancel_hms) is None
  31. def test_unknown_module_0x0c_code_returns_none() -> None:
  32. """Any module-0x0C code we don't have an explicit short-code mapping for must
  33. leave failure_reason=None — being honest beats guessing."""
  34. unknown_hms = [{"code": "0x4099", "attr": 0x0C00_0000, "module": 0x0C, "severity": 2}]
  35. assert derive_failure_reason("failed", unknown_hms) is None
  36. # ---------------------------------------------------------------------------
  37. # Genuine failure modes still classified correctly
  38. # ---------------------------------------------------------------------------
  39. def test_real_layer_shift_short_code_detected() -> None:
  40. """0300_4057 ("Z-axis step loss") is a real layer-shift code from the wiki."""
  41. hms = [{"code": "0x4057", "attr": 0x0300_0000, "module": 0x03, "severity": 1}]
  42. assert derive_failure_reason("failed", hms) == "Layer shift"
  43. def test_real_filament_runout_short_code_detected() -> None:
  44. """07FF_8011 = external filament runout."""
  45. hms = [{"code": "0x8011", "attr": 0x07FF_0000, "module": 0x07, "severity": 2}]
  46. assert derive_failure_reason("failed", hms) == "Filament runout"
  47. def test_real_clogged_nozzle_short_code_detected() -> None:
  48. """0300_4006 = "The nozzle is clogged"."""
  49. hms = [{"code": "0x4006", "attr": 0x0300_0000, "module": 0x03, "severity": 1}]
  50. assert derive_failure_reason("failed", hms) == "Clogged nozzle"
  51. def test_first_matching_code_wins() -> None:
  52. """When multiple known codes are present, the first one in the list wins."""
  53. hms = [
  54. {"code": "0x4057", "attr": 0x0300_0000, "module": 0x03, "severity": 1}, # layer shift
  55. {"code": "0x8011", "attr": 0x07FF_0000, "module": 0x07, "severity": 2}, # filament runout
  56. ]
  57. assert derive_failure_reason("failed", hms) == "Layer shift"
  58. def test_failed_with_no_hms_returns_none() -> None:
  59. assert derive_failure_reason("failed", None) is None
  60. assert derive_failure_reason("failed", []) is None
  61. # ---------------------------------------------------------------------------
  62. # Code-format tolerance (MQTT may send int or hex string)
  63. # ---------------------------------------------------------------------------
  64. def test_int_code_field_accepted() -> None:
  65. """The MQTT parser sometimes leaves `code` as an int rather than a hex string."""
  66. hms = [{"code": 0x4057, "attr": 0x0300_0000, "module": 0x03, "severity": 1}]
  67. assert derive_failure_reason("failed", hms) == "Layer shift"