pattern_manager.py 80 KB

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