2
0
mirror of https://github.com/frappe/books.git synced 2025-01-25 16:18:33 +00:00
books/frappe/core/docHandler.ts
2022-05-23 16:18:21 +05:30

181 lines
3.9 KiB
TypeScript

import Doc from 'frappe/model/doc';
import { DocMap, ModelMap } from 'frappe/model/types';
import { getRandomString } from 'frappe/utils';
import Observable from 'frappe/utils/observable';
import { Frappe } from '..';
import { DocValue, DocValueMap } from './types';
export class DocHandler {
frappe: Frappe;
singles: DocMap = {};
docs: Observable<DocMap> = new Observable();
models: ModelMap = {};
constructor(frappe: Frappe) {
this.frappe = frappe;
}
init() {
this.models = {};
this.docs = new Observable();
}
registerModels(models: ModelMap) {
for (const schemaName in models) {
this.models[schemaName] = models[schemaName];
}
}
/**
* 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;
}
getModel(schemaName: string): typeof Doc {
const Model = this.models[schemaName];
if (Model === undefined) {
return Doc;
}
return Model;
}
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;
}
getEmptyDoc(schemaName: string, cacheDoc: boolean = true): Doc {
const doc = this.getNewDoc(schemaName);
doc.name = getRandomString();
if (cacheDoc) {
this.addToCache(doc);
}
return doc;
}
getNewDoc(schemaName: string, data: DocValueMap = {}): Doc {
const Model = this.getModel(schemaName);
const schema = this.frappe.schemaMap[schemaName];
if (schema === undefined) {
throw new Error(`Schema not found for ${schemaName}`);
}
const doc = new Model(schema, data);
doc.setDefaults();
return doc;
}
async syncDoc(schemaName: string, data: DocValueMap) {
const name = data.name as string | undefined;
if (name === undefined) {
return;
}
const docExists = await this.frappe.db.exists(schemaName, name);
if (!docExists) {
const doc = this.getNewDoc(schemaName, data);
await doc.insert;
return;
}
const doc = await this.getDoc(schemaName, name);
await doc.setMultiple(data);
await doc.update();
}
}