print_queue.py 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639
  1. """API routes for print queue management."""
  2. import json
  3. import logging
  4. import xml.etree.ElementTree as ET
  5. import zipfile
  6. from datetime import datetime
  7. from pathlib import Path
  8. from fastapi import APIRouter, Depends, HTTPException, Query
  9. from sqlalchemy import func, select
  10. from sqlalchemy.ext.asyncio import AsyncSession
  11. from sqlalchemy.orm import selectinload
  12. from backend.app.core.auth import require_auth_if_enabled
  13. from backend.app.core.config import settings
  14. from backend.app.core.database import get_db
  15. from backend.app.models.archive import PrintArchive
  16. from backend.app.models.library import LibraryFile
  17. from backend.app.models.print_queue import PrintQueueItem
  18. from backend.app.models.printer import Printer
  19. from backend.app.models.user import User
  20. from backend.app.schemas.print_queue import (
  21. PrintQueueBulkUpdate,
  22. PrintQueueBulkUpdateResponse,
  23. PrintQueueItemCreate,
  24. PrintQueueItemResponse,
  25. PrintQueueItemUpdate,
  26. PrintQueueReorder,
  27. )
  28. from backend.app.services.notification_service import notification_service
  29. from backend.app.utils.printer_models import normalize_printer_model, normalize_printer_model_id
  30. logger = logging.getLogger(__name__)
  31. router = APIRouter(prefix="/queue", tags=["queue"])
  32. def _extract_filament_types_from_3mf(file_path: Path, plate_id: int | None = None) -> list[str]:
  33. """Extract unique filament types from a 3MF file.
  34. Args:
  35. file_path: Path to the 3MF file
  36. plate_id: Optional plate index to filter for (for multi-plate files)
  37. Returns:
  38. List of unique filament types (e.g., ["PLA", "PETG"])
  39. """
  40. types: set[str] = set()
  41. try:
  42. with zipfile.ZipFile(file_path, "r") as zf:
  43. if "Metadata/slice_info.config" not in zf.namelist():
  44. return []
  45. content = zf.read("Metadata/slice_info.config").decode()
  46. root = ET.fromstring(content)
  47. if plate_id is not None:
  48. # Find the plate element with matching index
  49. for plate_elem in root.findall(".//plate"):
  50. plate_index = None
  51. for meta in plate_elem.findall("metadata"):
  52. if meta.get("key") == "index":
  53. try:
  54. plate_index = int(meta.get("value", "0"))
  55. except ValueError:
  56. pass
  57. break
  58. if plate_index == plate_id:
  59. for filament_elem in plate_elem.findall("filament"):
  60. filament_type = filament_elem.get("type", "")
  61. used_g = filament_elem.get("used_g", "0")
  62. try:
  63. used_grams = float(used_g)
  64. except (ValueError, TypeError):
  65. used_grams = 0
  66. if used_grams > 0 and filament_type:
  67. types.add(filament_type)
  68. break
  69. else:
  70. # No plate_id specified - extract all filaments with used_g > 0
  71. for filament_elem in root.findall(".//filament"):
  72. filament_type = filament_elem.get("type", "")
  73. used_g = filament_elem.get("used_g", "0")
  74. try:
  75. used_grams = float(used_g)
  76. except (ValueError, TypeError):
  77. used_grams = 0
  78. if used_grams > 0 and filament_type:
  79. types.add(filament_type)
  80. except Exception as e:
  81. logger.warning(f"Failed to extract filament types from {file_path}: {e}")
  82. return sorted(types)
  83. def _enrich_response(item: PrintQueueItem) -> PrintQueueItemResponse:
  84. """Add nested archive/printer/library_file info to response."""
  85. # Parse ams_mapping from JSON string BEFORE model_validate
  86. ams_mapping_parsed = None
  87. if item.ams_mapping:
  88. try:
  89. ams_mapping_parsed = json.loads(item.ams_mapping)
  90. except json.JSONDecodeError:
  91. ams_mapping_parsed = None
  92. # Parse required_filament_types from JSON string
  93. required_filament_types_parsed = None
  94. if item.required_filament_types:
  95. try:
  96. required_filament_types_parsed = json.loads(item.required_filament_types)
  97. except json.JSONDecodeError:
  98. required_filament_types_parsed = None
  99. # Create response with parsed ams_mapping
  100. item_dict = {
  101. "id": item.id,
  102. "printer_id": item.printer_id,
  103. "target_model": item.target_model,
  104. "required_filament_types": required_filament_types_parsed,
  105. "waiting_reason": item.waiting_reason,
  106. "archive_id": item.archive_id,
  107. "library_file_id": item.library_file_id,
  108. "position": item.position,
  109. "scheduled_time": item.scheduled_time,
  110. "require_previous_success": item.require_previous_success,
  111. "auto_off_after": item.auto_off_after,
  112. "manual_start": item.manual_start,
  113. "ams_mapping": ams_mapping_parsed,
  114. "plate_id": item.plate_id,
  115. "bed_levelling": item.bed_levelling,
  116. "flow_cali": item.flow_cali,
  117. "vibration_cali": item.vibration_cali,
  118. "layer_inspect": item.layer_inspect,
  119. "timelapse": item.timelapse,
  120. "use_ams": item.use_ams,
  121. "status": item.status,
  122. "started_at": item.started_at,
  123. "completed_at": item.completed_at,
  124. "error_message": item.error_message,
  125. "created_at": item.created_at,
  126. # User tracking (Issue #206)
  127. "created_by_id": item.created_by_id,
  128. "created_by_username": item.created_by.username if item.created_by else None,
  129. }
  130. response = PrintQueueItemResponse(**item_dict)
  131. if item.archive:
  132. response.archive_name = item.archive.print_name or item.archive.filename
  133. response.archive_thumbnail = item.archive.thumbnail_path
  134. response.print_time_seconds = item.archive.print_time_seconds
  135. if item.library_file:
  136. response.library_file_name = (
  137. item.library_file.file_metadata.get("print_name") if item.library_file.file_metadata else None
  138. )
  139. if not response.library_file_name:
  140. response.library_file_name = item.library_file.filename
  141. response.library_file_thumbnail = item.library_file.thumbnail_path
  142. # Get print time from library file metadata if no archive
  143. if not item.archive and item.library_file.file_metadata:
  144. response.print_time_seconds = item.library_file.file_metadata.get("print_time_seconds")
  145. if item.printer:
  146. response.printer_name = item.printer.name
  147. return response
  148. @router.get("/", response_model=list[PrintQueueItemResponse])
  149. async def list_queue(
  150. printer_id: int | None = Query(None, description="Filter by printer (-1 for unassigned)"),
  151. status: str | None = Query(None, description="Filter by status"),
  152. db: AsyncSession = Depends(get_db),
  153. ):
  154. """List all queue items, optionally filtered by printer or status."""
  155. query = (
  156. select(PrintQueueItem)
  157. .options(
  158. selectinload(PrintQueueItem.archive),
  159. selectinload(PrintQueueItem.printer),
  160. selectinload(PrintQueueItem.library_file),
  161. selectinload(PrintQueueItem.created_by),
  162. )
  163. .order_by(PrintQueueItem.printer_id.nulls_first(), PrintQueueItem.position)
  164. )
  165. if printer_id is not None:
  166. if printer_id == -1:
  167. # Special value: filter for unassigned items
  168. query = query.where(PrintQueueItem.printer_id.is_(None))
  169. else:
  170. query = query.where(PrintQueueItem.printer_id == printer_id)
  171. if status:
  172. query = query.where(PrintQueueItem.status == status)
  173. result = await db.execute(query)
  174. items = result.scalars().all()
  175. return [_enrich_response(item) for item in items]
  176. @router.post("/", response_model=PrintQueueItemResponse)
  177. async def add_to_queue(
  178. data: PrintQueueItemCreate,
  179. db: AsyncSession = Depends(get_db),
  180. current_user: User | None = Depends(require_auth_if_enabled),
  181. ):
  182. """Add an item to the print queue."""
  183. # Normalize target_model (e.g., "Bambu Lab X1E" / "C13" -> "X1E")
  184. target_model_norm = None
  185. if data.target_model:
  186. target_model_norm = (
  187. normalize_printer_model(data.target_model)
  188. or normalize_printer_model_id(data.target_model)
  189. or data.target_model
  190. )
  191. # Validate that either archive_id or library_file_id is provided
  192. if not data.archive_id and not data.library_file_id:
  193. raise HTTPException(400, "Either archive_id or library_file_id must be provided")
  194. # Cannot specify both printer_id and target_model
  195. if data.printer_id and target_model_norm:
  196. raise HTTPException(400, "Cannot specify both printer_id and target_model")
  197. # Validate printer exists (if assigned)
  198. if data.printer_id is not None:
  199. result = await db.execute(select(Printer).where(Printer.id == data.printer_id))
  200. if not result.scalar_one_or_none():
  201. raise HTTPException(400, "Printer not found")
  202. # Validate target_model has active printers
  203. if target_model_norm:
  204. result = await db.execute(
  205. select(Printer).where(Printer.model == target_model_norm).where(Printer.is_active == True) # noqa: E712
  206. )
  207. if not result.scalars().first():
  208. raise HTTPException(400, f"No active printers for model: {target_model_norm}")
  209. # Validate archive exists (if provided) and get it for filament extraction
  210. archive = None
  211. if data.archive_id:
  212. result = await db.execute(select(PrintArchive).where(PrintArchive.id == data.archive_id))
  213. archive = result.scalar_one_or_none()
  214. if not archive:
  215. raise HTTPException(400, "Archive not found")
  216. # Validate library file exists (if provided) and get it for filament extraction
  217. library_file = None
  218. if data.library_file_id:
  219. result = await db.execute(select(LibraryFile).where(LibraryFile.id == data.library_file_id))
  220. library_file = result.scalar_one_or_none()
  221. if not library_file:
  222. raise HTTPException(400, "Library file not found")
  223. # Extract filament types for model-based assignment (used by scheduler for validation)
  224. required_filament_types = None
  225. if target_model_norm:
  226. # Get file path from archive or library file
  227. file_path = None
  228. if archive:
  229. file_path = settings.base_dir / archive.file_path
  230. elif library_file:
  231. lib_path = Path(library_file.file_path)
  232. file_path = lib_path if lib_path.is_absolute() else settings.base_dir / library_file.file_path
  233. if file_path and file_path.exists():
  234. filament_types = _extract_filament_types_from_3mf(file_path, data.plate_id)
  235. if filament_types:
  236. required_filament_types = json.dumps(filament_types)
  237. logger.info(f"Extracted filament types for model-based queue: {filament_types}")
  238. # Get next position for this printer (or for unassigned/model-based items)
  239. if data.printer_id is not None:
  240. result = await db.execute(
  241. select(func.max(PrintQueueItem.position))
  242. .where(PrintQueueItem.printer_id == data.printer_id)
  243. .where(PrintQueueItem.status == "pending")
  244. )
  245. else:
  246. # For unassigned/model-based items, get max position across all unassigned
  247. result = await db.execute(
  248. select(func.max(PrintQueueItem.position))
  249. .where(PrintQueueItem.printer_id.is_(None))
  250. .where(PrintQueueItem.status == "pending")
  251. )
  252. max_pos = result.scalar() or 0
  253. item = PrintQueueItem(
  254. printer_id=data.printer_id,
  255. target_model=target_model_norm,
  256. required_filament_types=required_filament_types,
  257. archive_id=data.archive_id,
  258. library_file_id=data.library_file_id,
  259. scheduled_time=data.scheduled_time,
  260. require_previous_success=data.require_previous_success,
  261. auto_off_after=data.auto_off_after,
  262. manual_start=data.manual_start,
  263. ams_mapping=json.dumps(data.ams_mapping) if data.ams_mapping else None,
  264. plate_id=data.plate_id,
  265. bed_levelling=data.bed_levelling,
  266. flow_cali=data.flow_cali,
  267. vibration_cali=data.vibration_cali,
  268. layer_inspect=data.layer_inspect,
  269. timelapse=data.timelapse,
  270. use_ams=data.use_ams,
  271. position=max_pos + 1,
  272. status="pending",
  273. created_by_id=current_user.id if current_user else None,
  274. )
  275. db.add(item)
  276. await db.commit()
  277. await db.refresh(item)
  278. # Load relationships for response
  279. await db.refresh(item, ["archive", "printer", "library_file", "created_by"])
  280. source_name = f"archive {data.archive_id}" if data.archive_id else f"library file {data.library_file_id}"
  281. target_desc = data.printer_id or (f"model {target_model_norm}" if target_model_norm else "unassigned")
  282. logger.info(f"Added {source_name} to queue for {target_desc}")
  283. # MQTT relay - publish queue job added
  284. try:
  285. from backend.app.services.mqtt_relay import mqtt_relay
  286. await mqtt_relay.on_queue_job_added(
  287. job_id=item.id,
  288. filename=item.archive.filename if item.archive else "",
  289. printer_id=item.printer_id,
  290. printer_name=item.printer.name if item.printer else None,
  291. )
  292. except Exception:
  293. pass # Don't fail queue add if MQTT fails
  294. # Send notification for job added
  295. try:
  296. job_name = (
  297. item.archive.filename
  298. if item.archive
  299. else item.library_file.filename
  300. if item.library_file
  301. else f"Job #{item.id}"
  302. )
  303. job_name = job_name.replace(".gcode.3mf", "").replace(".3mf", "")
  304. target = (
  305. item.printer.name if item.printer else (f"Any {item.target_model}" if target_model_norm else "Unassigned")
  306. )
  307. await notification_service.on_queue_job_added(
  308. job_name=job_name,
  309. target=target,
  310. db=db,
  311. printer_id=item.printer_id,
  312. printer_name=item.printer.name if item.printer else None,
  313. )
  314. except Exception:
  315. pass # Don't fail queue add if notification fails
  316. return _enrich_response(item)
  317. @router.patch("/bulk", response_model=PrintQueueBulkUpdateResponse)
  318. async def bulk_update_queue_items(
  319. data: PrintQueueBulkUpdate,
  320. db: AsyncSession = Depends(get_db),
  321. ):
  322. """Bulk update multiple queue items with the same values.
  323. Only pending items can be updated. Non-pending items are skipped.
  324. """
  325. if not data.item_ids:
  326. raise HTTPException(400, "No item IDs provided")
  327. # Get fields to update (exclude item_ids and unset fields)
  328. update_data = data.model_dump(exclude={"item_ids"}, exclude_unset=True)
  329. if not update_data:
  330. raise HTTPException(400, "No fields to update")
  331. # Validate printer_id if being changed
  332. if "printer_id" in update_data and update_data["printer_id"] is not None:
  333. result = await db.execute(select(Printer).where(Printer.id == update_data["printer_id"]))
  334. if not result.scalar_one_or_none():
  335. raise HTTPException(400, "Printer not found")
  336. # Fetch all items
  337. result = await db.execute(select(PrintQueueItem).where(PrintQueueItem.id.in_(data.item_ids)))
  338. items = result.scalars().all()
  339. updated_count = 0
  340. skipped_count = 0
  341. for item in items:
  342. if item.status != "pending":
  343. skipped_count += 1
  344. continue
  345. for field, value in update_data.items():
  346. setattr(item, field, value)
  347. updated_count += 1
  348. await db.commit()
  349. logger.info(f"Bulk updated {updated_count} queue items, skipped {skipped_count}")
  350. return PrintQueueBulkUpdateResponse(
  351. updated_count=updated_count,
  352. skipped_count=skipped_count,
  353. message=f"Updated {updated_count} items" + (f", skipped {skipped_count} non-pending" if skipped_count else ""),
  354. )
  355. @router.get("/{item_id}", response_model=PrintQueueItemResponse)
  356. async def get_queue_item(item_id: int, db: AsyncSession = Depends(get_db)):
  357. """Get a specific queue item."""
  358. result = await db.execute(
  359. select(PrintQueueItem)
  360. .options(
  361. selectinload(PrintQueueItem.archive),
  362. selectinload(PrintQueueItem.printer),
  363. selectinload(PrintQueueItem.library_file),
  364. selectinload(PrintQueueItem.created_by),
  365. )
  366. .where(PrintQueueItem.id == item_id)
  367. )
  368. item = result.scalar_one_or_none()
  369. if not item:
  370. raise HTTPException(404, "Queue item not found")
  371. return _enrich_response(item)
  372. @router.patch("/{item_id}", response_model=PrintQueueItemResponse)
  373. async def update_queue_item(
  374. item_id: int,
  375. data: PrintQueueItemUpdate,
  376. db: AsyncSession = Depends(get_db),
  377. ):
  378. """Update a queue item."""
  379. result = await db.execute(select(PrintQueueItem).where(PrintQueueItem.id == item_id))
  380. item = result.scalar_one_or_none()
  381. if not item:
  382. raise HTTPException(404, "Queue item not found")
  383. if item.status != "pending":
  384. raise HTTPException(400, "Can only update pending items")
  385. update_data = data.model_dump(exclude_unset=True)
  386. # Normalize target_model if being updated
  387. if "target_model" in update_data and update_data["target_model"]:
  388. update_data["target_model"] = (
  389. normalize_printer_model(update_data["target_model"])
  390. or normalize_printer_model_id(update_data["target_model"])
  391. or update_data["target_model"]
  392. )
  393. # Cannot specify both printer_id and target_model
  394. new_printer_id = update_data.get("printer_id", item.printer_id)
  395. new_target_model = update_data.get("target_model", item.target_model)
  396. if new_printer_id and new_target_model:
  397. raise HTTPException(400, "Cannot specify both printer_id and target_model")
  398. # Validate new printer_id if being changed (and not None)
  399. if "printer_id" in update_data and update_data["printer_id"] is not None:
  400. result = await db.execute(select(Printer).where(Printer.id == update_data["printer_id"]))
  401. if not result.scalar_one_or_none():
  402. raise HTTPException(400, "Printer not found")
  403. # Validate target_model has active printers
  404. if "target_model" in update_data and update_data["target_model"]:
  405. result = await db.execute(
  406. select(Printer).where(Printer.model == update_data["target_model"]).where(Printer.is_active == True) # noqa: E712
  407. )
  408. if not result.scalars().first():
  409. raise HTTPException(400, f"No active printers for model: {update_data['target_model']}")
  410. # Serialize ams_mapping to JSON for TEXT column storage
  411. if "ams_mapping" in update_data:
  412. update_data["ams_mapping"] = json.dumps(update_data["ams_mapping"]) if update_data["ams_mapping"] else None
  413. for field, value in update_data.items():
  414. setattr(item, field, value)
  415. await db.commit()
  416. await db.refresh(item, ["archive", "printer", "library_file", "created_by"])
  417. logger.info(f"Updated queue item {item_id}")
  418. return _enrich_response(item)
  419. @router.delete("/{item_id}")
  420. async def delete_queue_item(item_id: int, db: AsyncSession = Depends(get_db)):
  421. """Remove an item from the queue."""
  422. result = await db.execute(select(PrintQueueItem).where(PrintQueueItem.id == item_id))
  423. item = result.scalar_one_or_none()
  424. if not item:
  425. raise HTTPException(404, "Queue item not found")
  426. if item.status == "printing":
  427. raise HTTPException(400, "Cannot delete item that is currently printing")
  428. await db.delete(item)
  429. await db.commit()
  430. logger.info(f"Deleted queue item {item_id}")
  431. return {"message": "Queue item deleted"}
  432. @router.post("/reorder")
  433. async def reorder_queue(
  434. data: PrintQueueReorder,
  435. db: AsyncSession = Depends(get_db),
  436. ):
  437. """Bulk update positions for queue items."""
  438. for reorder_item in data.items:
  439. result = await db.execute(select(PrintQueueItem).where(PrintQueueItem.id == reorder_item.id))
  440. item = result.scalar_one_or_none()
  441. if item and item.status == "pending":
  442. item.position = reorder_item.position
  443. await db.commit()
  444. logger.info(f"Reordered {len(data.items)} queue items")
  445. return {"message": f"Reordered {len(data.items)} items"}
  446. @router.post("/{item_id}/cancel")
  447. async def cancel_queue_item(item_id: int, db: AsyncSession = Depends(get_db)):
  448. """Cancel a pending queue item."""
  449. result = await db.execute(select(PrintQueueItem).where(PrintQueueItem.id == item_id))
  450. item = result.scalar_one_or_none()
  451. if not item:
  452. raise HTTPException(404, "Queue item not found")
  453. if item.status not in ("pending",):
  454. raise HTTPException(400, f"Cannot cancel item with status '{item.status}'")
  455. item.status = "cancelled"
  456. item.completed_at = datetime.now()
  457. await db.commit()
  458. logger.info(f"Cancelled queue item {item_id}")
  459. return {"message": "Queue item cancelled"}
  460. @router.post("/{item_id}/stop")
  461. async def stop_queue_item(
  462. item_id: int,
  463. db: AsyncSession = Depends(get_db),
  464. ):
  465. """Stop an actively printing queue item."""
  466. import asyncio
  467. from backend.app.models.smart_plug import SmartPlug
  468. from backend.app.services.printer_manager import printer_manager
  469. from backend.app.services.tasmota import tasmota_service
  470. result = await db.execute(select(PrintQueueItem).where(PrintQueueItem.id == item_id))
  471. item = result.scalar_one_or_none()
  472. if not item:
  473. raise HTTPException(404, "Queue item not found")
  474. if item.status != "printing":
  475. raise HTTPException(400, f"Can only stop items that are printing, current status: '{item.status}'")
  476. # Capture values we need for background task
  477. printer_id = item.printer_id
  478. auto_off_after = item.auto_off_after
  479. # Try to send stop command to printer
  480. stop_sent = False
  481. try:
  482. stop_sent = printer_manager.stop_print(printer_id)
  483. if not stop_sent:
  484. logger.warning(f"stop_print returned False for printer {printer_id} - printer may not be connected")
  485. except Exception as e:
  486. logger.error(f"Error sending stop command for queue item {item_id}: {e}")
  487. # Update queue item status regardless - if printer is off, print is already stopped
  488. item.status = "cancelled"
  489. item.completed_at = datetime.now()
  490. item.error_message = "Stopped by user" if stop_sent else "Stopped by user (printer was offline)"
  491. await db.commit()
  492. # Get smart plug info if auto-off is enabled
  493. plug_ip = None
  494. if auto_off_after:
  495. result = await db.execute(select(SmartPlug).where(SmartPlug.printer_id == printer_id))
  496. plug = result.scalar_one_or_none()
  497. if plug and plug.enabled:
  498. plug_ip = plug.ip_address
  499. logger.info(f"Stopped printing queue item {item_id} (stop command sent: {stop_sent})")
  500. # Schedule background task for cooldown + power off
  501. if plug_ip:
  502. async def cooldown_and_poweroff():
  503. logger.info(f"Auto-off: Waiting for printer {printer_id} to cool down before power off...")
  504. await printer_manager.wait_for_cooldown(printer_id, target_temp=50.0, timeout=600)
  505. # Re-fetch plug since we're in a new async context
  506. from backend.app.core.database import async_session
  507. async with async_session() as new_db:
  508. result = await new_db.execute(select(SmartPlug).where(SmartPlug.printer_id == printer_id))
  509. plug = result.scalar_one_or_none()
  510. if plug and plug.enabled:
  511. logger.info(f"Auto-off: Powering off printer {printer_id}")
  512. await tasmota_service.turn_off(plug)
  513. asyncio.create_task(cooldown_and_poweroff())
  514. return {"message": "Print stopped" if stop_sent else "Queue item cancelled (printer was offline)"}
  515. @router.post("/{item_id}/start")
  516. async def start_queue_item(
  517. item_id: int,
  518. db: AsyncSession = Depends(get_db),
  519. ):
  520. """Manually start a staged (manual_start) queue item.
  521. This clears the manual_start flag so the scheduler will pick it up,
  522. or starts immediately if the printer is ready.
  523. """
  524. result = await db.execute(
  525. select(PrintQueueItem)
  526. .options(selectinload(PrintQueueItem.archive), selectinload(PrintQueueItem.printer))
  527. .where(PrintQueueItem.id == item_id)
  528. )
  529. item = result.scalar_one_or_none()
  530. if not item:
  531. raise HTTPException(404, "Queue item not found")
  532. if item.status != "pending":
  533. raise HTTPException(400, f"Can only start pending items, current status: '{item.status}'")
  534. # Clear manual_start flag so scheduler picks it up
  535. item.manual_start = False
  536. await db.commit()
  537. await db.refresh(item, ["archive", "printer", "library_file", "created_by"])
  538. logger.info(f"Manually started queue item {item_id} (cleared manual_start flag)")
  539. return _enrich_response(item)