jupyter-server-ydoc 1.0.1__tar.gz → 1.1.0a0__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.
Files changed (29) hide show
  1. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/PKG-INFO +2 -1
  2. jupyter_server_ydoc-1.1.0a0/jupyter_server_ydoc/_version.py +1 -0
  3. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/jupyter_server_ydoc/app.py +10 -0
  4. jupyter_server_ydoc-1.1.0a0/jupyter_server_ydoc/events/fork.yaml +56 -0
  5. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/jupyter_server_ydoc/handlers.py +100 -0
  6. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/jupyter_server_ydoc/pytest_plugin.py +59 -0
  7. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/jupyter_server_ydoc/utils.py +2 -0
  8. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/pyproject.toml +1 -0
  9. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/tests/test_handlers.py +146 -0
  10. jupyter_server_ydoc-1.0.1/jupyter_server_ydoc/_version.py +0 -1
  11. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/.gitignore +0 -0
  12. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/LICENSE +0 -0
  13. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/README.md +0 -0
  14. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/jupyter-config/jupyter_server_ydoc.json +0 -0
  15. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/jupyter_server_ydoc/__init__.py +0 -0
  16. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/jupyter_server_ydoc/events/awareness.yaml +0 -0
  17. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/jupyter_server_ydoc/events/session.yaml +0 -0
  18. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/jupyter_server_ydoc/loaders.py +0 -0
  19. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/jupyter_server_ydoc/rooms.py +0 -0
  20. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/jupyter_server_ydoc/stores.py +0 -0
  21. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/jupyter_server_ydoc/test_utils.py +0 -0
  22. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/jupyter_server_ydoc/websocketserver.py +0 -0
  23. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/setup.py +0 -0
  24. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/tests/__init__.py +0 -0
  25. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/tests/conftest.py +0 -0
  26. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/tests/test_app.py +0 -0
  27. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/tests/test_documents.py +0 -0
  28. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/tests/test_loaders.py +0 -0
  29. {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0a0}/tests/test_rooms.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.3
2
2
  Name: jupyter-server-ydoc
3
- Version: 1.0.1
3
+ Version: 1.1.0a0
4
4
  Summary: jupyter-server extension integrating collaborative shared models.
5
5
  Author-email: Jupyter Development Team <jupyter@googlegroups.com>
6
6
  License: # Licensing terms
@@ -84,6 +84,7 @@ Requires-Dist: pycrdt-websocket<0.16.0,>=0.15.0
84
84
  Provides-Extra: test
85
85
  Requires-Dist: anyio; extra == 'test'
86
86
  Requires-Dist: coverage; extra == 'test'
87
+ Requires-Dist: dirty-equals; extra == 'test'
87
88
  Requires-Dist: httpx-ws>=0.5.2; extra == 'test'
88
89
  Requires-Dist: importlib-metadata>=4.8.3; (python_version < '3.10') and extra == 'test'
89
90
  Requires-Dist: jupyter-server-fileid[test]; extra == 'test'
@@ -0,0 +1 @@
1
+ __version__ = "1.1.0.a0"
@@ -14,6 +14,7 @@ from pycrdt_websocket.ystore import BaseYStore
14
14
  from traitlets import Bool, Float, Type
15
15
 
16
16
  from .handlers import (
17
+ DocForkHandler,
17
18
  DocSessionHandler,
18
19
  TimelineHandler,
19
20
  UndoRedoHandler,
@@ -25,6 +26,7 @@ from .stores import SQLiteYStore
25
26
  from .utils import (
26
27
  AWARENESS_EVENTS_SCHEMA_PATH,
27
28
  EVENTS_SCHEMA_PATH,
29
+ FORK_EVENTS_SCHEMA_PATH,
28
30
  encode_file_path,
29
31
  room_id_from_encoded_path,
30
32
  )
@@ -85,6 +87,7 @@ class YDocExtension(ExtensionApp):
85
87
  super().initialize()
86
88
  self.serverapp.event_logger.register_event_schema(EVENTS_SCHEMA_PATH)
87
89
  self.serverapp.event_logger.register_event_schema(AWARENESS_EVENTS_SCHEMA_PATH)
90
+ self.serverapp.event_logger.register_event_schema(FORK_EVENTS_SCHEMA_PATH)
88
91
 
89
92
  def initialize_settings(self):
90
93
  self.settings.update(
@@ -123,6 +126,13 @@ class YDocExtension(ExtensionApp):
123
126
 
124
127
  self.handlers.extend(
125
128
  [
129
+ (
130
+ r"/api/collaboration/fork/(.*)",
131
+ DocForkHandler,
132
+ {
133
+ "ywebsocket_server": self.ywebsocket_server,
134
+ },
135
+ ),
126
136
  (
127
137
  r"/api/collaboration/room/(.*)",
128
138
  YDocWebSocketHandler,
@@ -0,0 +1,56 @@
1
+ "$id": https://schema.jupyter.org/jupyter_collaboration/fork/v1
2
+ "$schema": "http://json-schema.org/draft-07/schema"
3
+ version: 1
4
+ title: Collaborative fork events
5
+ personal-data: true
6
+ description: |
7
+ Fork events emitted from server-side during a collaborative session.
8
+ type: object
9
+ required:
10
+ - fork_roomid
11
+ - fork_info
12
+ - username
13
+ - action
14
+ properties:
15
+ fork_roomid:
16
+ type: string
17
+ description: |
18
+ Fork root room ID.
19
+ fork_info:
20
+ type: object
21
+ description: |
22
+ Fork root room information.
23
+ required:
24
+ - root_roomid
25
+ - synchronize
26
+ - title
27
+ - description
28
+ properties:
29
+ root_roomid:
30
+ type: string
31
+ description: |
32
+ Root room ID. Usually composed by the file type, format and ID.
33
+ synchronize:
34
+ type: boolean
35
+ description: |
36
+ Whether the fork is kept in sync with the root.
37
+ title:
38
+ type: string
39
+ description: |
40
+ The title of the fork.
41
+ description:
42
+ type: string
43
+ description: |
44
+ The description of the fork.
45
+ username:
46
+ type: string
47
+ description: |
48
+ The name of the user who created or deleted the fork.
49
+ action:
50
+ enum:
51
+ - create
52
+ - delete
53
+ description: |
54
+ Possible values:
55
+ 1. create
56
+ 2. delete
@@ -26,6 +26,7 @@ from .rooms import DocumentRoom, TransientRoom
26
26
  from .utils import (
27
27
  JUPYTER_COLLABORATION_AWARENESS_EVENTS_URI,
28
28
  JUPYTER_COLLABORATION_EVENTS_URI,
29
+ JUPYTER_COLLABORATION_FORK_EVENTS_URI,
29
30
  LogLevel,
30
31
  MessageType,
31
32
  decode_file_path,
@@ -39,6 +40,7 @@ YFILE = YDOCS["file"]
39
40
 
40
41
  SERVER_SESSION = str(uuid.uuid4())
41
42
  FORK_DOCUMENTS = {}
43
+ FORK_ROOMS: dict[str, dict[str, str]] = {}
42
44
 
43
45
 
44
46
  class YDocWebSocketHandler(WebSocketHandler, JupyterHandler):
@@ -600,3 +602,101 @@ class UndoRedoHandler(APIHandler):
600
602
  if room_id in FORK_DOCUMENTS:
601
603
  del FORK_DOCUMENTS[room_id]
602
604
  self.log.info(f"Fork Document for {room_id} has been removed.")
605
+
606
+
607
+ class DocForkHandler(APIHandler):
608
+ """
609
+ Jupyter Server handler to:
610
+ - create a fork of a root document (optionally synchronizing with the root document),
611
+ - delete a fork of a root document (optionally merging back in the root document).
612
+ - get fork IDs of a root document.
613
+ """
614
+
615
+ auth_resource = "contents"
616
+
617
+ def initialize(
618
+ self,
619
+ ywebsocket_server: JupyterWebsocketServer,
620
+ ) -> None:
621
+ self._websocket_server = ywebsocket_server
622
+
623
+ @web.authenticated
624
+ @authorized
625
+ async def get(self, root_roomid):
626
+ """
627
+ Returns a dictionary of fork room ID to fork room information for the given root room ID.
628
+ """
629
+ self.write(
630
+ {
631
+ fork_roomid: fork_info
632
+ for fork_roomid, fork_info in FORK_ROOMS.items()
633
+ if fork_info["root_roomid"] == root_roomid
634
+ }
635
+ )
636
+
637
+ @web.authenticated
638
+ @authorized
639
+ async def put(self, root_roomid):
640
+ """
641
+ Creates a fork of a root document and returns its ID.
642
+ Optionally keeps the fork in sync with the root.
643
+ """
644
+ fork_roomid = uuid4().hex
645
+ root_room = await self._websocket_server.get_room(root_roomid)
646
+ update = root_room.ydoc.get_update()
647
+ fork_ydoc = Doc()
648
+ fork_ydoc.apply_update(update)
649
+ model = self.get_json_body()
650
+ synchronize = model.get("synchronize", False)
651
+ if synchronize:
652
+ root_room.ydoc.observe(lambda event: fork_ydoc.apply_update(event.update))
653
+ FORK_ROOMS[fork_roomid] = fork_info = {
654
+ "root_roomid": root_roomid,
655
+ "synchronize": synchronize,
656
+ "title": model.get("title", ""),
657
+ "description": model.get("description", ""),
658
+ }
659
+ fork_room = YRoom(ydoc=fork_ydoc)
660
+ self._websocket_server.rooms[fork_roomid] = fork_room
661
+ await self._websocket_server.start_room(fork_room)
662
+ self._emit_fork_event(self.current_user.username, fork_roomid, fork_info, "create")
663
+ data = json.dumps(
664
+ {
665
+ "sessionId": SERVER_SESSION,
666
+ "fork_roomid": fork_roomid,
667
+ "fork_info": fork_info,
668
+ }
669
+ )
670
+ self.set_status(201)
671
+ return self.finish(data)
672
+
673
+ @web.authenticated
674
+ @authorized
675
+ async def delete(self, fork_roomid):
676
+ """
677
+ Deletes a forked document, and optionally merges it back in the root document.
678
+ """
679
+ fork_info = FORK_ROOMS[fork_roomid]
680
+ root_roomid = fork_info["root_roomid"]
681
+ del FORK_ROOMS[fork_roomid]
682
+ if self.get_query_argument("merge") == "true":
683
+ root_room = await self._websocket_server.get_room(root_roomid)
684
+ root_ydoc = root_room.ydoc
685
+ fork_room = await self._websocket_server.get_room(fork_roomid)
686
+ fork_ydoc = fork_room.ydoc
687
+ fork_update = fork_ydoc.get_update()
688
+ root_ydoc.apply_update(fork_update)
689
+ await self._websocket_server.delete_room(name=fork_roomid)
690
+ self._emit_fork_event(self.current_user.username, fork_roomid, fork_info, "delete")
691
+ self.set_status(200)
692
+
693
+ def _emit_fork_event(
694
+ self, username: str, fork_roomid: str, fork_info: dict[str, str], action: str
695
+ ) -> None:
696
+ data = {
697
+ "username": username,
698
+ "fork_roomid": fork_roomid,
699
+ "fork_info": fork_info,
700
+ "action": action,
701
+ }
702
+ self.event_logger.emit(schema_id=JUPYTER_COLLABORATION_FORK_EVENTS_URI, data=data)
@@ -154,6 +154,65 @@ def rtc_connect_doc_client(jp_http_port, jp_base_url, rtc_fetch_session):
154
154
  return _inner
155
155
 
156
156
 
157
+ @pytest.fixture
158
+ def rtc_connect_fork_client(jp_http_port, jp_base_url, rtc_fetch_session):
159
+ async def _inner(room_id: str) -> Any:
160
+ return aconnect_ws(
161
+ f"http://127.0.0.1:{jp_http_port}{jp_base_url}api/collaboration/room/{room_id}"
162
+ )
163
+
164
+ return _inner
165
+
166
+
167
+ @pytest.fixture
168
+ def rtc_get_forks_client(jp_fetch):
169
+ async def _inner(root_roomid: str) -> Any:
170
+ return await jp_fetch(
171
+ "/api/collaboration/fork",
172
+ root_roomid,
173
+ method="GET",
174
+ )
175
+
176
+ return _inner
177
+
178
+
179
+ @pytest.fixture
180
+ def rtc_create_fork_client(jp_fetch):
181
+ async def _inner(
182
+ root_roomid: str,
183
+ synchronize: bool,
184
+ title: str | None = None,
185
+ description: str | None = None,
186
+ ) -> Any:
187
+ return await jp_fetch(
188
+ "/api/collaboration/fork",
189
+ root_roomid,
190
+ method="PUT",
191
+ body=json.dumps(
192
+ {
193
+ "synchronize": synchronize,
194
+ "title": title,
195
+ "description": description,
196
+ }
197
+ ),
198
+ )
199
+
200
+ return _inner
201
+
202
+
203
+ @pytest.fixture
204
+ def rtc_delete_fork_client(jp_fetch):
205
+ async def _inner(fork_roomid: str, merge: bool) -> Any:
206
+ return await jp_fetch(
207
+ "/api/collaboration/fork",
208
+ fork_roomid,
209
+ method="DELETE",
210
+ params={"merge": str(merge).lower()},
211
+ )
212
+
213
+ return _inner
214
+
215
+
157
216
  @pytest.fixture
158
217
  def rtc_add_doc_to_store(rtc_connect_doc_client):
159
218
  event = Event()
@@ -11,7 +11,9 @@ EVENTS_SCHEMA_PATH = EVENTS_FOLDER_PATH / "session.yaml"
11
11
  JUPYTER_COLLABORATION_AWARENESS_EVENTS_URI = (
12
12
  "https://schema.jupyter.org/jupyter_collaboration/awareness/v1"
13
13
  )
14
+ JUPYTER_COLLABORATION_FORK_EVENTS_URI = "https://schema.jupyter.org/jupyter_collaboration/fork/v1"
14
15
  AWARENESS_EVENTS_SCHEMA_PATH = EVENTS_FOLDER_PATH / "awareness.yaml"
16
+ FORK_EVENTS_SCHEMA_PATH = EVENTS_FOLDER_PATH / "fork.yaml"
15
17
 
16
18
 
17
19
  class MessageType(IntEnum):
@@ -41,6 +41,7 @@ dynamic = ["version"]
41
41
  [project.optional-dependencies]
42
42
  test = [
43
43
  "coverage",
44
+ "dirty-equals",
44
45
  "jupyter_server[test]>=2.4.0",
45
46
  "jupyter_server_fileid[test]",
46
47
  "pytest>=7.0",
@@ -7,9 +7,11 @@ import json
7
7
  from asyncio import Event, sleep
8
8
  from typing import Any
9
9
 
10
+ from dirty_equals import IsStr
10
11
  from jupyter_events.logger import EventLogger
11
12
  from jupyter_server_ydoc.test_utils import Websocket
12
13
  from jupyter_ydoc import YUnicode
14
+ from pycrdt import Text
13
15
  from pycrdt_websocket import WebsocketProvider
14
16
 
15
17
 
@@ -211,3 +213,147 @@ async def test_room_handler_doc_client_should_cleanup_room_file(
211
213
 
212
214
  await jp_serverapp.web_app.settings["jupyter_server_ydoc"].stop_extension()
213
215
  del jp_serverapp.web_app.settings["file_id_manager"]
216
+
217
+
218
+ async def test_fork_handler(
219
+ jp_serverapp,
220
+ rtc_create_file,
221
+ rtc_connect_doc_client,
222
+ rtc_connect_fork_client,
223
+ rtc_get_forks_client,
224
+ rtc_create_fork_client,
225
+ rtc_delete_fork_client,
226
+ rtc_fetch_session,
227
+ ):
228
+ collected_data = []
229
+
230
+ async def my_listener(logger: EventLogger, schema_id: str, data: dict) -> None:
231
+ collected_data.append(data)
232
+
233
+ event_logger = jp_serverapp.event_logger
234
+ event_logger.add_listener(
235
+ schema_id="https://schema.jupyter.org/jupyter_collaboration/fork/v1",
236
+ listener=my_listener,
237
+ )
238
+
239
+ path, _ = await rtc_create_file("test.txt", "Hello")
240
+
241
+ root_connect_event = Event()
242
+
243
+ def _on_root_change(topic: str, event: Any) -> None:
244
+ if topic == "source":
245
+ root_connect_event.set()
246
+
247
+ root_ydoc = YUnicode()
248
+ root_ydoc.observe(_on_root_change)
249
+
250
+ resp = await rtc_fetch_session("text", "file", path)
251
+ data = json.loads(resp.body.decode("utf-8"))
252
+ file_id = data["fileId"]
253
+ root_roomid = f"text:file:{file_id}"
254
+
255
+ websocket, room_name = await rtc_connect_doc_client("text", "file", path)
256
+ async with websocket as ws, WebsocketProvider(root_ydoc.ydoc, Websocket(ws, room_name)):
257
+ await root_connect_event.wait()
258
+
259
+ resp = await rtc_create_fork_client(root_roomid, False, "my fork0", "is awesome0")
260
+ data = json.loads(resp.body.decode("utf-8"))
261
+ fork_roomid0 = data["fork_roomid"]
262
+
263
+ resp = await rtc_get_forks_client(root_roomid)
264
+ data = json.loads(resp.body.decode("utf-8"))
265
+ expected_data0 = {
266
+ fork_roomid0: {
267
+ "root_roomid": root_roomid,
268
+ "synchronize": False,
269
+ "title": "my fork0",
270
+ "description": "is awesome0",
271
+ }
272
+ }
273
+ assert data == expected_data0
274
+
275
+ assert collected_data == [
276
+ {
277
+ "username": IsStr(),
278
+ "fork_roomid": fork_roomid0,
279
+ "fork_info": expected_data0[fork_roomid0],
280
+ "action": "create",
281
+ }
282
+ ]
283
+
284
+ resp = await rtc_create_fork_client(root_roomid, True, "my fork1", "is awesome1")
285
+ data = json.loads(resp.body.decode("utf-8"))
286
+ fork_roomid1 = data["fork_roomid"]
287
+
288
+ resp = await rtc_get_forks_client(root_roomid)
289
+ data = json.loads(resp.body.decode("utf-8"))
290
+ expected_data1 = {
291
+ fork_roomid1: {
292
+ "root_roomid": root_roomid,
293
+ "synchronize": True,
294
+ "title": "my fork1",
295
+ "description": "is awesome1",
296
+ }
297
+ }
298
+ expected_data = dict(**expected_data0, **expected_data1)
299
+ assert data == expected_data
300
+
301
+ assert len(collected_data) == 2
302
+ assert collected_data[1] == {
303
+ "username": IsStr(),
304
+ "fork_roomid": fork_roomid1,
305
+ "fork_info": expected_data[fork_roomid1],
306
+ "action": "create",
307
+ }
308
+
309
+ fork_ydoc = YUnicode()
310
+ fork_connect_event = Event()
311
+
312
+ def _on_fork_change(topic: str, event: Any) -> None:
313
+ if topic == "source":
314
+ fork_connect_event.set()
315
+
316
+ fork_ydoc.observe(_on_fork_change)
317
+ fork_text = fork_ydoc.ydoc.get("source", type=Text)
318
+
319
+ async with await rtc_connect_fork_client(fork_roomid1) as ws, WebsocketProvider(
320
+ fork_ydoc.ydoc, Websocket(ws, fork_roomid1)
321
+ ):
322
+ await fork_connect_event.wait()
323
+ root_text = root_ydoc.ydoc.get("source", type=Text)
324
+ root_text += ", World!"
325
+ await sleep(0.1)
326
+ assert str(fork_text) == "Hello, World!"
327
+ fork_text += " Hi!"
328
+ await sleep(0.1)
329
+
330
+ await sleep(0.1)
331
+ assert str(root_text) == "Hello, World!"
332
+
333
+ await rtc_delete_fork_client(fork_roomid0, True)
334
+ await sleep(0.1)
335
+ assert str(root_text) == "Hello, World!"
336
+ resp = await rtc_get_forks_client(root_roomid)
337
+ data = json.loads(resp.body.decode("utf-8"))
338
+ assert data == expected_data1
339
+ assert len(collected_data) == 3
340
+ assert collected_data[2] == {
341
+ "username": IsStr(),
342
+ "fork_roomid": fork_roomid0,
343
+ "fork_info": expected_data[fork_roomid0],
344
+ "action": "delete",
345
+ }
346
+
347
+ await rtc_delete_fork_client(fork_roomid1, True)
348
+ await sleep(0.1)
349
+ assert str(root_text) == "Hello, World! Hi!"
350
+ resp = await rtc_get_forks_client(root_roomid)
351
+ data = json.loads(resp.body.decode("utf-8"))
352
+ assert data == {}
353
+ assert len(collected_data) == 4
354
+ assert collected_data[3] == {
355
+ "username": IsStr(),
356
+ "fork_roomid": fork_roomid1,
357
+ "fork_info": expected_data[fork_roomid1],
358
+ "action": "delete",
359
+ }
@@ -1 +0,0 @@
1
- __version__ = "1.0.1"