1
0

pattern_manager.py 90 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972
  1. import os
  2. from zoneinfo import ZoneInfo
  3. import threading
  4. import time
  5. import random
  6. import logging
  7. from datetime import datetime, time as datetime_time
  8. from tqdm import tqdm
  9. from modules.connection import connection_manager
  10. from modules.core.state import state
  11. from math import pi, isnan, isinf
  12. import asyncio
  13. import json
  14. from modules.led.idle_timeout_manager import idle_timeout_manager
  15. import queue
  16. from dataclasses import dataclass
  17. from typing import Optional, Callable, Literal
  18. # Configure logging
  19. logger = logging.getLogger(__name__)
  20. # Global state
  21. THETA_RHO_DIR = './patterns'
  22. os.makedirs(THETA_RHO_DIR, exist_ok=True)
  23. # Execution time log file (JSON Lines format - one JSON object per line)
  24. EXECUTION_LOG_FILE = './execution_times.jsonl'
  25. async def wait_with_interrupt(
  26. condition_fn: Callable[[], bool],
  27. check_stop: bool = True,
  28. check_skip: bool = True,
  29. interval: float = 1.0,
  30. ) -> Literal['completed', 'stopped', 'skipped']:
  31. """
  32. Wait while condition_fn() returns True, with instant interrupt support.
  33. Uses asyncio.Event for instant response to stop/skip requests rather than
  34. polling at fixed intervals. This ensures users get immediate feedback when
  35. pressing stop or skip buttons.
  36. Args:
  37. condition_fn: Function that returns True while waiting should continue
  38. check_stop: Whether to respond to stop requests (default True)
  39. check_skip: Whether to respond to skip requests (default True)
  40. interval: How often to re-check condition_fn in seconds (default 1.0)
  41. Returns:
  42. 'completed' - condition_fn() returned False (normal completion)
  43. 'stopped' - stop was requested
  44. 'skipped' - skip was requested
  45. Example:
  46. result = await wait_with_interrupt(
  47. lambda: state.pause_requested or is_in_scheduled_pause_period()
  48. )
  49. if result == 'stopped':
  50. return # Exit pattern execution
  51. if result == 'skipped':
  52. break # Skip to next pattern
  53. """
  54. while condition_fn():
  55. result = await state.wait_for_interrupt(
  56. timeout=interval,
  57. check_stop=check_stop,
  58. check_skip=check_skip,
  59. )
  60. if result == 'stopped':
  61. return 'stopped'
  62. if result == 'skipped':
  63. return 'skipped'
  64. # 'timeout' means we should re-check condition_fn
  65. return 'completed'
  66. def log_execution_time(pattern_name: str, table_type: str, speed: int, actual_time: float,
  67. total_coordinates: int, was_completed: bool):
  68. """Log pattern execution time to JSON Lines file for analysis.
  69. Args:
  70. pattern_name: Name of the pattern file
  71. table_type: Type of table (e.g., 'dune_weaver', 'dune_weaver_mini')
  72. speed: Speed setting used (0-255)
  73. actual_time: Actual execution time in seconds (excluding pauses)
  74. total_coordinates: Total number of coordinates in the pattern
  75. was_completed: Whether the pattern completed normally (not stopped/skipped)
  76. """
  77. # Format time as HH:MM:SS
  78. hours, remainder = divmod(int(actual_time), 3600)
  79. minutes, seconds = divmod(remainder, 60)
  80. time_formatted = f"{hours:02d}:{minutes:02d}:{seconds:02d}"
  81. log_entry = {
  82. "timestamp": datetime.now().isoformat(),
  83. "pattern_name": pattern_name,
  84. "table_type": table_type or "unknown",
  85. "speed": speed,
  86. "actual_time_seconds": round(actual_time, 2),
  87. "actual_time_formatted": time_formatted,
  88. "total_coordinates": total_coordinates,
  89. "completed": was_completed
  90. }
  91. try:
  92. with open(EXECUTION_LOG_FILE, 'a') as f:
  93. f.write(json.dumps(log_entry) + '\n')
  94. logger.info(f"Execution time logged: {pattern_name} - {time_formatted} (speed: {speed}, table: {table_type})")
  95. except Exception as e:
  96. logger.error(f"Failed to log execution time: {e}")
  97. def get_last_completed_execution_time(pattern_name: str, speed: float) -> Optional[dict]:
  98. """Get the last completed execution time for a pattern at a specific speed.
  99. Args:
  100. pattern_name: Name of the pattern file (e.g., 'circle.thr')
  101. speed: Speed setting to match
  102. Returns:
  103. Dict with execution time info if found, None otherwise.
  104. Format: {"actual_time_seconds": float, "actual_time_formatted": str, "timestamp": str}
  105. """
  106. if not os.path.exists(EXECUTION_LOG_FILE):
  107. return None
  108. try:
  109. matching_entry = None
  110. with open(EXECUTION_LOG_FILE, 'r') as f:
  111. for line in f:
  112. line = line.strip()
  113. if not line:
  114. continue
  115. try:
  116. entry = json.loads(line)
  117. # Only consider fully completed patterns (100% finished)
  118. if (entry.get('completed', False) and
  119. entry.get('pattern_name') == pattern_name and
  120. entry.get('speed') == speed):
  121. # Keep the most recent match (last one in file)
  122. matching_entry = entry
  123. except json.JSONDecodeError:
  124. continue
  125. if matching_entry:
  126. return {
  127. "actual_time_seconds": matching_entry.get('actual_time_seconds'),
  128. "actual_time_formatted": matching_entry.get('actual_time_formatted'),
  129. "timestamp": matching_entry.get('timestamp')
  130. }
  131. return None
  132. except Exception as e:
  133. logger.error(f"Failed to read execution time log: {e}")
  134. return None
  135. def get_pattern_execution_history(pattern_name: str) -> Optional[dict]:
  136. """Get the most recent completed execution for a pattern (any speed).
  137. Args:
  138. pattern_name: Name of the pattern file (e.g., 'circle.thr')
  139. Returns:
  140. Dict with execution time info if found, None otherwise.
  141. Format: {"actual_time_seconds": float, "actual_time_formatted": str,
  142. "speed": int, "timestamp": str}
  143. """
  144. if not os.path.exists(EXECUTION_LOG_FILE):
  145. return None
  146. try:
  147. matching_entry = None
  148. with open(EXECUTION_LOG_FILE, 'r') as f:
  149. for line in f:
  150. line = line.strip()
  151. if not line:
  152. continue
  153. try:
  154. entry = json.loads(line)
  155. # Only consider fully completed patterns
  156. if (entry.get('completed', False) and
  157. entry.get('pattern_name') == pattern_name):
  158. # Keep the most recent match (last one in file)
  159. matching_entry = entry
  160. except json.JSONDecodeError:
  161. continue
  162. if matching_entry:
  163. return {
  164. "actual_time_seconds": matching_entry.get('actual_time_seconds'),
  165. "actual_time_formatted": matching_entry.get('actual_time_formatted'),
  166. "speed": matching_entry.get('speed'),
  167. "timestamp": matching_entry.get('timestamp')
  168. }
  169. return None
  170. except Exception as e:
  171. logger.error(f"Failed to read execution time log: {e}")
  172. return None
  173. # Asyncio primitives - initialized lazily to avoid event loop issues
  174. # These must be created in the context of the running event loop
  175. pause_event: Optional[asyncio.Event] = None
  176. pattern_lock: Optional[asyncio.Lock] = None
  177. progress_update_task = None
  178. def get_pause_event() -> asyncio.Event:
  179. """Get or create the pause event in the current event loop."""
  180. global pause_event
  181. if pause_event is None:
  182. pause_event = asyncio.Event()
  183. pause_event.set() # Initially not paused
  184. return pause_event
  185. def get_pattern_lock() -> asyncio.Lock:
  186. """Get or create the pattern lock in the current event loop."""
  187. global pattern_lock
  188. if pattern_lock is None:
  189. pattern_lock = asyncio.Lock()
  190. return pattern_lock
  191. # Cache timezone at module level - read once per session (cleared when user changes timezone)
  192. _cached_timezone = None
  193. _cached_zoneinfo = None
  194. def _get_timezone():
  195. """Get and cache the timezone for Still Sands. Uses user-selected timezone if set, otherwise system timezone."""
  196. global _cached_timezone, _cached_zoneinfo
  197. if _cached_timezone is not None:
  198. return _cached_zoneinfo
  199. user_tz = 'UTC' # Default fallback
  200. # First, check if user has selected a specific timezone in settings
  201. if state.scheduled_pause_timezone:
  202. user_tz = state.scheduled_pause_timezone
  203. logger.info(f"Still Sands using timezone: {user_tz} (user-selected)")
  204. else:
  205. # Fall back to system timezone detection
  206. try:
  207. if os.path.exists('/etc/host-timezone'):
  208. with open('/etc/host-timezone', 'r') as f:
  209. user_tz = f.read().strip()
  210. logger.info(f"Still Sands using timezone: {user_tz} (from host system)")
  211. # Fallback to /etc/timezone if host-timezone doesn't exist
  212. elif os.path.exists('/etc/timezone'):
  213. with open('/etc/timezone', 'r') as f:
  214. user_tz = f.read().strip()
  215. logger.info(f"Still Sands using timezone: {user_tz} (from container)")
  216. # Fallback to TZ environment variable
  217. elif os.environ.get('TZ'):
  218. user_tz = os.environ.get('TZ')
  219. logger.info(f"Still Sands using timezone: {user_tz} (from environment)")
  220. else:
  221. logger.info("Still Sands using timezone: UTC (system default)")
  222. except Exception as e:
  223. logger.debug(f"Could not read timezone: {e}")
  224. # Cache the timezone
  225. _cached_timezone = user_tz
  226. try:
  227. _cached_zoneinfo = ZoneInfo(user_tz)
  228. except Exception as e:
  229. logger.warning(f"Invalid timezone '{user_tz}', falling back to system time: {e}")
  230. _cached_zoneinfo = None
  231. return _cached_zoneinfo
  232. def is_in_scheduled_pause_period():
  233. """Check if current time falls within any scheduled pause period."""
  234. if not state.scheduled_pause_enabled or not state.scheduled_pause_time_slots:
  235. return False
  236. # Get cached timezone (user-selected or system default)
  237. tz_info = _get_timezone()
  238. try:
  239. # Get current time in user's timezone
  240. if tz_info:
  241. now = datetime.now(tz_info)
  242. else:
  243. now = datetime.now()
  244. except Exception as e:
  245. logger.warning(f"Error getting current time: {e}")
  246. now = datetime.now()
  247. current_time = now.time()
  248. current_weekday = now.strftime("%A").lower() # monday, tuesday, etc.
  249. for slot in state.scheduled_pause_time_slots:
  250. # Parse start and end times
  251. try:
  252. start_time = datetime_time.fromisoformat(slot['start_time'])
  253. end_time = datetime_time.fromisoformat(slot['end_time'])
  254. except (ValueError, KeyError):
  255. logger.warning(f"Invalid time format in scheduled pause slot: {slot}")
  256. continue
  257. # Check if this slot applies to today
  258. slot_applies_today = False
  259. days_setting = slot.get('days', 'daily')
  260. if days_setting == 'daily':
  261. slot_applies_today = True
  262. elif days_setting == 'weekdays':
  263. slot_applies_today = current_weekday in ['monday', 'tuesday', 'wednesday', 'thursday', 'friday']
  264. elif days_setting == 'weekends':
  265. slot_applies_today = current_weekday in ['saturday', 'sunday']
  266. elif days_setting == 'custom':
  267. custom_days = slot.get('custom_days', [])
  268. slot_applies_today = current_weekday in custom_days
  269. if not slot_applies_today:
  270. continue
  271. # Check if current time is within the pause period
  272. if start_time <= end_time:
  273. # Normal case: start and end are on the same day
  274. if start_time <= current_time <= end_time:
  275. return True
  276. else:
  277. # Time spans midnight: start is before midnight, end is after midnight
  278. if current_time >= start_time or current_time <= end_time:
  279. return True
  280. return False
  281. async def check_table_is_idle() -> bool:
  282. """
  283. Check if the table is currently idle by querying actual machine status.
  284. Returns True if idle, False if playing/moving.
  285. This checks the real machine state rather than relying on state variables,
  286. making it more reliable for detecting when table is truly idle.
  287. """
  288. # Use the connection_manager's is_machine_idle() function
  289. # Run it in a thread since it's a synchronous function
  290. return await asyncio.to_thread(connection_manager.is_machine_idle)
  291. async def start_idle_led_timeout(check_still_sands: bool = True):
  292. """
  293. Set LED to idle state and start timeout if enabled.
  294. Handles Still Sands: if in scheduled pause period with LED control enabled,
  295. turns off LEDs instead of showing idle effect.
  296. Should be called whenever the table goes idle.
  297. Args:
  298. check_still_sands: If True, checks Still Sands period and turns off LEDs if applicable.
  299. Set to False when caller already handles Still Sands logic
  300. (e.g., during pause with "finish pattern first" mode).
  301. """
  302. if not state.led_controller:
  303. return
  304. # Still Sands with LED control: turn off instead of idle effect
  305. if check_still_sands and is_in_scheduled_pause_period() and state.scheduled_pause_control_wled:
  306. logger.info("Turning off LED lights during Still Sands period")
  307. await state.led_controller.set_power_async(0)
  308. return
  309. # Normal flow: show idle effect (only if one is configured)
  310. if not state.dw_led_idle_effect:
  311. logger.debug("No idle effect configured, leaving LEDs unchanged")
  312. return
  313. await state.led_controller.effect_idle_async(state.dw_led_idle_effect)
  314. # Start timeout if enabled
  315. if not state.dw_led_idle_timeout_enabled:
  316. logger.debug("Idle LED timeout not enabled")
  317. return
  318. timeout_minutes = state.dw_led_idle_timeout_minutes
  319. if timeout_minutes <= 0:
  320. logger.debug("Idle LED timeout not configured (timeout <= 0)")
  321. return
  322. logger.debug(f"Starting idle LED timeout: {timeout_minutes} minutes")
  323. idle_timeout_manager.start_idle_timeout(
  324. timeout_minutes=timeout_minutes,
  325. state=state,
  326. check_idle_callback=check_table_is_idle
  327. )
  328. # Motion Control Thread Infrastructure
  329. @dataclass
  330. class MotionCommand:
  331. """Represents a motion command for the motion control thread."""
  332. command_type: str # 'move', 'stop', 'pause', 'resume', 'shutdown'
  333. theta: Optional[float] = None
  334. rho: Optional[float] = None
  335. speed: Optional[float] = None
  336. callback: Optional[Callable] = None
  337. future: Optional[asyncio.Future] = None
  338. class MotionControlThread:
  339. """Dedicated thread for hardware motion control operations."""
  340. def __init__(self):
  341. self.command_queue = queue.Queue()
  342. self.thread = None
  343. self.running = False
  344. self.paused = False
  345. def start(self):
  346. """Start the motion control thread with elevated priority."""
  347. if self.thread and self.thread.is_alive():
  348. return
  349. self.running = True
  350. self.thread = threading.Thread(target=self._motion_loop, daemon=True)
  351. self.thread.start()
  352. logger.info("Motion control thread started")
  353. def stop(self):
  354. """Stop the motion control thread."""
  355. if not self.running:
  356. return
  357. self.running = False
  358. # Send shutdown command
  359. self.command_queue.put(MotionCommand('shutdown'))
  360. if self.thread and self.thread.is_alive():
  361. self.thread.join(timeout=5.0)
  362. logger.info("Motion control thread stopped")
  363. def _motion_loop(self):
  364. """Main loop for the motion control thread."""
  365. logger.info("Motion control thread loop started")
  366. while self.running:
  367. try:
  368. # Get command with timeout to allow periodic checks
  369. command = self.command_queue.get(timeout=1.0)
  370. if command.command_type == 'shutdown':
  371. break
  372. elif command.command_type == 'move':
  373. self._execute_move(command)
  374. elif command.command_type == 'pause':
  375. self.paused = True
  376. elif command.command_type == 'resume':
  377. self.paused = False
  378. elif command.command_type == 'stop':
  379. # Clear any pending commands
  380. while not self.command_queue.empty():
  381. try:
  382. self.command_queue.get_nowait()
  383. except queue.Empty:
  384. break
  385. self.command_queue.task_done()
  386. except queue.Empty:
  387. # Timeout - continue loop for shutdown check
  388. continue
  389. except Exception as e:
  390. logger.error(f"Error in motion control thread: {e}")
  391. logger.info("Motion control thread loop ended")
  392. def _execute_move(self, command: MotionCommand):
  393. """Execute a move command in the motion thread."""
  394. try:
  395. # Wait if paused
  396. while self.paused and self.running:
  397. time.sleep(0.1)
  398. if not self.running:
  399. return
  400. # Execute the actual motion using sync version
  401. self._move_polar_sync(command.theta, command.rho, command.speed)
  402. # Signal completion if future provided
  403. if command.future and not command.future.done():
  404. command.future.get_loop().call_soon_threadsafe(
  405. command.future.set_result, None
  406. )
  407. except Exception as e:
  408. logger.error(f"Error executing move command: {e}")
  409. if command.future and not command.future.done():
  410. command.future.get_loop().call_soon_threadsafe(
  411. command.future.set_exception, e
  412. )
  413. def _move_polar_sync(self, theta: float, rho: float, speed: Optional[float] = None):
  414. """Synchronous version of move_polar for use in motion thread."""
  415. # Check for valid machine position (can be None if homing failed)
  416. if state.machine_x is None or state.machine_y is None:
  417. logger.error("Cannot execute move: machine position unknown (homing may have failed)")
  418. logger.error("Please home the machine before running patterns")
  419. state.stop_requested = True
  420. return
  421. # This is the original sync logic but running in dedicated thread
  422. if state.table_type == 'dune_weaver_mini':
  423. x_scaling_factor = 2
  424. y_scaling_factor = 3.7
  425. else:
  426. x_scaling_factor = 2
  427. y_scaling_factor = 5
  428. delta_theta = theta - state.current_theta
  429. delta_rho = rho - state.current_rho
  430. x_increment = delta_theta * 100 / (2 * pi * x_scaling_factor)
  431. y_increment = delta_rho * 100 / y_scaling_factor
  432. x_total_steps = state.x_steps_per_mm * (100/x_scaling_factor)
  433. y_total_steps = state.y_steps_per_mm * (100/y_scaling_factor)
  434. offset = x_increment * (x_total_steps * x_scaling_factor / (state.gear_ratio * y_total_steps * y_scaling_factor))
  435. if state.table_type == 'dune_weaver_mini' or state.y_steps_per_mm == 546:
  436. y_increment -= offset
  437. else:
  438. y_increment += offset
  439. new_x_abs = state.machine_x + x_increment
  440. new_y_abs = state.machine_y + y_increment
  441. # Use provided speed or fall back to state.speed
  442. actual_speed = speed if speed is not None else state.speed
  443. # Validate coordinates before sending to prevent GRBL error:2
  444. if isnan(new_x_abs) or isnan(new_y_abs) or isinf(new_x_abs) or isinf(new_y_abs):
  445. logger.error(f"Motion thread: Invalid coordinates detected - X:{new_x_abs}, Y:{new_y_abs}")
  446. logger.error(f" theta:{theta}, rho:{rho}, current_theta:{state.current_theta}, current_rho:{state.current_rho}")
  447. logger.error(f" x_steps_per_mm:{state.x_steps_per_mm}, y_steps_per_mm:{state.y_steps_per_mm}, gear_ratio:{state.gear_ratio}")
  448. state.stop_requested = True
  449. return
  450. # Call sync version of send_grbl_coordinates in this thread
  451. # Use 2 decimal precision to reduce GRBL parsing overhead
  452. self._send_grbl_coordinates_sync(round(new_x_abs, 2), round(new_y_abs, 2), actual_speed)
  453. # Update state
  454. state.current_theta = theta
  455. state.current_rho = rho
  456. state.machine_x = new_x_abs
  457. state.machine_y = new_y_abs
  458. def _send_grbl_coordinates_sync(self, x: float, y: float, speed: int = 600, timeout: int = 2, home: bool = False):
  459. """Synchronous version of send_grbl_coordinates for motion thread.
  460. Waits for 'ok' with a timeout. GRBL sends 'ok' after the move completes,
  461. which can take many seconds at slow speeds. We use a generous timeout
  462. (120 seconds) to handle slow movements, but prevent indefinite hangs.
  463. Includes retry logic for serial corruption errors (common on Pi 3B+).
  464. """
  465. gcode = f"$J=G91 G21 Y{y:.2f} F{speed}" if home else f"G1 X{x:.2f} Y{y:.2f} F{speed}"
  466. max_wait_time = 120 # Maximum seconds to wait for 'ok' response
  467. max_corruption_retries = 10 # Max retries for corruption-type errors
  468. max_timeout_retries = 10 # Max retries for timeout (lost 'ok' response)
  469. corruption_retry_count = 0
  470. timeout_retry_count = 0
  471. # GRBL error codes that indicate likely serial corruption (syntax errors)
  472. # These are recoverable by resending the command
  473. corruption_error_codes = {
  474. 'error:1', # Expected command letter
  475. 'error:2', # Bad number format
  476. 'error:20', # Invalid gcode ID (e.g., G5s instead of G53)
  477. 'error:21', # Invalid gcode command value
  478. 'error:22', # Invalid gcode command value in negative
  479. 'error:23', # Invalid gcode command value in decimal
  480. }
  481. while True:
  482. # Check stop_requested at the start of each iteration
  483. if state.stop_requested:
  484. logger.debug("Motion thread: Stop requested, aborting command")
  485. return False
  486. try:
  487. # Clear any stale input data before sending to prevent interleaving
  488. # This helps with timing issues on slower UARTs like Pi 3B+
  489. if hasattr(state.conn, 'reset_input_buffer'):
  490. state.conn.reset_input_buffer()
  491. logger.debug(f"Motion thread sending G-code: {gcode}")
  492. state.conn.send(gcode + "\n")
  493. # Small delay for serial buffer to stabilize on slower UARTs
  494. # Prevents timing-related corruption on Pi 3B+
  495. time.sleep(0.005)
  496. # Wait for 'ok' with timeout
  497. wait_start = time.time()
  498. while True:
  499. # Check stop_requested while waiting
  500. if state.stop_requested:
  501. logger.debug("Motion thread: Stop requested while waiting for response")
  502. return False
  503. # Check for timeout
  504. elapsed = time.time() - wait_start
  505. if elapsed > max_wait_time:
  506. logger.warning(f"Motion thread: Timeout ({max_wait_time}s) waiting for 'ok' response")
  507. logger.warning(f"Motion thread: Failed command was: {gcode}")
  508. # Attempt to recover by checking machine status
  509. # The 'ok' might have been lost but command may have executed
  510. logger.info("Motion thread: Attempting timeout recovery - checking machine status")
  511. logger.info(f"Motion thread: Current retry counts - timeout: {timeout_retry_count}/{max_timeout_retries}, corruption: {corruption_retry_count}/{max_corruption_retries}")
  512. try:
  513. # Check connection state first
  514. conn_type = type(state.conn).__name__ if state.conn else "None"
  515. logger.info(f"Motion thread: Connection type: {conn_type}")
  516. if not state.conn:
  517. logger.error("Motion thread: Connection object is None!")
  518. raise Exception("Connection is None")
  519. # Clear buffer first
  520. if hasattr(state.conn, 'reset_input_buffer'):
  521. state.conn.reset_input_buffer()
  522. logger.info("Motion thread: Input buffer cleared")
  523. else:
  524. logger.warning("Motion thread: Connection has no reset_input_buffer method")
  525. # Check if there's data waiting before we send
  526. if hasattr(state.conn, 'in_waiting'):
  527. waiting = state.conn.in_waiting()
  528. logger.info(f"Motion thread: Bytes waiting in buffer after clear: {waiting}")
  529. # Send status query
  530. logger.info("Motion thread: Sending status query '?'...")
  531. state.conn.send("?\n")
  532. time.sleep(0.2)
  533. logger.info("Motion thread: Status query sent, reading responses...")
  534. # Try to read status response
  535. status_response = None
  536. responses_received = []
  537. for i in range(10):
  538. resp = state.conn.readline()
  539. if resp:
  540. responses_received.append(resp)
  541. logger.info(f"Motion thread: Recovery response [{i+1}/10]: '{resp}'")
  542. if '<' in resp or 'Idle' in resp or 'Run' in resp or 'Hold' in resp or 'Alarm' in resp:
  543. status_response = resp
  544. logger.info(f"Motion thread: Found valid status response: '{resp}'")
  545. break
  546. # Also check for 'ok' that might have been delayed
  547. if resp.lower() == 'ok':
  548. logger.info("Motion thread: Received delayed 'ok' during recovery - SUCCESS")
  549. return True
  550. else:
  551. logger.debug(f"Motion thread: Recovery read [{i+1}/10]: no data (timeout)")
  552. time.sleep(0.05)
  553. # Log summary of what we received
  554. if responses_received:
  555. logger.info(f"Motion thread: Total responses received during recovery: {len(responses_received)}")
  556. logger.info(f"Motion thread: All responses: {responses_received}")
  557. else:
  558. logger.warning("Motion thread: No responses received during recovery - connection may be dead")
  559. if status_response:
  560. if 'Idle' in status_response:
  561. # Machine is idle - command likely completed, 'ok' was lost
  562. logger.info("Motion thread: Machine is Idle - assuming command completed (ok was lost) - SUCCESS")
  563. return True
  564. elif 'Run' in status_response:
  565. # Machine still running - extend timeout
  566. logger.info("Motion thread: Machine still running, extending wait time")
  567. wait_start = time.time() # Reset timeout
  568. continue
  569. elif 'Hold' in status_response:
  570. # Machine is in Hold state - attempt to resume
  571. logger.warning(f"Motion thread: Machine in Hold state: '{status_response}'")
  572. logger.info("Motion thread: Sending cycle start command '~' to resume from Hold...")
  573. # Send cycle start command to resume
  574. state.conn.send("~\n")
  575. time.sleep(0.3) # Give time for resume to process
  576. # Re-check status after resume attempt
  577. state.conn.send("?\n")
  578. time.sleep(0.2)
  579. # Read new status
  580. resume_response = None
  581. for _ in range(5):
  582. resp = state.conn.readline()
  583. if resp:
  584. logger.info(f"Motion thread: Post-resume response: '{resp}'")
  585. if '<' in resp:
  586. resume_response = resp
  587. break
  588. time.sleep(0.05)
  589. if resume_response:
  590. if 'Idle' in resume_response:
  591. logger.info("Motion thread: Machine resumed and is now Idle - SUCCESS")
  592. return True
  593. elif 'Run' in resume_response:
  594. logger.info("Motion thread: Machine resumed and running, extending wait time")
  595. wait_start = time.time()
  596. continue
  597. elif 'Hold' in resume_response:
  598. # Still in Hold - may need user intervention
  599. logger.warning(f"Motion thread: Still in Hold after resume: '{resume_response}'")
  600. else:
  601. logger.warning("Motion thread: No response after resume attempt")
  602. elif 'Alarm' in status_response:
  603. # Machine is in Alarm state - attempt to unlock
  604. logger.warning(f"Motion thread: Machine in ALARM state: '{status_response}'")
  605. logger.info("Motion thread: Sending $X to unlock from Alarm...")
  606. # Send unlock command
  607. state.conn.send("$X\n")
  608. time.sleep(0.5) # Give time for unlock to process
  609. # Re-check status after unlock attempt
  610. state.conn.send("?\n")
  611. time.sleep(0.2)
  612. # Read new status
  613. unlock_response = None
  614. for _ in range(5):
  615. resp = state.conn.readline()
  616. if resp:
  617. logger.info(f"Motion thread: Post-unlock response: '{resp}'")
  618. if '<' in resp:
  619. unlock_response = resp
  620. break
  621. time.sleep(0.05)
  622. if unlock_response:
  623. if 'Idle' in unlock_response:
  624. logger.info("Motion thread: Machine unlocked and is now Idle - retrying command")
  625. # Don't return True - we need to resend the failed command
  626. break # Break inner loop to retry the command
  627. elif 'Alarm' in unlock_response:
  628. # Still in Alarm - underlying issue persists (e.g., sensor triggered)
  629. logger.error(f"Motion thread: Still in ALARM after unlock: '{unlock_response}'")
  630. logger.error("Motion thread: Machine may need physical attention")
  631. state.stop_requested = True
  632. return False
  633. else:
  634. logger.warning("Motion thread: No response after unlock attempt")
  635. else:
  636. logger.warning(f"Motion thread: Unrecognized status response: '{status_response}'")
  637. else:
  638. logger.warning("Motion thread: No valid status response found in any received data")
  639. # No valid status response - connection may be dead
  640. timeout_retry_count += 1
  641. if timeout_retry_count <= max_timeout_retries:
  642. logger.warning(f"Motion thread: Recovery failed, will retry command ({timeout_retry_count}/{max_timeout_retries})")
  643. time.sleep(0.1)
  644. break # Break inner loop to resend command
  645. else:
  646. logger.error(f"Motion thread: Max timeout retries ({max_timeout_retries}) exceeded")
  647. except Exception as e:
  648. logger.error(f"Motion thread: Error during timeout recovery: {e}")
  649. import traceback
  650. logger.error(f"Motion thread: Traceback: {traceback.format_exc()}")
  651. # Max retries exceeded or recovery failed
  652. logger.error("=" * 60)
  653. logger.error("Motion thread: TIMEOUT RECOVERY FAILED - STOPPING PATTERN")
  654. logger.error(f" Failed command: {gcode}")
  655. logger.error(f" Timeout retries used: {timeout_retry_count}/{max_timeout_retries}")
  656. logger.error(f" Corruption retries used: {corruption_retry_count}/{max_corruption_retries}")
  657. logger.error(" Possible causes:")
  658. logger.error(" - Serial connection lost or unstable")
  659. logger.error(" - Hardware controller unresponsive")
  660. logger.error(" - USB power issue (try powered hub)")
  661. logger.error("=" * 60)
  662. state.stop_requested = True
  663. return False
  664. response = state.conn.readline()
  665. if response:
  666. logger.debug(f"Motion thread response: {response}")
  667. if response.lower() == "ok":
  668. logger.debug("Motion thread: Command execution confirmed.")
  669. # Reset corruption retry count on success
  670. if corruption_retry_count > 0:
  671. logger.info(f"Motion thread: Command succeeded after {corruption_retry_count} corruption retry(ies)")
  672. return True
  673. # Handle GRBL errors
  674. if response.lower().startswith("error"):
  675. error_code = response.lower().split()[0] if response else ""
  676. # Check if this is a corruption-type error (recoverable)
  677. if error_code in corruption_error_codes:
  678. corruption_retry_count += 1
  679. if corruption_retry_count <= max_corruption_retries:
  680. logger.warning(f"Motion thread: Likely serial corruption detected ({response})")
  681. logger.warning(f"Motion thread: Retrying command ({corruption_retry_count}/{max_corruption_retries}): {gcode}")
  682. # Clear buffer and wait longer before retry
  683. if hasattr(state.conn, 'reset_input_buffer'):
  684. state.conn.reset_input_buffer()
  685. time.sleep(0.02) # 20ms delay before retry
  686. break # Break inner loop to retry send
  687. else:
  688. logger.error(f"Motion thread: Max corruption retries ({max_corruption_retries}) exceeded")
  689. logger.error(f"Motion thread: GRBL error received: {response}")
  690. logger.error(f"Failed command: {gcode}")
  691. logger.error("Stopping pattern due to persistent serial corruption")
  692. state.stop_requested = True
  693. return False
  694. else:
  695. # Non-corruption error - stop immediately
  696. logger.error(f"Motion thread: GRBL error received: {response}")
  697. logger.error(f"Failed command: {gcode}")
  698. logger.error("Stopping pattern due to GRBL error")
  699. state.stop_requested = True
  700. return False
  701. # Handle GRBL alarms - machine needs attention
  702. if "alarm" in response.lower():
  703. logger.error(f"Motion thread: GRBL ALARM: {response}")
  704. logger.error("Machine alarm triggered - stopping pattern")
  705. state.stop_requested = True
  706. return False
  707. # FluidNC may echo commands back before sending 'ok'
  708. # Silently ignore echoed G-code commands (G0, G1, $J, etc.)
  709. if response.startswith(('G0', 'G1', 'G2', 'G3', '$J', 'M')):
  710. logger.debug(f"Motion thread: Ignoring echoed command: {response}")
  711. continue # Read next line to get 'ok'
  712. # Check for corruption indicator in MSG:ERR responses
  713. if 'MSG:ERR' in response and 'Bad GCode' in response:
  714. corruption_retry_count += 1
  715. if corruption_retry_count <= max_corruption_retries:
  716. logger.warning(f"Motion thread: Corrupted command detected: {response}")
  717. logger.warning(f"Motion thread: Retrying command ({corruption_retry_count}/{max_corruption_retries}): {gcode}")
  718. # Don't break yet - wait for the error:XX that follows
  719. continue
  720. # If we've exceeded retries, the error:XX handler above will catch it
  721. # Log truly unexpected responses
  722. logger.warning(f"Motion thread: Unexpected response: '{response}'")
  723. else:
  724. # Log periodically when waiting for response (every 30s)
  725. if int(elapsed) > 0 and int(elapsed) % 30 == 0 and elapsed - int(elapsed) < 0.1:
  726. logger.warning(f"Motion thread: Still waiting for 'ok' after {int(elapsed)}s for command: {gcode}")
  727. else:
  728. # Inner while loop completed without break - shouldn't happen normally
  729. # This means we hit timeout, which is handled above
  730. continue
  731. except Exception as e:
  732. error_str = str(e)
  733. logger.warning(f"Motion thread error sending command: {error_str}")
  734. # Immediately return for device not configured errors
  735. if "Device not configured" in error_str or "Errno 6" in error_str:
  736. logger.error(f"Motion thread: Device configuration error detected: {error_str}")
  737. state.stop_requested = True
  738. state.conn = None
  739. state.is_connected = False
  740. logger.info("Connection marked as disconnected due to device error")
  741. return False
  742. # Retry on exception or corruption error
  743. logger.warning(f"Motion thread: Retrying {gcode}...")
  744. time.sleep(0.1)
  745. # Global motion control thread instance
  746. motion_controller = MotionControlThread()
  747. async def cleanup_pattern_manager():
  748. """Clean up pattern manager resources"""
  749. global progress_update_task, pattern_lock, pause_event
  750. try:
  751. # Signal stop to allow any running pattern to exit gracefully
  752. state.stop_requested = True
  753. # Stop motion control thread
  754. motion_controller.stop()
  755. # Cancel progress update task if running
  756. if progress_update_task and not progress_update_task.done():
  757. try:
  758. progress_update_task.cancel()
  759. # Wait for task to actually cancel
  760. try:
  761. await progress_update_task
  762. except asyncio.CancelledError:
  763. pass
  764. except Exception as e:
  765. logger.error(f"Error cancelling progress update task: {e}")
  766. # Clean up pattern lock - wait for it to be released naturally, don't force release
  767. # Force releasing an asyncio.Lock can corrupt internal state if held by another coroutine
  768. current_lock = pattern_lock
  769. if current_lock and current_lock.locked():
  770. logger.info("Pattern lock is held, waiting for release (max 5s)...")
  771. try:
  772. # Wait with timeout for the lock to become available
  773. # Use wait_for for Python 3.9 compatibility (asyncio.timeout is 3.11+)
  774. async def acquire_lock():
  775. async with current_lock:
  776. pass # Lock acquired means previous holder released it
  777. await asyncio.wait_for(acquire_lock(), timeout=5.0)
  778. logger.info("Pattern lock released normally")
  779. except asyncio.TimeoutError:
  780. logger.warning("Timed out waiting for pattern lock - creating fresh lock")
  781. except Exception as e:
  782. logger.error(f"Error waiting for pattern lock: {e}")
  783. # Clean up pause event - wake up any waiting tasks, then create fresh event
  784. current_event = pause_event
  785. if current_event:
  786. try:
  787. current_event.set() # Wake up any waiting tasks
  788. except Exception as e:
  789. logger.error(f"Error setting pause event: {e}")
  790. # Clean up pause condition from state
  791. if state.pause_condition:
  792. try:
  793. with state.pause_condition:
  794. state.pause_condition.notify_all()
  795. state.pause_condition = threading.Condition()
  796. except Exception as e:
  797. logger.error(f"Error cleaning up pause condition: {e}")
  798. # Clear all state variables
  799. state.current_playing_file = None
  800. state.execution_progress = 0
  801. state.is_running = False
  802. state.pause_requested = False
  803. state.stop_requested = True
  804. state.is_clearing = False
  805. # Reset machine position
  806. await connection_manager.update_machine_position()
  807. logger.info("Pattern manager resources cleaned up")
  808. except Exception as e:
  809. logger.error(f"Error during pattern manager cleanup: {e}")
  810. finally:
  811. # Reset to fresh instances instead of None to allow continued operation
  812. progress_update_task = None
  813. pattern_lock = asyncio.Lock() # Fresh lock instead of None
  814. pause_event = asyncio.Event() # Fresh event instead of None
  815. pause_event.set() # Initially not paused
  816. def list_theta_rho_files():
  817. files = []
  818. for root, dirs, filenames in os.walk(THETA_RHO_DIR):
  819. # Skip cached_images directories to avoid scanning thousands of WebP files
  820. if 'cached_images' in dirs:
  821. dirs.remove('cached_images')
  822. # Filter .thr files during traversal for better performance
  823. thr_files = [f for f in filenames if f.endswith('.thr')]
  824. for file in thr_files:
  825. relative_path = os.path.relpath(os.path.join(root, file), THETA_RHO_DIR)
  826. # Normalize path separators to always use forward slashes for consistency across platforms
  827. relative_path = relative_path.replace(os.sep, '/')
  828. files.append(relative_path)
  829. logger.debug(f"Found {len(files)} theta-rho files")
  830. return files
  831. def parse_theta_rho_file(file_path):
  832. """Parse a theta-rho file and return a list of (theta, rho) pairs."""
  833. coordinates = []
  834. try:
  835. logger.debug(f"Parsing theta-rho file: {file_path}")
  836. with open(file_path, 'r', encoding='utf-8') as file:
  837. for line in file:
  838. line = line.strip()
  839. if not line or line.startswith("#"):
  840. continue
  841. try:
  842. theta, rho = map(float, line.split())
  843. coordinates.append((theta, rho))
  844. except ValueError:
  845. logger.warning(f"Skipping invalid line: {line}")
  846. continue
  847. except Exception as e:
  848. logger.error(f"Error reading file: {e}")
  849. return coordinates
  850. logger.debug(f"Parsed {len(coordinates)} coordinates from {file_path}")
  851. return coordinates
  852. def get_first_rho_from_cache(file_path, cache_data=None):
  853. """Get the first rho value from cached metadata, falling back to file parsing if needed.
  854. Args:
  855. file_path: Path to the pattern file
  856. cache_data: Optional pre-loaded cache data dict to avoid repeated disk I/O
  857. """
  858. try:
  859. # Import cache_manager locally to avoid circular import
  860. from modules.core import cache_manager
  861. # Try to get from metadata cache first
  862. # Use relative path from THETA_RHO_DIR to match cache keys (which include subdirectories)
  863. file_name = os.path.relpath(file_path, THETA_RHO_DIR)
  864. # Use provided cache_data if available, otherwise load from disk
  865. if cache_data is not None:
  866. # Extract metadata directly from provided cache
  867. data_section = cache_data.get('data', {})
  868. if file_name in data_section:
  869. cached_entry = data_section[file_name]
  870. metadata = cached_entry.get('metadata')
  871. # When cache_data is provided, trust it without checking mtime
  872. # This significantly speeds up bulk operations (playlists with 1000+ patterns)
  873. # by avoiding 1000+ os.path.getmtime() calls on slow storage (e.g., Pi SD cards)
  874. if metadata and 'first_coordinate' in metadata:
  875. return metadata['first_coordinate']['y']
  876. else:
  877. # Fall back to loading cache from disk (original behavior)
  878. metadata = cache_manager.get_pattern_metadata(file_name)
  879. if metadata and 'first_coordinate' in metadata:
  880. # In the cache, 'x' is theta and 'y' is rho
  881. return metadata['first_coordinate']['y']
  882. # Fallback to parsing the file if not in cache
  883. logger.debug(f"Metadata not cached for {file_name}, parsing file")
  884. coordinates = parse_theta_rho_file(file_path)
  885. if coordinates:
  886. return coordinates[0][1] # Return rho value
  887. return None
  888. except Exception as e:
  889. logger.warning(f"Error getting first rho from cache for {file_path}: {str(e)}")
  890. return None
  891. def get_clear_pattern_file(clear_pattern_mode, path=None, cache_data=None):
  892. """Return a .thr file path based on pattern_name and table type.
  893. Args:
  894. clear_pattern_mode: The clear pattern mode to use
  895. path: Optional path to the pattern file for adaptive mode
  896. cache_data: Optional pre-loaded cache data dict to avoid repeated disk I/O
  897. """
  898. if not clear_pattern_mode or clear_pattern_mode == 'none':
  899. return
  900. # Define patterns for each table type
  901. clear_patterns = {
  902. 'dune_weaver': {
  903. 'clear_from_out': './patterns/clear_from_out.thr',
  904. 'clear_from_in': './patterns/clear_from_in.thr',
  905. 'clear_sideway': './patterns/clear_sideway.thr'
  906. },
  907. 'dune_weaver_mini': {
  908. 'clear_from_out': './patterns/clear_from_out_mini.thr',
  909. 'clear_from_in': './patterns/clear_from_in_mini.thr',
  910. 'clear_sideway': './patterns/clear_sideway_mini.thr'
  911. },
  912. 'dune_weaver_mini_pro': {
  913. 'clear_from_out': './patterns/clear_from_out_mini.thr',
  914. 'clear_from_in': './patterns/clear_from_in_mini.thr',
  915. 'clear_sideway': './patterns/clear_sideway_mini.thr'
  916. },
  917. 'dune_weaver_pro': {
  918. 'clear_from_out': './patterns/clear_from_out_pro.thr',
  919. 'clear_from_out_Ultra': './patterns/clear_from_out_Ultra.thr',
  920. 'clear_from_in': './patterns/clear_from_in_pro.thr',
  921. 'clear_from_in_Ultra': './patterns/clear_from_in_Ultra.thr',
  922. 'clear_sideway': './patterns/clear_sideway_pro.thr'
  923. }
  924. }
  925. # Get patterns for current table type, fallback to standard patterns if type not found
  926. table_patterns = clear_patterns.get(state.table_type, clear_patterns['dune_weaver'])
  927. # Check for custom patterns first
  928. if state.custom_clear_from_out and clear_pattern_mode in ['clear_from_out', 'adaptive']:
  929. if clear_pattern_mode == 'adaptive':
  930. # For adaptive mode, use cached metadata to check first rho
  931. if path:
  932. first_rho = get_first_rho_from_cache(path, cache_data)
  933. if first_rho is not None and first_rho < 0.5:
  934. # Use custom clear_from_out if set
  935. custom_path = os.path.join('./patterns', state.custom_clear_from_out)
  936. if os.path.exists(custom_path):
  937. logger.debug(f"Using custom clear_from_out: {custom_path}")
  938. return custom_path
  939. elif clear_pattern_mode == 'clear_from_out':
  940. custom_path = os.path.join('./patterns', state.custom_clear_from_out)
  941. if os.path.exists(custom_path):
  942. logger.debug(f"Using custom clear_from_out: {custom_path}")
  943. return custom_path
  944. if state.custom_clear_from_in and clear_pattern_mode in ['clear_from_in', 'adaptive']:
  945. if clear_pattern_mode == 'adaptive':
  946. # For adaptive mode, use cached metadata to check first rho
  947. if path:
  948. first_rho = get_first_rho_from_cache(path, cache_data)
  949. if first_rho is not None and first_rho >= 0.5:
  950. # Use custom clear_from_in if set
  951. custom_path = os.path.join('./patterns', state.custom_clear_from_in)
  952. if os.path.exists(custom_path):
  953. logger.debug(f"Using custom clear_from_in: {custom_path}")
  954. return custom_path
  955. elif clear_pattern_mode == 'clear_from_in':
  956. custom_path = os.path.join('./patterns', state.custom_clear_from_in)
  957. if os.path.exists(custom_path):
  958. logger.debug(f"Using custom clear_from_in: {custom_path}")
  959. return custom_path
  960. logger.debug(f"Clear pattern mode: {clear_pattern_mode} for table type: {state.table_type}")
  961. if clear_pattern_mode == "random":
  962. return random.choice(list(table_patterns.values()))
  963. if clear_pattern_mode == 'adaptive':
  964. if not path:
  965. logger.warning("No path provided for adaptive clear pattern")
  966. return random.choice(list(table_patterns.values()))
  967. # Use cached metadata to get first rho value
  968. first_rho = get_first_rho_from_cache(path, cache_data)
  969. if first_rho is None:
  970. logger.warning("Could not determine first rho value for adaptive clear pattern")
  971. return random.choice(list(table_patterns.values()))
  972. if first_rho < 0.5:
  973. return table_patterns['clear_from_out']
  974. else:
  975. return table_patterns['clear_from_in']
  976. else:
  977. if clear_pattern_mode not in table_patterns:
  978. return False
  979. return table_patterns[clear_pattern_mode]
  980. def is_clear_pattern(file_path):
  981. """Check if a file path is a clear pattern file."""
  982. # Get all possible clear pattern files for all table types
  983. clear_patterns = []
  984. for table_type in ['dune_weaver', 'dune_weaver_mini', 'dune_weaver_pro']:
  985. clear_patterns.extend([
  986. f'./patterns/clear_from_out{("_" + table_type.split("_")[-1]) if table_type != "dune_weaver" else ""}.thr',
  987. f'./patterns/clear_from_in{("_" + table_type.split("_")[-1]) if table_type != "dune_weaver" else ""}.thr',
  988. f'./patterns/clear_sideway{("_" + table_type.split("_")[-1]) if table_type != "dune_weaver" else ""}.thr'
  989. ])
  990. # Normalize paths for comparison
  991. normalized_path = os.path.normpath(file_path)
  992. normalized_clear_patterns = [os.path.normpath(p) for p in clear_patterns]
  993. # Check if the file path matches any clear pattern path
  994. return normalized_path in normalized_clear_patterns
  995. async def _execute_pattern_internal(file_path):
  996. """Internal function to execute a pattern file. Must be called with lock already held.
  997. Args:
  998. file_path: Path to the .thr file to execute
  999. Returns:
  1000. True if pattern completed successfully, False if stopped/skipped
  1001. """
  1002. # Run file parsing in thread to avoid blocking the event loop
  1003. coordinates = await asyncio.to_thread(parse_theta_rho_file, file_path)
  1004. total_coordinates = len(coordinates)
  1005. # Cache coordinates in state for frontend preview (avoids re-parsing large files)
  1006. state._current_coordinates = coordinates
  1007. if total_coordinates < 2:
  1008. logger.warning("Not enough coordinates for interpolation")
  1009. return False
  1010. # Pre-calculate rho-based weights for more accurate time estimation
  1011. # Moves near center (low rho) are slower than perimeter moves due to
  1012. # polar geometry - less linear distance per theta change at low rho
  1013. def calc_move_weight(rho):
  1014. # Weight inversely proportional to rho, with floor to avoid extreme values
  1015. # At rho=0: weight≈6.7, at rho=0.5: weight≈1.5, at rho=1.0: weight≈0.87
  1016. return 1.0 / (rho + 0.15)
  1017. coord_weights = [calc_move_weight(rho) for _, rho in coordinates]
  1018. total_weight = sum(coord_weights)
  1019. # Determine if this is a clearing pattern
  1020. is_clear_file = is_clear_pattern(file_path)
  1021. if is_clear_file:
  1022. initial_speed = state.clear_pattern_speed if state.clear_pattern_speed is not None else state.speed
  1023. logger.info(f"Running clearing pattern at initial speed {initial_speed}")
  1024. else:
  1025. logger.info(f"Running normal pattern at initial speed {state.speed}")
  1026. state.execution_progress = (0, total_coordinates, None, 0)
  1027. # stop actions without resetting the playlist, and don't wait for lock (we already have it)
  1028. # Preserve is_clearing flag since stop_actions resets it
  1029. was_clearing = state.is_clearing
  1030. await stop_actions(clear_playlist=False, wait_for_lock=False)
  1031. state.is_clearing = was_clearing
  1032. state.current_playing_file = file_path
  1033. state.stop_requested = False
  1034. # Reset LED idle timeout activity time when pattern starts
  1035. import time as time_module
  1036. state.dw_led_last_activity_time = time_module.time()
  1037. logger.info(f"Starting pattern execution: {file_path}")
  1038. logger.info(f"t: {state.current_theta}, r: {state.current_rho}")
  1039. await reset_theta()
  1040. start_time = time.time()
  1041. total_pause_time = 0 # Track total time spent paused (manual + scheduled)
  1042. completed_weight = 0.0 # Track rho-weighted progress
  1043. smoothed_rate = None # For exponential smoothing of time-per-unit-weight rate
  1044. if state.led_controller and state.dw_led_playing_effect:
  1045. logger.info(f"Setting LED to playing effect: {state.dw_led_playing_effect}")
  1046. await state.led_controller.effect_playing_async(state.dw_led_playing_effect)
  1047. # Cancel idle timeout when playing starts
  1048. idle_timeout_manager.cancel_timeout()
  1049. with tqdm(
  1050. total=total_coordinates,
  1051. unit="coords",
  1052. desc=f"Executing Pattern {file_path}",
  1053. dynamic_ncols=True,
  1054. disable=False,
  1055. mininterval=1.0
  1056. ) as pbar:
  1057. for i, coordinate in enumerate(coordinates):
  1058. theta, rho = coordinate
  1059. if state.stop_requested:
  1060. logger.info("Execution stopped by user")
  1061. await start_idle_led_timeout()
  1062. break
  1063. if state.skip_requested:
  1064. logger.info("Skipping pattern...")
  1065. await connection_manager.check_idle_async()
  1066. await start_idle_led_timeout()
  1067. break
  1068. # Wait for resume if paused (manual or scheduled)
  1069. manual_pause = state.pause_requested
  1070. # Only check scheduled pause during pattern if "finish pattern first" is NOT enabled
  1071. scheduled_pause = is_in_scheduled_pause_period() if not state.scheduled_pause_finish_pattern else False
  1072. if manual_pause or scheduled_pause:
  1073. pause_start = time.time() # Track when pause started
  1074. if manual_pause and scheduled_pause:
  1075. logger.info("Execution paused (manual + scheduled pause active)...")
  1076. elif manual_pause:
  1077. logger.info("Execution paused (manual)...")
  1078. else:
  1079. logger.info("Execution paused (scheduled pause period)...")
  1080. # Turn off LED controller if scheduled pause and control_wled is enabled
  1081. if state.scheduled_pause_control_wled and state.led_controller:
  1082. logger.info("Turning off LED lights during Still Sands period")
  1083. await state.led_controller.set_power_async(0)
  1084. # Show idle effect for manual pause or scheduled pause without LED control
  1085. # (skip Still Sands check since we handle it above with local scheduled_pause variable)
  1086. if not (scheduled_pause and state.scheduled_pause_control_wled):
  1087. await start_idle_led_timeout(check_still_sands=False)
  1088. # Remember if we turned off LED controller for scheduled pause
  1089. wled_was_off_for_scheduled = scheduled_pause and state.scheduled_pause_control_wled and not manual_pause
  1090. # Wait until both manual pause is released AND we're outside scheduled pause period
  1091. # Also check for stop/skip requests to allow immediate interruption
  1092. interrupted = False
  1093. while state.pause_requested or is_in_scheduled_pause_period():
  1094. # Check for stop/skip first
  1095. if state.stop_requested:
  1096. logger.info("Stop requested during pause, exiting")
  1097. interrupted = True
  1098. break
  1099. if state.skip_requested:
  1100. logger.info("Skip requested during pause, skipping pattern")
  1101. interrupted = True
  1102. break
  1103. if state.pause_requested:
  1104. # For manual pause, wait on multiple events for immediate response
  1105. # Wake on: resume, stop, skip, or timeout (for flag polling fallback)
  1106. pause_event = get_pause_event()
  1107. stop_event = state.get_stop_event()
  1108. skip_event = state.get_skip_event()
  1109. wait_tasks = [asyncio.create_task(pause_event.wait(), name='pause')]
  1110. if stop_event:
  1111. wait_tasks.append(asyncio.create_task(stop_event.wait(), name='stop'))
  1112. if skip_event:
  1113. wait_tasks.append(asyncio.create_task(skip_event.wait(), name='skip'))
  1114. # Add timeout to ensure we periodically check flags even if events aren't set
  1115. # This handles the case where stop is called from sync context (no event loop)
  1116. timeout_task = asyncio.create_task(asyncio.sleep(1.0), name='timeout')
  1117. wait_tasks.append(timeout_task)
  1118. try:
  1119. done, pending = await asyncio.wait(
  1120. wait_tasks, return_when=asyncio.FIRST_COMPLETED
  1121. )
  1122. finally:
  1123. for task in pending:
  1124. task.cancel()
  1125. for task in pending:
  1126. try:
  1127. await task
  1128. except asyncio.CancelledError:
  1129. pass
  1130. else:
  1131. # For scheduled pause, use wait_for_interrupt for instant response
  1132. result = await state.wait_for_interrupt(timeout=1.0)
  1133. if result in ('stopped', 'skipped'):
  1134. interrupted = True
  1135. break
  1136. total_pause_time += time.time() - pause_start # Add pause duration
  1137. if interrupted:
  1138. # Exit the coordinate loop if we were interrupted
  1139. break
  1140. logger.info("Execution resumed...")
  1141. if state.led_controller:
  1142. # Turn LED controller back on if it was turned off for scheduled pause
  1143. # Only power on if a playing effect is configured, otherwise leave LEDs off
  1144. if wled_was_off_for_scheduled:
  1145. if state.dw_led_playing_effect:
  1146. logger.info("Turning LED lights back on as Still Sands period ended")
  1147. await state.led_controller.set_power_async(1)
  1148. # CRITICAL: Give LED controller time to fully power on before sending more commands
  1149. # Without this delay, rapid-fire requests can crash controllers on resource-constrained Pis
  1150. await asyncio.sleep(0.5)
  1151. else:
  1152. logger.info("No playing effect configured, keeping LEDs off after Still Sands")
  1153. if state.dw_led_playing_effect:
  1154. await state.led_controller.effect_playing_async(state.dw_led_playing_effect)
  1155. # Cancel idle timeout when resuming from pause
  1156. idle_timeout_manager.cancel_timeout()
  1157. # Dynamically determine the speed for each movement
  1158. # Use clear_pattern_speed if it's set and this is a clear file, otherwise use state.speed
  1159. if is_clear_file and state.clear_pattern_speed is not None:
  1160. current_speed = state.clear_pattern_speed
  1161. else:
  1162. current_speed = state.speed
  1163. await move_polar(theta, rho, current_speed)
  1164. # Update progress for all coordinates including the first one
  1165. pbar.update(1)
  1166. elapsed_time = time.time() - start_time
  1167. coords_done = i + 1
  1168. # Track rho-weighted progress for accurate time estimation
  1169. completed_weight += coord_weights[i]
  1170. remaining_weight = total_weight - completed_weight
  1171. # Calculate actual execution time (excluding pauses)
  1172. active_time = elapsed_time - total_pause_time
  1173. # Need minimum samples for stable estimate (at least 100 coords and 10 seconds)
  1174. if coords_done >= 100 and active_time > 10:
  1175. # Rate is time per unit weight (accounts for slower moves near center)
  1176. current_rate = active_time / completed_weight
  1177. # Smooth the RATE for stability
  1178. if smoothed_rate is not None:
  1179. alpha = 0.02 # Very smooth - 2% new, 98% old
  1180. smoothed_rate = alpha * current_rate + (1 - alpha) * smoothed_rate
  1181. else:
  1182. smoothed_rate = current_rate
  1183. # Remaining time based on weighted remaining work
  1184. estimated_remaining_time = smoothed_rate * remaining_weight
  1185. else:
  1186. estimated_remaining_time = None
  1187. state.execution_progress = (coords_done, total_coordinates, estimated_remaining_time, elapsed_time)
  1188. # Add a small delay to allow other async operations
  1189. await asyncio.sleep(0.001)
  1190. # Update progress one last time to show 100%
  1191. elapsed_time = time.time() - start_time
  1192. actual_execution_time = elapsed_time - total_pause_time
  1193. state.execution_progress = (total_coordinates, total_coordinates, 0, elapsed_time)
  1194. # Give WebSocket a chance to send the final update
  1195. await asyncio.sleep(0.1)
  1196. # Log execution time (only for completed patterns, not stopped/skipped)
  1197. was_completed = not state.stop_requested and not state.skip_requested
  1198. pattern_name = os.path.basename(file_path)
  1199. effective_speed = state.clear_pattern_speed if (is_clear_file and state.clear_pattern_speed is not None) else state.speed
  1200. log_execution_time(
  1201. pattern_name=pattern_name,
  1202. table_type=state.table_type,
  1203. speed=effective_speed,
  1204. actual_time=actual_execution_time,
  1205. total_coordinates=total_coordinates,
  1206. was_completed=was_completed
  1207. )
  1208. if not state.conn:
  1209. logger.error("Device is not connected. Stopping pattern execution.")
  1210. return False
  1211. await connection_manager.check_idle_async()
  1212. # Set LED back to idle when pattern completes normally (not stopped early)
  1213. # This also handles Still Sands: turns off LEDs if in scheduled pause period with LED control
  1214. if not state.stop_requested:
  1215. await start_idle_led_timeout()
  1216. return was_completed
  1217. async def run_theta_rho_file(file_path, is_playlist=False, clear_pattern=None, cache_data=None):
  1218. """Run a theta-rho file with optional pre-execution clear pattern.
  1219. Args:
  1220. file_path: Path to the main .thr file to execute
  1221. is_playlist: True if running as part of a playlist
  1222. clear_pattern: Clear pattern mode ('adaptive', 'clear_from_in', 'clear_from_out', 'none', or None)
  1223. cache_data: Pre-loaded metadata cache for adaptive clear pattern selection
  1224. """
  1225. lock = get_pattern_lock()
  1226. if lock.locked():
  1227. logger.warning("Another pattern is already running. Cannot start a new one.")
  1228. return
  1229. async with lock: # This ensures only one pattern can run at a time
  1230. # Clear any stale pause state from previous playlist
  1231. state.pause_time_remaining = 0
  1232. state.original_pause_time = None
  1233. # Start progress update task only if not part of a playlist
  1234. global progress_update_task
  1235. if not is_playlist and not progress_update_task:
  1236. progress_update_task = asyncio.create_task(broadcast_progress())
  1237. # Run clear pattern first if specified
  1238. if clear_pattern and clear_pattern != 'none':
  1239. clear_file_path = get_clear_pattern_file(clear_pattern, file_path, cache_data)
  1240. if clear_file_path:
  1241. logger.info(f"Running pre-execution clear pattern: {clear_file_path}")
  1242. state.is_clearing = True
  1243. await _execute_pattern_internal(clear_file_path)
  1244. state.is_clearing = False
  1245. # Reset skip flag after clear pattern (if user skipped clear, continue to main)
  1246. state.skip_requested = False
  1247. # Check if stopped during clear pattern
  1248. if state.stop_requested:
  1249. logger.info("Execution stopped during clear pattern")
  1250. if not is_playlist:
  1251. state.current_playing_file = None
  1252. state.execution_progress = None
  1253. return
  1254. # Run the main pattern
  1255. await _execute_pattern_internal(file_path)
  1256. # Only clear state if not part of a playlist
  1257. if not is_playlist:
  1258. state.current_playing_file = None
  1259. state.execution_progress = None
  1260. logger.info("Pattern execution completed and state cleared")
  1261. # Only cancel progress update task if not part of a playlist
  1262. if progress_update_task:
  1263. progress_update_task.cancel()
  1264. try:
  1265. await progress_update_task
  1266. except asyncio.CancelledError:
  1267. pass
  1268. progress_update_task = None
  1269. else:
  1270. logger.info("Pattern execution completed, maintaining state for playlist")
  1271. async def run_theta_rho_files(file_paths, pause_time=0, clear_pattern=None, run_mode="single", shuffle=False):
  1272. """Run multiple .thr files in sequence with options.
  1273. The playlist now stores only main patterns. Clear patterns are executed dynamically
  1274. before each main pattern based on the clear_pattern option.
  1275. """
  1276. state.stop_requested = False
  1277. # Reset LED idle timeout activity time when playlist starts
  1278. import time as time_module
  1279. state.dw_led_last_activity_time = time_module.time()
  1280. # Set initial playlist state only if not already set by caller (playlist_manager).
  1281. # This ensures backward compatibility when this function is called directly.
  1282. if state.playlist_mode is None:
  1283. state.playlist_mode = run_mode
  1284. if state.current_playlist_index is None:
  1285. state.current_playlist_index = 0
  1286. # Start progress update task for the playlist
  1287. global progress_update_task
  1288. if not progress_update_task:
  1289. progress_update_task = asyncio.create_task(broadcast_progress())
  1290. # Shuffle main patterns if requested (before starting)
  1291. if shuffle:
  1292. random.shuffle(file_paths)
  1293. logger.info("Playlist shuffled")
  1294. # Store patterns in state only if not already set by caller.
  1295. # The caller (playlist_manager.run_playlist) sets this before creating the task.
  1296. if state.current_playlist is None:
  1297. state.current_playlist = file_paths
  1298. try:
  1299. while True:
  1300. # Load metadata cache once per playlist iteration (for adaptive clear patterns)
  1301. cache_data = None
  1302. if clear_pattern and clear_pattern in ['adaptive', 'clear_from_in', 'clear_from_out']:
  1303. from modules.core import cache_manager
  1304. cache_data = await asyncio.to_thread(cache_manager.load_metadata_cache)
  1305. logger.info(f"Loaded metadata cache for {len(cache_data.get('data', {}))} patterns")
  1306. # Reset pattern counter at the start of the playlist
  1307. state.patterns_since_last_home = 0
  1308. # Execute main patterns using index-based access
  1309. # This allows the playlist to be reordered during execution
  1310. idx = 0
  1311. while state.current_playlist and idx < len(state.current_playlist):
  1312. state.current_playlist_index = idx
  1313. if state.stop_requested or not state.current_playlist:
  1314. logger.info("Execution stopped")
  1315. return
  1316. # Get the pattern at the current index (may have changed due to reordering)
  1317. file_path = state.current_playlist[idx]
  1318. logger.info(f"Running pattern {idx + 1}/{len(state.current_playlist)}: {file_path}")
  1319. # Clear pause state when starting a new pattern (prevents stale "waiting" UI)
  1320. state.pause_time_remaining = 0
  1321. state.original_pause_time = None
  1322. # Execute the pattern with optional clear pattern
  1323. await run_theta_rho_file(
  1324. file_path,
  1325. is_playlist=True,
  1326. clear_pattern=clear_pattern,
  1327. cache_data=cache_data
  1328. )
  1329. # Increment pattern counter (auto-home check happens after pause time)
  1330. state.patterns_since_last_home += 1
  1331. logger.debug(f"Patterns since last home: {state.patterns_since_last_home}")
  1332. # Check for scheduled pause after pattern completes (when "finish pattern first" is enabled)
  1333. if state.scheduled_pause_finish_pattern and is_in_scheduled_pause_period() and not state.stop_requested and not state.skip_requested:
  1334. logger.info("Pattern completed. Entering Still Sands period (finish pattern first mode)...")
  1335. wled_was_off_for_scheduled = False
  1336. if state.scheduled_pause_control_wled and state.led_controller:
  1337. logger.info("Turning off LED lights during Still Sands period")
  1338. await state.led_controller.set_power_async(0)
  1339. wled_was_off_for_scheduled = True
  1340. else:
  1341. # Show idle effect (WLED control not enabled)
  1342. await start_idle_led_timeout(check_still_sands=False)
  1343. # Wait for scheduled pause to end, but allow stop/skip to interrupt
  1344. result = await wait_with_interrupt(
  1345. is_in_scheduled_pause_period,
  1346. check_stop=True,
  1347. check_skip=True,
  1348. )
  1349. if result == 'completed':
  1350. logger.info("Still Sands period ended. Resuming playlist...")
  1351. if state.led_controller:
  1352. # Only power on if a playing effect is configured, otherwise leave LEDs off
  1353. if wled_was_off_for_scheduled:
  1354. if state.dw_led_playing_effect:
  1355. logger.info("Turning LED lights back on as Still Sands period ended")
  1356. await state.led_controller.set_power_async(1)
  1357. await asyncio.sleep(0.5)
  1358. else:
  1359. logger.info("No playing effect configured, keeping LEDs off after Still Sands")
  1360. if state.dw_led_playing_effect:
  1361. await state.led_controller.effect_playing_async(state.dw_led_playing_effect)
  1362. idle_timeout_manager.cancel_timeout()
  1363. # Handle pause between patterns
  1364. if state.current_playlist and idx < len(state.current_playlist) - 1 and not state.stop_requested and pause_time > 0 and not state.skip_requested:
  1365. logger.info(f"Pausing for {pause_time} seconds")
  1366. state.original_pause_time = pause_time
  1367. pause_start = time.time()
  1368. while time.time() - pause_start < pause_time:
  1369. state.pause_time_remaining = pause_start + pause_time - time.time()
  1370. if state.skip_requested:
  1371. logger.info("Pause interrupted by skip request")
  1372. break
  1373. await asyncio.sleep(1)
  1374. # Clear both pause state vars immediately (so UI updates right away)
  1375. state.pause_time_remaining = 0
  1376. state.original_pause_time = None
  1377. # Auto-home after pause time, before next clear pattern starts
  1378. # Only home if there's a next pattern and we haven't been stopped
  1379. if (state.auto_home_enabled and
  1380. state.patterns_since_last_home >= state.auto_home_after_patterns and
  1381. state.current_playlist and idx < len(state.current_playlist) - 1 and
  1382. not state.stop_requested):
  1383. logger.info(f"Auto-homing triggered after {state.patterns_since_last_home} patterns (before next clear pattern)")
  1384. try:
  1385. success = await asyncio.to_thread(connection_manager.home)
  1386. if success:
  1387. logger.info("Auto-homing completed successfully")
  1388. state.patterns_since_last_home = 0
  1389. else:
  1390. logger.warning("Auto-homing failed, continuing with playlist")
  1391. except Exception as e:
  1392. logger.error(f"Error during auto-homing: {e}")
  1393. state.skip_requested = False
  1394. idx += 1
  1395. if run_mode == "indefinite":
  1396. logger.info("Playlist completed. Restarting as per 'indefinite' run mode")
  1397. if pause_time > 0:
  1398. pause_start = time.time()
  1399. while time.time() - pause_start < pause_time:
  1400. state.pause_time_remaining = pause_start + pause_time - time.time()
  1401. if state.skip_requested:
  1402. logger.info("Pause interrupted by skip request")
  1403. break
  1404. await asyncio.sleep(1)
  1405. # Clear both pause state vars immediately (so UI updates right away)
  1406. state.pause_time_remaining = 0
  1407. state.original_pause_time = None
  1408. continue
  1409. else:
  1410. logger.info("Playlist completed")
  1411. break
  1412. except asyncio.CancelledError:
  1413. # Task was cancelled externally (e.g., by TestClient cleanup, or explicit cancellation).
  1414. # Do NOT clear playlist state - preserve what the caller set.
  1415. logger.info("Playlist task was cancelled externally, preserving state")
  1416. if progress_update_task:
  1417. progress_update_task.cancel()
  1418. try:
  1419. await progress_update_task
  1420. except asyncio.CancelledError:
  1421. pass
  1422. progress_update_task = None
  1423. raise # Re-raise to signal cancellation
  1424. finally:
  1425. if progress_update_task:
  1426. progress_update_task.cancel()
  1427. try:
  1428. await progress_update_task
  1429. except asyncio.CancelledError:
  1430. pass
  1431. progress_update_task = None
  1432. # Check if we're exiting due to CancelledError - if so, don't clear state.
  1433. # State should only be cleared when:
  1434. # 1. Task completed normally (all patterns executed)
  1435. # 2. Task was stopped by user request (stop_requested)
  1436. # NOT when task was cancelled externally (CancelledError)
  1437. import sys
  1438. exc_type = sys.exc_info()[0]
  1439. if exc_type is asyncio.CancelledError:
  1440. logger.info("Task exiting due to cancellation, state preserved for caller")
  1441. else:
  1442. # Normal completion or user-requested stop - clear state
  1443. state.current_playing_file = None
  1444. state.execution_progress = None
  1445. state.current_playlist = None
  1446. state.current_playlist_index = None
  1447. state.playlist_mode = None
  1448. state.pause_time_remaining = 0
  1449. await start_idle_led_timeout()
  1450. logger.info("All requested patterns completed (or stopped) and state cleared")
  1451. async def stop_actions(clear_playlist = True, wait_for_lock = True):
  1452. """Stop all current actions and wait for pattern to fully release.
  1453. Args:
  1454. clear_playlist: Whether to clear playlist state
  1455. wait_for_lock: Whether to wait for pattern_lock to be released. Set to False when
  1456. called from within pattern execution to avoid deadlock.
  1457. Returns:
  1458. True if stopped cleanly, False if timed out waiting for pattern lock
  1459. """
  1460. timed_out = False
  1461. try:
  1462. with state.pause_condition:
  1463. state.pause_requested = False
  1464. state.stop_requested = True
  1465. state.is_clearing = False
  1466. # Always clear pause time between patterns on stop
  1467. state.pause_time_remaining = 0
  1468. state.original_pause_time = None
  1469. if clear_playlist:
  1470. # Clear playlist state
  1471. state.current_playlist = None
  1472. state.current_playlist_index = None
  1473. state.playlist_mode = None
  1474. # Cancel progress update task if we're clearing the playlist
  1475. global progress_update_task
  1476. if progress_update_task and not progress_update_task.done():
  1477. progress_update_task.cancel()
  1478. # Cancel the playlist task itself (late import to avoid circular dependency)
  1479. from modules.core import playlist_manager
  1480. await playlist_manager.cancel_current_playlist()
  1481. state.pause_condition.notify_all()
  1482. # Also set the pause event to wake up any paused patterns
  1483. get_pause_event().set()
  1484. # Send stop command to motion thread to clear its queue
  1485. if motion_controller.running:
  1486. motion_controller.command_queue.put(MotionCommand('stop'))
  1487. # Wait for the pattern lock to be released before continuing
  1488. # This ensures that when stop_actions completes, the pattern has fully stopped
  1489. # Skip this if called from within pattern execution to avoid deadlock
  1490. lock = get_pattern_lock()
  1491. if wait_for_lock and lock.locked():
  1492. logger.info("Waiting for pattern to fully stop...")
  1493. # Use a timeout to prevent hanging forever
  1494. # Use wait_for for Python 3.9 compatibility (asyncio.timeout is 3.11+)
  1495. try:
  1496. async def acquire_stop_lock():
  1497. async with lock:
  1498. logger.info("Pattern lock acquired - pattern has fully stopped")
  1499. await asyncio.wait_for(acquire_stop_lock(), timeout=10.0)
  1500. except asyncio.TimeoutError:
  1501. logger.warning("Timeout waiting for pattern to stop - forcing cleanup")
  1502. timed_out = True
  1503. # Force cleanup of state even if pattern didn't release lock gracefully
  1504. state.current_playing_file = None
  1505. state.execution_progress = None
  1506. state.is_running = False
  1507. # Clear current playing file only when clearing the entire playlist.
  1508. # When clear_playlist=False (called from within pattern execution), the caller
  1509. # will set current_playing_file to the new pattern immediately after.
  1510. if clear_playlist:
  1511. state.current_playing_file = None
  1512. state.execution_progress = None
  1513. # Clear stop_requested now that the pattern has stopped - this allows
  1514. # check_idle_async to work (it exits early if stop_requested is True)
  1515. state.stop_requested = False
  1516. # Wait for hardware to reach idle state before returning
  1517. # This ensures the machine has physically stopped moving
  1518. if not timed_out:
  1519. idle = await connection_manager.check_idle_async(timeout=30.0)
  1520. if not idle:
  1521. logger.warning("Machine did not reach idle after stop")
  1522. # Call async function directly since we're in async context
  1523. await connection_manager.update_machine_position()
  1524. return not timed_out
  1525. except Exception as e:
  1526. logger.error(f"Error during stop_actions: {e}")
  1527. # Force cleanup state on error
  1528. state.current_playing_file = None
  1529. state.execution_progress = None
  1530. state.is_running = False
  1531. # Ensure we still update machine position even if there's an error
  1532. try:
  1533. await connection_manager.update_machine_position()
  1534. except Exception as update_err:
  1535. logger.error(f"Error updating machine position on error: {update_err}")
  1536. return False
  1537. async def move_polar(theta, rho, speed=None):
  1538. """
  1539. Queue a motion command to be executed in the dedicated motion control thread.
  1540. This makes motion control non-blocking for API endpoints.
  1541. Args:
  1542. theta (float): Target theta coordinate
  1543. rho (float): Target rho coordinate
  1544. speed (int, optional): Speed override. If None, uses state.speed
  1545. """
  1546. # Note: stop_requested is cleared once at pattern start (execute_theta_rho_file line 890)
  1547. # Don't clear it here on every coordinate - causes performance issues with event system
  1548. # Ensure motion control thread is running
  1549. if not motion_controller.running:
  1550. motion_controller.start()
  1551. # Create future for async/await pattern
  1552. loop = asyncio.get_event_loop()
  1553. future = loop.create_future()
  1554. # Create and queue motion command
  1555. command = MotionCommand(
  1556. command_type='move',
  1557. theta=theta,
  1558. rho=rho,
  1559. speed=speed,
  1560. future=future
  1561. )
  1562. motion_controller.command_queue.put(command)
  1563. logger.debug(f"Queued motion command: theta={theta}, rho={rho}, speed={speed}")
  1564. # Wait for command completion
  1565. await future
  1566. def pause_execution():
  1567. """Pause pattern execution using asyncio Event."""
  1568. logger.info("Pausing pattern execution")
  1569. state.pause_requested = True
  1570. get_pause_event().clear() # Clear the event to pause execution
  1571. return True
  1572. def resume_execution():
  1573. """Resume pattern execution using asyncio Event."""
  1574. logger.info("Resuming pattern execution")
  1575. state.pause_requested = False
  1576. get_pause_event().set() # Set the event to resume execution
  1577. return True
  1578. async def reset_theta():
  1579. """
  1580. Reset theta to [0, 2π) range and optionally hard reset machine position using $Bye.
  1581. When state.hard_reset_theta is True:
  1582. - $Bye sends a soft reset to FluidNC which resets the controller and clears
  1583. all position counters to 0. This is more reliable than G92 which only sets
  1584. a work coordinate offset without changing the actual machine position (MPos).
  1585. - We wait for machine to be idle before sending $Bye to avoid error:25
  1586. When state.hard_reset_theta is False (default):
  1587. - Only normalizes theta to [0, 2π) range without affecting machine position
  1588. - Faster and doesn't interrupt machine state
  1589. """
  1590. logger.info('Resetting Theta')
  1591. # Always normalize theta to [0, 2π) range
  1592. state.current_theta = state.current_theta % (2 * pi)
  1593. logger.info(f'Theta normalized to {state.current_theta:.4f} radians')
  1594. # Only perform hard reset if enabled
  1595. if state.hard_reset_theta:
  1596. logger.info('Hard reset enabled - performing machine soft reset')
  1597. # Wait for machine to be idle before reset to prevent error:25
  1598. if state.conn and state.conn.is_connected():
  1599. logger.info("Waiting for machine to be idle before reset...")
  1600. idle = await connection_manager.check_idle_async(timeout=30)
  1601. if not idle:
  1602. logger.warning("Machine not idle after 30s, proceeding with reset anyway")
  1603. # Hard reset machine position using $Bye via connection_manager
  1604. success = await connection_manager.perform_soft_reset()
  1605. if not success:
  1606. logger.error("Soft reset failed - theta reset may be unreliable")
  1607. else:
  1608. logger.info('Hard reset disabled - skipping machine soft reset')
  1609. def set_speed(new_speed):
  1610. state.speed = new_speed
  1611. logger.info(f'Set new state.speed {new_speed}')
  1612. def get_status():
  1613. """Get the current status of pattern execution."""
  1614. status = {
  1615. "current_file": state.current_playing_file,
  1616. "is_paused": state.pause_requested or is_in_scheduled_pause_period(),
  1617. "manual_pause": state.pause_requested,
  1618. "scheduled_pause": is_in_scheduled_pause_period(),
  1619. "is_running": bool(state.current_playing_file and not state.stop_requested),
  1620. "is_homing": state.is_homing,
  1621. "sensor_homing_failed": state.sensor_homing_failed,
  1622. "is_clearing": state.is_clearing,
  1623. "progress": None,
  1624. "playlist": None,
  1625. "speed": state.speed,
  1626. "pause_time_remaining": state.pause_time_remaining,
  1627. "original_pause_time": getattr(state, 'original_pause_time', None),
  1628. "connection_status": state.conn.is_connected() if state.conn else False,
  1629. "current_theta": state.current_theta,
  1630. "current_rho": state.current_rho
  1631. }
  1632. # Add playlist information if available
  1633. if state.current_playlist and state.current_playlist_index is not None:
  1634. # When a clear pattern is running, the "next" pattern is the current main pattern
  1635. # (since the clear pattern runs before the main pattern at current_playlist_index)
  1636. if state.is_clearing:
  1637. next_file = state.current_playlist[state.current_playlist_index]
  1638. else:
  1639. next_index = state.current_playlist_index + 1
  1640. next_file = state.current_playlist[next_index] if next_index < len(state.current_playlist) else None
  1641. status["playlist"] = {
  1642. "current_index": state.current_playlist_index,
  1643. "total_files": len(state.current_playlist),
  1644. "mode": state.playlist_mode,
  1645. "next_file": next_file,
  1646. "files": state.current_playlist,
  1647. "name": state.current_playlist_name
  1648. }
  1649. if state.execution_progress:
  1650. current, total, remaining_time, elapsed_time = state.execution_progress
  1651. status["progress"] = {
  1652. "current": current,
  1653. "total": total,
  1654. "remaining_time": remaining_time,
  1655. "elapsed_time": elapsed_time,
  1656. "percentage": (current / total * 100) if total > 0 else 0
  1657. }
  1658. # Add historical execution time if available for this pattern at current speed
  1659. if state.current_playing_file:
  1660. pattern_name = os.path.basename(state.current_playing_file)
  1661. historical_time = get_last_completed_execution_time(pattern_name, state.speed)
  1662. if historical_time:
  1663. status["progress"]["last_completed_time"] = historical_time
  1664. return status
  1665. async def broadcast_progress():
  1666. """Background task to broadcast progress updates."""
  1667. from main import broadcast_status_update
  1668. while True:
  1669. # Send status updates regardless of pattern_lock state
  1670. status = get_status()
  1671. # Use the existing broadcast function from main.py
  1672. await broadcast_status_update(status)
  1673. # Check if we should stop broadcasting
  1674. if not state.current_playlist:
  1675. # If no playlist, only stop if no pattern is being executed
  1676. if not get_pattern_lock().locked():
  1677. logger.info("No playlist or pattern running, stopping broadcast")
  1678. break
  1679. # Wait before next update
  1680. await asyncio.sleep(1)