mirror of
https://github.com/streetwriters/notesnook.git
synced 2025-12-23 23:19:40 +01:00
223 lines
6.2 KiB
JavaScript
223 lines
6.2 KiB
JavaScript
/*
|
|
This file is part of the Notesnook project (https://notesnook.com/)
|
|
|
|
Copyright (C) 2023 Streetwriters (Private) Limited
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
import Collection from "./collection";
|
|
import getId from "../utils/id";
|
|
import { getContentFromData } from "../content-types";
|
|
import { hasItem } from "../utils/array";
|
|
|
|
export default class Content extends Collection {
|
|
async add(content) {
|
|
if (!content) return;
|
|
|
|
if (typeof content.data === "object") {
|
|
if (typeof content.data.data === "string")
|
|
content.data = content.data.data;
|
|
else if (!content.data.iv && !content.data.cipher)
|
|
content.data = `<p>Content is invalid: ${JSON.stringify(
|
|
content.data
|
|
)}</p>`;
|
|
}
|
|
|
|
if (content.remote || content.deleted)
|
|
return await this._collection.addItem(
|
|
await this.extractAttachments(content)
|
|
);
|
|
|
|
const oldContent = await this.raw(content.id);
|
|
if (content.id && oldContent) {
|
|
content = {
|
|
...oldContent,
|
|
...content,
|
|
dateEdited: content.dateEdited || Date.now()
|
|
};
|
|
}
|
|
|
|
const id = content.id || getId();
|
|
|
|
const contentItem = await this.extractAttachments({
|
|
noteId: content.noteId,
|
|
id,
|
|
type: content.type,
|
|
data: content.data || content,
|
|
dateEdited: content.dateEdited,
|
|
dateCreated: content.dateCreated,
|
|
dateModified: content.dateModified,
|
|
localOnly: !!content.localOnly,
|
|
conflicted: content.conflicted,
|
|
dateResolved: content.dateResolved
|
|
});
|
|
await this._collection.addItem(contentItem);
|
|
|
|
if (content.sessionId) {
|
|
await this._db.noteHistory.add(contentItem.noteId, content.sessionId, {
|
|
data: contentItem.data,
|
|
type: contentItem.type
|
|
});
|
|
}
|
|
return id;
|
|
}
|
|
|
|
async get(id) {
|
|
const content = await this.raw(id);
|
|
if (!content) return;
|
|
return content.data;
|
|
}
|
|
|
|
async raw(id) {
|
|
const content = await this._collection.getItem(id);
|
|
if (!content) return;
|
|
return content;
|
|
}
|
|
|
|
remove(id) {
|
|
if (!id) return;
|
|
return this._collection.removeItem(id);
|
|
}
|
|
|
|
multi(ids) {
|
|
return this._collection.getItems(ids);
|
|
}
|
|
|
|
exists(id) {
|
|
return this._collection.exists(id);
|
|
}
|
|
|
|
async all() {
|
|
return Object.values(
|
|
await this._collection.getItems(this._collection.indexer.indices)
|
|
);
|
|
}
|
|
|
|
insertMedia(contentItem) {
|
|
return this._insert(contentItem, this._db.attachments.read);
|
|
}
|
|
|
|
insertPlaceholders(contentItem, placeholder) {
|
|
return this._insert(contentItem, () => placeholder);
|
|
}
|
|
|
|
async downloadMedia(groupId, contentItem, notify = true) {
|
|
const content = getContentFromData(contentItem.type, contentItem.data);
|
|
contentItem.data = await content.insertMedia((hash, { total, current }) => {
|
|
const attachment = this._db.attachments.attachment(hash);
|
|
if (!attachment) return;
|
|
|
|
const progressData = {
|
|
total,
|
|
current,
|
|
groupId
|
|
};
|
|
|
|
return this._db.attachments._download(attachment, progressData, notify);
|
|
});
|
|
return contentItem;
|
|
}
|
|
|
|
async _insert(contentItem, getData) {
|
|
if (!contentItem || !getData) return;
|
|
const content = getContentFromData(contentItem.type, contentItem.data);
|
|
contentItem.data = await content.insertMedia(getData);
|
|
return contentItem;
|
|
}
|
|
|
|
/**
|
|
*
|
|
* @param {string} id
|
|
* @param {string[]} hashes
|
|
* @returns {Promise<any>}
|
|
*/
|
|
async removeAttachments(id, hashes) {
|
|
const contentItem = await this.raw(id);
|
|
const content = getContentFromData(contentItem.type, contentItem.data);
|
|
contentItem.data = content.removeAttachments(hashes);
|
|
await this.add(contentItem);
|
|
}
|
|
|
|
/**
|
|
*
|
|
* @param {any} contentItem
|
|
* @returns {Promise<any>}
|
|
*/
|
|
async extractAttachments(contentItem) {
|
|
if (contentItem.localOnly || typeof contentItem.data !== "string")
|
|
return contentItem;
|
|
|
|
const allAttachments = this._db.attachments.all;
|
|
const content = getContentFromData(contentItem.type, contentItem.data);
|
|
if (!content) return contentItem;
|
|
const { data, attachments } = await content.extractAttachments(
|
|
(data, type, mimeType) => this._db.attachments.save(data, type, mimeType)
|
|
);
|
|
|
|
const noteAttachments = allAttachments.filter((attachment) =>
|
|
hasItem(attachment.noteIds, contentItem.noteId)
|
|
);
|
|
|
|
const toDelete = noteAttachments.filter((attachment) => {
|
|
return attachments.every(
|
|
(a) => a.hash && a.hash !== attachment.metadata.hash
|
|
);
|
|
});
|
|
|
|
const toAdd = attachments.filter((attachment) => {
|
|
return (
|
|
attachment.hash &&
|
|
noteAttachments.every((a) => attachment.hash !== a.metadata.hash)
|
|
);
|
|
});
|
|
|
|
for (let attachment of toDelete) {
|
|
await this._db.attachments.delete(
|
|
attachment.metadata.hash,
|
|
contentItem.noteId
|
|
);
|
|
}
|
|
|
|
for (let attachment of toAdd) {
|
|
await this._db.attachments.add(attachment, contentItem.noteId);
|
|
}
|
|
|
|
if (toAdd.length > 0) {
|
|
contentItem.dateModified = Date.now();
|
|
}
|
|
contentItem.data = data;
|
|
return contentItem;
|
|
}
|
|
|
|
async cleanup() {
|
|
const indices = this._collection.indexer.indices;
|
|
await this._db.notes.init();
|
|
const notes = this._db.notes._collection.getRaw();
|
|
if (!notes.length && indices.length > 0) return [];
|
|
let ids = [];
|
|
for (let contentId of indices) {
|
|
const noteIndex = notes.findIndex((note) => note.contentId === contentId);
|
|
const isOrphaned = noteIndex === -1;
|
|
if (isOrphaned) {
|
|
ids.push(contentId);
|
|
await this._collection.deleteItem(contentId);
|
|
} else if (notes[noteIndex].localOnly) {
|
|
ids.push(contentId);
|
|
}
|
|
}
|
|
return ids;
|
|
}
|
|
}
|