2
0
mirror of https://github.com/frappe/books.git synced 2024-09-20 19:29:02 +00:00
books/fyo/core/docHandler.ts

193 lines
4.3 KiB
TypeScript

import { Doc } from 'fyo/model/doc';
import { DocMap, ModelMap, SinglesMap } from 'fyo/model/types';
import { coreModels } from 'fyo/models';
import Observable from 'fyo/utils/observable';
import { Schema } from 'schemas/types';
import { getRandomString } from 'utils';
import { Fyo } from '..';
import { DocValue, DocValueMap } from './types';
export class DocHandler {
fyo: Fyo;
models: ModelMap = {};
singles: SinglesMap = {};
docs: Observable<DocMap> = new Observable();
observer: Observable<never> = new Observable();
constructor(fyo: Fyo) {
this.fyo = fyo;
}
init() {
this.models = {};
this.singles = {};
this.docs = new Observable();
this.observer = new Observable();
}
purgeCache() {
this.init();
}
registerModels(models: ModelMap, regionalModels: ModelMap = {}) {
for (const schemaName in this.fyo.db.schemaMap) {
if (coreModels[schemaName] !== undefined) {
this.models[schemaName] = coreModels[schemaName];
} else if (regionalModels[schemaName] !== undefined) {
this.models[schemaName] = regionalModels[schemaName];
} else if (models[schemaName] !== undefined) {
this.models[schemaName] = models[schemaName];
} else {
this.models[schemaName] = Doc;
}
}
}
/**
* Cache operations
*/
addToCache(doc: Doc) {
if (!this.docs) {
return;
}
// add to `docs` cache
const name = doc.name;
const schemaName = doc.schemaName;
if (!name) {
return;
}
if (!this.docs[schemaName]) {
this.docs[schemaName] = {};
}
(this.docs[schemaName] as DocMap)[name] = doc;
// singles available as first level objects too
if (schemaName === doc.name) {
this.singles[name] = doc;
}
// propagate change to `docs`
doc.on('change', (params: unknown) => {
this.docs!.trigger('change', params);
});
}
removeFromCache(schemaName: string, name: string) {
const docMap = this.docs[schemaName] as DocMap | undefined;
delete docMap?.[name];
}
getFromCache(schemaName: string, name: string): Doc | undefined {
const docMap = this.docs[schemaName] as DocMap | undefined;
return docMap?.[name];
}
getCachedValue(
schemaName: string,
name: string,
fieldname: string
): DocValue | Doc[] | undefined {
const docMap = this.docs[schemaName] as DocMap;
const doc = docMap[name];
if (doc === undefined) {
return;
}
return doc.get(fieldname);
}
isDirty(schemaName: string, name: string): boolean {
const doc = (this.docs?.[schemaName] as DocMap)?.[name];
if (doc === undefined) {
return false;
}
return doc.dirty;
}
/**
* Doc Operations
*/
async getDoc(
schemaName: string,
name: string,
options = { skipDocumentCache: false }
) {
let doc: Doc | undefined;
if (!options?.skipDocumentCache) {
doc = this.getFromCache(schemaName, name);
}
if (doc) {
return doc;
}
doc = this.getNewDoc(schemaName, { name });
await doc.load();
this.addToCache(doc);
return doc;
}
async getSingle(schemaName: string) {
return await this.getDoc(schemaName, schemaName);
}
async getDuplicate(doc: Doc): Promise<Doc> {
const newDoc = await doc.duplicate(false);
delete newDoc.name;
return newDoc;
}
getNewDoc(
schemaName: string,
data: DocValueMap = {},
cacheDoc: boolean = true,
schema?: Schema,
Model?: typeof Doc
): Doc {
if (!this.models[schemaName] && Model) {
this.models[schemaName] = Model;
}
Model ??= this.models[schemaName];
schema ??= this.fyo.schemaMap[schemaName];
if (schema === undefined) {
throw new Error(`Schema not found for ${schemaName}`);
}
const doc = new Model!(schema, data, this.fyo);
doc.name ??= getRandomString();
if (cacheDoc) {
this.addToCache(doc);
}
return doc;
}
async syncDoc(schemaName: string, data: DocValueMap) {
const name = data.name as string | undefined;
if (name === undefined) {
return;
}
const docExists = await this.fyo.db.exists(schemaName, name);
if (!docExists) {
const doc = this.getNewDoc(schemaName, data);
await doc.sync();
return;
}
const doc = await this.getDoc(schemaName, name);
await doc.setMultiple(data);
await doc.sync();
}
}