* WIP * feat: Add collection.memberships endpoint * feat: Add ability to filter collection.memberships with query * WIP * Merge stashed work * feat: Add ability to filter memberships by permission * continued refactoring * paginated list component * Collection member management * fix: Incorrect policy data sent down after collection.update * Reduce duplication, add empty state * cleanup * fix: Modal close should be a real button * fix: Allow opening edit from modal * fix: remove unused methods * test: fix * Passing test suite * Refactor * fix: Flow UI errors * test: Add collections.update tests * lint * test: moar tests * fix: Missing scopes, more missing tests * fix: Handle collection privacy change over socket * fix: More membership scopes * fix: view endpoint permissions * fix: respond to privacy change on socket event * policy driven menus * fix: share endpoint policies * chore: Use policies to drive documents UI * alignment * fix: Header height * fix: Correct behavior when collection becomes private * fix: Header height for read-only collection * send id's over socket instead of serialized objects * fix: Remote policy change * fix: reduce collection fetching * More websocket efficiencies * fix: Document collection pinning * fix: Restored ability to edit drafts fix: Removed ability to star drafts * fix: Require write permissions to pin doc to collection * fix: Header title overlaying document actions at small screen sizes * fix: Jank on load caused by previous commit * fix: Double collection fetch post-publish * fix: Hide publish button if draft is in no longer accessible collection * fix: Always allow deleting drafts fix: Improved handling of deleted documents * feat: Show collections in drafts view feat: Show more obvious 'draft' badge on documents * fix: incorrect policies after publish to private collection * fix: Duplicating a draft publishes it
226 lines
6.4 KiB
JavaScript
226 lines
6.4 KiB
JavaScript
// @flow
|
|
import type { Event } from '../events';
|
|
import { Document, Collection } from '../models';
|
|
import { socketio } from '../';
|
|
|
|
export default class Websockets {
|
|
async on(event: Event) {
|
|
if (process.env.WEBSOCKETS_ENABLED !== 'true' || !socketio) return;
|
|
|
|
switch (event.name) {
|
|
case 'documents.publish':
|
|
case 'documents.restore':
|
|
case 'documents.archive':
|
|
case 'documents.unarchive': {
|
|
const document = await Document.findByPk(event.documentId, {
|
|
paranoid: false,
|
|
});
|
|
|
|
return socketio
|
|
.to(`collection-${document.collectionId}`)
|
|
.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.pin':
|
|
case 'documents.unpin':
|
|
case 'documents.update': {
|
|
const document = await Document.findByPk(event.documentId, {
|
|
paranoid: false,
|
|
});
|
|
|
|
return socketio
|
|
.to(`collection-${document.collectionId}`)
|
|
.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.private
|
|
? `collection-${collection.id}`
|
|
: `team-${collection.teamId}`
|
|
)
|
|
.emit('entities', {
|
|
event: event.name,
|
|
collectionIds: [
|
|
{
|
|
id: collection.id,
|
|
updatedAt: collection.updatedAt,
|
|
},
|
|
],
|
|
});
|
|
return socketio
|
|
.to(
|
|
collection.private
|
|
? `collection-${collection.id}`
|
|
: `team-${collection.teamId}`
|
|
)
|
|
.emit('join', {
|
|
event: event.name,
|
|
roomId: 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.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,
|
|
roomId: event.collectionId,
|
|
});
|
|
}
|
|
case 'collections.remove_user': {
|
|
// let everyone with access to the collection know a user was removed
|
|
socketio.to(`collection-${event.collectionId}`).emit(event.name, {
|
|
event: event.name,
|
|
userId: event.userId,
|
|
collectionId: event.collectionId,
|
|
});
|
|
|
|
// tell any user clients to disconnect from the websocket channel for the collection
|
|
return socketio.to(`user-${event.userId}`).emit('leave', {
|
|
event: event.name,
|
|
roomId: event.collectionId,
|
|
});
|
|
}
|
|
default:
|
|
}
|
|
}
|
|
}
|