2017-11-25 17:43:05 -05:00
|
|
|
const WebSocket = require('ws');
|
2017-11-30 23:50:42 -05:00
|
|
|
const utils = require('./utils');
|
|
|
|
const log = require('./log');
|
2017-12-19 23:22:21 -05:00
|
|
|
const sql = require('./sql');
|
2020-01-31 22:32:24 +01:00
|
|
|
const cls = require('./cls');
|
2020-08-29 00:11:50 +02:00
|
|
|
const config = require('./config');
|
2019-10-22 21:59:51 +02:00
|
|
|
const syncMutexService = require('./sync_mutex');
|
2020-02-01 22:29:32 +01:00
|
|
|
const protectedSessionService = require('./protected_session');
|
2021-07-25 21:25:06 +02:00
|
|
|
const becca = require("../becca/becca");
|
2023-01-03 13:52:37 +01:00
|
|
|
const AbstractBeccaEntity = require("../becca/entities/abstract_becca_entity");
|
2017-11-25 17:43:05 -05:00
|
|
|
|
2023-04-15 17:31:55 +08:00
|
|
|
const env = require('./env');
|
|
|
|
if (env.isDev()) {
|
|
|
|
const chokidar = require('chokidar');
|
|
|
|
const debounce = require('debounce');
|
2023-04-15 19:42:09 +08:00
|
|
|
const debouncedReloadFrontend = debounce(reloadFrontend, 200);
|
2023-04-15 17:31:55 +08:00
|
|
|
chokidar
|
|
|
|
.watch('src/public')
|
2023-04-15 19:42:09 +08:00
|
|
|
.on('add', debouncedReloadFrontend)
|
|
|
|
.on('change', debouncedReloadFrontend)
|
|
|
|
.on('unlink', debouncedReloadFrontend);
|
2023-04-15 17:31:55 +08:00
|
|
|
}
|
|
|
|
|
2017-11-25 17:43:05 -05:00
|
|
|
let webSocketServer;
|
2021-03-21 22:43:41 +01:00
|
|
|
let lastSyncedPush = null;
|
2017-11-25 17:43:05 -05:00
|
|
|
|
2017-11-30 23:50:42 -05:00
|
|
|
function init(httpServer, sessionParser) {
|
|
|
|
webSocketServer = new WebSocket.Server({
|
|
|
|
verifyClient: (info, done) => {
|
|
|
|
sessionParser(info.req, {}, () => {
|
2020-08-29 00:11:50 +02:00
|
|
|
const allowed = utils.isElectron()
|
|
|
|
|| info.req.session.loggedIn
|
|
|
|
|| (config.General && config.General.noAuthentication);
|
2017-11-30 23:50:42 -05:00
|
|
|
|
|
|
|
if (!allowed) {
|
|
|
|
log.error("WebSocket connection not allowed because session is neither electron nor logged in.");
|
|
|
|
}
|
|
|
|
|
|
|
|
done(allowed)
|
|
|
|
});
|
|
|
|
},
|
|
|
|
server: httpServer
|
|
|
|
});
|
|
|
|
|
2017-12-01 22:28:22 -05:00
|
|
|
webSocketServer.on('connection', (ws, req) => {
|
2019-12-02 22:27:06 +01:00
|
|
|
ws.id = utils.randomString(10);
|
|
|
|
|
|
|
|
console.log(`websocket client connected`);
|
2017-12-01 22:28:22 -05:00
|
|
|
|
2020-06-20 12:31:38 +02:00
|
|
|
ws.on('message', async messageJson => {
|
2017-12-01 22:28:22 -05:00
|
|
|
const message = JSON.parse(messageJson);
|
|
|
|
|
|
|
|
if (message.type === 'log-error') {
|
2022-12-21 15:19:05 +01:00
|
|
|
log.info(`JS Error: ${message.error}\r
|
|
|
|
Stack: ${message.stack}`);
|
2017-12-01 22:28:22 -05:00
|
|
|
}
|
2021-09-17 22:34:23 +02:00
|
|
|
else if (message.type === 'log-info') {
|
2022-12-21 15:19:05 +01:00
|
|
|
log.info(`JS Info: ${message.info}`);
|
2021-09-17 22:34:23 +02:00
|
|
|
}
|
2017-12-19 23:22:21 -05:00
|
|
|
else if (message.type === 'ping') {
|
2020-06-20 12:31:38 +02:00
|
|
|
await syncMutexService.doExclusively(() => sendPing(ws));
|
2017-12-19 23:22:21 -05:00
|
|
|
}
|
2017-12-01 22:28:22 -05:00
|
|
|
else {
|
|
|
|
log.error('Unrecognized message: ');
|
|
|
|
log.error(message);
|
|
|
|
}
|
|
|
|
});
|
2017-11-25 17:43:05 -05:00
|
|
|
});
|
2022-12-07 23:58:22 +01:00
|
|
|
|
|
|
|
webSocketServer.on('error', error => {
|
|
|
|
// https://github.com/zadam/trilium/issues/3374#issuecomment-1341053765
|
|
|
|
console.log(error);
|
|
|
|
});
|
2017-11-25 17:43:05 -05:00
|
|
|
}
|
|
|
|
|
2019-10-28 18:42:22 +01:00
|
|
|
function sendMessage(client, message) {
|
2017-12-19 23:22:21 -05:00
|
|
|
const jsonStr = JSON.stringify(message);
|
|
|
|
|
|
|
|
if (client.readyState === WebSocket.OPEN) {
|
|
|
|
client.send(jsonStr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-28 18:42:22 +01:00
|
|
|
function sendMessageToAllClients(message) {
|
2017-11-25 17:43:05 -05:00
|
|
|
const jsonStr = JSON.stringify(message);
|
|
|
|
|
2019-01-02 22:36:06 +01:00
|
|
|
if (webSocketServer) {
|
2022-09-17 23:06:17 +02:00
|
|
|
if (message.type !== 'sync-failed' && message.type !== 'api-log-messages') {
|
2022-12-21 15:19:05 +01:00
|
|
|
log.info(`Sending message to all clients: ${jsonStr}`);
|
2021-03-21 22:43:41 +01:00
|
|
|
}
|
2018-01-01 19:41:22 -05:00
|
|
|
|
2019-01-02 22:36:06 +01:00
|
|
|
webSocketServer.clients.forEach(function each(client) {
|
|
|
|
if (client.readyState === WebSocket.OPEN) {
|
|
|
|
client.send(jsonStr);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2017-11-25 17:43:05 -05:00
|
|
|
}
|
|
|
|
|
2020-12-16 22:17:42 +01:00
|
|
|
function fillInAdditionalProperties(entityChange) {
|
2021-07-03 20:52:40 +02:00
|
|
|
if (entityChange.isErased) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-01-18 08:48:36 +01:00
|
|
|
// fill in some extra data needed by the frontend
|
2023-05-29 23:42:08 +02:00
|
|
|
// first try to use becca, which works for non-deleted entities
|
|
|
|
// only when that fails, try to load from the database
|
2020-12-16 22:17:42 +01:00
|
|
|
if (entityChange.entityName === 'attributes') {
|
2021-07-25 21:25:06 +02:00
|
|
|
entityChange.entity = becca.getAttribute(entityChange.entityId);
|
|
|
|
|
|
|
|
if (!entityChange.entity) {
|
|
|
|
entityChange.entity = sql.getRow(`SELECT * FROM attributes WHERE attributeId = ?`, [entityChange.entityId]);
|
|
|
|
}
|
2020-12-16 22:17:42 +01:00
|
|
|
} else if (entityChange.entityName === 'branches') {
|
2021-07-25 21:25:06 +02:00
|
|
|
entityChange.entity = becca.getBranch(entityChange.entityId);
|
|
|
|
|
|
|
|
if (!entityChange.entity) {
|
|
|
|
entityChange.entity = sql.getRow(`SELECT * FROM branches WHERE branchId = ?`, [entityChange.entityId]);
|
|
|
|
}
|
2020-12-16 22:17:42 +01:00
|
|
|
} else if (entityChange.entityName === 'notes') {
|
2021-07-25 21:25:06 +02:00
|
|
|
entityChange.entity = becca.getNote(entityChange.entityId);
|
|
|
|
|
|
|
|
if (!entityChange.entity) {
|
|
|
|
entityChange.entity = sql.getRow(`SELECT * FROM notes WHERE noteId = ?`, [entityChange.entityId]);
|
2020-12-16 22:17:42 +01:00
|
|
|
|
2021-07-25 21:25:06 +02:00
|
|
|
if (entityChange.entity.isProtected) {
|
|
|
|
entityChange.entity.title = protectedSessionService.decryptString(entityChange.entity.title);
|
|
|
|
}
|
2020-02-01 22:29:32 +01:00
|
|
|
}
|
2023-06-04 23:01:40 +02:00
|
|
|
} else if (entityChange.entityName === 'revisions') {
|
2020-12-16 22:17:42 +01:00
|
|
|
entityChange.noteId = sql.getValue(`SELECT noteId
|
2023-06-04 23:01:40 +02:00
|
|
|
FROM revisions
|
|
|
|
WHERE revisionId = ?`, [entityChange.entityId]);
|
2020-12-16 22:17:42 +01:00
|
|
|
} else if (entityChange.entityName === 'note_reordering') {
|
2021-07-25 21:25:06 +02:00
|
|
|
entityChange.positions = {};
|
|
|
|
|
|
|
|
const parentNote = becca.getNote(entityChange.entityId);
|
|
|
|
|
|
|
|
if (parentNote) {
|
|
|
|
for (const childBranch of parentNote.getChildBranches()) {
|
|
|
|
entityChange.positions[childBranch.branchId] = childBranch.notePosition;
|
|
|
|
}
|
|
|
|
}
|
2023-03-15 22:44:08 +01:00
|
|
|
} else if (entityChange.entityName === 'options') {
|
2021-07-25 21:25:06 +02:00
|
|
|
entityChange.entity = becca.getOption(entityChange.entityId);
|
|
|
|
|
|
|
|
if (!entityChange.entity) {
|
|
|
|
entityChange.entity = sql.getRow(`SELECT * FROM options WHERE name = ?`, [entityChange.entityId]);
|
|
|
|
}
|
2023-03-16 12:11:00 +01:00
|
|
|
} else if (entityChange.entityName === 'blobs') {
|
2023-03-15 22:44:08 +01:00
|
|
|
entityChange.noteIds = sql.getColumn("SELECT noteId FROM notes WHERE blobId = ? AND isDeleted = 0", [entityChange.entityId]);
|
2023-04-01 23:55:04 +02:00
|
|
|
} else if (entityChange.entityName === 'attachments') {
|
2023-05-29 23:42:08 +02:00
|
|
|
entityChange.entity = becca.getAttachment(entityChange.entityId, {includeContentLength: true});
|
2021-07-25 21:25:06 +02:00
|
|
|
}
|
|
|
|
|
2023-01-03 13:52:37 +01:00
|
|
|
if (entityChange.entity instanceof AbstractBeccaEntity) {
|
2021-07-25 21:25:06 +02:00
|
|
|
entityChange.entity = entityChange.entity.getPojo();
|
2020-02-05 22:08:45 +01:00
|
|
|
}
|
2020-01-18 08:48:36 +01:00
|
|
|
}
|
|
|
|
|
2021-10-02 23:26:18 +02:00
|
|
|
// entities with higher number can reference the entities with lower number
|
|
|
|
const ORDERING = {
|
2022-01-10 17:09:20 +01:00
|
|
|
"etapi_tokens": 0,
|
2023-03-16 11:02:07 +01:00
|
|
|
"attributes": 2,
|
|
|
|
"branches": 2,
|
|
|
|
"blobs": 0,
|
|
|
|
"note_reordering": 2,
|
2023-06-04 23:01:40 +02:00
|
|
|
"revisions": 2,
|
2023-03-16 12:17:55 +01:00
|
|
|
"attachments": 3,
|
2023-03-16 11:02:07 +01:00
|
|
|
"notes": 1,
|
2021-10-02 23:26:18 +02:00
|
|
|
"options": 0
|
|
|
|
};
|
|
|
|
|
2021-09-16 15:02:20 +02:00
|
|
|
function sendPing(client, entityChangeIds = []) {
|
|
|
|
if (entityChangeIds.length === 0) {
|
2021-09-26 15:37:18 +02:00
|
|
|
sendMessage(client, { type: 'ping' });
|
|
|
|
|
2021-09-16 15:02:20 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const entityChanges = sql.getManyRows(`SELECT * FROM entity_changes WHERE id IN (???)`, entityChangeIds);
|
|
|
|
|
2021-10-02 23:26:18 +02:00
|
|
|
// sort entity changes since froca expects "referential order", i.e. referenced entities should already exist
|
|
|
|
// in froca.
|
|
|
|
// Froca needs this since it is incomplete copy, it can't create "skeletons" like becca.
|
|
|
|
entityChanges.sort((a, b) => ORDERING[a.entityName] - ORDERING[b.entityName]);
|
|
|
|
|
2021-03-21 22:43:41 +01:00
|
|
|
for (const entityChange of entityChanges) {
|
2020-01-18 08:48:36 +01:00
|
|
|
try {
|
2021-03-21 22:43:41 +01:00
|
|
|
fillInAdditionalProperties(entityChange);
|
2019-06-01 12:14:09 +02:00
|
|
|
}
|
2020-01-18 08:48:36 +01:00
|
|
|
catch (e) {
|
2022-12-21 15:19:05 +01:00
|
|
|
log.error(`Could not fill additional properties for entity change ${JSON.stringify(entityChange)} because of error: ${e.message}: ${e.stack}`);
|
2019-10-20 12:29:34 +02:00
|
|
|
}
|
2019-06-01 12:14:09 +02:00
|
|
|
}
|
|
|
|
|
2019-10-28 18:42:22 +01:00
|
|
|
sendMessage(client, {
|
2021-03-21 22:43:41 +01:00
|
|
|
type: 'frontend-update',
|
|
|
|
data: {
|
|
|
|
lastSyncedPush,
|
|
|
|
entityChanges
|
|
|
|
}
|
2017-12-19 23:22:21 -05:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-03-21 22:43:41 +01:00
|
|
|
function sendTransactionEntityChangesToAllClients() {
|
2019-12-02 22:27:06 +01:00
|
|
|
if (webSocketServer) {
|
2021-09-16 15:02:20 +02:00
|
|
|
const entityChangeIds = cls.getAndClearEntityChangeIds();
|
2020-06-21 13:44:47 +02:00
|
|
|
|
2021-09-16 15:02:20 +02:00
|
|
|
webSocketServer.clients.forEach(client => sendPing(client, entityChangeIds));
|
2019-12-02 22:27:06 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-28 18:42:22 +01:00
|
|
|
function syncPullInProgress() {
|
2021-05-15 22:00:53 +02:00
|
|
|
sendMessageToAllClients({ type: 'sync-pull-in-progress', lastSyncedPush });
|
2019-10-25 22:20:14 +02:00
|
|
|
}
|
|
|
|
|
2021-03-21 00:01:28 +01:00
|
|
|
function syncPushInProgress() {
|
2021-05-15 22:00:53 +02:00
|
|
|
sendMessageToAllClients({ type: 'sync-push-in-progress', lastSyncedPush });
|
2021-03-21 00:01:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
function syncFinished() {
|
2021-05-15 22:00:53 +02:00
|
|
|
sendMessageToAllClients({ type: 'sync-finished', lastSyncedPush });
|
2021-03-21 00:01:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
function syncFailed() {
|
2021-05-15 22:00:53 +02:00
|
|
|
sendMessageToAllClients({ type: 'sync-failed', lastSyncedPush });
|
2019-10-25 22:20:14 +02:00
|
|
|
}
|
|
|
|
|
2021-09-12 11:18:06 +02:00
|
|
|
function reloadFrontend() {
|
|
|
|
sendMessageToAllClients({ type: 'reload-frontend' });
|
|
|
|
}
|
|
|
|
|
2021-03-21 22:43:41 +01:00
|
|
|
function setLastSyncedPush(entityChangeId) {
|
|
|
|
lastSyncedPush = entityChangeId;
|
|
|
|
}
|
|
|
|
|
2017-11-25 17:43:05 -05:00
|
|
|
module.exports = {
|
|
|
|
init,
|
2019-02-10 16:59:50 +01:00
|
|
|
sendMessageToAllClients,
|
2021-03-21 00:01:28 +01:00
|
|
|
syncPushInProgress,
|
2019-10-25 22:20:14 +02:00
|
|
|
syncPullInProgress,
|
2021-03-21 00:01:28 +01:00
|
|
|
syncFinished,
|
|
|
|
syncFailed,
|
2021-03-21 22:43:41 +01:00
|
|
|
sendTransactionEntityChangesToAllClients,
|
2021-09-12 11:18:06 +02:00
|
|
|
setLastSyncedPush,
|
|
|
|
reloadFrontend
|
2020-06-10 00:10:27 +02:00
|
|
|
};
|