tcp_proxy.py 39 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007
  1. """TLS proxy for slicer-to-printer communication.
  2. This module provides a TLS terminating proxy that forwards data between
  3. a slicer and a real Bambu printer, enabling remote printing over
  4. any network connection.
  5. Unlike a transparent TCP proxy, this terminates TLS on both ends:
  6. - Slicer connects to Bambuddy using Bambuddy's certificate
  7. - Bambuddy connects to printer using printer's certificate
  8. - Data is decrypted, forwarded, and re-encrypted
  9. """
  10. import asyncio
  11. import logging
  12. import random
  13. import re
  14. import ssl
  15. import subprocess
  16. from collections.abc import Callable
  17. from pathlib import Path
  18. logger = logging.getLogger(__name__)
  19. def detect_port_redirect(port: int) -> int | None:
  20. """Detect if iptables redirects a port to another port.
  21. When iptables NAT REDIRECT rules exist (e.g. 990→9990), connections
  22. to the original port never reach our socket because iptables intercepts
  23. them in PREROUTING. We must listen on the redirect target instead.
  24. Returns the redirect target port, or None if no redirect is active.
  25. """
  26. # Method 1: Read persistent rules file (doesn't require root)
  27. for rules_path in ("/etc/iptables/rules.v4", "/etc/iptables.rules"):
  28. try:
  29. with open(rules_path) as f:
  30. content = f.read()
  31. match = re.search(rf"--dport {port}\b.*?--to-ports\s+(\d+)", content)
  32. if match:
  33. target = int(match.group(1))
  34. if target != port:
  35. return target
  36. except (FileNotFoundError, PermissionError, OSError):
  37. continue
  38. # Method 2: Query live iptables rules (may require root)
  39. try:
  40. result = subprocess.run( # noqa: S603, S607
  41. ["iptables-save", "-t", "nat"],
  42. capture_output=True,
  43. text=True,
  44. timeout=5,
  45. )
  46. if result.returncode == 0:
  47. match = re.search(rf"--dport {port}\b.*?--to-ports\s+(\d+)", result.stdout)
  48. if match:
  49. target = int(match.group(1))
  50. if target != port:
  51. return target
  52. except (FileNotFoundError, subprocess.TimeoutExpired, OSError):
  53. pass
  54. return None
  55. class TLSProxy:
  56. """TLS terminating proxy that forwards data between client and target.
  57. This proxy terminates TLS on both ends, allowing the slicer to connect
  58. to Bambuddy's certificate while Bambuddy connects to the real printer.
  59. """
  60. def __init__(
  61. self,
  62. name: str,
  63. listen_port: int,
  64. target_host: str,
  65. target_port: int,
  66. server_cert_path: Path,
  67. server_key_path: Path,
  68. on_connect: Callable[[str], None] | None = None,
  69. on_disconnect: Callable[[str], None] | None = None,
  70. ):
  71. """Initialize the TLS proxy.
  72. Args:
  73. name: Friendly name for logging (e.g., "FTP", "MQTT")
  74. listen_port: Port to listen on for incoming connections
  75. target_host: Target printer IP/hostname
  76. target_port: Target printer port
  77. server_cert_path: Path to server certificate (for accepting slicer connections)
  78. server_key_path: Path to server private key
  79. on_connect: Optional callback when client connects (receives client_id)
  80. on_disconnect: Optional callback when client disconnects (receives client_id)
  81. """
  82. self.name = name
  83. self.listen_port = listen_port
  84. self.target_host = target_host
  85. self.target_port = target_port
  86. self.server_cert_path = server_cert_path
  87. self.server_key_path = server_key_path
  88. self.on_connect = on_connect
  89. self.on_disconnect = on_disconnect
  90. self._server: asyncio.Server | None = None
  91. self._running = False
  92. self._active_connections: dict[str, tuple[asyncio.Task, asyncio.Task]] = {}
  93. self._server_ssl_context: ssl.SSLContext | None = None
  94. self._client_ssl_context: ssl.SSLContext | None = None
  95. def _create_server_ssl_context(self) -> ssl.SSLContext:
  96. """Create SSL context for accepting client (slicer) connections."""
  97. ctx = ssl.SSLContext(ssl.PROTOCOL_TLS_SERVER)
  98. ctx.load_cert_chain(self.server_cert_path, self.server_key_path)
  99. # Allow older TLS versions for compatibility with slicers
  100. ctx.minimum_version = ssl.TLSVersion.TLSv1_2
  101. # Don't require client certificates
  102. ctx.verify_mode = ssl.CERT_NONE
  103. return ctx
  104. def _create_client_ssl_context(self) -> ssl.SSLContext:
  105. """Create SSL context for connecting to printer."""
  106. ctx = ssl.SSLContext(ssl.PROTOCOL_TLS_CLIENT)
  107. # Don't verify printer's certificate (self-signed)
  108. ctx.check_hostname = False
  109. ctx.verify_mode = ssl.CERT_NONE
  110. ctx.minimum_version = ssl.TLSVersion.TLSv1_2
  111. return ctx
  112. async def start(self) -> None:
  113. """Start the TLS proxy server."""
  114. if self._running:
  115. return
  116. logger.info(
  117. f"Starting {self.name} TLS proxy: 0.0.0.0:{self.listen_port} → {self.target_host}:{self.target_port}"
  118. )
  119. try:
  120. self._running = True
  121. # Create SSL contexts
  122. self._server_ssl_context = self._create_server_ssl_context()
  123. self._client_ssl_context = self._create_client_ssl_context()
  124. # Start server with TLS
  125. self._server = await asyncio.start_server(
  126. self._handle_client,
  127. "0.0.0.0", # nosec B104
  128. self.listen_port,
  129. ssl=self._server_ssl_context,
  130. )
  131. logger.info("%s TLS proxy listening on port %s", self.name, self.listen_port)
  132. async with self._server:
  133. await self._server.serve_forever()
  134. except OSError as e:
  135. if e.errno == 98: # Address already in use
  136. logger.error("%s proxy port %s is already in use", self.name, self.listen_port)
  137. elif e.errno == 13: # Permission denied
  138. logger.error(
  139. "%s proxy: cannot bind to port %s (permission denied). "
  140. "Port %s requires root or CAP_NET_BIND_SERVICE. "
  141. "Docker: add 'cap_add: [NET_BIND_SERVICE]' to docker-compose.yml. "
  142. "Native: use 'sudo setcap cap_net_bind_service=+ep $(which python3)' "
  143. "or redirect with iptables.",
  144. self.name,
  145. self.listen_port,
  146. self.listen_port,
  147. )
  148. else:
  149. logger.error("%s proxy error: %s", self.name, e)
  150. except asyncio.CancelledError:
  151. logger.debug("%s proxy task cancelled", self.name)
  152. except Exception as e:
  153. logger.error("%s proxy error: %s", self.name, e)
  154. finally:
  155. await self.stop()
  156. async def stop(self) -> None:
  157. """Stop the TLS proxy server."""
  158. logger.info("Stopping %s proxy", self.name)
  159. self._running = False
  160. # Cancel all active connection tasks
  161. for client_id, (task1, task2) in list(self._active_connections.items()):
  162. task1.cancel()
  163. task2.cancel()
  164. if self.on_disconnect:
  165. try:
  166. self.on_disconnect(client_id)
  167. except Exception:
  168. pass # Ignore disconnect callback errors during shutdown
  169. self._active_connections.clear()
  170. if self._server:
  171. try:
  172. self._server.close()
  173. await self._server.wait_closed()
  174. except OSError as e:
  175. logger.debug("Error closing %s proxy server: %s", self.name, e)
  176. self._server = None
  177. async def _handle_client(
  178. self,
  179. client_reader: asyncio.StreamReader,
  180. client_writer: asyncio.StreamWriter,
  181. ) -> None:
  182. """Handle a new client connection by proxying to target."""
  183. peername = client_writer.get_extra_info("peername")
  184. client_id = f"{peername[0]}:{peername[1]}" if peername else "unknown"
  185. logger.info("%s proxy: client connected from %s", self.name, client_id)
  186. if self.on_connect:
  187. try:
  188. self.on_connect(client_id)
  189. except Exception:
  190. pass # Ignore connect callback errors; connection proceeds regardless
  191. # Connect to target printer with TLS
  192. try:
  193. printer_reader, printer_writer = await asyncio.wait_for(
  194. asyncio.open_connection(
  195. self.target_host,
  196. self.target_port,
  197. ssl=self._client_ssl_context,
  198. ),
  199. timeout=10.0,
  200. )
  201. logger.info("%s proxy: connected to printer %s:%s", self.name, self.target_host, self.target_port)
  202. except TimeoutError:
  203. logger.error("%s proxy: timeout connecting to %s:%s", self.name, self.target_host, self.target_port)
  204. client_writer.close()
  205. await client_writer.wait_closed()
  206. return
  207. except ssl.SSLError as e:
  208. logger.error(
  209. "%s proxy: SSL error connecting to %s:%s: %s", self.name, self.target_host, self.target_port, e
  210. )
  211. client_writer.close()
  212. await client_writer.wait_closed()
  213. return
  214. except OSError as e:
  215. logger.error("%s proxy: failed to connect to %s:%s: %s", self.name, self.target_host, self.target_port, e)
  216. client_writer.close()
  217. await client_writer.wait_closed()
  218. return
  219. # Create bidirectional forwarding tasks
  220. client_to_printer = asyncio.create_task(
  221. self._forward(client_reader, printer_writer, f"{client_id}→printer"),
  222. name=f"{self.name}_c2p_{client_id}",
  223. )
  224. printer_to_client = asyncio.create_task(
  225. self._forward(printer_reader, client_writer, f"printer→{client_id}"),
  226. name=f"{self.name}_p2c_{client_id}",
  227. )
  228. self._active_connections[client_id] = (client_to_printer, printer_to_client)
  229. try:
  230. # Wait for either direction to complete (connection closed)
  231. done, pending = await asyncio.wait(
  232. [client_to_printer, printer_to_client],
  233. return_when=asyncio.FIRST_COMPLETED,
  234. )
  235. # Cancel the other direction
  236. for task in pending:
  237. task.cancel()
  238. try:
  239. await task
  240. except asyncio.CancelledError:
  241. pass # Expected when cancelling the other forwarding direction
  242. except Exception as e:
  243. logger.debug("%s proxy connection error: %s", self.name, e)
  244. finally:
  245. # Clean up
  246. self._active_connections.pop(client_id, None)
  247. for writer in [client_writer, printer_writer]:
  248. try:
  249. writer.close()
  250. await writer.wait_closed()
  251. except OSError:
  252. pass # Best-effort connection cleanup; peer may have disconnected
  253. logger.info("%s proxy: client %s disconnected", self.name, client_id)
  254. if self.on_disconnect:
  255. try:
  256. self.on_disconnect(client_id)
  257. except Exception:
  258. pass # Ignore disconnect callback errors; cleanup continues
  259. async def _forward(
  260. self,
  261. reader: asyncio.StreamReader,
  262. writer: asyncio.StreamWriter,
  263. direction: str,
  264. ) -> None:
  265. """Forward data from reader to writer.
  266. Args:
  267. reader: Source stream (already TLS-decrypted)
  268. writer: Destination stream (will be TLS-encrypted by the stream)
  269. direction: Description for logging (e.g., "client→printer")
  270. """
  271. total_bytes = 0
  272. try:
  273. while self._running:
  274. # Read chunk - use reasonable buffer size
  275. data = await reader.read(65536)
  276. if not data:
  277. # Connection closed
  278. break
  279. # Forward to destination
  280. writer.write(data)
  281. await writer.drain()
  282. total_bytes += len(data)
  283. logger.debug("%s proxy %s: %s bytes", self.name, direction, len(data))
  284. except asyncio.CancelledError:
  285. pass # Expected when the other forwarding direction closes first
  286. except ConnectionResetError:
  287. logger.debug("%s proxy %s: connection reset", self.name, direction)
  288. except BrokenPipeError:
  289. logger.debug("%s proxy %s: broken pipe", self.name, direction)
  290. except OSError as e:
  291. logger.debug("%s proxy %s error: %s", self.name, direction, e)
  292. logger.debug("%s proxy %s: total %s bytes", self.name, direction, total_bytes)
  293. class FTPTLSProxy(TLSProxy):
  294. """FTP-aware TLS proxy that handles passive data connections.
  295. Extends TLSProxy to intercept PASV/EPSV responses on the FTP control
  296. channel, dynamically create TLS data proxies on local ports, and rewrite
  297. the responses so the slicer connects to the proxy instead of the printer.
  298. Without this, FTP passive data connections bypass the proxy and go directly
  299. to the printer, which fails when the slicer can't reach the printer's IP.
  300. """
  301. PASV_PORT_MIN = 50000
  302. PASV_PORT_MAX = 50100
  303. async def stop(self) -> None:
  304. """Stop proxy and clean up data connection servers."""
  305. # Close all data servers first
  306. for server in list(self._data_servers):
  307. try:
  308. server.close()
  309. await server.wait_closed()
  310. except OSError:
  311. pass # Best-effort cleanup of data proxy servers
  312. self._data_servers.clear()
  313. await super().stop()
  314. async def start(self) -> None:
  315. """Start the FTP TLS proxy."""
  316. self._data_servers: list[asyncio.Server] = []
  317. await super().start()
  318. async def _handle_client(
  319. self,
  320. client_reader: asyncio.StreamReader,
  321. client_writer: asyncio.StreamWriter,
  322. ) -> None:
  323. """Handle FTP client with PASV/EPSV-aware response forwarding."""
  324. peername = client_writer.get_extra_info("peername")
  325. client_id = f"{peername[0]}:{peername[1]}" if peername else "unknown"
  326. logger.info("%s proxy: client connected from %s", self.name, client_id)
  327. if self.on_connect:
  328. try:
  329. self.on_connect(client_id)
  330. except Exception:
  331. pass # Ignore connect callback errors; connection proceeds regardless
  332. # Determine our local IP from the control connection socket
  333. sockname = client_writer.get_extra_info("sockname")
  334. local_ip = sockname[0] if sockname else "0.0.0.0"
  335. if local_ip in ("0.0.0.0", "::"):
  336. local_ip = "127.0.0.1"
  337. # Connect to target printer with TLS
  338. try:
  339. printer_reader, printer_writer = await asyncio.wait_for(
  340. asyncio.open_connection(
  341. self.target_host,
  342. self.target_port,
  343. ssl=self._client_ssl_context,
  344. ),
  345. timeout=10.0,
  346. )
  347. logger.info("%s proxy: connected to printer %s:%s", self.name, self.target_host, self.target_port)
  348. except TimeoutError:
  349. logger.error("%s proxy: timeout connecting to %s:%s", self.name, self.target_host, self.target_port)
  350. client_writer.close()
  351. await client_writer.wait_closed()
  352. return
  353. except ssl.SSLError as e:
  354. logger.error(
  355. "%s proxy: SSL error connecting to %s:%s: %s", self.name, self.target_host, self.target_port, e
  356. )
  357. client_writer.close()
  358. await client_writer.wait_closed()
  359. return
  360. except OSError as e:
  361. logger.error("%s proxy: failed to connect to %s:%s: %s", self.name, self.target_host, self.target_port, e)
  362. client_writer.close()
  363. await client_writer.wait_closed()
  364. return
  365. # Track data channel protection level per session.
  366. # PROT C = cleartext data, PROT P = TLS data.
  367. # Default to cleartext — many Bambu printers (A1, H2D) use PROT C.
  368. # If the slicer sends PROT P, we switch to TLS for data connections.
  369. session_state: dict[str, str] = {"prot": "C"}
  370. # Client→Printer: intercept EPSV and replace with PASV
  371. # EPSV responses only contain a port (no IP), so the slicer reuses
  372. # the control connection IP. If that IP is the real printer (via
  373. # iptables REDIRECT), the data connection bypasses the proxy.
  374. # PASV responses include an explicit IP that we can rewrite.
  375. client_to_printer = asyncio.create_task(
  376. self._forward_ftp_commands(client_reader, printer_writer, f"{client_id}→printer", session_state),
  377. name=f"{self.name}_c2p_{client_id}",
  378. )
  379. # Printer→Client: intercept PASV/EPSV responses
  380. printer_to_client = asyncio.create_task(
  381. self._forward_ftp_control(printer_reader, client_writer, f"printer→{client_id}", local_ip, session_state),
  382. name=f"{self.name}_p2c_{client_id}",
  383. )
  384. self._active_connections[client_id] = (client_to_printer, printer_to_client)
  385. try:
  386. done, pending = await asyncio.wait(
  387. [client_to_printer, printer_to_client],
  388. return_when=asyncio.FIRST_COMPLETED,
  389. )
  390. for task in pending:
  391. task.cancel()
  392. try:
  393. await task
  394. except asyncio.CancelledError:
  395. pass # Expected when cancelling the other forwarding direction
  396. except Exception as e:
  397. logger.debug("%s proxy connection error: %s", self.name, e)
  398. finally:
  399. self._active_connections.pop(client_id, None)
  400. for writer in [client_writer, printer_writer]:
  401. try:
  402. writer.close()
  403. await writer.wait_closed()
  404. except OSError:
  405. pass # Best-effort connection cleanup; peer may have disconnected
  406. logger.info("%s proxy: client %s disconnected", self.name, client_id)
  407. if self.on_disconnect:
  408. try:
  409. self.on_disconnect(client_id)
  410. except Exception:
  411. pass # Ignore disconnect callback errors; cleanup continues
  412. async def _forward_ftp_commands(
  413. self,
  414. reader: asyncio.StreamReader,
  415. writer: asyncio.StreamWriter,
  416. direction: str,
  417. session_state: dict[str, str],
  418. ) -> None:
  419. """Forward FTP client commands, replacing EPSV with PASV.
  420. EPSV responses only contain a port number — the client reuses the
  421. control connection IP for data. When the control IP is the real
  422. printer (due to iptables REDIRECT), EPSV data connections bypass
  423. the proxy. PASV responses include an explicit IP that the proxy
  424. can rewrite to its own address.
  425. Also tracks PROT P/C commands to know whether data connections
  426. should use TLS or cleartext.
  427. """
  428. buffer = b""
  429. total_bytes = 0
  430. try:
  431. while self._running:
  432. data = await reader.read(65536)
  433. if not data:
  434. break
  435. total_bytes += len(data)
  436. buffer += data
  437. output = b""
  438. while b"\r\n" in buffer:
  439. idx = buffer.index(b"\r\n")
  440. line = buffer[:idx]
  441. buffer = buffer[idx + 2 :]
  442. cmd_upper = line.strip().upper()
  443. # Log all FTP commands from slicer
  444. try:
  445. logger.info("FTP cmd >>> %s", line.decode("utf-8", errors="replace"))
  446. except Exception:
  447. pass
  448. # Replace EPSV with PASV so response includes an IP
  449. if cmd_upper == b"EPSV":
  450. line = b"PASV"
  451. logger.info("FTP command rewrite: EPSV → PASV")
  452. # Track PROT level for data channel encryption
  453. elif cmd_upper == b"PROT P":
  454. session_state["prot"] = "P"
  455. logger.info("FTP data protection: PROT P (TLS)")
  456. elif cmd_upper == b"PROT C":
  457. session_state["prot"] = "C"
  458. logger.info("FTP data protection: PROT C (cleartext)")
  459. output += line + b"\r\n"
  460. if output:
  461. writer.write(output)
  462. await writer.drain()
  463. logger.debug("%s proxy %s: %s bytes", self.name, direction, len(data))
  464. except asyncio.CancelledError:
  465. pass # Expected when the other forwarding direction closes first
  466. except ConnectionResetError:
  467. logger.debug("%s proxy %s: connection reset", self.name, direction)
  468. except BrokenPipeError:
  469. logger.debug("%s proxy %s: broken pipe", self.name, direction)
  470. except OSError as e:
  471. logger.debug("%s proxy %s error: %s", self.name, direction, e)
  472. if buffer:
  473. try:
  474. writer.write(buffer)
  475. await writer.drain()
  476. except OSError:
  477. pass # Best-effort flush of remaining FTP command data
  478. logger.debug("%s proxy %s: total %s bytes", self.name, direction, total_bytes)
  479. async def _forward_ftp_control(
  480. self,
  481. reader: asyncio.StreamReader,
  482. writer: asyncio.StreamWriter,
  483. direction: str,
  484. local_ip: str,
  485. session_state: dict[str, str],
  486. ) -> None:
  487. """Forward FTP control channel responses, rewriting PASV/EPSV.
  488. FTP control channel is line-based (\\r\\n terminated). We buffer data
  489. and process complete lines, intercepting 227 (PASV) and 229 (EPSV)
  490. responses to create local data proxies.
  491. """
  492. buffer = b""
  493. total_bytes = 0
  494. try:
  495. while self._running:
  496. data = await reader.read(65536)
  497. if not data:
  498. break
  499. total_bytes += len(data)
  500. buffer += data
  501. output = b""
  502. # Process all complete lines
  503. while b"\r\n" in buffer:
  504. idx = buffer.index(b"\r\n")
  505. line = buffer[:idx]
  506. buffer = buffer[idx + 2 :]
  507. # Log all FTP responses from printer
  508. try:
  509. logger.info("FTP resp <<< %s", line.decode("utf-8", errors="replace"))
  510. except Exception:
  511. pass
  512. rewritten = await self._maybe_rewrite_pasv(line, local_ip, session_state)
  513. output += rewritten + b"\r\n"
  514. if output:
  515. writer.write(output)
  516. await writer.drain()
  517. logger.debug("%s proxy %s: %s bytes", self.name, direction, len(data))
  518. except asyncio.CancelledError:
  519. pass # Expected when the other forwarding direction closes first
  520. except ConnectionResetError:
  521. logger.debug("%s proxy %s: connection reset", self.name, direction)
  522. except BrokenPipeError:
  523. logger.debug("%s proxy %s: broken pipe", self.name, direction)
  524. except OSError as e:
  525. logger.debug("%s proxy %s error: %s", self.name, direction, e)
  526. # Flush any remaining buffered data
  527. if buffer:
  528. try:
  529. writer.write(buffer)
  530. await writer.drain()
  531. except OSError:
  532. pass # Best-effort flush of remaining FTP control data
  533. logger.debug("%s proxy %s: total %s bytes", self.name, direction, total_bytes)
  534. async def _maybe_rewrite_pasv(self, line: bytes, local_ip: str, session_state: dict[str, str]) -> bytes:
  535. """Rewrite PASV/EPSV response to point to a local data proxy."""
  536. try:
  537. text = line.decode("utf-8")
  538. except UnicodeDecodeError:
  539. return line
  540. # 227 Entering Passive Mode (h1,h2,h3,h4,p1,p2)
  541. if text.startswith("227 "):
  542. match = re.search(r"\(\s*(\d+)\s*,\s*(\d+)\s*,\s*(\d+)\s*,\s*(\d+)\s*,\s*(\d+)\s*,\s*(\d+)\s*\)", text)
  543. if match:
  544. h1, h2, h3, h4, p1, p2 = (int(x) for x in match.groups())
  545. printer_ip = f"{h1}.{h2}.{h3}.{h4}"
  546. printer_port = p1 * 256 + p2
  547. local_port = await self._create_data_proxy(printer_ip, printer_port, session_state)
  548. if local_port:
  549. ip_parts = local_ip.split(".")
  550. lp1 = local_port // 256
  551. lp2 = local_port % 256
  552. rewritten = (
  553. f"227 Entering Passive Mode "
  554. f"({ip_parts[0]},{ip_parts[1]},{ip_parts[2]},{ip_parts[3]},{lp1},{lp2})"
  555. )
  556. logger.info("FTP PASV rewrite: %s:%s → %s:%s", printer_ip, printer_port, local_ip, local_port)
  557. return rewritten.encode("utf-8")
  558. else:
  559. logger.error("FTP PASV: failed to create data proxy for %s:%s", printer_ip, printer_port)
  560. else:
  561. logger.warning("FTP PASV: 227 response didn't match expected format: %s", text[:100])
  562. # 229 Entering Extended Passive Mode (|||port|)
  563. elif text.startswith("229 "):
  564. match = re.search(r"\(\|\|\|(\d+)\|\)", text)
  565. if match:
  566. printer_port = int(match.group(1))
  567. local_port = await self._create_data_proxy(self.target_host, printer_port, session_state)
  568. if local_port:
  569. rewritten = f"229 Entering Extended Passive Mode (|||{local_port}|)"
  570. logger.info("FTP EPSV rewrite: port %s → %s", printer_port, local_port)
  571. return rewritten.encode("utf-8")
  572. else:
  573. logger.error("FTP EPSV: failed to create data proxy for port %s", printer_port)
  574. else:
  575. logger.warning("FTP EPSV: 229 response didn't match expected format: %s", text[:100])
  576. return line
  577. async def _create_data_proxy(self, printer_ip: str, printer_port: int, session_state: dict[str, str]) -> int | None:
  578. """Create a one-shot proxy for an FTP data connection.
  579. Prefers the printer's original passive port so the port number stays
  580. the same in the rewritten PASV/EPSV response. This is critical when
  581. the slicer's FTP bounce-attack protection overrides the IP in the PASV
  582. response: the slicer connects to <control_IP>:<port>, and if iptables
  583. REDIRECT maps that port to the local machine, the data proxy must be
  584. listening on the *same* port number.
  585. Falls back to a random port if the original is unavailable.
  586. Uses TLS or cleartext based on the session's PROT level:
  587. - PROT P: TLS on both slicer and printer data connections
  588. - PROT C: cleartext on both sides (common for A1/H2D printers)
  589. Returns the local port number, or None if binding failed.
  590. """
  591. use_tls = session_state.get("prot") == "P"
  592. logger.info(
  593. "FTP data proxy: creating data proxy for %s:%s (printer-side %s)",
  594. printer_ip,
  595. printer_port,
  596. "TLS" if use_tls else "cleartext",
  597. )
  598. # Try the printer's original port first — this ensures the port
  599. # matches even when bounce protection or iptables REDIRECT is in play.
  600. try:
  601. await self._start_data_proxy_server(printer_port, printer_ip, printer_port, use_tls)
  602. logger.info("FTP data proxy: using printer's port %s", printer_port)
  603. return printer_port
  604. except OSError as e:
  605. logger.debug(
  606. "FTP data proxy: printer port %s unavailable (%s), trying random",
  607. printer_port,
  608. e,
  609. )
  610. for _attempt in range(10):
  611. port = random.randint(self.PASV_PORT_MIN, self.PASV_PORT_MAX)
  612. try:
  613. await self._start_data_proxy_server(port, printer_ip, printer_port, use_tls)
  614. logger.info("FTP data proxy: using random port %s", port)
  615. return port
  616. except OSError:
  617. continue
  618. logger.error("Failed to bind FTP data proxy port after 10 attempts")
  619. return None
  620. async def _start_data_proxy_server(self, port: int, printer_ip: str, printer_port: int, use_tls: bool) -> None:
  621. """Start a one-shot server for one FTP data connection.
  622. The slicer-side listener is ALWAYS cleartext. Even when the slicer
  623. sends PROT P on the control channel, Bambu Studio does not perform
  624. a TLS handshake on the data connection — it relies on the implicit
  625. FTPS control channel for authentication and sends data unencrypted.
  626. The printer-side outbound connection follows the PROT level:
  627. - PROT P (use_tls=True): TLS to the printer's data port
  628. - PROT C (use_tls=False): cleartext to the printer's data port
  629. This mirrors the control channel's TLS-termination architecture.
  630. Raises OSError if the port is already in use.
  631. """
  632. connected = asyncio.Event()
  633. server_holder: list[asyncio.Server] = []
  634. # Slicer side: ALWAYS cleartext — Bambu Studio does not do TLS on
  635. # the data channel even after sending PROT P.
  636. # Printer side: TLS if PROT P, cleartext if PROT C.
  637. client_ssl = self._client_ssl_context if use_tls else None
  638. printer_mode = "TLS" if use_tls else "cleartext"
  639. async def handle_data(
  640. client_reader: asyncio.StreamReader,
  641. client_writer: asyncio.StreamWriter,
  642. ) -> None:
  643. """Handle one FTP data connection, then close the server."""
  644. peername = client_writer.get_extra_info("peername")
  645. data_client = f"{peername[0]}:{peername[1]}" if peername else "unknown"
  646. logger.info(
  647. "FTP data proxy port %s (slicer=cleartext, printer=%s): client connected from %s, bridging to %s:%s",
  648. port,
  649. printer_mode,
  650. data_client,
  651. printer_ip,
  652. printer_port,
  653. )
  654. connected.set()
  655. # One-shot: close server after accepting first connection
  656. if server_holder:
  657. server_holder[0].close()
  658. printer_writer = None
  659. try:
  660. # Connect to printer's data port
  661. printer_reader, printer_writer = await asyncio.wait_for(
  662. asyncio.open_connection(
  663. printer_ip,
  664. printer_port,
  665. ssl=client_ssl,
  666. ),
  667. timeout=10.0,
  668. )
  669. logger.info(
  670. "FTP data proxy port %s (printer=%s): connected to printer %s:%s",
  671. port,
  672. printer_mode,
  673. printer_ip,
  674. printer_port,
  675. )
  676. # Bidirectional data forwarding
  677. c2p = asyncio.create_task(self._forward(client_reader, printer_writer, "data_c2p"))
  678. p2c = asyncio.create_task(self._forward(printer_reader, client_writer, "data_p2c"))
  679. done, pending = await asyncio.wait([c2p, p2c], return_when=asyncio.FIRST_COMPLETED)
  680. for task in pending:
  681. task.cancel()
  682. try:
  683. await task
  684. except asyncio.CancelledError:
  685. pass # Expected when other data direction closes
  686. except TimeoutError:
  687. logger.error("FTP data proxy port %s: timeout connecting to printer", port)
  688. except ssl.SSLError as e:
  689. logger.error("FTP data proxy port %s: SSL error to printer: %s", port, e)
  690. except Exception as e:
  691. logger.error("FTP data proxy port %s: error: %s", port, e)
  692. finally:
  693. for w in [client_writer, printer_writer]:
  694. if w:
  695. try:
  696. w.close()
  697. await w.wait_closed()
  698. except OSError:
  699. pass # Best-effort data connection cleanup
  700. logger.info("FTP data proxy port %s: connection closed", port)
  701. server = await asyncio.start_server(
  702. handle_data,
  703. "0.0.0.0", # nosec B104
  704. port,
  705. # No TLS on slicer side — Bambu Studio doesn't do TLS on data
  706. # channel even after PROT P. The proxy terminates TLS only on
  707. # the printer side (inside handle_data).
  708. )
  709. server_holder.append(server)
  710. self._data_servers.append(server)
  711. # Auto-close after 60s if no connection arrives
  712. async def auto_close() -> None:
  713. try:
  714. await asyncio.wait_for(connected.wait(), timeout=60.0)
  715. except TimeoutError:
  716. logger.debug("FTP data proxy on port %s timed out, closing", port)
  717. try:
  718. server.close()
  719. await server.wait_closed()
  720. except OSError:
  721. pass # Best-effort timeout cleanup
  722. finally:
  723. if server in self._data_servers:
  724. self._data_servers.remove(server)
  725. asyncio.create_task(auto_close(), name=f"ftp_data_timeout_{port}")
  726. logger.debug("FTP data proxy: port %s → %s:%s", port, printer_ip, printer_port)
  727. class SlicerProxyManager:
  728. """Manages FTP and MQTT TLS proxies for a single printer target."""
  729. # Bambu printer ports
  730. PRINTER_FTP_PORT = 990
  731. PRINTER_MQTT_PORT = 8883
  732. # Local listen ports - must match what Bambu Studio expects
  733. # Note: Port 990 requires root or CAP_NET_BIND_SERVICE capability
  734. LOCAL_FTP_PORT = 990
  735. LOCAL_MQTT_PORT = 8883
  736. def __init__(
  737. self,
  738. target_host: str,
  739. cert_path: Path,
  740. key_path: Path,
  741. on_activity: Callable[[str, str], None] | None = None,
  742. ):
  743. """Initialize the slicer proxy manager.
  744. Args:
  745. target_host: Target printer IP address
  746. cert_path: Path to server certificate
  747. key_path: Path to server private key
  748. on_activity: Optional callback for activity logging (name, message)
  749. """
  750. self.target_host = target_host
  751. self.cert_path = cert_path
  752. self.key_path = key_path
  753. self.on_activity = on_activity
  754. self._ftp_proxy: TLSProxy | None = None
  755. self._mqtt_proxy: TLSProxy | None = None
  756. self._tasks: list[asyncio.Task] = []
  757. async def start(self) -> None:
  758. """Start FTP and MQTT TLS proxies."""
  759. logger.info("Starting slicer TLS proxy to %s", self.target_host)
  760. # Detect iptables port redirect (e.g. 990→9990 for non-root installs).
  761. # If active, connections to port 990 get intercepted by iptables PREROUTING
  762. # and sent to the redirect target — our socket on 990 never sees them.
  763. ftp_listen_port = self.LOCAL_FTP_PORT
  764. redirect_target = detect_port_redirect(self.LOCAL_FTP_PORT)
  765. if redirect_target:
  766. logger.info(
  767. "Detected iptables redirect: port %d → %d. FTP proxy will listen on %d.",
  768. self.LOCAL_FTP_PORT,
  769. redirect_target,
  770. redirect_target,
  771. )
  772. ftp_listen_port = redirect_target
  773. # Create FTP proxy with PASV/EPSV awareness for data connections
  774. self._ftp_proxy = FTPTLSProxy(
  775. name="FTP",
  776. listen_port=ftp_listen_port,
  777. target_host=self.target_host,
  778. target_port=self.PRINTER_FTP_PORT,
  779. server_cert_path=self.cert_path,
  780. server_key_path=self.key_path,
  781. on_connect=lambda cid: self._log_activity("FTP", f"connected: {cid}"),
  782. on_disconnect=lambda cid: self._log_activity("FTP", f"disconnected: {cid}"),
  783. )
  784. self._mqtt_proxy = TLSProxy(
  785. name="MQTT",
  786. listen_port=self.LOCAL_MQTT_PORT,
  787. target_host=self.target_host,
  788. target_port=self.PRINTER_MQTT_PORT,
  789. server_cert_path=self.cert_path,
  790. server_key_path=self.key_path,
  791. on_connect=lambda cid: self._log_activity("MQTT", f"connected: {cid}"),
  792. on_disconnect=lambda cid: self._log_activity("MQTT", f"disconnected: {cid}"),
  793. )
  794. # Start as background tasks
  795. async def run_with_logging(proxy: TLSProxy) -> None:
  796. try:
  797. await proxy.start()
  798. except Exception as e:
  799. logger.error("Slicer proxy %s failed: %s", proxy.name, e)
  800. self._tasks = [
  801. asyncio.create_task(
  802. run_with_logging(self._ftp_proxy),
  803. name="slicer_proxy_ftp",
  804. ),
  805. asyncio.create_task(
  806. run_with_logging(self._mqtt_proxy),
  807. name="slicer_proxy_mqtt",
  808. ),
  809. ]
  810. logger.info("Slicer TLS proxy started for %s", self.target_host)
  811. # Wait for tasks to complete (they run until cancelled)
  812. # This keeps the start() coroutine alive so the parent task doesn't complete
  813. try:
  814. await asyncio.gather(*self._tasks)
  815. except asyncio.CancelledError:
  816. logger.debug("Slicer proxy start cancelled")
  817. async def stop(self) -> None:
  818. """Stop all proxies."""
  819. logger.info("Stopping slicer proxy")
  820. # Stop proxies
  821. if self._ftp_proxy:
  822. await self._ftp_proxy.stop()
  823. self._ftp_proxy = None
  824. if self._mqtt_proxy:
  825. await self._mqtt_proxy.stop()
  826. self._mqtt_proxy = None
  827. # Cancel tasks
  828. for task in self._tasks:
  829. task.cancel()
  830. if self._tasks:
  831. try:
  832. await asyncio.wait_for(
  833. asyncio.gather(*self._tasks, return_exceptions=True),
  834. timeout=2.0,
  835. )
  836. except TimeoutError:
  837. logger.debug("Some proxy tasks didn't stop in time")
  838. self._tasks = []
  839. logger.info("Slicer proxy stopped")
  840. def _log_activity(self, name: str, message: str) -> None:
  841. """Log activity via callback if configured."""
  842. if self.on_activity:
  843. try:
  844. self.on_activity(name, message)
  845. except Exception:
  846. pass # Ignore activity callback errors; logging is non-critical
  847. @property
  848. def is_running(self) -> bool:
  849. """Check if proxies are running."""
  850. return len(self._tasks) > 0 and all(not t.done() for t in self._tasks)
  851. def get_status(self) -> dict:
  852. """Get proxy status."""
  853. return {
  854. "running": self.is_running,
  855. "target_host": self.target_host,
  856. "ftp_port": self.LOCAL_FTP_PORT,
  857. "mqtt_port": self.LOCAL_MQTT_PORT,
  858. "ftp_connections": (len(self._ftp_proxy._active_connections) if self._ftp_proxy else 0),
  859. "mqtt_connections": (len(self._mqtt_proxy._active_connections) if self._mqtt_proxy else 0),
  860. }