| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071 |
- """Unit tests for the per-run filament helper (#1378, #1390).
- The helper computes what value to write into PrintLogEntry.filament_used_grams
- for a given print event — partial-aware so failed / cancelled / stopped prints
- don't inflate stats with the full slicer estimate, and tracker-aware so
- completed prints agree with the per-spool counter on the Inventory page.
- """
- from backend.app.main import _compute_run_filament_grams
- class TestComputeRunFilamentGrams:
- def test_completed_no_tracker_returns_archive_estimate(self):
- # Completed print without inventory tracking: the slicer estimate is
- # the canonical "this print used X" value.
- assert _compute_run_filament_grams("completed", 100.0, 100, []) == 100.0
- def test_completed_prefers_tracked_over_estimate(self):
- # #1390: when inventory tracked the AMS weight delta, Stats should
- # reflect that — same source that drives "Total Consumed" on the
- # Inventory page. Two halves of the app must show the same number.
- assert _compute_run_filament_grams("completed", 100.0, 100, [{"weight_used": 96.5}]) == 96.5
- def test_failed_uses_tracked_spool_delta(self):
- # Failed reprint at 10g actual: inventory tracked the spool delta.
- # The estimate was 100g; we want 10g recorded for stats.
- assert _compute_run_filament_grams("failed", 100.0, 10, [{"weight_used": 10.0}]) == 10.0
- def test_cancelled_uses_tracked_spool_delta(self):
- # Same logic for cancelled.
- assert _compute_run_filament_grams("cancelled", 100.0, 12, [{"weight_used": 8.5}]) == 8.5
- def test_stopped_uses_tracked_spool_delta(self):
- assert _compute_run_filament_grams("stopped", 100.0, 15, [{"weight_used": 12.0}]) == 12.0
- def test_failed_with_no_tracked_falls_back_to_progress_scale(self):
- # No inventory tracking: scale estimate by progress% (10% of 100g = 10g).
- assert _compute_run_filament_grams("failed", 100.0, 10, []) == 10.0
- def test_failed_with_no_tracked_and_no_progress_returns_none(self):
- # Nothing to infer from — return None rather than guess the estimate.
- assert _compute_run_filament_grams("failed", 100.0, 0, []) is None
- def test_failed_with_partial_progress_rounds_correctly(self):
- # 100g × 33% = 33.0g (rounded to 1 decimal)
- assert _compute_run_filament_grams("failed", 100.0, 33, []) == 33.0
- def test_failed_with_no_estimate_returns_none(self):
- # No estimate, no tracked usage → can't compute anything.
- assert _compute_run_filament_grams("failed", None, 50, []) is None
- def test_failed_with_no_estimate_but_tracked_uses_tracked(self):
- # Tracked spool delta is authoritative even without an estimate.
- assert _compute_run_filament_grams("failed", None, 50, [{"weight_used": 5.0}]) == 5.0
- def test_tracked_overrides_progress_scale_when_both_available(self):
- # If inventory says 8g but progress says 15g, trust inventory (it's measured).
- assert _compute_run_filament_grams("failed", 100.0, 15, [{"weight_used": 8.0}]) == 8.0
- def test_progress_above_100_clamps_to_full_estimate(self):
- # Defensive: progress overshoot doesn't multiply past the estimate.
- assert _compute_run_filament_grams("failed", 100.0, 150, []) == 100.0
- def test_multiple_tracked_slots_summed(self):
- # Multi-filament print, two slots tracked.
- usage = [{"weight_used": 5.0}, {"weight_used": 3.5}, {"weight_used": 1.0}]
- assert _compute_run_filament_grams("failed", 100.0, 20, usage) == 9.5
- def test_completed_with_none_estimate_returns_none(self):
- # Archive somehow has no estimate (rare; archive_print parsed nothing).
- assert _compute_run_filament_grams("completed", None, 100, []) is None
|