import { SingleValue } from 'backend/database/types'; import { Fyo } from 'fyo'; import { DatabaseDemux } from 'fyo/demux/db'; import Observable from 'fyo/utils/observable'; import { Field, RawValue, SchemaMap } from 'schemas/types'; import { getMapFromList } from 'utils'; import { DatabaseBase, DatabaseDemuxBase, GetAllOptions } from 'utils/db/types'; import { Converter } from './converter'; import { DatabaseDemuxConstructor, DocValue, DocValueMap, RawValueMap, } from './types'; // Return types of Bespoke Queries type TopExpenses = { account: string; total: number }[]; type TotalOutstanding = { total: number; outstanding: number }; type Cashflow = { inflow: number; outflow: number; 'month-year': string }[]; export class DatabaseHandler extends DatabaseBase { #fyo: Fyo; converter: Converter; #demux: DatabaseDemuxBase; dbPath?: string; schemaMap: Readonly = {}; observer: Observable = new Observable(); fieldValueMap: Record> = {}; constructor(fyo: Fyo, Demux?: DatabaseDemuxConstructor) { super(); this.#fyo = fyo; this.converter = new Converter(this, this.#fyo); if (Demux !== undefined) { this.#demux = new Demux(fyo.isElectron); } else { this.#demux = new DatabaseDemux(fyo.isElectron); } } get isConnected() { return !!this.dbPath; } async createNewDatabase(dbPath: string, countryCode: string) { countryCode = await this.#demux.createNewDatabase(dbPath, countryCode); await this.init(); this.dbPath = dbPath; return countryCode; } async connectToDatabase(dbPath: string, countryCode?: string) { countryCode = await this.#demux.connectToDatabase(dbPath, countryCode); await this.init(); this.dbPath = dbPath; return countryCode; } async init() { this.schemaMap = (await this.#demux.getSchemaMap()) as Readonly; for (const schemaName in this.schemaMap) { const fields = this.schemaMap[schemaName]!.fields!; this.fieldValueMap[schemaName] = getMapFromList(fields, 'fieldname'); } this.observer = new Observable(); } purgeCache() { this.dbPath = undefined; this.schemaMap = {}; this.fieldValueMap = {}; } async insert( schemaName: string, docValueMap: DocValueMap ): Promise { let rawValueMap = this.converter.toRawValueMap( schemaName, docValueMap ) as RawValueMap; rawValueMap = (await this.#demux.call( 'insert', schemaName, rawValueMap )) as RawValueMap; this.observer.trigger(`insert:${schemaName}`, docValueMap); return this.converter.toDocValueMap(schemaName, rawValueMap) as DocValueMap; } // Read async get( schemaName: string, name: string, fields?: string | string[] ): Promise { const rawValueMap = (await this.#demux.call( 'get', schemaName, name, fields )) as RawValueMap; this.observer.trigger(`get:${schemaName}`, { name, fields }); return this.converter.toDocValueMap(schemaName, rawValueMap) as DocValueMap; } async getAll( schemaName: string, options: GetAllOptions = {} ): Promise { const rawValueMap = await this.#getAll(schemaName, options); this.observer.trigger(`getAll:${schemaName}`, options); return this.converter.toDocValueMap( schemaName, rawValueMap ) as DocValueMap[]; } async getAllRaw( schemaName: string, options: GetAllOptions = {} ): Promise { const all = await this.#getAll(schemaName, options); this.observer.trigger(`getAllRaw:${schemaName}`, options); return all; } async getSingleValues( ...fieldnames: ({ fieldname: string; parent?: string } | string)[] ): Promise> { const rawSingleValue = (await this.#demux.call( 'getSingleValues', ...fieldnames )) as SingleValue; const docSingleValue: SingleValue = []; for (const sv of rawSingleValue) { const field = this.fieldValueMap[sv.parent][sv.fieldname]; const value = Converter.toDocValue(sv.value, field, this.#fyo); docSingleValue.push({ value, parent: sv.parent, fieldname: sv.fieldname, }); } this.observer.trigger(`getSingleValues`, fieldnames); return docSingleValue; } async count( schemaName: string, options: GetAllOptions = {} ): Promise { const rawValueMap = await this.#getAll(schemaName, options); const count = rawValueMap.length; this.observer.trigger(`count:${schemaName}`, options); return count; } // Update async rename( schemaName: string, oldName: string, newName: string ): Promise { await this.#demux.call('rename', schemaName, oldName, newName); this.observer.trigger(`rename:${schemaName}`, { oldName, newName }); } async update(schemaName: string, docValueMap: DocValueMap): Promise { const rawValueMap = this.converter.toRawValueMap(schemaName, docValueMap); await this.#demux.call('update', schemaName, rawValueMap); this.observer.trigger(`update:${schemaName}`, docValueMap); } // Delete async delete(schemaName: string, name: string): Promise { await this.#demux.call('delete', schemaName, name); this.observer.trigger(`delete:${schemaName}`, name); } // Other async exists(schemaName: string, name?: string): Promise { const doesExist = (await this.#demux.call( 'exists', schemaName, name )) as boolean; this.observer.trigger(`exists:${schemaName}`, name); return doesExist; } async close(): Promise { await this.#demux.call('close'); this.purgeCache(); } /** * Bespoke function * * These are functions to run custom queries that are too complex for * DatabaseCore and require use of knex or raw queries. The output * of these is not converted to DocValue and is used as is (RawValue). * * The query logic for these is in backend/database/bespoke.ts */ async getTopExpenses(fromDate: string, toDate: string): Promise { return (await this.#demux.callBespoke( 'getTopExpenses', fromDate, toDate )) as TopExpenses; } async getTotalOutstanding( schemaName: string, fromDate: string, toDate: string ): Promise { return (await this.#demux.callBespoke( 'getTotalOutstanding', schemaName, fromDate, toDate )) as TotalOutstanding; } async getCashflow(fromDate: string, toDate: string): Promise { return (await this.#demux.callBespoke( 'getCashflow', fromDate, toDate )) as Cashflow; } /** * Internal methods */ async #getAll( schemaName: string, options: GetAllOptions = {} ): Promise { return (await this.#demux.call( 'getAll', schemaName, options )) as RawValueMap[]; } }