annotate blender/time_sync/time_from_graph.py @ 2454:405abed9a45c

fix up asyncio-in-bg-thread sorcery
author drewp@bigasterisk.com
date Mon, 19 May 2025 21:25:32 -0700
parents b23afde50bc2
children 2d454737a916
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
2454
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
1 import asyncio
2432
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
2 import threading
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
3 import time
2454
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
4 from typing import Coroutine, cast
2432
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
5
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
6 import bpy # type: ignore
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
7 from bpy.app.handlers import persistent # type: ignore
2454
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
8 from ..asyncio_thread import startLoopInThread
2432
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
9 from rdfdb.syncedgraph.syncedgraph import SyncedGraph
2454
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
10 from rdflib import URIRef
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
11 from rdflib.graph import _ContextType
2432
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
12
2454
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
13 from light9 import networking, showconfig
2432
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
14 from light9.namespaces import L9
2454
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
15 from light9.newtypes import decimalLiteral
2432
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
16 from light9.run_local import log
2453
b23afde50bc2 blender addons get thier own pdm setup for now. fix time_from_graph startup race
drewp@bigasterisk.com
parents: 2436
diff changeset
17 from light9.typedgraph import typedValue
2432
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
18
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
19
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
20 def clamp(lo, hi, x):
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
21 return max(lo, min(hi, x))
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
22
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
23
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
24 UPDATE_PERIOD = 1 / 20
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
25
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
26
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
27 class Sync:
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
28 lock = threading.Lock()
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
29 duration: float = 1
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
30 wallStartTime: float | None = None
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
31 pausedSongTime: float | None = None
2433
ffa69645e9dc refactor
drewp@bigasterisk.com
parents: 2432
diff changeset
32 playing = False
2432
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
33 latestTime: float
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
34
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
35 def __init__(self):
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
36 # main thread
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
37 self.lastSetFrame = -1
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
38 self.lastGraphFrame = -1
2454
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
39 show = showconfig.showUri()
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
40 self.ctx = cast(_ContextType, (URIRef(show + '/ascoltami')))
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
41 log.debug('🚋3 startLoopInThread')
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
42 self._loop = startLoopInThread(self.task())
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
43
2453
b23afde50bc2 blender addons get thier own pdm setup for now. fix time_from_graph startup race
drewp@bigasterisk.com
parents: 2436
diff changeset
44 # need persistent because `blender --addons ...` seems to start addon
b23afde50bc2 blender addons get thier own pdm setup for now. fix time_from_graph startup race
drewp@bigasterisk.com
parents: 2436
diff changeset
45 # before loading scene.
b23afde50bc2 blender addons get thier own pdm setup for now. fix time_from_graph startup race
drewp@bigasterisk.com
parents: 2436
diff changeset
46 bpy.app.timers.register(self.update, persistent=True)
2432
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
47 bpy.app.handlers.frame_change_post.append(self.on_frame_change_post)
2454
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
48 log.info('🚋10 Sync initd')
2432
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
49
2433
ffa69645e9dc refactor
drewp@bigasterisk.com
parents: 2432
diff changeset
50 ## updates from graph -> self
2432
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
51
2454
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
52 def runInBackgroundLoop(self, f: Coroutine):
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
53 asyncio.run_coroutine_threadsafe(f, self._loop)
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
54
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
55
2432
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
56 async def task(self):
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
57 # bg thread with asyncio loop
2454
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
58 log.info('🚋11 start SyncedGraph')
2432
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
59 self.graph = SyncedGraph(networking.rdfdb.url, "time_sync")
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
60 self.graph.addHandler(self.syncFromGraph)
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
61
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
62 def syncFromGraph(self):
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
63 # bg thread
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
64 with self.lock:
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
65 asco = L9['ascoltami']
2433
ffa69645e9dc refactor
drewp@bigasterisk.com
parents: 2432
diff changeset
66 self.wallStartTime = typedValue(float | None, self.graph, asco, L9['wallStartTime'])
ffa69645e9dc refactor
drewp@bigasterisk.com
parents: 2432
diff changeset
67 self.pausedSongTime = typedValue(float | None, self.graph, asco, L9['pausedSongTime'])
ffa69645e9dc refactor
drewp@bigasterisk.com
parents: 2432
diff changeset
68 self.duration = typedValue(float | None, self.graph, asco, L9['duration']) or 1.0
2436
e683b449506b blender effect that sets lights to match blender lights
drewp@bigasterisk.com
parents: 2434
diff changeset
69 self.playing = typedValue(bool | None, self.graph, asco, L9['playing']) or False
2432
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
70
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
71 def currentTime(self) -> float | None:
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
72 if self.wallStartTime is not None:
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
73 return time.time() - self.wallStartTime
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
74 if self.pausedSongTime is not None:
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
75 return self.pausedSongTime
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
76 log.warn('no time data')
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
77 return None
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
78
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
79 ## graph time -> blender time
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
80
2454
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
81 @persistent
2432
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
82 def update(self):
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
83 # main thread? wherever blender runs timers
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
84 if self.playing:
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
85 with self.lock:
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
86 t = self.currentTime()
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
87 if t is not None:
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
88 self.setBlenderTime(t, self.duration)
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
89 return UPDATE_PERIOD
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
90
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
91 def setBlenderTime(self, t: float, duration: float):
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
92 scene = bpy.context.scene
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
93 fps = scene.render.fps
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
94 scene.frame_start = 0
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
95 scene.frame_end = int(duration * fps)
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
96 fr = int(clamp(t, 0, duration) * fps)
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
97 self.lastSetFrame = fr
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
98 scene.frame_set(fr)
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
99
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
100 ## blender time -> graph time
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
101
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
102 @persistent
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
103 def on_frame_change_post(self, scene, deps):
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
104 if scene.frame_current != self.lastSetFrame:
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
105 # self.setGraphTime(scene.frame_current / scene.render.fps, self.duration)
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
106 self.lastSetFrame = scene.frame_current
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
107 t = scene.frame_current / scene.render.fps
2454
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
108 self.runInBackgroundLoop(self.setInGraph(t, bpy.context.screen.is_animation_playing))
2432
b8a408caf115 start blender sync
drewp@bigasterisk.com
parents:
diff changeset
109
2454
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
110 async def setInGraph(self, t: float, isBlenderPlaying: bool):
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
111 log.info(f'set graph time to {t:.2f}')
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
112 if isBlenderPlaying:
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
113 log.info(' playing mode')
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
114 p = self.graph.getObjectPatch(self.ctx, L9['ascoltami'], L9['wallStartTime'], decimalLiteral(t))
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
115 else:
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
116 log.info(' paused mode')
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
117 p = self.graph.getObjectPatch(self.ctx, L9['ascoltami'], L9['pausedSongTime'], decimalLiteral(t))
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
118
405abed9a45c fix up asyncio-in-bg-thread sorcery
drewp@bigasterisk.com
parents: 2453
diff changeset
119 await self.graph.patch(p)