import { getMoneyMaker } from 'pesa'; import { markRaw } from 'vue'; import * as errors from './utils/errors'; import { asyncHandler, getDuplicates, getRandomString } from './utils'; import { DEFAULT_DISPLAY_PRECISION, DEFAULT_INTERNAL_PRECISION, } from './utils/consts'; import { format } from './utils/format'; import Observable from './utils/observable'; import { t, T } from './utils/translation'; class Frappe { t = t; T = T; format = format; getRandomString = getRandomString; errors = errors; isElectron = false; isServer = false; async initializeAndRegister(customModels = {}, force = false) { this.init(force); this.Meta = (await import('frappe/model/meta')).default; this.Document = (await import('frappe/model/document')).default; const coreModels = await import('frappe/models'); this.registerModels(coreModels.default); this.registerModels(customModels); } async initializeMoneyMaker(currency) { currency ??= 'XXX'; // to be called after db initialization const values = (await frappe.db?.getSingleValues( { fieldname: 'internalPrecision', parent: 'SystemSettings', }, { fieldname: 'displayPrecision', parent: 'SystemSettings', } )) ?? []; let { internalPrecision: precision, displayPrecision: display } = values.reduce((acc, { fieldname, value }) => { acc[fieldname] = value; return acc; }, {}); if (typeof precision === 'undefined') { precision = DEFAULT_INTERNAL_PRECISION; } if (typeof precision === 'string') { precision = parseInt(precision); } if (typeof display === 'undefined') { display = DEFAULT_DISPLAY_PRECISION; } if (typeof display === 'string') { display = parseInt(display); } this.pesa = getMoneyMaker({ currency, precision, display, wrapper: markRaw, }); } init(force) { if (this._initialized && !force) return; // Initialize Config this.config = { serverURL: '', backend: 'sqlite', port: 8000, }; // Initialize Globals this.metaCache = {}; this.models = {}; this.forms = {}; this.views = {}; this.flags = {}; this.methods = {}; this.errorLog = []; // temp params while calling routes this.temp = {}; this.params = {}; this.docs = new Observable(); this.events = new Observable(); this._initialized = true; } registerModels(models) { // register models from app/models/index.js for (let doctype in models) { let metaDefinition = models[doctype]; if (!metaDefinition.name) { throw new Error(`Name is mandatory for ${doctype}`); } if (metaDefinition.name !== doctype) { throw new Error( `Model name mismatch for ${doctype}: ${metaDefinition.name}` ); } let fieldnames = (metaDefinition.fields || []) .map((df) => df.fieldname) .sort(); let duplicateFieldnames = getDuplicates(fieldnames); if (duplicateFieldnames.length > 0) { throw new Error( `Duplicate fields in ${doctype}: ${duplicateFieldnames.join(', ')}` ); } this.models[doctype] = metaDefinition; } } getModels(filterFunction) { let models = []; for (let doctype in this.models) { models.push(this.models[doctype]); } return filterFunction ? models.filter(filterFunction) : models; } registerView(view, name, module) { if (!this.views[view]) this.views[view] = {}; this.views[view][name] = module; } registerMethod({ method, handler }) { this.methods[method] = handler; if (this.app) { // add to router if client-server this.app.post( `/api/method/${method}`, asyncHandler(async function (request, response) { let data = await handler(request.body); if (data === undefined) { data = {}; } return response.json(data); }) ); } } async call({ method, args }) { if (this.isServer) { if (this.methods[method]) { return await this.methods[method](args); } else { throw new Error(`${method} not found`); } } let url = `/api/method/${method}`; let response = await fetch(url, { method: 'POST', headers: { Accept: 'application/json', 'Content-Type': 'application/json', }, body: JSON.stringify(args || {}), }); return await response.json(); } addToCache(doc) { if (!this.docs) return; // add to `docs` cache if (doc.doctype && doc.name) { if (!this.docs[doc.doctype]) { this.docs[doc.doctype] = {}; } this.docs[doc.doctype][doc.name] = doc; // singles available as first level objects too if (doc.doctype === doc.name) { this[doc.name] = doc; } // propogate change to `docs` doc.on('change', (params) => { this.docs.trigger('change', params); }); } } removeFromCache(doctype, name) { try { delete this.docs[doctype][name]; } catch (e) { console.warn(`Document ${doctype} ${name} does not exist`); } } isDirty(doctype, name) { return ( (this.docs && this.docs[doctype] && this.docs[doctype][name] && this.docs[doctype][name]._dirty) || false ); } getDocFromCache(doctype, name) { if (this.docs && this.docs[doctype] && this.docs[doctype][name]) { return this.docs[doctype][name]; } } getMeta(doctype) { if (!this.metaCache[doctype]) { let model = this.models[doctype]; if (!model) { throw new Error(`${doctype} is not a registered doctype`); } let metaClass = model.metaClass || this.Meta; this.metaCache[doctype] = new metaClass(model); } return this.metaCache[doctype]; } async getDoc(doctype, name, options = { skipDocumentCache: false }) { let doc = options.skipDocumentCache ? null : this.getDocFromCache(doctype, name); if (!doc) { doc = new (this.getDocumentClass(doctype))({ doctype: doctype, name: name, }); await doc.load(); this.addToCache(doc); } return doc; } getDocumentClass(doctype) { const meta = this.getMeta(doctype); return meta.documentClass || this.Document; } async getSingle(doctype) { return await this.getDoc(doctype, doctype); } async getDuplicate(doc) { const newDoc = await this.getNewDoc(doc.doctype); for (let field of this.getMeta(doc.doctype).getValidFields()) { if (['name', 'submitted'].includes(field.fieldname)) continue; if (field.fieldtype === 'Table') { newDoc[field.fieldname] = (doc[field.fieldname] || []).map((d) => { let newd = Object.assign({}, d); newd.name = ''; return newd; }); } else { newDoc[field.fieldname] = doc[field.fieldname]; } } return newDoc; } getNewDoc(doctype, cacheDoc = true) { let doc = this.newDoc({ doctype: doctype }); doc._notInserted = true; doc.name = frappe.getRandomString(); if (cacheDoc) { this.addToCache(doc); } return doc; } async newCustomDoc(fields) { let doc = new this.Document({ isCustom: 1, fields }); doc._notInserted = true; doc.name = this.getRandomString(); this.addToCache(doc); return doc; } createMeta(fields) { let meta = new this.Meta({ isCustom: 1, fields }); return meta; } newDoc(data) { let doc = new (this.getDocumentClass(data.doctype))(data); doc.setDefaults(); return doc; } async insert(data) { return await this.newDoc(data).insert(); } async syncDoc(data) { let doc; if (await this.db.exists(data.doctype, data.name)) { doc = await this.getDoc(data.doctype, data.name); Object.assign(doc, data); await doc.update(); } else { doc = this.newDoc(data); await doc.insert(); } } // only for client side async login(email, password) { if (email === 'Administrator') { this.session = { user: 'Administrator', }; return; } let response = await fetch(this.getServerURL() + '/api/login', { method: 'POST', headers: { Accept: 'application/json', 'Content-Type': 'application/json', }, body: JSON.stringify({ email, password }), }); if (response.status === 200) { const res = await response.json(); this.session = { user: email, token: res.token, }; return res; } return response; } async signup(email, fullName, password) { let response = await fetch(this.getServerURL() + '/api/signup', { method: 'POST', headers: { Accept: 'application/json', 'Content-Type': 'application/json', }, body: JSON.stringify({ email, fullName, password }), }); if (response.status === 200) { return await response.json(); } return response; } getServerURL() { return this.config.serverURL || ''; } close() { this.db.close(); if (this.server) { this.server.close(); } } store = { isDevelopment: false, appVersion: '', }; } export { T, t }; export default new Frappe();