jupyter-server-ydoc 1.0.0rc1__py3-none-any.whl → 1.1.0__py3-none-any.whl

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.
@@ -1 +1 @@
1
- __version__ = "1.0.0rc1"
1
+ __version__ = "1.1.0"
@@ -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)
@@ -9,14 +9,19 @@ from typing import Any
9
9
 
10
10
  import nbformat
11
11
  import pytest
12
+ from httpx_ws import aconnect_ws
12
13
  from jupyter_server_ydoc.loaders import FileLoader
13
14
  from jupyter_server_ydoc.rooms import DocumentRoom
14
15
  from jupyter_server_ydoc.stores import SQLiteYStore
15
16
  from jupyter_ydoc import YNotebook, YUnicode
16
17
  from pycrdt_websocket import WebsocketProvider
17
- from websockets import connect
18
18
 
19
- from .test_utils import FakeContentsManager, FakeEventLogger, FakeFileIDManager
19
+ from .test_utils import (
20
+ FakeContentsManager,
21
+ FakeEventLogger,
22
+ FakeFileIDManager,
23
+ Websocket,
24
+ )
20
25
 
21
26
 
22
27
  @pytest.fixture
@@ -126,8 +131,8 @@ def rtc_fetch_session(jp_fetch):
126
131
  @pytest.fixture
127
132
  def rtc_connect_awareness_client(jp_http_port, jp_base_url):
128
133
  async def _inner(room_id: str) -> Any:
129
- return connect(
130
- f"ws://127.0.0.1:{jp_http_port}{jp_base_url}api/collaboration/room/{room_id}"
134
+ return aconnect_ws(
135
+ f"http://127.0.0.1:{jp_http_port}{jp_base_url}api/collaboration/room/{room_id}"
131
136
  )
132
137
 
133
138
  return _inner
@@ -138,8 +143,71 @@ def rtc_connect_doc_client(jp_http_port, jp_base_url, rtc_fetch_session):
138
143
  async def _inner(format: str, type: str, path: str) -> Any:
139
144
  resp = await rtc_fetch_session(format, type, path)
140
145
  data = json.loads(resp.body.decode("utf-8"))
141
- return connect(
142
- f"ws://127.0.0.1:{jp_http_port}{jp_base_url}api/collaboration/room/{data['format']}:{data['type']}:{data['fileId']}?sessionId={data['sessionId']}"
146
+ room_name = f"{data['format']}:{data['type']}:{data['fileId']}"
147
+ return (
148
+ aconnect_ws(
149
+ f"http://127.0.0.1:{jp_http_port}{jp_base_url}api/collaboration/room/{room_name}?sessionId={data['sessionId']}"
150
+ ),
151
+ room_name,
152
+ )
153
+
154
+ return _inner
155
+
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()},
143
211
  )
144
212
 
145
213
  return _inner
@@ -162,9 +230,8 @@ def rtc_add_doc_to_store(rtc_connect_doc_client):
162
230
 
163
231
  doc.observe(_on_document_change)
164
232
 
165
- async with await rtc_connect_doc_client(format, type, path) as ws, WebsocketProvider(
166
- doc.ydoc, ws
167
- ):
233
+ websocket, room_name = await rtc_connect_doc_client(format, type, path)
234
+ async with websocket as ws, WebsocketProvider(doc.ydoc, Websocket(ws, room_name)):
168
235
  await event.wait()
169
236
  await sleep(0.1)
170
237
 
@@ -6,6 +6,7 @@ from __future__ import annotations
6
6
  from datetime import datetime
7
7
  from typing import Any
8
8
 
9
+ from anyio import Lock
9
10
  from jupyter_server import _tz as tz
10
11
 
11
12
 
@@ -55,3 +56,32 @@ class FakeContentsManager:
55
56
  class FakeEventLogger:
56
57
  def emit(self, schema_id: str, data: dict) -> None:
57
58
  print(data)
59
+
60
+
61
+ class Websocket:
62
+ def __init__(self, websocket: Any, path: str):
63
+ self._websocket = websocket
64
+ self._path = path
65
+ self._send_lock = Lock()
66
+
67
+ @property
68
+ def path(self) -> str:
69
+ return self._path
70
+
71
+ def __aiter__(self):
72
+ return self
73
+
74
+ async def __anext__(self) -> bytes:
75
+ try:
76
+ message = await self.recv()
77
+ except Exception:
78
+ raise StopAsyncIteration()
79
+ return message
80
+
81
+ async def send(self, message: bytes) -> None:
82
+ async with self._send_lock:
83
+ await self._websocket.send_bytes(message)
84
+
85
+ async def recv(self) -> bytes:
86
+ b = await self._websocket.receive_bytes()
87
+ return bytes(b)
@@ -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):
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.3
2
2
  Name: jupyter-server-ydoc
3
- Version: 1.0.0rc1
3
+ Version: 1.1.0
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
@@ -62,7 +62,6 @@ License: # Licensing terms
62
62
 
63
63
  # Copyright (c) Jupyter Development Team.
64
64
  # Distributed under the terms of the Modified BSD License.
65
- License-File: LICENSE
66
65
  Classifier: Framework :: Jupyter
67
66
  Classifier: Intended Audience :: Developers
68
67
  Classifier: Intended Audience :: Science/Research
@@ -79,17 +78,19 @@ Requires-Dist: jsonschema>=4.18.0
79
78
  Requires-Dist: jupyter-events>=0.10.0
80
79
  Requires-Dist: jupyter-server-fileid<1,>=0.7.0
81
80
  Requires-Dist: jupyter-server<3.0.0,>=2.11.1
82
- Requires-Dist: jupyter-ydoc<4.0.0,>=2.1.2
81
+ Requires-Dist: jupyter-ydoc!=3.0.0,!=3.0.1,<4.0.0,>=2.1.2
83
82
  Requires-Dist: pycrdt
84
83
  Requires-Dist: pycrdt-websocket<0.16.0,>=0.15.0
85
84
  Provides-Extra: test
85
+ Requires-Dist: anyio; extra == 'test'
86
86
  Requires-Dist: coverage; extra == 'test'
87
+ Requires-Dist: dirty-equals; extra == 'test'
88
+ Requires-Dist: httpx-ws>=0.5.2; extra == 'test'
87
89
  Requires-Dist: importlib-metadata>=4.8.3; (python_version < '3.10') and extra == 'test'
88
90
  Requires-Dist: jupyter-server-fileid[test]; extra == 'test'
89
91
  Requires-Dist: jupyter-server[test]>=2.4.0; extra == 'test'
90
92
  Requires-Dist: pytest-cov; extra == 'test'
91
93
  Requires-Dist: pytest>=7.0; extra == 'test'
92
- Requires-Dist: websockets; extra == 'test'
93
94
  Description-Content-Type: text/markdown
94
95
 
95
96
  # jupyter-server-ydoc
@@ -0,0 +1,19 @@
1
+ jupyter_server_ydoc/__init__.py,sha256=B8H7XLhzgrTCQD8304Lx91FYXslwabsnV9OuYu4M4Hw,346
2
+ jupyter_server_ydoc/_version.py,sha256=LGVQyDsWifdACo7qztwb8RWWHds1E7uQ-ZqD8SAjyw4,22
3
+ jupyter_server_ydoc/app.py,sha256=JqkpijoPdo_qum0CKqbG-I6W8fpC3-v2cFA3kFxK3mg,8111
4
+ jupyter_server_ydoc/handlers.py,sha256=5ywD2yuwXNQPnEFO_bkogxk_jPxuCZ-LhyzrkAFRmpE,27668
5
+ jupyter_server_ydoc/loaders.py,sha256=TijilImdgYk9K91cXEIP_DzkOr6phSddwQFpLI5l_RA,10564
6
+ jupyter_server_ydoc/pytest_plugin.py,sha256=1Y-iNZnEyhajx4HU-40aZ9iRVWcC5ikC5Y8JJHCH0So,8419
7
+ jupyter_server_ydoc/rooms.py,sha256=szOAfMldhQIrmVpqoF75O0_KXY54X_TrzJz6vpjR6kE,12254
8
+ jupyter_server_ydoc/stores.py,sha256=_5J6eNs3R5Tv88PCc-GGuszxQstfvNoBCYABqzBzJXA,1004
9
+ jupyter_server_ydoc/test_utils.py,sha256=utUwB5FThc_SCQshhUbLNih9GUa5qBcmMgU6-jx0ZnA,2275
10
+ jupyter_server_ydoc/utils.py,sha256=EgKC15js8VOS8-5jGMs4pfHQfV9drnNT2Gew5UlyXZc,2171
11
+ jupyter_server_ydoc/websocketserver.py,sha256=7fLPJcWczD-4R_-LXtfvNxM_pUXFasZWDmT4RIrOQHE,5150
12
+ jupyter_server_ydoc/events/awareness.yaml,sha256=9isoK58uue7lqMnlHqyfQt29z16Otkh14oRe1k5vbKM,753
13
+ jupyter_server_ydoc/events/fork.yaml,sha256=v7WQX2tkD6Zoh9a9qiJpq6wsU4UVnhdZfZSxdnukk_4,1301
14
+ jupyter_server_ydoc/events/session.yaml,sha256=A7Wt7czyx38MXp5fpDbH7HLS0QNkeOqaEhHdP2x-0Mo,1594
15
+ jupyter_server_ydoc-1.1.0.data/data/etc/jupyter/jupyter_server_config.d/jupyter_collaboration.json,sha256=0thh2hJUxAKkZSmneJMG0U6QJRjdM6zGlwrTedEt-Jk,94
16
+ jupyter_server_ydoc-1.1.0.dist-info/METADATA,sha256=eZBtRd3rSEhhqPITzo0bSPbO9zcrzNIALgPmaypFNKo,5092
17
+ jupyter_server_ydoc-1.1.0.dist-info/WHEEL,sha256=C2FUgwZgiLbznR-k0b_5k3Ai_1aASOXDss3lzCUsUug,87
18
+ jupyter_server_ydoc-1.1.0.dist-info/licenses/LICENSE,sha256=mhO0ZW9EiWOPg0dUgB-lNbJ0CGwRmTdbeAg_se1SOnY,2833
19
+ jupyter_server_ydoc-1.1.0.dist-info/RECORD,,
@@ -1,4 +1,4 @@
1
1
  Wheel-Version: 1.0
2
- Generator: hatchling 1.25.0
2
+ Generator: hatchling 1.26.3
3
3
  Root-Is-Purelib: true
4
4
  Tag: py3-none-any
@@ -1,18 +0,0 @@
1
- jupyter_server_ydoc/__init__.py,sha256=B8H7XLhzgrTCQD8304Lx91FYXslwabsnV9OuYu4M4Hw,346
2
- jupyter_server_ydoc/_version.py,sha256=iwDi0TUz45SMYIlshEAh-UPerqIe7nxUavbLYwEgSR8,25
3
- jupyter_server_ydoc/app.py,sha256=8vOKWLYa4OKu8Pw24TLWtbgs2SfH0_R64dPrErdWyQM,7739
4
- jupyter_server_ydoc/handlers.py,sha256=xfcTzrOV08wJVIwxHhp9DdVH3ogEY5E_ktLpzBXV5qE,24016
5
- jupyter_server_ydoc/loaders.py,sha256=TijilImdgYk9K91cXEIP_DzkOr6phSddwQFpLI5l_RA,10564
6
- jupyter_server_ydoc/pytest_plugin.py,sha256=pSw5KGHid4qRgu1PqQ-GiNOH7IBSWudXQX21J43cB3o,6805
7
- jupyter_server_ydoc/rooms.py,sha256=szOAfMldhQIrmVpqoF75O0_KXY54X_TrzJz6vpjR6kE,12254
8
- jupyter_server_ydoc/stores.py,sha256=_5J6eNs3R5Tv88PCc-GGuszxQstfvNoBCYABqzBzJXA,1004
9
- jupyter_server_ydoc/test_utils.py,sha256=IFXUPf1efHd4DgC1GT7ZMJMhKryKlB0Lx4vU2-mhz4Q,1540
10
- jupyter_server_ydoc/utils.py,sha256=yQC-uRdLyFDYbt2Zms_hA1HyjlwznMK4yQ3_FUwTlnQ,2013
11
- jupyter_server_ydoc/websocketserver.py,sha256=7fLPJcWczD-4R_-LXtfvNxM_pUXFasZWDmT4RIrOQHE,5150
12
- jupyter_server_ydoc/events/awareness.yaml,sha256=9isoK58uue7lqMnlHqyfQt29z16Otkh14oRe1k5vbKM,753
13
- jupyter_server_ydoc/events/session.yaml,sha256=A7Wt7czyx38MXp5fpDbH7HLS0QNkeOqaEhHdP2x-0Mo,1594
14
- jupyter_server_ydoc-1.0.0rc1.data/data/etc/jupyter/jupyter_server_config.d/jupyter_collaboration.json,sha256=0thh2hJUxAKkZSmneJMG0U6QJRjdM6zGlwrTedEt-Jk,94
15
- jupyter_server_ydoc-1.0.0rc1.dist-info/METADATA,sha256=8MBKlcFq26cR8DCRI2rNc4ENZLCLoT-SR6ktn9FpbX0,5013
16
- jupyter_server_ydoc-1.0.0rc1.dist-info/WHEEL,sha256=1yFddiXMmvYK7QYTqtRNtX66WJ0Mz8PYEiEUoOUUxRY,87
17
- jupyter_server_ydoc-1.0.0rc1.dist-info/licenses/LICENSE,sha256=mhO0ZW9EiWOPg0dUgB-lNbJ0CGwRmTdbeAg_se1SOnY,2833
18
- jupyter_server_ydoc-1.0.0rc1.dist-info/RECORD,,