jupyter-server-ydoc 1.0.1__tar.gz → 1.1.0b0__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.
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/PKG-INFO +3 -2
- jupyter_server_ydoc-1.1.0b0/jupyter_server_ydoc/_version.py +1 -0
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/jupyter_server_ydoc/app.py +10 -0
- jupyter_server_ydoc-1.1.0b0/jupyter_server_ydoc/events/fork.yaml +56 -0
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/jupyter_server_ydoc/handlers.py +100 -0
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/jupyter_server_ydoc/pytest_plugin.py +59 -0
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/jupyter_server_ydoc/utils.py +2 -0
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/pyproject.toml +2 -1
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/tests/test_handlers.py +146 -0
- jupyter_server_ydoc-1.0.1/jupyter_server_ydoc/_version.py +0 -1
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/.gitignore +0 -0
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/LICENSE +0 -0
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/README.md +0 -0
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/jupyter-config/jupyter_server_ydoc.json +0 -0
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/jupyter_server_ydoc/__init__.py +0 -0
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/jupyter_server_ydoc/events/awareness.yaml +0 -0
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/jupyter_server_ydoc/events/session.yaml +0 -0
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/jupyter_server_ydoc/loaders.py +0 -0
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/jupyter_server_ydoc/rooms.py +0 -0
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/jupyter_server_ydoc/stores.py +0 -0
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/jupyter_server_ydoc/test_utils.py +0 -0
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/jupyter_server_ydoc/websocketserver.py +0 -0
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/setup.py +0 -0
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/tests/__init__.py +0 -0
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/tests/conftest.py +0 -0
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/tests/test_app.py +0 -0
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/tests/test_documents.py +0 -0
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/tests/test_loaders.py +0 -0
- {jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/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.
|
|
3
|
+
Version: 1.1.0b0
|
|
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
|
|
@@ -78,12 +78,13 @@ Requires-Dist: jsonschema>=4.18.0
|
|
|
78
78
|
Requires-Dist: jupyter-events>=0.10.0
|
|
79
79
|
Requires-Dist: jupyter-server-fileid<1,>=0.7.0
|
|
80
80
|
Requires-Dist: jupyter-server<3.0.0,>=2.11.1
|
|
81
|
-
Requires-Dist: jupyter-ydoc
|
|
81
|
+
Requires-Dist: jupyter-ydoc!=3.0.0,!=3.0.1,<4.0.0,>=2.1.2
|
|
82
82
|
Requires-Dist: pycrdt
|
|
83
83
|
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.0b0"
|
|
@@ -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)
|
{jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/jupyter_server_ydoc/pytest_plugin.py
RENAMED
|
@@ -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):
|
|
@@ -29,7 +29,7 @@ authors = [
|
|
|
29
29
|
]
|
|
30
30
|
dependencies = [
|
|
31
31
|
"jupyter_server>=2.11.1,<3.0.0",
|
|
32
|
-
"jupyter_ydoc>=2.1.2,<4.0.0",
|
|
32
|
+
"jupyter_ydoc>=2.1.2,<4.0.0,!=3.0.0,!=3.0.1",
|
|
33
33
|
"pycrdt",
|
|
34
34
|
"pycrdt-websocket>=0.15.0,<0.16.0",
|
|
35
35
|
"jupyter_events>=0.10.0",
|
|
@@ -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"
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
{jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/jupyter-config/jupyter_server_ydoc.json
RENAMED
|
File without changes
|
|
File without changes
|
{jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/jupyter_server_ydoc/events/awareness.yaml
RENAMED
|
File without changes
|
{jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/jupyter_server_ydoc/events/session.yaml
RENAMED
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
{jupyter_server_ydoc-1.0.1 → jupyter_server_ydoc-1.1.0b0}/jupyter_server_ydoc/websocketserver.py
RENAMED
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|