mirror of
https://github.com/outline/outline.git
synced 2026-01-27 05:39:29 -06:00
119 lines
3.3 KiB
TypeScript
119 lines
3.3 KiB
TypeScript
import isEqual from "fast-deep-equal";
|
|
import uniq from "lodash/uniq";
|
|
import { yDocToProsemirrorJSON } from "y-prosemirror";
|
|
import * as Y from "yjs";
|
|
import { ProsemirrorData } from "@shared/types";
|
|
import Logger from "@server/logging/Logger";
|
|
import { Document, Event } from "@server/models";
|
|
import { sequelize } from "@server/storage/database";
|
|
import { AuthenticationType } from "@server/types";
|
|
import semver from "semver";
|
|
|
|
type Props = {
|
|
/** The document ID to update. */
|
|
documentId: string;
|
|
/** Current collaobrative state. */
|
|
ydoc: Y.Doc;
|
|
/** The user IDs that have modified the document since it was last persisted. */
|
|
sessionCollaboratorIds: string[];
|
|
/** Whether the last connection to the document left. */
|
|
isLastConnection: boolean;
|
|
/** The client version, if available. */
|
|
clientVersion: string | null;
|
|
};
|
|
|
|
export default async function documentCollaborativeUpdater({
|
|
documentId,
|
|
ydoc,
|
|
sessionCollaboratorIds,
|
|
isLastConnection,
|
|
clientVersion,
|
|
}: Props) {
|
|
return sequelize.transaction(async (transaction) => {
|
|
await sequelize.query(`SET LOCAL lock_timeout = '15s';`, {
|
|
transaction,
|
|
});
|
|
|
|
const document = await Document.unscoped()
|
|
.scope("withoutState")
|
|
.findOne({
|
|
where: {
|
|
id: documentId,
|
|
},
|
|
transaction,
|
|
lock: {
|
|
of: Document,
|
|
level: transaction.LOCK.UPDATE,
|
|
},
|
|
rejectOnEmpty: true,
|
|
paranoid: false,
|
|
});
|
|
|
|
const state = Y.encodeStateAsUpdate(ydoc);
|
|
const content = yDocToProsemirrorJSON(ydoc, "default") as ProsemirrorData;
|
|
const isUnchanged = isEqual(document.content, content);
|
|
const isDeleted = !!document.deletedAt;
|
|
const lastModifiedById = isDeleted
|
|
? document.lastModifiedById
|
|
: (sessionCollaboratorIds[sessionCollaboratorIds.length - 1] ??
|
|
document.lastModifiedById);
|
|
|
|
if (isUnchanged) {
|
|
return;
|
|
}
|
|
|
|
Logger.info(
|
|
"multiplayer",
|
|
`Persisting ${documentId}, attributed to ${lastModifiedById}`
|
|
);
|
|
|
|
// extract collaborators from doc user data
|
|
const pud = new Y.PermanentUserData(ydoc);
|
|
const pudIds = Array.from(pud.clients.values());
|
|
const collaboratorIds = uniq([
|
|
...document.collaboratorIds,
|
|
...sessionCollaboratorIds,
|
|
...pudIds,
|
|
]);
|
|
|
|
// Either the client or server version could be null, or they could both be
|
|
// set. In that case we want to use the greater (newer) version.
|
|
const editorVersion =
|
|
document.editorVersion && clientVersion
|
|
? semver.gt(clientVersion, document.editorVersion)
|
|
? clientVersion
|
|
: document.editorVersion
|
|
: clientVersion
|
|
? clientVersion
|
|
: document.editorVersion;
|
|
|
|
await document.update(
|
|
{
|
|
content,
|
|
state: Buffer.from(state),
|
|
lastModifiedById,
|
|
collaboratorIds,
|
|
editorVersion,
|
|
},
|
|
{
|
|
transaction,
|
|
hooks: false,
|
|
}
|
|
);
|
|
|
|
await Event.schedule({
|
|
name: "documents.update",
|
|
documentId: document.id,
|
|
collectionId: document.collectionId,
|
|
teamId: document.teamId,
|
|
actorId: lastModifiedById,
|
|
authType: AuthenticationType.APP,
|
|
data: {
|
|
multiplayer: true,
|
|
title: document.title,
|
|
done: isLastConnection,
|
|
},
|
|
});
|
|
});
|
|
}
|