mirror of
https://github.com/gristlabs/grist-core.git
synced 2024-10-27 20:44:07 +00:00
4f1cb53b29
Summary: - Add app/common/CommTypes.ts to define types shared by client and server. - Include @types/ws npm package Test Plan: Intended to have no changes in behavior Reviewers: paulfitz Reviewed By: paulfitz Differential Revision: https://phab.getgrist.com/D3467
144 lines
5.5 KiB
TypeScript
144 lines
5.5 KiB
TypeScript
/**
|
|
* Module to manage the clients of an ActiveDoc. It keeps track of how many clients have the doc
|
|
* open, and what FD they are using.
|
|
*/
|
|
|
|
import {CommDocEventType} from 'app/common/CommTypes';
|
|
import {arrayRemove} from 'app/common/gutil';
|
|
import {ActiveDoc} from 'app/server/lib/ActiveDoc';
|
|
import {Authorizer} from 'app/server/lib/Authorizer';
|
|
import {Client} from 'app/server/lib/Client';
|
|
import {sendDocMessage} from 'app/server/lib/Comm';
|
|
import {DocSession, OptDocSession} from 'app/server/lib/DocSession';
|
|
import {LogMethods} from "app/server/lib/LogMethods";
|
|
|
|
// Allow tests to impose a serial order for broadcasts if they need that for repeatability.
|
|
export const Deps = {
|
|
BROADCAST_ORDER: 'parallel' as 'parallel' | 'series',
|
|
};
|
|
|
|
export class DocClients {
|
|
private _docSessions: DocSession[] = [];
|
|
private _log = new LogMethods('DocClients ', (s: DocSession|null) => this.activeDoc.getLogMeta(s));
|
|
|
|
constructor(
|
|
public readonly activeDoc: ActiveDoc
|
|
) {}
|
|
|
|
/**
|
|
* Returns the number of connected clients.
|
|
*/
|
|
public clientCount(): number {
|
|
return this._docSessions.length;
|
|
}
|
|
|
|
/**
|
|
* Adds a client's open file to the list of connected clients.
|
|
*/
|
|
public addClient(client: Client, authorizer: Authorizer): DocSession {
|
|
const docSession = client.addDocSession(this.activeDoc, authorizer);
|
|
this._docSessions.push(docSession);
|
|
this._log.debug(docSession, "now %d clients; new client is %s (fd %s)",
|
|
this._docSessions.length, client.clientId, docSession.fd);
|
|
return docSession;
|
|
}
|
|
|
|
/**
|
|
* Removes a client from the list of connected clients for this document. In other words, closes
|
|
* this DocSession.
|
|
*/
|
|
public removeClient(docSession: DocSession): void {
|
|
this._log.debug(docSession, "removeClient", docSession.client.clientId);
|
|
docSession.client.removeDocSession(docSession.fd);
|
|
|
|
if (arrayRemove(this._docSessions, docSession)) {
|
|
this._log.debug(docSession, "now %d clients", this._docSessions.length);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Removes all active clients from this document, i.e. closes all DocSessions.
|
|
*/
|
|
public removeAllClients(): void {
|
|
this._log.debug(null, "removeAllClients() removing %s docSessions", this._docSessions.length);
|
|
const docSessions = this._docSessions.splice(0);
|
|
for (const docSession of docSessions) {
|
|
docSession.client.removeDocSession(docSession.fd);
|
|
}
|
|
}
|
|
|
|
public interruptAllClients() {
|
|
this._log.debug(null, "interruptAllClients() interrupting %s docSessions", this._docSessions.length);
|
|
for (const docSession of this._docSessions) {
|
|
docSession.client.interruptConnection();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Broadcasts a message to all clients of this document using Comm.sendDocMessage. Also sends all
|
|
* docAction to active doc's plugin manager.
|
|
* @param {Object} client: Originating client used to set the `fromSelf` flag in the message.
|
|
* @param {String} type: The type of the message, e.g. 'docUserAction'.
|
|
* @param {Object} messageData: The data for this type of message.
|
|
* @param {Object} filterMessage: Optional callback to filter message per client.
|
|
*/
|
|
public async broadcastDocMessage(client: Client|null, type: CommDocEventType, messageData: any,
|
|
filterMessage?: (docSession: OptDocSession,
|
|
messageData: any) => Promise<any>): Promise<void> {
|
|
const send = (curr: DocSession) => this._send(curr, client, type, messageData, filterMessage);
|
|
if (Deps.BROADCAST_ORDER === 'parallel') {
|
|
await Promise.all(this._docSessions.map(send));
|
|
} else {
|
|
for (const session of this._docSessions) {
|
|
await send(session);
|
|
}
|
|
}
|
|
if (type === "docUserAction" && messageData.docActions) {
|
|
for (const action of messageData.docActions) {
|
|
this.activeDoc.docPluginManager.receiveAction(action);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Send a message to a single client. See broadcastDocMessage for parameters.
|
|
*/
|
|
private async _send(target: DocSession, client: Client|null, type: CommDocEventType, messageData: any,
|
|
filterMessage?: (docSession: OptDocSession,
|
|
messageData: any) => Promise<any>): Promise<void> {
|
|
const fromSelf = (target.client === client);
|
|
try {
|
|
// Make sure user still has view access.
|
|
await target.authorizer.assertAccess('viewers');
|
|
if (!filterMessage) {
|
|
sendDocMessage(target.client, target.fd, type, messageData, fromSelf);
|
|
} else {
|
|
try {
|
|
const filteredMessageData = await filterMessage(target, messageData);
|
|
if (filteredMessageData) {
|
|
sendDocMessage(target.client, target.fd, type, filteredMessageData, fromSelf);
|
|
} else {
|
|
this._log.debug(target, 'skip broadcastDocMessage because it is not allowed for this client');
|
|
}
|
|
} catch (e) {
|
|
if (e.code && e.code === 'NEED_RELOAD') {
|
|
sendDocMessage(target.client, target.fd, 'docShutdown', null, fromSelf);
|
|
} else {
|
|
sendDocMessage(target.client, target.fd, 'docUserAction', {error: String(e)}, fromSelf);
|
|
}
|
|
}
|
|
}
|
|
} catch (e) {
|
|
if (e.code === 'AUTH_NO_VIEW') {
|
|
// Skip sending data to this user, they have no view access.
|
|
this._log.debug(target, 'skip broadcastDocMessage because AUTH_NO_VIEW');
|
|
// Go further and trigger a shutdown for this user, in case they are granted
|
|
// access again later.
|
|
sendDocMessage(target.client, target.fd, 'docShutdown', null, fromSelf);
|
|
} else {
|
|
throw(e);
|
|
}
|
|
}
|
|
}
|
|
}
|