| 0b4b582 | | | 1 | import http from "http"; |
| 0b4b582 | | | 2 | import { Server } from "socket.io"; |
| 0b4b582 | | | 3 | import * as Y from "yjs"; |
| 0b4b582 | | | 4 | import { socketAuth } from "./collab-auth"; |
| 0b4b582 | | | 5 | import { |
| 0b4b582 | | | 6 | getRoom, |
| 0b4b582 | | | 7 | roomKey, |
| 0b4b582 | | | 8 | loadRoom, |
| 0b4b582 | | | 9 | persistRoom, |
| 0b4b582 | | | 10 | getYDoc, |
| 0b4b582 | | | 11 | persistYDoc, |
| 0b4b582 | | | 12 | canAccessRepo, |
| 0b4b582 | | | 13 | pickColor, |
| 0b4b582 | | | 14 | } from "./collab-rooms"; |
| 0b4b582 | | | 15 | |
| 0b4b582 | | | 16 | const PORT = parseInt(process.env.COLLAB_SOCKET_PORT || "3334", 10); |
| 0b4b582 | | | 17 | const CORS_ORIGIN = process.env.CORS_ORIGIN || true; // true = reflect request origin |
| 0b4b582 | | | 18 | |
| 0b4b582 | | | 19 | const server = http.createServer((_req, res) => { |
| 0b4b582 | | | 20 | res.writeHead(200); |
| 0b4b582 | | | 21 | res.end("ok"); |
| 0b4b582 | | | 22 | }); |
| 0b4b582 | | | 23 | |
| 0b4b582 | | | 24 | const io = new Server(server, { |
| 0b4b582 | | | 25 | cors: { origin: CORS_ORIGIN, credentials: true }, |
| 0b4b582 | | | 26 | }); |
| 0b4b582 | | | 27 | |
| 0b4b582 | | | 28 | const collabNs = io.of("/collab"); |
| 0b4b582 | | | 29 | collabNs.use(socketAuth as any); |
| 0b4b582 | | | 30 | |
| 0b4b582 | | | 31 | collabNs.on("connection", (socket: any) => { |
| 0b4b582 | | | 32 | let currentOwner: string | null = null; |
| 0b4b582 | | | 33 | let currentRepo: string | null = null; |
| 0b4b582 | | | 34 | let userName: string | null = null; |
| 0b4b582 | | | 35 | |
| 0b4b582 | | | 36 | socket.on( |
| 0b4b582 | | | 37 | "join-room", |
| 0b4b582 | | | 38 | async ({ owner, repo }: { owner: string; repo: string }) => { |
| 0b4b582 | | | 39 | const allowed = await canAccessRepo(owner, repo, socket.token); |
| 0b4b582 | | | 40 | if (!allowed) { |
| 0b4b582 | | | 41 | socket.emit("error", { message: "Access denied" }); |
| 0b4b582 | | | 42 | return; |
| 0b4b582 | | | 43 | } |
| 0b4b582 | | | 44 | |
| 0b4b582 | | | 45 | currentOwner = owner; |
| 0b4b582 | | | 46 | currentRepo = repo; |
| 0b4b582 | | | 47 | userName = |
| 0b4b582 | | | 48 | socket.user.display_name || |
| 0b4b582 | | | 49 | socket.user.username || |
| 0b4b582 | | | 50 | `User-${socket.id.slice(0, 4)}`; |
| 0b4b582 | | | 51 | |
| 0b4b582 | | | 52 | const key = roomKey(owner, repo); |
| 0b4b582 | | | 53 | socket.join(key); |
| 0b4b582 | | | 54 | |
| 0b4b582 | | | 55 | const room = getRoom(key); |
| 0b4b582 | | | 56 | // Load persisted notes if room is fresh |
| 0b4b582 | | | 57 | if (Object.keys(room.notes).length === 0) { |
| 0b4b582 | | | 58 | room.notes = loadRoom(owner, repo); |
| 0b4b582 | | | 59 | } |
| 0b4b582 | | | 60 | |
| 0b4b582 | | | 61 | room.users[socket.id] = { |
| 0b4b582 | | | 62 | id: socket.id, |
| 0b4b582 | | | 63 | name: userName!, |
| 0b4b582 | | | 64 | color: pickColor(room), |
| 0b4b582 | | | 65 | cursor: null, |
| 0b4b582 | | | 66 | activeTab: null, |
| 0b4b582 | | | 67 | }; |
| 0b4b582 | | | 68 | |
| 0b4b582 | | | 69 | // Send current state to the joining user |
| 0b4b582 | | | 70 | socket.emit("room-state", { |
| 0b4b582 | | | 71 | notes: room.notes, |
| 0b4b582 | | | 72 | users: room.users, |
| 0b4b582 | | | 73 | }); |
| 0b4b582 | | | 74 | |
| 0b4b582 | | | 75 | // Notify others |
| 0b4b582 | | | 76 | collabNs.to(key).emit("users-updated", room.users); |
| 0b4b582 | | | 77 | } |
| 0b4b582 | | | 78 | ); |
| 0b4b582 | | | 79 | |
| 0b4b582 | | | 80 | // ── Yjs sync protocol over socket.io ── |
| 0b4b582 | | | 81 | socket.on("yjs-sync", ({ diagramId }: { diagramId: string }) => { |
| 0b4b582 | | | 82 | if (!currentOwner || !currentRepo) return; |
| 0b4b582 | | | 83 | const ydoc = getYDoc(currentOwner, currentRepo, diagramId); |
| 0b4b582 | | | 84 | const state = Y.encodeStateAsUpdate(ydoc); |
| 0b4b582 | | | 85 | socket.emit("yjs-sync", { |
| 0b4b582 | | | 86 | diagramId, |
| 0b4b582 | | | 87 | update: Buffer.from(state).toString("base64"), |
| 0b4b582 | | | 88 | }); |
| 0b4b582 | | | 89 | }); |
| 0b4b582 | | | 90 | |
| 0b4b582 | | | 91 | socket.on( |
| 0b4b582 | | | 92 | "yjs-update", |
| 0b4b582 | | | 93 | ({ diagramId, update }: { diagramId: string; update: string }) => { |
| 0b4b582 | | | 94 | if (!currentOwner || !currentRepo) return; |
| 0b4b582 | | | 95 | const key = roomKey(currentOwner, currentRepo); |
| 0b4b582 | | | 96 | const ydoc = getYDoc(currentOwner, currentRepo, diagramId); |
| 0b4b582 | | | 97 | const buf = Buffer.from(update, "base64"); |
| 0b4b582 | | | 98 | Y.applyUpdate(ydoc, new Uint8Array(buf)); |
| 0b4b582 | | | 99 | socket.to(key).emit("yjs-update", { diagramId, update }); |
| 0b4b582 | | | 100 | // Persist (debounced per diagram) |
| 0b4b582 | | | 101 | const owner = currentOwner; |
| 0b4b582 | | | 102 | const repo = currentRepo; |
| 0b4b582 | | | 103 | clearTimeout((ydoc as any)._persistTimer); |
| 0b4b582 | | | 104 | (ydoc as any)._persistTimer = setTimeout( |
| 0b4b582 | | | 105 | () => persistYDoc(owner, repo, diagramId), |
| 0b4b582 | | | 106 | 2000 |
| 0b4b582 | | | 107 | ); |
| 0b4b582 | | | 108 | } |
| 0b4b582 | | | 109 | ); |
| 0b4b582 | | | 110 | |
| 0b4b582 | | | 111 | socket.on( |
| 0b4b582 | | | 112 | "cursor-move", |
| 0b4b582 | | | 113 | ({ |
| 0b4b582 | | | 114 | x, |
| 0b4b582 | | | 115 | y, |
| 0b4b582 | | | 116 | activeTab, |
| 0b4b582 | | | 117 | }: { |
| 0b4b582 | | | 118 | x: number; |
| 0b4b582 | | | 119 | y: number; |
| 0b4b582 | | | 120 | activeTab: string; |
| 0b4b582 | | | 121 | }) => { |
| 0b4b582 | | | 122 | if (!currentOwner || !currentRepo) return; |
| 0b4b582 | | | 123 | const key = roomKey(currentOwner, currentRepo); |
| 0b4b582 | | | 124 | const room = getRoom(key); |
| 0b4b582 | | | 125 | if (room.users[socket.id]) { |
| 0b4b582 | | | 126 | room.users[socket.id].cursor = { x, y }; |
| 0b4b582 | | | 127 | room.users[socket.id].activeTab = activeTab; |
| 0b4b582 | | | 128 | } |
| 0b4b582 | | | 129 | socket.to(key).emit("cursor-updated", { |
| 0b4b582 | | | 130 | userId: socket.id, |
| 0b4b582 | | | 131 | x, |
| 0b4b582 | | | 132 | y, |
| 0b4b582 | | | 133 | activeTab, |
| 0b4b582 | | | 134 | name: userName, |
| 0b4b582 | | | 135 | color: room.users[socket.id]?.color, |
| 0b4b582 | | | 136 | }); |
| 0b4b582 | | | 137 | } |
| 0b4b582 | | | 138 | ); |
| 0b4b582 | | | 139 | |
| 0b4b582 | | | 140 | socket.on("add-note", (note: any) => { |
| 0b4b582 | | | 141 | if (!currentOwner || !currentRepo) return; |
| 0b4b582 | | | 142 | const key = roomKey(currentOwner, currentRepo); |
| 0b4b582 | | | 143 | const room = getRoom(key); |
| 0b4b582 | | | 144 | const diagramId = note.diagramId; |
| 0b4b582 | | | 145 | if (!room.notes[diagramId]) room.notes[diagramId] = []; |
| 0b4b582 | | | 146 | |
| 0b4b582 | | | 147 | const fullNote = { |
| 0b4b582 | | | 148 | id: `${Date.now()}-${Math.random().toString(36).slice(2, 7)}`, |
| 0b4b582 | | | 149 | author: userName!, |
| 0b4b582 | | | 150 | text: note.text, |
| 0b4b582 | | | 151 | x: note.x, |
| 0b4b582 | | | 152 | y: note.y, |
| 0b4b582 | | | 153 | diagramId, |
| 0b4b582 | | | 154 | diagramTitle: note.diagramTitle || diagramId, |
| 0b4b582 | | | 155 | targetNode: note.targetNode || null, |
| 0b4b582 | | | 156 | timestamp: new Date().toISOString(), |
| 0b4b582 | | | 157 | }; |
| 0b4b582 | | | 158 | |
| 0b4b582 | | | 159 | room.notes[diagramId].push(fullNote); |
| 0b4b582 | | | 160 | persistRoom(currentOwner, currentRepo); |
| 0b4b582 | | | 161 | collabNs.to(key).emit("note-added", fullNote); |
| 0b4b582 | | | 162 | }); |
| 0b4b582 | | | 163 | |
| 0b4b582 | | | 164 | socket.on( |
| 0b4b582 | | | 165 | "edit-note", |
| 0b4b582 | | | 166 | ({ |
| 0b4b582 | | | 167 | noteId, |
| 0b4b582 | | | 168 | diagramId, |
| 0b4b582 | | | 169 | text, |
| 0b4b582 | | | 170 | }: { |
| 0b4b582 | | | 171 | noteId: string; |
| 0b4b582 | | | 172 | diagramId: string; |
| 0b4b582 | | | 173 | text: string; |
| 0b4b582 | | | 174 | }) => { |
| 0b4b582 | | | 175 | if (!currentOwner || !currentRepo) return; |
| 0b4b582 | | | 176 | const key = roomKey(currentOwner, currentRepo); |
| 0b4b582 | | | 177 | const room = getRoom(key); |
| 0b4b582 | | | 178 | const notes = room.notes[diagramId]; |
| 0b4b582 | | | 179 | if (!notes) return; |
| 0b4b582 | | | 180 | const note = notes.find((n) => n.id === noteId); |
| 0b4b582 | | | 181 | if (!note) return; |
| 0b4b582 | | | 182 | note.text = text; |
| 0b4b582 | | | 183 | note.editedAt = new Date().toISOString(); |
| 0b4b582 | | | 184 | persistRoom(currentOwner, currentRepo); |
| 0b4b582 | | | 185 | collabNs.to(key).emit("note-edited", { |
| 0b4b582 | | | 186 | noteId, |
| 0b4b582 | | | 187 | diagramId, |
| 0b4b582 | | | 188 | text, |
| 0b4b582 | | | 189 | editedAt: note.editedAt, |
| 0b4b582 | | | 190 | }); |
| 0b4b582 | | | 191 | } |
| 0b4b582 | | | 192 | ); |
| 0b4b582 | | | 193 | |
| 0b4b582 | | | 194 | socket.on( |
| 0b4b582 | | | 195 | "diagram-code", |
| 0b4b582 | | | 196 | ({ diagramId, code }: { diagramId: string; code: string }) => { |
| 0b4b582 | | | 197 | if (!currentOwner || !currentRepo) return; |
| 0b4b582 | | | 198 | const key = roomKey(currentOwner, currentRepo); |
| 0b4b582 | | | 199 | socket |
| 0b4b582 | | | 200 | .to(key) |
| 0b4b582 | | | 201 | .emit("diagram-code", { diagramId, code, userId: socket.id }); |
| 0b4b582 | | | 202 | } |
| 0b4b582 | | | 203 | ); |
| 0b4b582 | | | 204 | |
| 0b4b582 | | | 205 | socket.on( |
| 0b4b582 | | | 206 | "delete-note", |
| 0b4b582 | | | 207 | ({ noteId, diagramId }: { noteId: string; diagramId: string }) => { |
| 0b4b582 | | | 208 | if (!currentOwner || !currentRepo) return; |
| 0b4b582 | | | 209 | const key = roomKey(currentOwner, currentRepo); |
| 0b4b582 | | | 210 | const room = getRoom(key); |
| 0b4b582 | | | 211 | if (!room.notes[diagramId]) return; |
| 0b4b582 | | | 212 | room.notes[diagramId] = room.notes[diagramId].filter( |
| 0b4b582 | | | 213 | (n) => n.id !== noteId |
| 0b4b582 | | | 214 | ); |
| 0b4b582 | | | 215 | persistRoom(currentOwner, currentRepo); |
| 0b4b582 | | | 216 | collabNs.to(key).emit("note-deleted", { noteId, diagramId }); |
| 0b4b582 | | | 217 | } |
| 0b4b582 | | | 218 | ); |
| 0b4b582 | | | 219 | |
| 0b4b582 | | | 220 | socket.on("disconnect", () => { |
| 0b4b582 | | | 221 | if (!currentOwner || !currentRepo) return; |
| 0b4b582 | | | 222 | const key = roomKey(currentOwner, currentRepo); |
| 0b4b582 | | | 223 | const room = getRoom(key); |
| 0b4b582 | | | 224 | delete room.users[socket.id]; |
| 0b4b582 | | | 225 | collabNs.to(key).emit("users-updated", room.users); |
| 0b4b582 | | | 226 | collabNs.to(key).emit("cursor-removed", { userId: socket.id }); |
| 0b4b582 | | | 227 | }); |
| 0b4b582 | | | 228 | }); |
| 0b4b582 | | | 229 | |
| 0b4b582 | | | 230 | server.listen(PORT, () => { |
| 0b4b582 | | | 231 | console.log(`Grove Collab socket server running on port ${PORT}`); |
| 0b4b582 | | | 232 | }); |