discovery.py 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169
  1. """
  2. Printer discovery API endpoints.
  3. Provides endpoints for discovering Bambu Lab printers on the local network.
  4. Supports both SSDP discovery (for native installs) and subnet scanning (for Docker).
  5. """
  6. import logging
  7. from fastapi import APIRouter
  8. from pydantic import BaseModel
  9. from backend.app.services.discovery import (
  10. discovery_service,
  11. is_running_in_docker,
  12. subnet_scanner,
  13. )
  14. logger = logging.getLogger(__name__)
  15. router = APIRouter(prefix="/discovery", tags=["discovery"])
  16. class DiscoveryStatus(BaseModel):
  17. """Discovery status response."""
  18. running: bool
  19. class DiscoveryInfo(BaseModel):
  20. """Discovery environment info."""
  21. is_docker: bool
  22. ssdp_running: bool
  23. scan_running: bool
  24. class SubnetScanRequest(BaseModel):
  25. """Request to scan a subnet."""
  26. subnet: str # CIDR notation, e.g., "192.168.1.0/24"
  27. timeout: float = 1.0 # Connection timeout per host
  28. class SubnetScanStatus(BaseModel):
  29. """Subnet scan status response."""
  30. running: bool
  31. scanned: int
  32. total: int
  33. class DiscoveredPrinterResponse(BaseModel):
  34. """Discovered printer response."""
  35. serial: str
  36. name: str
  37. ip_address: str
  38. model: str | None = None
  39. discovered_at: str | None = None
  40. @router.get("/info", response_model=DiscoveryInfo)
  41. async def get_discovery_info():
  42. """Get discovery environment info (Docker detection, etc.)."""
  43. return DiscoveryInfo(
  44. is_docker=is_running_in_docker(),
  45. ssdp_running=discovery_service.is_running,
  46. scan_running=subnet_scanner.is_running,
  47. )
  48. @router.get("/status", response_model=DiscoveryStatus)
  49. async def get_discovery_status():
  50. """Get the current SSDP discovery status."""
  51. return DiscoveryStatus(running=discovery_service.is_running)
  52. @router.post("/start", response_model=DiscoveryStatus)
  53. async def start_discovery(duration: float = 10.0):
  54. """Start SSDP printer discovery.
  55. Args:
  56. duration: Discovery duration in seconds (default 10)
  57. """
  58. await discovery_service.start(duration=duration)
  59. return DiscoveryStatus(running=discovery_service.is_running)
  60. @router.post("/stop", response_model=DiscoveryStatus)
  61. async def stop_discovery():
  62. """Stop SSDP printer discovery."""
  63. await discovery_service.stop()
  64. return DiscoveryStatus(running=discovery_service.is_running)
  65. @router.get("/printers", response_model=list[DiscoveredPrinterResponse])
  66. async def get_discovered_printers():
  67. """Get list of discovered printers (from both SSDP and subnet scan)."""
  68. # Combine results from both discovery methods
  69. printers = {}
  70. # Add SSDP discovered printers
  71. for p in discovery_service.discovered_printers:
  72. printers[p.ip_address] = p
  73. # Add subnet scan discovered printers (may override if same IP)
  74. for p in subnet_scanner.discovered_printers:
  75. if p.ip_address not in printers:
  76. printers[p.ip_address] = p
  77. return [
  78. DiscoveredPrinterResponse(
  79. serial=p.serial,
  80. name=p.name,
  81. ip_address=p.ip_address,
  82. model=p.model,
  83. discovered_at=p.discovered_at,
  84. )
  85. for p in printers.values()
  86. ]
  87. # Subnet scanning endpoints (for Docker environments)
  88. @router.post("/scan", response_model=SubnetScanStatus)
  89. async def start_subnet_scan(request: SubnetScanRequest):
  90. """Start a subnet scan for Bambu printers.
  91. Use this when running in Docker where SSDP multicast doesn't work.
  92. Args:
  93. request: Subnet to scan in CIDR notation (e.g., "192.168.1.0/24")
  94. """
  95. # Start scan in background
  96. import asyncio
  97. asyncio.create_task(subnet_scanner.scan_subnet(request.subnet, request.timeout))
  98. # Return immediate status
  99. scanned, total = subnet_scanner.progress
  100. return SubnetScanStatus(
  101. running=subnet_scanner.is_running,
  102. scanned=scanned,
  103. total=total,
  104. )
  105. @router.get("/scan/status", response_model=SubnetScanStatus)
  106. async def get_scan_status():
  107. """Get the current subnet scan status."""
  108. scanned, total = subnet_scanner.progress
  109. return SubnetScanStatus(
  110. running=subnet_scanner.is_running,
  111. scanned=scanned,
  112. total=total,
  113. )
  114. @router.post("/scan/stop", response_model=SubnetScanStatus)
  115. async def stop_subnet_scan():
  116. """Stop the current subnet scan."""
  117. subnet_scanner.stop()
  118. scanned, total = subnet_scanner.progress
  119. return SubnetScanStatus(
  120. running=subnet_scanner.is_running,
  121. scanned=scanned,
  122. total=total,
  123. )