Notes/src/becca/becca-interface.ts

313 lines
9.9 KiB
TypeScript
Raw Normal View History

2024-02-16 23:51:56 +02:00
import sql = require('../services/sql');
import NoteSet = require('../services/search/note_set');
import NotFoundError = require('../errors/not_found_error');
2024-02-17 01:00:38 +02:00
import BOption = require('./entities/boption');
import BNote = require('./entities/bnote');
2024-02-17 01:03:38 +02:00
import BEtapiToken = require('./entities/betapi_token');
2024-02-17 01:19:49 +02:00
import BAttribute = require('./entities/battribute');
import BBranch = require('./entities/bbranch');
import BRevision = require('./entities/brevision');
import BAttachment = require('./entities/battachment');
import { AttachmentRow, RevisionRow } from './entities/rows';
import BBlob = require('./entities/bblob');
import BRecentNote = require('./entities/brecent_note');
2024-03-30 10:49:40 +02:00
import AbstractBeccaEntity = require('./entities/abstract_becca_entity');
interface AttachmentOpts {
includeContentLength?: boolean;
}
2021-10-16 22:13:34 +02:00
/**
2023-06-29 23:32:19 +02:00
* Becca is a backend cache of all notes, branches, and attributes.
* There's a similar frontend cache Froca, and share cache Shaca.
2021-10-16 22:13:34 +02:00
*/
export default class Becca {
2024-02-17 01:24:37 +02:00
loaded!: boolean;
notes!: Record<string, BNote>;
branches!: Record<string, BBranch>;
childParentToBranch!: Record<string, BBranch>;
2024-02-17 01:19:49 +02:00
attributes!: Record<string, BAttribute>;
/** Points from attribute type-name to list of attributes */
attributeIndex!: Record<string, BAttribute[]>;
2024-02-17 01:00:38 +02:00
options!: Record<string, BOption>;
2024-02-17 01:03:38 +02:00
etapiTokens!: Record<string, BEtapiToken>;
allNoteSetCache: NoteSet | null;
2020-05-16 23:12:29 +02:00
constructor() {
2020-05-22 09:38:30 +02:00
this.reset();
this.allNoteSetCache = null;
2020-05-22 09:38:30 +02:00
}
reset() {
2021-04-30 23:10:25 +02:00
this.notes = {};
this.branches = {};
2020-05-16 23:12:29 +02:00
this.childParentToBranch = {};
2024-02-17 01:19:49 +02:00
this.attributes = {};
2020-05-22 09:38:30 +02:00
this.attributeIndex = {};
2021-04-30 23:10:25 +02:00
this.options = {};
2022-01-10 17:09:20 +01:00
this.etapiTokens = {};
2020-05-16 23:12:29 +02:00
this.dirtyNoteSetCache();
2020-05-16 23:12:29 +02:00
this.loaded = false;
}
2022-12-23 15:07:48 +01:00
getRoot() {
return this.getNote('root');
}
findAttributes(type: string, name: string): BAttribute[] {
name = name.trim().toLowerCase();
if (name.startsWith('#') || name.startsWith('~')) {
name = name.substr(1);
}
return this.attributeIndex[`${type}-${name}`] || [];
2020-05-16 23:12:29 +02:00
}
findAttributesWithPrefix(type: string, name: string): BAttribute[] {
2024-02-17 18:55:41 +02:00
const resArr: BAttribute[][] = [];
const key = `${type}-${name}`;
for (const idx in this.attributeIndex) {
if (idx.startsWith(key)) {
resArr.push(this.attributeIndex[idx]);
}
}
return resArr.flat();
}
2020-05-16 23:12:29 +02:00
decryptProtectedNotes() {
for (const note of Object.values(this.notes)) {
2020-05-17 09:48:24 +02:00
note.decrypt();
2020-05-16 23:12:29 +02:00
}
}
2020-05-17 09:48:24 +02:00
addNote(noteId: string, note: BNote) {
this.notes[noteId] = note;
this.dirtyNoteSetCache();
}
getNote(noteId: string): BNote | null {
return this.notes[noteId];
}
2024-03-30 10:49:40 +02:00
getNoteOrThrow(noteId: string): BNote {
const note = this.notes[noteId];
if (!note) {
throw new NotFoundError(`Note '${noteId}' doesn't exist.`);
}
return note;
}
getNotes(noteIds: string[], ignoreMissing: boolean = false): BNote[] {
2024-02-17 18:55:41 +02:00
const filteredNotes: BNote[] = [];
2021-05-08 11:20:20 +02:00
for (const noteId of noteIds) {
const note = this.notes[noteId];
if (!note) {
if (ignoreMissing) {
continue;
}
2021-05-08 11:20:20 +02:00
throw new Error(`Note '${noteId}' was not found in becca.`);
}
filteredNotes.push(note);
}
return filteredNotes;
2021-04-26 22:24:55 +02:00
}
getBranch(branchId: string): BBranch | null {
return this.branches[branchId];
}
2024-04-05 20:33:04 +03:00
getBranchOrThrow(branchId: string): BBranch {
const branch = this.getBranch(branchId);
if (!branch) {
throw new NotFoundError(`Branch '${branchId}' was not found in becca.`);
}
return branch;
}
getAttribute(attributeId: string): BAttribute | null {
return this.attributes[attributeId];
}
getAttributeOrThrow(attributeId: string): BAttribute {
const attribute = this.getAttribute(attributeId);
if (!attribute) {
throw new NotFoundError(`Attribute '${attributeId}' does not exist.`);
}
return attribute;
}
getBranchFromChildAndParent(childNoteId: string, parentNoteId: string): BBranch | null {
2020-05-17 09:48:24 +02:00
return this.childParentToBranch[`${childNoteId}-${parentNoteId}`];
}
getRevision(revisionId: string): BRevision | null {
const row = sql.getRow("SELECT * FROM revisions WHERE revisionId = ?", [revisionId]);
const BRevision = require('./entities/brevision'); // avoiding circular dependency problems
return row ? new BRevision(row) : null;
}
2021-05-01 21:52:22 +02:00
getRevisionOrThrow(revisionId: string): BRevision {
const revision = this.getRevision(revisionId);
if (!revision) {
throw new NotFoundError(`Revision '${revisionId}' has not been found.`);
}
return revision;
}
getAttachment(attachmentId: string, opts: AttachmentOpts = {}): BAttachment | null {
2023-05-21 18:14:17 +02:00
opts.includeContentLength = !!opts.includeContentLength;
const query = opts.includeContentLength
? `SELECT attachments.*, LENGTH(blobs.content) AS contentLength
FROM attachments
JOIN blobs USING (blobId)
WHERE attachmentId = ? AND isDeleted = 0`
: `SELECT * FROM attachments WHERE attachmentId = ? AND isDeleted = 0`;
const BAttachment = require('./entities/battachment'); // avoiding circular dependency problems
2023-05-21 18:14:17 +02:00
return sql.getRows(query, [attachmentId])
.map(row => new BAttachment(row))[0];
}
getAttachmentOrThrow(attachmentId: string, opts: AttachmentOpts = {}): BAttachment {
2023-05-21 18:14:17 +02:00
const attachment = this.getAttachment(attachmentId, opts);
if (!attachment) {
throw new NotFoundError(`Attachment '${attachmentId}' has not been found.`);
}
return attachment;
}
getAttachments(attachmentIds: string[]): BAttachment[] {
const BAttachment = require('./entities/battachment'); // avoiding circular dependency problems
return sql.getManyRows<AttachmentRow>("SELECT * FROM attachments WHERE attachmentId IN (???) AND isDeleted = 0", attachmentIds)
2023-04-25 00:01:58 +02:00
.map(row => new BAttachment(row));
}
2024-03-30 10:49:40 +02:00
getBlob(entity: { blobId?: string }): BBlob | null {
if (!entity.blobId) {
return null;
}
const row = sql.getRow("SELECT *, LENGTH(content) AS contentLength FROM blobs WHERE blobId = ?", [entity.blobId]);
2023-05-05 16:37:39 +02:00
const BBlob = require('./entities/bblob'); // avoiding circular dependency problems
2023-05-05 16:37:39 +02:00
return row ? new BBlob(row) : null;
}
getOption(name: string): BOption | null {
2021-05-01 21:52:22 +02:00
return this.options[name];
}
2021-05-02 12:02:32 +02:00
getEtapiTokens(): BEtapiToken[] {
2022-01-10 17:09:20 +01:00
return Object.values(this.etapiTokens);
}
getEtapiToken(etapiTokenId: string): BEtapiToken | null {
2022-01-10 17:09:20 +01:00
return this.etapiTokens[etapiTokenId];
}
2024-03-30 10:49:40 +02:00
getEntity<T extends AbstractBeccaEntity<T>>(entityName: string, entityId: string): AbstractBeccaEntity<T> | null {
2021-05-02 12:02:32 +02:00
if (!entityName || !entityId) {
return null;
}
if (entityName === 'revisions') {
return this.getRevision(entityId);
2023-03-16 12:17:55 +01:00
} else if (entityName === 'attachments') {
return this.getAttachment(entityId);
2021-08-07 21:21:30 +02:00
}
2021-05-02 12:02:32 +02:00
const camelCaseEntityName = entityName.toLowerCase().replace(/(_[a-z])/g,
group =>
group
.toUpperCase()
.replace('_', '')
);
if (!(camelCaseEntityName in this)) {
throw new Error(`Unknown entity name '${camelCaseEntityName}' (original argument '${entityName}')`);
}
return (this as any)[camelCaseEntityName][entityId];
2021-05-02 12:02:32 +02:00
}
2021-05-02 19:59:16 +02:00
getRecentNotesFromQuery(query: string, params: string[] = []): BRecentNote[] {
2021-05-02 19:59:16 +02:00
const rows = sql.getRows(query, params);
const BRecentNote = require('./entities/brecent_note'); // avoiding circular dependency problems
return rows.map(row => new BRecentNote(row));
2021-05-02 19:59:16 +02:00
}
getRevisionsFromQuery(query: string, params: string[] = []): BRevision[] {
const rows = sql.getRows<RevisionRow>(query, params);
2021-05-02 19:59:16 +02:00
const BRevision = require('./entities/brevision'); // avoiding circular dependency problems
return rows.map(row => new BRevision(row));
2021-05-02 19:59:16 +02:00
}
/** Should be called when the set of all non-skeleton notes changes (added/removed) */
dirtyNoteSetCache() {
this.allNoteSetCache = null;
}
getAllNoteSet() {
// caching this since it takes 10s of milliseconds to fill this initial NoteSet for many notes
if (!this.allNoteSetCache) {
const allNotes = [];
for (const noteId in this.notes) {
const note = this.notes[noteId];
// in the process of loading data sometimes we create "skeleton" note instances which are expected to be filled later
// in case of inconsistent data this might not work and search will then crash on these
if (note.type !== undefined) {
allNotes.push(note);
}
}
this.allNoteSetCache = new NoteSet(allNotes);
}
return this.allNoteSetCache;
}
2020-05-16 23:12:29 +02:00
}
/**
* This interface contains the data that is shared across all the objects of a given derived class of {@link AbstractBeccaEntity}.
* For example, all BAttributes will share their content, but all BBranches will have another set of this data.
*/
export interface ConstructorData<T extends AbstractBeccaEntity<T>> {
primaryKeyName: string;
entityName: string;
hashedProperties: (keyof T)[];
}
export interface NotePojo {
noteId: string;
title?: string;
isProtected?: boolean;
type: string;
mime: string;
blobId?: string;
isDeleted: boolean;
dateCreated?: string;
dateModified?: string;
utcDateCreated: string;
utcDateModified?: string;
}