// @flow import { subHours } from "date-fns"; import { Document, Collection, Group, CollectionGroup, GroupUser, } from "../../models"; import { Op } from "../../sequelize"; import type { Event } from "../../types"; export default class WebsocketsProcessor { async on(event: Event, socketio: any) { switch (event.name) { case "documents.publish": case "documents.restore": case "documents.archive": case "documents.unarchive": { const document = await Document.findByPk(event.documentId, { paranoid: false, }); const channel = document.publishedAt ? `collection-${document.collectionId}` : `user-${event.actorId}`; return socketio.to(channel).emit("entities", { event: event.name, documentIds: [ { id: document.id, updatedAt: document.updatedAt, }, ], collectionIds: [ { id: document.collectionId, }, ], }); } case "documents.delete": { const document = await Document.findByPk(event.documentId, { paranoid: false, }); if (!document.publishedAt) { return socketio.to(`user-${document.createdById}`).emit("entities", { event: event.name, documentIds: [ { id: document.id, updatedAt: document.updatedAt, }, ], }); } return socketio .to(`collection-${document.collectionId}`) .emit("entities", { event: event.name, documentIds: [ { id: document.id, updatedAt: document.updatedAt, }, ], collectionIds: [ { id: document.collectionId, }, ], }); } case "documents.permanent_delete": { return socketio .to(`collection-${event.collectionId}`) .emit(event.name, { documentId: event.documentId, }); } case "documents.pin": case "documents.unpin": case "documents.update": { const document = await Document.findByPk(event.documentId, { paranoid: false, }); const channel = document.publishedAt ? `collection-${document.collectionId}` : `user-${event.actorId}`; return socketio.to(channel).emit("entities", { event: event.name, documentIds: [ { id: document.id, updatedAt: document.updatedAt, }, ], }); } case "documents.create": { const document = await Document.findByPk(event.documentId); return socketio.to(`user-${event.actorId}`).emit("entities", { event: event.name, documentIds: [ { id: document.id, updatedAt: document.updatedAt, }, ], collectionIds: [ { id: document.collectionId, }, ], }); } case "documents.star": case "documents.unstar": { return socketio.to(`user-${event.actorId}`).emit(event.name, { documentId: event.documentId, }); } case "documents.move": { const documents = await Document.findAll({ where: { id: event.data.documentIds, }, paranoid: false, }); documents.forEach((document) => { socketio.to(`collection-${document.collectionId}`).emit("entities", { event: event.name, documentIds: [ { id: document.id, updatedAt: document.updatedAt, }, ], }); }); event.data.collectionIds.forEach((collectionId) => { socketio.to(`collection-${collectionId}`).emit("entities", { event: event.name, collectionIds: [{ id: collectionId }], }); }); return; } case "collections.create": { const collection = await Collection.findByPk(event.collectionId, { paranoid: false, }); socketio .to( collection.permission ? `team-${collection.teamId}` : `collection-${collection.id}` ) .emit("entities", { event: event.name, collectionIds: [ { id: collection.id, updatedAt: collection.updatedAt, }, ], }); return socketio .to( collection.permission ? `team-${collection.teamId}` : `collection-${collection.id}` ) .emit("join", { event: event.name, collectionId: collection.id, }); } case "collections.update": case "collections.delete": { const collection = await Collection.findByPk(event.collectionId, { paranoid: false, }); return socketio.to(`team-${collection.teamId}`).emit("entities", { event: event.name, collectionIds: [ { id: collection.id, updatedAt: collection.updatedAt, }, ], }); } case "collections.move": { return socketio .to(`collection-${event.collectionId}`) .emit("collections.update_index", { collectionId: event.collectionId, index: event.data.index, }); } case "collections.add_user": { // the user being added isn't yet in the websocket channel for the collection // so they need to be notified separately socketio.to(`user-${event.userId}`).emit(event.name, { event: event.name, userId: event.userId, collectionId: event.collectionId, }); // let everyone with access to the collection know a user was added socketio.to(`collection-${event.collectionId}`).emit(event.name, { event: event.name, userId: event.userId, collectionId: event.collectionId, }); // tell any user clients to connect to the websocket channel for the collection return socketio.to(`user-${event.userId}`).emit("join", { event: event.name, collectionId: event.collectionId, }); } case "collections.remove_user": { const membershipUserIds = await Collection.membershipUserIds( event.collectionId ); if (membershipUserIds.includes(event.userId)) { // Even though we just removed a user from the collection // the user still has access through some means // treat this like an add, so that the client re-syncs policies socketio.to(`user-${event.userId}`).emit("collections.add_user", { event: "collections.add_user", userId: event.userId, collectionId: event.collectionId, }); } else { // let everyone with access to the collection know a user was removed socketio .to(`collection-${event.collectionId}`) .emit("collections.remove_user", { event: event.name, userId: event.userId, collectionId: event.collectionId, }); // tell any user clients to disconnect from the websocket channel for the collection socketio.to(`user-${event.userId}`).emit("leave", { event: event.name, collectionId: event.collectionId, }); } return; } case "collections.add_group": { const group = await Group.findByPk(event.data.groupId); // the users being added are not yet in the websocket channel for the collection // so they need to be notified separately for (const groupMembership of group.groupMemberships) { socketio .to(`user-${groupMembership.userId}`) .emit("collections.add_user", { event: event.name, userId: groupMembership.userId, collectionId: event.collectionId, }); // tell any user clients to connect to the websocket channel for the collection socketio.to(`user-${groupMembership.userId}`).emit("join", { event: event.name, collectionId: event.collectionId, }); } return; } case "collections.remove_group": { const group = await Group.findByPk(event.data.groupId); const membershipUserIds = await Collection.membershipUserIds( event.collectionId ); for (const groupMembership of group.groupMemberships) { if (membershipUserIds.includes(groupMembership.userId)) { // the user still has access through some means... // treat this like an add, so that the client re-syncs policies socketio .to(`user-${groupMembership.userId}`) .emit("collections.add_user", { event: event.name, userId: groupMembership.userId, collectionId: event.collectionId, }); } else { // let users in the channel know they were removed socketio .to(`user-${groupMembership.userId}`) .emit("collections.remove_user", { event: event.name, userId: groupMembership.userId, collectionId: event.collectionId, }); // tell any user clients to disconnect to the websocket channel for the collection socketio.to(`user-${groupMembership.userId}`).emit("leave", { event: event.name, collectionId: event.collectionId, }); } } return; } case "fileOperations.update": { return socketio .to(`user-${event.actorId}`) .emit("fileOperations.update", event.data); } case "groups.create": case "groups.update": { const group = await Group.findByPk(event.modelId, { paranoid: false, }); return socketio.to(`team-${group.teamId}`).emit("entities", { event: event.name, groupIds: [ { id: group.id, updatedAt: group.updatedAt, }, ], }); } case "groups.add_user": { // do an add user for every collection that the group is a part of const collectionGroupMemberships = await CollectionGroup.findAll({ where: { groupId: event.modelId }, }); for (const collectionGroup of collectionGroupMemberships) { // the user being added isn't yet in the websocket channel for the collection // so they need to be notified separately socketio.to(`user-${event.userId}`).emit("collections.add_user", { event: event.name, userId: event.userId, collectionId: collectionGroup.collectionId, }); // let everyone with access to the collection know a user was added socketio .to(`collection-${collectionGroup.collectionId}`) .emit("collections.add_user", { event: event.name, userId: event.userId, collectionId: collectionGroup.collectionId, }); // tell any user clients to connect to the websocket channel for the collection return socketio.to(`user-${event.userId}`).emit("join", { event: event.name, collectionId: collectionGroup.collectionId, }); } return; } case "groups.remove_user": { const collectionGroupMemberships = await CollectionGroup.findAll({ where: { groupId: event.modelId }, }); for (const collectionGroup of collectionGroupMemberships) { // if the user has any memberships remaining on the collection // we need to emit add instead of remove const collection = await Collection.scope({ method: ["withMembership", event.userId], }).findByPk(collectionGroup.collectionId); if (!collection) { continue; } const hasMemberships = collection.memberships.length > 0 || collection.collectionGroupMemberships.length > 0; if (hasMemberships) { // the user still has access through some means... // treat this like an add, so that the client re-syncs policies socketio.to(`user-${event.userId}`).emit("collections.add_user", { event: event.name, userId: event.userId, collectionId: collectionGroup.collectionId, }); } else { // let everyone with access to the collection know a user was removed socketio .to(`collection-${collectionGroup.collectionId}`) .emit("collections.remove_user", { event: event.name, userId: event.userId, collectionId: collectionGroup.collectionId, }); // tell any user clients to disconnect from the websocket channel for the collection socketio.to(`user-${event.userId}`).emit("leave", { event: event.name, collectionId: collectionGroup.collectionId, }); } } return; } case "groups.delete": { const group = await Group.findByPk(event.modelId, { paranoid: false, }); socketio.to(`team-${group.teamId}`).emit("entities", { event: event.name, groupIds: [ { id: group.id, updatedAt: group.updatedAt, }, ], }); // we the users and collection relations that were just severed as a result of the group deletion // since there are cascading deletes, we approximate this by looking for the recently deleted // items in the GroupUser and CollectionGroup tables const groupUsers = await GroupUser.findAll({ paranoid: false, where: { groupId: event.modelId, deletedAt: { [Op.gt]: subHours(new Date(), 1), }, }, }); const collectionGroupMemberships = await CollectionGroup.findAll({ paranoid: false, where: { groupId: event.modelId, deletedAt: { [Op.gt]: subHours(new Date(), 1), }, }, }); for (const collectionGroup of collectionGroupMemberships) { const membershipUserIds = await Collection.membershipUserIds( collectionGroup.collectionId ); for (const groupUser of groupUsers) { if (membershipUserIds.includes(groupUser.userId)) { // the user still has access through some means... // treat this like an add, so that the client re-syncs policies socketio .to(`user-${groupUser.userId}`) .emit("collections.add_user", { event: event.name, userId: groupUser.userId, collectionId: collectionGroup.collectionId, }); } else { // let everyone with access to the collection know a user was removed socketio .to(`collection-${collectionGroup.collectionId}`) .emit("collections.remove_user", { event: event.name, userId: groupUser.userId, collectionId: collectionGroup.collectionId, }); // tell any user clients to disconnect from the websocket channel for the collection socketio.to(`user-${groupUser.userId}`).emit("leave", { event: event.name, collectionId: collectionGroup.collectionId, }); } } } return; } default: } } }