trilium/src/services/repository.js

152 lines
4.0 KiB
JavaScript

"use strict";
const sql = require('./sql');
const entityChangesService = require('./entity_changes.js');
const eventService = require('./events');
const cls = require('./cls');
const entityConstructor = require('../entities/entity_constructor');
function getEntityFromName(entityName, entityId) {
if (!entityName || !entityId) {
return null;
}
const constructor = entityConstructor.getEntityFromEntityName(entityName);
return getEntity(`SELECT * FROM ${constructor.entityName} WHERE ${constructor.primaryKeyName} = ?`, [entityId]);
}
function getEntities(query, params = []) {
const rows = sql.getRows(query, params);
return rows.map(entityConstructor.createEntityFromRow);
}
function getEntity(query, params = []) {
const row = sql.getRowOrNull(query, params);
if (!row) {
return null;
}
return entityConstructor.createEntityFromRow(row);
}
function getCachedEntity(entityName, entityId, query) {
let entity = cls.getEntityFromCache(entityName, entityId);
if (!entity) {
entity = getEntity(query, [entityId]);
cls.setEntityToCache(entityName, entityId, entity);
}
return entity;
}
/** @returns {Note|null} */
function getNote(noteId) {
return getCachedEntity('notes', noteId, "SELECT * FROM notes WHERE noteId = ?");
}
/** @returns {Note[]} */
function getNotes(noteIds) {
// this note might be optimised, but remember that it must keep the existing order of noteIds
// (important e.g. for @orderBy in search)
const notes = [];
for (const noteId of noteIds) {
const note = getNote(noteId);
notes.push(note);
}
return notes;
}
/** @returns {NoteRevision|null} */
function getNoteRevision(noteRevisionId) {
return getCachedEntity('note_revisions', noteRevisionId, "SELECT * FROM note_revisions WHERE noteRevisionId = ?");
}
/** @returns {Branch|null} */
function getBranch(branchId) {
return getCachedEntity('branches', branchId, "SELECT * FROM branches WHERE branchId = ?", [branchId]);
}
/** @returns {Attribute|null} */
function getAttribute(attributeId) {
return getCachedEntity('attributes', attributeId, "SELECT * FROM attributes WHERE attributeId = ?");
}
/** @returns {Option|null} */
function getOption(name) {
return getEntity("SELECT * FROM options WHERE name = ?", [name]);
}
function updateEntity(entity) {
const entityName = entity.constructor.entityName;
const primaryKeyName = entity.constructor.primaryKeyName;
const isNewEntity = !entity[primaryKeyName];
if (entity.beforeSaving) {
entity.beforeSaving();
}
const clone = Object.assign({}, entity);
// this check requires that updatePojo is not static
if (entity.updatePojo) {
entity.updatePojo(clone);
}
// indicates whether entity actually changed
delete clone.isChanged;
for (const key in clone) {
// !isBuffer is for images and attachments
if (clone[key] !== null && typeof clone[key] === 'object' && !Buffer.isBuffer(clone[key])) {
clone[key] = JSON.stringify(clone[key]);
}
}
sql.transactional(() => {
sql.upsert(entityName, primaryKeyName, clone);
const primaryKey = entity[primaryKeyName];
if (entity.isChanged) {
const isSynced = entityName !== 'options' || entity.isSynced;
entityChangesService.addEntityChange(entityName, primaryKey, null, isSynced);
if (!cls.isEntityEventsDisabled()) {
const eventPayload = {
entityName,
entity
};
if (isNewEntity && !entity.isDeleted) {
eventService.emit(eventService.ENTITY_CREATED, eventPayload);
}
eventService.emit(entity.isDeleted ? eventService.ENTITY_DELETED : eventService.ENTITY_CHANGED, eventPayload);
}
}
});
}
module.exports = {
getEntityFromName,
getEntities,
getEntity,
getNote,
getNotes,
getNoteRevision,
getBranch,
getAttribute,
getOption,
updateEntity
};