mirror of
https://github.com/laurent22/joplin.git
synced 2025-01-23 18:53:36 +02:00
157 lines
5.2 KiB
TypeScript
157 lines
5.2 KiB
TypeScript
import { ModelType } from '../../BaseModel';
|
|
import { FolderEntity, NoteEntity, ResourceEntity, TagEntity, UserData, UserDataValue } from '../../services/database/types';
|
|
import Note from '../Note';
|
|
import Folder from '../Folder';
|
|
import Resource from '../Resource';
|
|
import Tag from '../Tag';
|
|
import BaseItem from '../BaseItem';
|
|
import { LoadOptions } from './types';
|
|
|
|
const maxKeyLength = 255;
|
|
|
|
type SupportedEntity = NoteEntity | ResourceEntity | FolderEntity | TagEntity;
|
|
|
|
const unserializeUserData = (s: string): UserData => {
|
|
if (!s) return {};
|
|
|
|
try {
|
|
const r = JSON.parse(s);
|
|
return r as UserData;
|
|
} catch (error) {
|
|
error.message = `Could not unserialize user data: ${error.message}: ${s}`;
|
|
throw error;
|
|
}
|
|
};
|
|
|
|
const serializeUserData = (d: UserData): string => {
|
|
if (!d) return '';
|
|
return JSON.stringify(d);
|
|
};
|
|
|
|
export const setUserData = <T>(userData: UserData, namespace: string, key: string, value: T, deleted = false): UserData => {
|
|
if (key.length > maxKeyLength) new Error(`Key must no be longer than ${maxKeyLength} characters`);
|
|
if (!(namespace in userData)) userData[namespace] = {};
|
|
if (key in userData[namespace] && userData[namespace][key].v === value) return userData;
|
|
|
|
const newUserDataValue: UserDataValue = {
|
|
v: value,
|
|
t: Date.now(),
|
|
};
|
|
|
|
if (deleted) newUserDataValue.d = 1;
|
|
|
|
return {
|
|
...userData,
|
|
[namespace]: {
|
|
...userData[namespace],
|
|
[key]: newUserDataValue,
|
|
},
|
|
};
|
|
};
|
|
|
|
export const getUserData = <T>(userData: UserData, namespace: string, key: string): T|undefined => {
|
|
if (!hasUserData(userData, namespace, key)) return undefined;
|
|
return userData[namespace][key].v as T;
|
|
};
|
|
|
|
const checkIsSupportedItemType = (itemType: ModelType) => {
|
|
if (![ModelType.Note, ModelType.Folder, ModelType.Tag, ModelType.Resource].includes(itemType)) {
|
|
throw new Error(`Unsupported item type: ${itemType}`);
|
|
}
|
|
};
|
|
|
|
export const setItemUserData = async <T>(itemType: ModelType, itemId: string, namespace: string, key: string, value: T, deleted = false): Promise<SupportedEntity> => {
|
|
checkIsSupportedItemType(itemType);
|
|
|
|
interface ItemSlice {
|
|
user_data: string;
|
|
updated_time?: number;
|
|
id?: string;
|
|
parent_id?: string;
|
|
}
|
|
|
|
const options: LoadOptions = { fields: ['user_data'] };
|
|
if (itemType === ModelType.Note) (options.fields as string[]).push('parent_id');
|
|
|
|
const item = await BaseItem.loadItem(itemType, itemId, options) as ItemSlice;
|
|
|
|
const userData = unserializeUserData(item.user_data);
|
|
const newUserData = setUserData(userData, namespace, key, value, deleted);
|
|
|
|
const itemToSave: ItemSlice = {
|
|
id: itemId,
|
|
user_data: serializeUserData(newUserData),
|
|
updated_time: Date.now(),
|
|
};
|
|
|
|
if (itemType === ModelType.Note) itemToSave.parent_id = item.parent_id;
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
const saveOptions: any = { autoTimestamp: false };
|
|
|
|
if (itemType === ModelType.Note) return Note.save(itemToSave, saveOptions);
|
|
if (itemType === ModelType.Folder) return Folder.save(itemToSave, saveOptions);
|
|
if (itemType === ModelType.Resource) return Resource.save(itemToSave, saveOptions);
|
|
if (itemType === ModelType.Tag) return Tag.save(itemToSave, saveOptions);
|
|
|
|
throw new Error('Unreachable');
|
|
};
|
|
|
|
// Deprecated - don't use
|
|
export const setNoteUserData = async <T>(note: NoteEntity, namespace: string, key: string, value: T, deleted = false): Promise<NoteEntity> => {
|
|
return setItemUserData(ModelType.Note, note.id, namespace, key, value, deleted);
|
|
};
|
|
|
|
const hasUserData = (userData: UserData, namespace: string, key: string) => {
|
|
if (!(namespace in userData)) return false;
|
|
if (!(key in userData[namespace])) return false;
|
|
if (userData[namespace][key].d) return false;
|
|
return true;
|
|
};
|
|
|
|
export const getItemUserData = async <T>(itemType: ModelType, itemId: string, namespace: string, key: string): Promise<T|undefined> => {
|
|
checkIsSupportedItemType(itemType);
|
|
|
|
interface ItemSlice {
|
|
user_data: string;
|
|
}
|
|
|
|
const item = await BaseItem.loadItem(itemType, itemId, { fields: ['user_data'] }) as ItemSlice;
|
|
const userData = unserializeUserData(item.user_data);
|
|
return getUserData(userData, namespace, key);
|
|
};
|
|
|
|
// Deprecated - don't use
|
|
export const getNoteUserData = async <T>(note: NoteEntity, namespace: string, key: string): Promise<T|undefined> => {
|
|
return getItemUserData<T>(ModelType.Note, note.id, namespace, key);
|
|
};
|
|
|
|
export const deleteItemUserData = async (itemType: ModelType, itemId: string, namespace: string, key: string): Promise<SupportedEntity> => {
|
|
return setItemUserData(itemType, itemId, namespace, key, 0, true);
|
|
};
|
|
|
|
// Deprecated - don't use
|
|
export const deleteNoteUserData = async (note: NoteEntity, namespace: string, key: string): Promise<NoteEntity> => {
|
|
return setNoteUserData(note, namespace, key, 0, true);
|
|
};
|
|
|
|
export const mergeUserData = (target: UserData, source: UserData): UserData => {
|
|
const output: UserData = { ...target };
|
|
|
|
for (const namespaceName of Object.keys(source)) {
|
|
if (!(namespaceName in output)) output[namespaceName] = source[namespaceName];
|
|
const namespace = source[namespaceName];
|
|
for (const [key, value] of Object.entries(namespace)) {
|
|
// Keep ours
|
|
if (output[namespaceName][key] && output[namespaceName][key].t >= value.t) continue;
|
|
|
|
// Use theirs
|
|
output[namespaceName][key] = {
|
|
...value,
|
|
};
|
|
}
|
|
}
|
|
|
|
return output;
|
|
};
|