pyview-web 0.2.2__tar.gz → 0.2.4__tar.gz
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Potentially problematic release.
This version of pyview-web might be problematic. Click here for more details.
- {pyview_web-0.2.2 → pyview_web-0.2.4}/PKG-INFO +1 -1
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyproject.toml +1 -1
- pyview_web-0.2.4/pyview/async_stream_runner.py +65 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/events/BaseEventHandler.py +28 -2
- pyview_web-0.2.4/pyview/events/__init__.py +4 -0
- pyview_web-0.2.4/pyview/events/info_event.py +16 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/live_socket.py +11 -4
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/ws_handler.py +8 -14
- pyview_web-0.2.2/pyview/events/__init__.py +0 -4
- pyview_web-0.2.2/pyview/events/info_event.py +0 -8
- {pyview_web-0.2.2 → pyview_web-0.2.4}/LICENSE +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/__init__.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/assets/js/app.js +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/assets/package-lock.json +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/assets/package.json +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/auth/__init__.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/auth/provider.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/auth/required.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/changesets/__init__.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/changesets/changesets.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/csrf.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/js.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/live_routes.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/live_view.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/meta.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/phx_message.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/pyview.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/secret.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/session.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/static/assets/app.js +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/template/__init__.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/template/context_processor.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/template/live_template.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/template/render_diff.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/template/root_template.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/template/serializer.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/template/utils.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/test_csrf.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/uploads.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/vendor/__init__.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/vendor/flet/pubsub/__init__.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/vendor/flet/pubsub/pub_sub.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/vendor/ibis/__init__.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/vendor/ibis/compiler.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/vendor/ibis/context.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/vendor/ibis/errors.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/vendor/ibis/filters.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/vendor/ibis/loaders.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/vendor/ibis/nodes.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/vendor/ibis/template.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/vendor/ibis/tree.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/pyview/vendor/ibis/utils.py +0 -0
- {pyview_web-0.2.2 → pyview_web-0.2.4}/readme.md +0 -0
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
import asyncio
|
|
2
|
+
import uuid
|
|
3
|
+
import logging
|
|
4
|
+
from typing import Any, AsyncGenerator, Callable, Optional
|
|
5
|
+
from pyview.events.info_event import InfoEvent, InfoEventScheduler
|
|
6
|
+
|
|
7
|
+
|
|
8
|
+
class AsyncStreamRunner:
|
|
9
|
+
def __init__(self, scheduler: InfoEventScheduler):
|
|
10
|
+
self._stream_tasks: dict[str, asyncio.Task] = {}
|
|
11
|
+
self._scheduler = scheduler
|
|
12
|
+
|
|
13
|
+
def start_stream(
|
|
14
|
+
self,
|
|
15
|
+
gen: AsyncGenerator[Any, None],
|
|
16
|
+
*,
|
|
17
|
+
on_yield: Callable[[Any], InfoEvent],
|
|
18
|
+
on_done: Optional[InfoEvent] = None,
|
|
19
|
+
on_error: Optional[Callable[[Exception], InfoEvent]] = None,
|
|
20
|
+
on_cancel: Optional[InfoEvent] = None,
|
|
21
|
+
) -> str:
|
|
22
|
+
"""
|
|
23
|
+
Run `gen` in the background, returning an op_id you can later use
|
|
24
|
+
to cancel. Hooks:
|
|
25
|
+
|
|
26
|
+
- on_yield(item) → scheduled per chunk
|
|
27
|
+
- on_done → scheduled once at normal completion
|
|
28
|
+
- on_error(exc) → scheduled on unexpected exception
|
|
29
|
+
- on_cancel → scheduled if the task is cancelled
|
|
30
|
+
"""
|
|
31
|
+
task_id = uuid.uuid4().hex
|
|
32
|
+
|
|
33
|
+
async def driver():
|
|
34
|
+
try:
|
|
35
|
+
async for item in gen:
|
|
36
|
+
self._scheduler.schedule_info_once(on_yield(item))
|
|
37
|
+
except asyncio.CancelledError:
|
|
38
|
+
# user-requested cancellation
|
|
39
|
+
if on_cancel:
|
|
40
|
+
self._scheduler.schedule_info_once(on_cancel)
|
|
41
|
+
# swallow so it doesn’t log as an “error”
|
|
42
|
+
except Exception as exc:
|
|
43
|
+
if on_error:
|
|
44
|
+
self._scheduler.schedule_info_once(on_error(exc))
|
|
45
|
+
else:
|
|
46
|
+
logging.exception(f"Error in stream {task_id}", exc_info=True)
|
|
47
|
+
else:
|
|
48
|
+
if on_done:
|
|
49
|
+
self._scheduler.schedule_info_once(on_done)
|
|
50
|
+
finally:
|
|
51
|
+
self._stream_tasks.pop(task_id, None)
|
|
52
|
+
|
|
53
|
+
task = asyncio.create_task(driver())
|
|
54
|
+
self._stream_tasks[task_id] = task
|
|
55
|
+
return task_id
|
|
56
|
+
|
|
57
|
+
def cancel_stream(self, task_id: str) -> bool:
|
|
58
|
+
"""
|
|
59
|
+
Cancel a running stream. Returns True if a task was found & cancelled.
|
|
60
|
+
"""
|
|
61
|
+
task = self._stream_tasks.get(task_id)
|
|
62
|
+
if not task:
|
|
63
|
+
return False
|
|
64
|
+
task.cancel()
|
|
65
|
+
return True
|
|
@@ -1,6 +1,9 @@
|
|
|
1
|
-
from typing import Callable
|
|
1
|
+
from typing import Callable, TYPE_CHECKING
|
|
2
2
|
import logging
|
|
3
3
|
|
|
4
|
+
if TYPE_CHECKING:
|
|
5
|
+
from pyview.live_view import InfoEvent
|
|
6
|
+
|
|
4
7
|
|
|
5
8
|
def event(*event_names):
|
|
6
9
|
"""Decorator that marks methods as event handlers."""
|
|
@@ -12,22 +15,37 @@ def event(*event_names):
|
|
|
12
15
|
return decorator
|
|
13
16
|
|
|
14
17
|
|
|
18
|
+
def info(*info_names):
|
|
19
|
+
"""Decorator that marks methods as info handlers."""
|
|
20
|
+
|
|
21
|
+
def decorator(func):
|
|
22
|
+
func._info_names = info_names
|
|
23
|
+
return func
|
|
24
|
+
|
|
25
|
+
return decorator
|
|
26
|
+
|
|
27
|
+
|
|
15
28
|
class BaseEventHandler:
|
|
16
|
-
"""Base class for event handlers to handle dispatching events."""
|
|
29
|
+
"""Base class for event handlers to handle dispatching events and info."""
|
|
17
30
|
|
|
18
31
|
_event_handlers: dict[str, Callable] = {}
|
|
32
|
+
_info_handlers: dict[str, Callable] = {}
|
|
19
33
|
|
|
20
34
|
def __init_subclass__(cls, **kwargs):
|
|
21
35
|
super().__init_subclass__(**kwargs)
|
|
22
36
|
|
|
23
37
|
# Find all decorated methods and register them
|
|
24
38
|
cls._event_handlers = {}
|
|
39
|
+
cls._info_handlers = {}
|
|
25
40
|
for attr_name in dir(cls):
|
|
26
41
|
if not attr_name.startswith("_"):
|
|
27
42
|
attr = getattr(cls, attr_name)
|
|
28
43
|
if hasattr(attr, "_event_names"):
|
|
29
44
|
for event_name in attr._event_names:
|
|
30
45
|
cls._event_handlers[event_name] = attr
|
|
46
|
+
if hasattr(attr, "_info_names"):
|
|
47
|
+
for info_name in attr._info_names:
|
|
48
|
+
cls._info_handlers[info_name] = attr
|
|
31
49
|
|
|
32
50
|
async def handle_event(self, event: str, payload: dict, socket):
|
|
33
51
|
handler = self._event_handlers.get(event)
|
|
@@ -36,3 +54,11 @@ class BaseEventHandler:
|
|
|
36
54
|
return await handler(self, event, payload, socket)
|
|
37
55
|
else:
|
|
38
56
|
logging.warning(f"Unhandled event: {event} {payload}")
|
|
57
|
+
|
|
58
|
+
async def handle_info(self, info: "InfoEvent", socket):
|
|
59
|
+
handler = self._info_handlers.get(info.name)
|
|
60
|
+
|
|
61
|
+
if handler:
|
|
62
|
+
return await handler(self, info, socket)
|
|
63
|
+
else:
|
|
64
|
+
logging.warning(f"Unhandled info: {info.name} {info}")
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
from dataclasses import dataclass
|
|
2
|
+
from typing import Any, Optional, Protocol
|
|
3
|
+
|
|
4
|
+
|
|
5
|
+
@dataclass
|
|
6
|
+
class InfoEvent:
|
|
7
|
+
name: str
|
|
8
|
+
payload: Any = None
|
|
9
|
+
|
|
10
|
+
|
|
11
|
+
class InfoEventScheduler(Protocol):
|
|
12
|
+
def schedule_info(self, event: InfoEvent, seconds: float):
|
|
13
|
+
pass
|
|
14
|
+
|
|
15
|
+
def schedule_info_once(self, event: InfoEvent, seconds: Optional[float] = None):
|
|
16
|
+
pass
|
|
@@ -17,7 +17,9 @@ from pyview.vendor.flet.pubsub import PubSubHub, PubSub
|
|
|
17
17
|
from pyview.events import InfoEvent
|
|
18
18
|
from pyview.uploads import UploadConstraints, UploadConfig, UploadManager
|
|
19
19
|
from pyview.meta import PyViewMeta
|
|
20
|
+
from pyview.template.render_diff import calc_diff
|
|
20
21
|
import datetime
|
|
22
|
+
from pyview.async_stream_runner import AsyncStreamRunner
|
|
21
23
|
|
|
22
24
|
|
|
23
25
|
if TYPE_CHECKING:
|
|
@@ -50,8 +52,8 @@ class ConnectedLiveViewSocket(Generic[T]):
|
|
|
50
52
|
context: T
|
|
51
53
|
live_title: Optional[str] = None
|
|
52
54
|
pending_events: list[tuple[str, Any]]
|
|
53
|
-
|
|
54
55
|
upload_manager: UploadManager
|
|
56
|
+
prev_rendered: Optional[dict[str, Any]] = None
|
|
55
57
|
|
|
56
58
|
def __init__(self, websocket: WebSocket, topic: str, liveview: LiveView):
|
|
57
59
|
self.websocket = websocket
|
|
@@ -62,6 +64,7 @@ class ConnectedLiveViewSocket(Generic[T]):
|
|
|
62
64
|
self.pub_sub = PubSub(pub_sub_hub, topic)
|
|
63
65
|
self.pending_events = []
|
|
64
66
|
self.upload_manager = UploadManager()
|
|
67
|
+
self.stream_runner = AsyncStreamRunner(self)
|
|
65
68
|
|
|
66
69
|
@property
|
|
67
70
|
def meta(self) -> PyViewMeta:
|
|
@@ -93,9 +96,13 @@ class ConnectedLiveViewSocket(Generic[T]):
|
|
|
93
96
|
)
|
|
94
97
|
|
|
95
98
|
def diff(self, render: dict[str, Any]) -> dict[str, Any]:
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
+
if self.prev_rendered:
|
|
100
|
+
diff = calc_diff(self.prev_rendered, render)
|
|
101
|
+
else:
|
|
102
|
+
diff = render
|
|
103
|
+
|
|
104
|
+
self.prev_rendered = render
|
|
105
|
+
return diff
|
|
99
106
|
|
|
100
107
|
async def send_info(self, event: InfoEvent):
|
|
101
108
|
await self.liveview.handle_info(event, self)
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
from typing import Optional
|
|
1
|
+
from typing import Optional
|
|
2
2
|
import json
|
|
3
3
|
from starlette.websockets import WebSocket, WebSocketDisconnect
|
|
4
4
|
from urllib.parse import urlparse, parse_qs
|
|
@@ -8,7 +8,6 @@ from pyview.csrf import validate_csrf_token
|
|
|
8
8
|
from pyview.session import deserialize_session
|
|
9
9
|
from pyview.auth import AuthProviderFactory
|
|
10
10
|
from pyview.phx_message import parse_message
|
|
11
|
-
from pyview.template.render_diff import calc_diff
|
|
12
11
|
|
|
13
12
|
|
|
14
13
|
class AuthException(Exception):
|
|
@@ -60,6 +59,7 @@ class LiveSocketHandler:
|
|
|
60
59
|
await lv.handle_params(url, merged_params, socket)
|
|
61
60
|
|
|
62
61
|
rendered = await _render(socket)
|
|
62
|
+
socket.prev_rendered = rendered
|
|
63
63
|
|
|
64
64
|
resp = [
|
|
65
65
|
joinRef,
|
|
@@ -70,7 +70,7 @@ class LiveSocketHandler:
|
|
|
70
70
|
]
|
|
71
71
|
|
|
72
72
|
await self.manager.send_personal_message(json.dumps(resp), websocket)
|
|
73
|
-
await self.handle_connected(topic, socket
|
|
73
|
+
await self.handle_connected(topic, socket)
|
|
74
74
|
|
|
75
75
|
except WebSocketDisconnect:
|
|
76
76
|
if socket:
|
|
@@ -80,9 +80,7 @@ class LiveSocketHandler:
|
|
|
80
80
|
await websocket.close()
|
|
81
81
|
self.sessions -= 1
|
|
82
82
|
|
|
83
|
-
async def handle_connected(
|
|
84
|
-
self, myJoinId, socket: ConnectedLiveViewSocket, prev_rendered: dict[str, Any]
|
|
85
|
-
):
|
|
83
|
+
async def handle_connected(self, myJoinId, socket: ConnectedLiveViewSocket):
|
|
86
84
|
while True:
|
|
87
85
|
message = await socket.websocket.receive()
|
|
88
86
|
[joinRef, mesageRef, topic, event, payload] = parse_message(message)
|
|
@@ -114,8 +112,7 @@ class LiveSocketHandler:
|
|
|
114
112
|
{} if not socket.pending_events else {"e": socket.pending_events}
|
|
115
113
|
)
|
|
116
114
|
|
|
117
|
-
diff =
|
|
118
|
-
prev_rendered = rendered
|
|
115
|
+
diff = socket.diff(rendered)
|
|
119
116
|
|
|
120
117
|
socket.pending_events = []
|
|
121
118
|
|
|
@@ -150,8 +147,7 @@ class LiveSocketHandler:
|
|
|
150
147
|
|
|
151
148
|
await lv.handle_params(url, merged_params, socket)
|
|
152
149
|
rendered = await _render(socket)
|
|
153
|
-
diff =
|
|
154
|
-
prev_rendered = rendered
|
|
150
|
+
diff = socket.diff(rendered)
|
|
155
151
|
|
|
156
152
|
resp = [
|
|
157
153
|
joinRef,
|
|
@@ -171,8 +167,7 @@ class LiveSocketHandler:
|
|
|
171
167
|
)
|
|
172
168
|
|
|
173
169
|
rendered = await _render(socket)
|
|
174
|
-
diff =
|
|
175
|
-
prev_rendered = rendered
|
|
170
|
+
diff = socket.diff(rendered)
|
|
176
171
|
|
|
177
172
|
resp = [
|
|
178
173
|
joinRef,
|
|
@@ -238,8 +233,7 @@ class LiveSocketHandler:
|
|
|
238
233
|
if event == "progress":
|
|
239
234
|
socket.upload_manager.update_progress(joinRef, payload)
|
|
240
235
|
rendered = await _render(socket)
|
|
241
|
-
diff =
|
|
242
|
-
prev_rendered = rendered
|
|
236
|
+
diff = socket.diff(rendered)
|
|
243
237
|
|
|
244
238
|
resp = [
|
|
245
239
|
joinRef,
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|