cloud.py 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137
  1. from typing import Literal
  2. from pydantic import BaseModel, Field
  3. Region = Literal["global", "china"]
  4. class CloudLoginRequest(BaseModel):
  5. """Request to initiate cloud login."""
  6. email: str = Field(..., description="Bambu Lab account email")
  7. password: str = Field(..., description="Account password")
  8. region: Region = Field(default="global", description="Region: 'global' or 'china'")
  9. class CloudVerifyRequest(BaseModel):
  10. """Request to verify login with 2FA code (email or TOTP)."""
  11. email: str = Field(..., description="Bambu Lab account email")
  12. code: str = Field(..., description="6-digit verification code")
  13. tfa_key: str | None = Field(None, description="TFA key for TOTP verification (from login response)")
  14. region: Region = Field(default="global", description="Region: 'global' or 'china'")
  15. class CloudLoginResponse(BaseModel):
  16. """Response from login attempt."""
  17. success: bool
  18. needs_verification: bool = False
  19. message: str
  20. verification_type: str | None = None # "email" or "totp"
  21. tfa_key: str | None = None # Key needed for TOTP verification
  22. class CloudAuthStatus(BaseModel):
  23. """Current authentication status."""
  24. is_authenticated: bool
  25. email: str | None = None
  26. region: Region | None = None
  27. class CloudTokenRequest(BaseModel):
  28. """Request to set access token directly."""
  29. access_token: str = Field(..., description="Bambu Lab access token")
  30. region: Region = Field(default="global", description="Region: 'global' or 'china'")
  31. class SlicerSetting(BaseModel):
  32. """A slicer setting/preset."""
  33. setting_id: str
  34. name: str
  35. type: str # filament, printer, process
  36. version: str | None = None
  37. user_id: str | None = None
  38. updated_time: str | None = None
  39. is_custom: bool = False
  40. class SlicerSettingsResponse(BaseModel):
  41. """Response containing slicer settings."""
  42. filament: list[SlicerSetting] = []
  43. printer: list[SlicerSetting] = []
  44. process: list[SlicerSetting] = []
  45. class CloudDevice(BaseModel):
  46. """A bound printer device."""
  47. dev_id: str
  48. name: str
  49. dev_model_name: str | None = None
  50. dev_product_name: str | None = None
  51. online: bool = False
  52. class SlicerSettingCreate(BaseModel):
  53. """Request to create a new slicer preset."""
  54. type: str = Field(..., description="Preset type: 'filament', 'print', or 'printer'")
  55. name: str = Field(..., description="Display name for the preset")
  56. base_id: str = Field(..., description="Base preset ID to inherit from")
  57. version: str = Field(default="2.0.0.0", description="Version string for the preset")
  58. setting: dict = Field(default_factory=dict, description="Setting key-value pairs (delta from base)")
  59. class SlicerSettingUpdate(BaseModel):
  60. """Request to update an existing slicer preset."""
  61. name: str | None = Field(None, description="New display name")
  62. setting: dict | None = Field(None, description="Setting key-value pairs to update")
  63. class SlicerSettingDetail(BaseModel):
  64. """Detailed slicer setting/preset response."""
  65. message: str | None = None
  66. code: str | None = None
  67. error: str | None = None
  68. public: bool = False
  69. version: str | None = None
  70. type: str
  71. name: str
  72. update_time: str | None = None
  73. nickname: str | None = None
  74. base_id: str | None = None
  75. setting: dict = Field(default_factory=dict)
  76. filament_id: str | None = None
  77. setting_id: str | None = None # For response after create
  78. class SlicerSettingDeleteResponse(BaseModel):
  79. """Response from deleting a preset."""
  80. success: bool
  81. message: str
  82. class FirmwareUpdateInfo(BaseModel):
  83. """Firmware update information for a device."""
  84. device_id: str = Field(..., description="Device ID")
  85. device_name: str = Field(..., description="Device name")
  86. current_version: str | None = Field(None, description="Currently installed firmware version")
  87. latest_version: str | None = Field(None, description="Latest available firmware version")
  88. update_available: bool = Field(False, description="Whether an update is available")
  89. release_notes: str | None = Field(None, description="Release notes for the latest version")
  90. class FirmwareUpdatesResponse(BaseModel):
  91. """Response containing firmware updates for all devices."""
  92. updates: list[FirmwareUpdateInfo] = Field(default_factory=list)
  93. updates_available: int = Field(0, description="Total number of devices with updates available")