mirror of
https://github.com/docmost/docmost.git
synced 2025-11-13 02:52:36 +10:00
* Switch to httpOnly cookie * create endpoint to retrieve temporary collaboration token * cleanups
91 lines
2.4 KiB
TypeScript
91 lines
2.4 KiB
TypeScript
import {
|
|
MessageBody,
|
|
OnGatewayConnection,
|
|
SubscribeMessage,
|
|
WebSocketGateway,
|
|
WebSocketServer,
|
|
} from '@nestjs/websockets';
|
|
import { Server, Socket } from 'socket.io';
|
|
import { TokenService } from '../core/auth/services/token.service';
|
|
import { JwtType } from '../core/auth/dto/jwt-payload';
|
|
import { OnModuleDestroy } from '@nestjs/common';
|
|
import { SpaceMemberRepo } from '@docmost/db/repos/space/space-member.repo';
|
|
import * as cookie from 'cookie';
|
|
|
|
@WebSocketGateway({
|
|
cors: { origin: '*' },
|
|
transports: ['websocket'],
|
|
})
|
|
export class WsGateway implements OnGatewayConnection, OnModuleDestroy {
|
|
@WebSocketServer()
|
|
server: Server;
|
|
constructor(
|
|
private tokenService: TokenService,
|
|
private spaceMemberRepo: SpaceMemberRepo,
|
|
) {}
|
|
|
|
async handleConnection(client: Socket, ...args: any[]): Promise<void> {
|
|
try {
|
|
const cookies = cookie.parse(client.handshake.headers.cookie);
|
|
const token = await this.tokenService.verifyJwt(cookies['authToken']);
|
|
|
|
if (token.type !== JwtType.ACCESS) {
|
|
client.emit('Unauthorized');
|
|
client.disconnect();
|
|
}
|
|
|
|
const userId = token.sub;
|
|
const workspaceId = token.workspaceId;
|
|
|
|
const userSpaceIds = await this.spaceMemberRepo.getUserSpaceIds(userId);
|
|
|
|
const workspaceRoom = `workspace-${workspaceId}`;
|
|
const spaceRooms = userSpaceIds.map((id) => this.getSpaceRoomName(id));
|
|
|
|
client.join([workspaceRoom, ...spaceRooms]);
|
|
} catch (err) {
|
|
client.emit('Unauthorized');
|
|
client.disconnect();
|
|
}
|
|
}
|
|
|
|
@SubscribeMessage('message')
|
|
handleMessage(client: Socket, data: any): void {
|
|
const spaceEvents = [
|
|
'updateOne',
|
|
'addTreeNode',
|
|
'moveTreeNode',
|
|
'deleteTreeNode',
|
|
];
|
|
|
|
if (spaceEvents.includes(data?.operation) && data?.spaceId) {
|
|
const room = this.getSpaceRoomName(data.spaceId);
|
|
client.broadcast.to(room).emit('message', data);
|
|
return;
|
|
}
|
|
|
|
client.broadcast.emit('message', data);
|
|
}
|
|
|
|
@SubscribeMessage('join-room')
|
|
handleJoinRoom(client: Socket, @MessageBody() roomName: string): void {
|
|
// if room is a space, check if user has permissions
|
|
//client.join(roomName);
|
|
}
|
|
|
|
@SubscribeMessage('leave-room')
|
|
handleLeaveRoom(client: Socket, @MessageBody() roomName: string): void {
|
|
client.leave(roomName);
|
|
}
|
|
|
|
onModuleDestroy() {
|
|
if (this.server) {
|
|
this.server.close();
|
|
}
|
|
}
|
|
|
|
getSpaceRoomName(spaceId: string): string {
|
|
return `space-${spaceId}`;
|
|
}
|
|
}
|