2022-04-19 11:29:36 +05:30
|
|
|
import { Fyo } from 'fyo';
|
2022-04-24 12:18:44 +05:30
|
|
|
import { Doc } from 'fyo/model/doc';
|
2022-04-23 14:53:44 +05:30
|
|
|
import { isPesa } from 'fyo/utils';
|
|
|
|
import { ValueError } from 'fyo/utils/errors';
|
|
|
|
import { DateTime } from 'luxon';
|
2022-05-23 11:00:54 +05:30
|
|
|
import { Money } from 'pesa';
|
2022-04-25 12:03:31 +05:30
|
|
|
import { Field, FieldTypeEnum, RawValue, TargetField } from 'schemas/types';
|
2022-04-23 14:53:44 +05:30
|
|
|
import { getIsNullOrUndef } from 'utils';
|
2022-04-18 10:37:36 +05:30
|
|
|
import { DatabaseHandler } from './dbHandler';
|
|
|
|
import { DocValue, DocValueMap, RawValueMap } from './types';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* # Converter
|
|
|
|
*
|
|
|
|
* Basically converts serializable RawValues from the db to DocValues used
|
|
|
|
* by the frontend and vice versa.
|
|
|
|
*
|
|
|
|
* ## Value Conversion
|
|
|
|
* It exposes two static methods: `toRawValue` and `toDocValue` that can be
|
|
|
|
* used elsewhere given the fieldtype.
|
|
|
|
*
|
|
|
|
* ## Map Conversion
|
|
|
|
* Two methods `toDocValueMap` and `toRawValueMap` are exposed but should be
|
|
|
|
* used only from the `dbHandler`.
|
|
|
|
*/
|
|
|
|
|
|
|
|
export class Converter {
|
|
|
|
db: DatabaseHandler;
|
2022-04-19 11:29:36 +05:30
|
|
|
fyo: Fyo;
|
2022-04-18 10:37:36 +05:30
|
|
|
|
2022-04-19 11:29:36 +05:30
|
|
|
constructor(db: DatabaseHandler, fyo: Fyo) {
|
2022-04-18 10:37:36 +05:30
|
|
|
this.db = db;
|
2022-04-19 11:29:36 +05:30
|
|
|
this.fyo = fyo;
|
2022-04-18 10:37:36 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
toDocValueMap(
|
|
|
|
schemaName: string,
|
|
|
|
rawValueMap: RawValueMap | RawValueMap[]
|
|
|
|
): DocValueMap | DocValueMap[] {
|
2022-08-24 16:06:48 +05:30
|
|
|
rawValueMap ??= {};
|
2022-04-18 10:37:36 +05:30
|
|
|
if (Array.isArray(rawValueMap)) {
|
|
|
|
return rawValueMap.map((dv) => this.#toDocValueMap(schemaName, dv));
|
|
|
|
} else {
|
|
|
|
return this.#toDocValueMap(schemaName, rawValueMap);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
toRawValueMap(
|
|
|
|
schemaName: string,
|
|
|
|
docValueMap: DocValueMap | DocValueMap[]
|
|
|
|
): RawValueMap | RawValueMap[] {
|
2022-08-24 16:06:48 +05:30
|
|
|
docValueMap ??= {};
|
2022-04-18 10:37:36 +05:30
|
|
|
if (Array.isArray(docValueMap)) {
|
|
|
|
return docValueMap.map((dv) => this.#toRawValueMap(schemaName, dv));
|
|
|
|
} else {
|
|
|
|
return this.#toRawValueMap(schemaName, docValueMap);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-23 14:53:44 +05:30
|
|
|
static toDocValue(value: RawValue, field: Field, fyo: Fyo): DocValue {
|
|
|
|
switch (field.fieldtype) {
|
2022-04-18 10:37:36 +05:30
|
|
|
case FieldTypeEnum.Currency:
|
2022-04-23 14:53:44 +05:30
|
|
|
return toDocCurrency(value, field, fyo);
|
2022-04-18 10:37:36 +05:30
|
|
|
case FieldTypeEnum.Date:
|
2022-04-23 14:53:44 +05:30
|
|
|
return toDocDate(value, field);
|
2022-04-18 10:37:36 +05:30
|
|
|
case FieldTypeEnum.Datetime:
|
2022-04-23 14:53:44 +05:30
|
|
|
return toDocDate(value, field);
|
2022-04-18 10:37:36 +05:30
|
|
|
case FieldTypeEnum.Int:
|
2022-04-23 14:53:44 +05:30
|
|
|
return toDocInt(value, field);
|
2022-04-18 10:37:36 +05:30
|
|
|
case FieldTypeEnum.Float:
|
2022-04-23 14:53:44 +05:30
|
|
|
return toDocFloat(value, field);
|
2022-04-18 10:37:36 +05:30
|
|
|
case FieldTypeEnum.Check:
|
2022-04-23 14:53:44 +05:30
|
|
|
return toDocCheck(value, field);
|
2022-04-18 10:37:36 +05:30
|
|
|
default:
|
2022-04-25 12:03:31 +05:30
|
|
|
return toDocString(value, field);
|
2022-04-18 10:37:36 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-23 14:53:44 +05:30
|
|
|
static toRawValue(value: DocValue, field: Field, fyo: Fyo): RawValue {
|
|
|
|
switch (field.fieldtype) {
|
2022-04-18 10:37:36 +05:30
|
|
|
case FieldTypeEnum.Currency:
|
2022-04-23 14:53:44 +05:30
|
|
|
return toRawCurrency(value, fyo, field);
|
2022-04-18 10:37:36 +05:30
|
|
|
case FieldTypeEnum.Date:
|
2022-04-23 14:53:44 +05:30
|
|
|
return toRawDate(value, field);
|
2022-04-18 10:37:36 +05:30
|
|
|
case FieldTypeEnum.Datetime:
|
2022-04-23 14:53:44 +05:30
|
|
|
return toRawDateTime(value, field);
|
|
|
|
case FieldTypeEnum.Int:
|
|
|
|
return toRawInt(value, field);
|
|
|
|
case FieldTypeEnum.Float:
|
|
|
|
return toRawFloat(value, field);
|
2022-04-18 10:37:36 +05:30
|
|
|
case FieldTypeEnum.Check:
|
2022-04-23 14:53:44 +05:30
|
|
|
return toRawCheck(value, field);
|
2022-04-28 18:07:07 +05:30
|
|
|
case FieldTypeEnum.Link:
|
|
|
|
return toRawLink(value, field);
|
2022-04-18 10:37:36 +05:30
|
|
|
default:
|
2022-04-23 14:53:44 +05:30
|
|
|
return toRawString(value, field);
|
2022-04-18 10:37:36 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#toDocValueMap(schemaName: string, rawValueMap: RawValueMap): DocValueMap {
|
|
|
|
const fieldValueMap = this.db.fieldValueMap[schemaName];
|
|
|
|
const docValueMap: DocValueMap = {};
|
|
|
|
|
|
|
|
for (const fieldname in rawValueMap) {
|
|
|
|
const field = fieldValueMap[fieldname];
|
|
|
|
const rawValue = rawValueMap[fieldname];
|
2022-05-17 11:39:08 +05:30
|
|
|
if (!field) {
|
|
|
|
continue;
|
|
|
|
}
|
2022-04-18 10:37:36 +05:30
|
|
|
|
|
|
|
if (Array.isArray(rawValue)) {
|
2022-04-25 12:03:31 +05:30
|
|
|
const parentSchemaName = (field as TargetField).target;
|
2022-04-18 10:37:36 +05:30
|
|
|
docValueMap[fieldname] = rawValue.map((rv) =>
|
2022-04-25 12:03:31 +05:30
|
|
|
this.#toDocValueMap(parentSchemaName, rv)
|
2022-04-18 10:37:36 +05:30
|
|
|
);
|
|
|
|
} else {
|
|
|
|
docValueMap[fieldname] = Converter.toDocValue(
|
|
|
|
rawValue,
|
2022-04-23 14:53:44 +05:30
|
|
|
field,
|
2022-04-19 11:29:36 +05:30
|
|
|
this.fyo
|
2022-04-18 10:37:36 +05:30
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return docValueMap;
|
|
|
|
}
|
|
|
|
|
|
|
|
#toRawValueMap(schemaName: string, docValueMap: DocValueMap): RawValueMap {
|
|
|
|
const fieldValueMap = this.db.fieldValueMap[schemaName];
|
|
|
|
const rawValueMap: RawValueMap = {};
|
|
|
|
|
|
|
|
for (const fieldname in docValueMap) {
|
|
|
|
const field = fieldValueMap[fieldname];
|
|
|
|
const docValue = docValueMap[fieldname];
|
|
|
|
|
|
|
|
if (Array.isArray(docValue)) {
|
2022-04-25 12:03:31 +05:30
|
|
|
const parentSchemaName = (field as TargetField).target;
|
|
|
|
|
2022-04-18 10:37:36 +05:30
|
|
|
rawValueMap[fieldname] = docValue.map((value) => {
|
|
|
|
if (value instanceof Doc) {
|
2022-04-25 12:03:31 +05:30
|
|
|
return this.#toRawValueMap(parentSchemaName, value.getValidDict());
|
2022-04-18 10:37:36 +05:30
|
|
|
}
|
|
|
|
|
2022-04-25 12:03:31 +05:30
|
|
|
return this.#toRawValueMap(parentSchemaName, value as DocValueMap);
|
2022-04-18 10:37:36 +05:30
|
|
|
});
|
|
|
|
} else {
|
|
|
|
rawValueMap[fieldname] = Converter.toRawValue(
|
|
|
|
docValue,
|
2022-04-23 14:53:44 +05:30
|
|
|
field,
|
|
|
|
this.fyo
|
2022-04-18 10:37:36 +05:30
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return rawValueMap;
|
|
|
|
}
|
|
|
|
}
|
2022-04-23 14:53:44 +05:30
|
|
|
|
2022-04-25 12:03:31 +05:30
|
|
|
function toDocString(value: RawValue, field: Field) {
|
2022-05-02 20:47:14 +05:30
|
|
|
if (value === null) {
|
|
|
|
return null;
|
2022-04-25 12:03:31 +05:30
|
|
|
}
|
|
|
|
|
2022-05-11 16:14:31 +05:30
|
|
|
if (value === undefined) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2022-05-02 20:47:14 +05:30
|
|
|
if (typeof value === 'string') {
|
2022-04-25 12:03:31 +05:30
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
throwError(value, field, 'doc');
|
|
|
|
}
|
|
|
|
|
2022-04-23 14:53:44 +05:30
|
|
|
function toDocDate(value: RawValue, field: Field) {
|
2022-05-10 14:56:17 +05:30
|
|
|
if ((value as any) instanceof Date) {
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
2022-05-05 16:14:26 +05:30
|
|
|
if (value === null || value === '') {
|
2022-05-02 20:47:14 +05:30
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2022-04-23 14:53:44 +05:30
|
|
|
if (typeof value !== 'number' && typeof value !== 'string') {
|
|
|
|
throwError(value, field, 'doc');
|
|
|
|
}
|
|
|
|
|
|
|
|
const date = new Date(value);
|
|
|
|
if (date.toString() === 'Invalid Date') {
|
|
|
|
throwError(value, field, 'doc');
|
|
|
|
}
|
|
|
|
|
|
|
|
return date;
|
|
|
|
}
|
|
|
|
|
|
|
|
function toDocCurrency(value: RawValue, field: Field, fyo: Fyo) {
|
2022-05-10 14:56:17 +05:30
|
|
|
if (isPesa(value)) {
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
2022-05-05 16:14:26 +05:30
|
|
|
if (value === '') {
|
|
|
|
return fyo.pesa(0);
|
|
|
|
}
|
|
|
|
|
2022-04-23 14:53:44 +05:30
|
|
|
if (typeof value === 'string') {
|
|
|
|
return fyo.pesa(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof value === 'number') {
|
|
|
|
return fyo.pesa(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof value === 'boolean') {
|
|
|
|
return fyo.pesa(Number(value));
|
|
|
|
}
|
|
|
|
|
2022-05-02 20:47:14 +05:30
|
|
|
if (value === null) {
|
|
|
|
return fyo.pesa(0);
|
|
|
|
}
|
|
|
|
|
2022-04-23 14:53:44 +05:30
|
|
|
throwError(value, field, 'doc');
|
|
|
|
}
|
|
|
|
|
|
|
|
function toDocInt(value: RawValue, field: Field): number {
|
2022-05-05 16:14:26 +05:30
|
|
|
if (value === '') {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2022-04-23 14:53:44 +05:30
|
|
|
if (typeof value === 'string') {
|
|
|
|
value = parseInt(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
return toDocFloat(value, field);
|
|
|
|
}
|
|
|
|
|
|
|
|
function toDocFloat(value: RawValue, field: Field): number {
|
2022-05-05 16:14:26 +05:30
|
|
|
if (value === '') {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2022-04-23 14:53:44 +05:30
|
|
|
if (typeof value === 'boolean') {
|
|
|
|
return Number(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof value === 'string') {
|
|
|
|
value = parseFloat(value);
|
|
|
|
}
|
|
|
|
|
2022-05-02 20:47:14 +05:30
|
|
|
if (value === null) {
|
|
|
|
value = 0;
|
|
|
|
}
|
|
|
|
|
2022-04-23 14:53:44 +05:30
|
|
|
if (typeof value === 'number' && !Number.isNaN(value)) {
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
throwError(value, field, 'doc');
|
|
|
|
}
|
|
|
|
|
|
|
|
function toDocCheck(value: RawValue, field: Field): boolean {
|
|
|
|
if (typeof value === 'boolean') {
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof value === 'string') {
|
2022-05-20 16:42:32 +05:30
|
|
|
return !!parseFloat(value);
|
2022-04-23 14:53:44 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof value === 'number') {
|
|
|
|
return Boolean(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
throwError(value, field, 'doc');
|
|
|
|
}
|
|
|
|
|
|
|
|
function toRawCurrency(value: DocValue, fyo: Fyo, field: Field): string {
|
|
|
|
if (isPesa(value)) {
|
|
|
|
return (value as Money).store;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (getIsNullOrUndef(value)) {
|
|
|
|
return fyo.pesa(0).store;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof value === 'number') {
|
|
|
|
return fyo.pesa(value).store;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof value === 'string') {
|
|
|
|
return fyo.pesa(value).store;
|
|
|
|
}
|
|
|
|
|
|
|
|
throwError(value, field, 'raw');
|
|
|
|
}
|
|
|
|
|
|
|
|
function toRawInt(value: DocValue, field: Field): number {
|
|
|
|
if (typeof value === 'string') {
|
|
|
|
return parseInt(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (getIsNullOrUndef(value)) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof value === 'number') {
|
|
|
|
return Math.floor(value as number);
|
|
|
|
}
|
|
|
|
|
|
|
|
throwError(value, field, 'raw');
|
|
|
|
}
|
|
|
|
|
|
|
|
function toRawFloat(value: DocValue, field: Field): number {
|
|
|
|
if (typeof value === 'string') {
|
|
|
|
return parseFloat(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (getIsNullOrUndef(value)) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof value === 'number') {
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
throwError(value, field, 'raw');
|
|
|
|
}
|
|
|
|
|
2022-04-24 12:18:44 +05:30
|
|
|
function toRawDate(value: DocValue, field: Field): string | null {
|
2022-04-23 14:53:44 +05:30
|
|
|
const dateTime = toRawDateTime(value, field);
|
2022-04-24 12:18:44 +05:30
|
|
|
if (dateTime === null) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2022-04-23 14:53:44 +05:30
|
|
|
return dateTime.split('T')[0];
|
|
|
|
}
|
|
|
|
|
2022-04-24 12:18:44 +05:30
|
|
|
function toRawDateTime(value: DocValue, field: Field): string | null {
|
|
|
|
if (value === null) {
|
|
|
|
return null;
|
2022-04-23 14:53:44 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof value === 'string') {
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (value instanceof Date) {
|
|
|
|
return (value as Date).toISOString();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (value instanceof DateTime) {
|
|
|
|
return (value as DateTime).toISO();
|
|
|
|
}
|
|
|
|
|
|
|
|
throwError(value, field, 'raw');
|
|
|
|
}
|
|
|
|
|
|
|
|
function toRawCheck(value: DocValue, field: Field): number {
|
|
|
|
if (typeof value === 'number') {
|
|
|
|
value = Boolean(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof value === 'boolean') {
|
|
|
|
return Number(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
throwError(value, field, 'raw');
|
|
|
|
}
|
|
|
|
|
2022-04-24 12:18:44 +05:30
|
|
|
function toRawString(value: DocValue, field: Field): string | null {
|
|
|
|
if (value === null) {
|
|
|
|
return null;
|
2022-04-23 14:53:44 +05:30
|
|
|
}
|
|
|
|
|
2022-05-11 16:14:31 +05:30
|
|
|
if (value === undefined) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2022-04-23 14:53:44 +05:30
|
|
|
if (typeof value === 'string') {
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
throwError(value, field, 'raw');
|
|
|
|
}
|
|
|
|
|
2022-04-28 18:07:07 +05:30
|
|
|
function toRawLink(value: DocValue, field: Field): string | null {
|
|
|
|
if (value === null || !(value as string)?.length) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof value === 'string') {
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
throwError(value, field, 'raw');
|
|
|
|
}
|
|
|
|
|
2022-04-23 14:53:44 +05:30
|
|
|
function throwError<T>(value: T, field: Field, type: 'raw' | 'doc'): never {
|
|
|
|
throw new ValueError(
|
|
|
|
`invalid ${type} conversion '${value}' of type ${typeof value} found, field: ${JSON.stringify(
|
|
|
|
field
|
|
|
|
)}`
|
|
|
|
);
|
|
|
|
}
|