2022-04-19 05:59:36 +00:00
|
|
|
import { Fyo } from 'fyo';
|
2022-08-29 12:35:33 +00:00
|
|
|
import { DocValue, DocValueMap } from 'fyo/core/types';
|
2022-04-19 05:59:36 +00:00
|
|
|
import { isPesa } from 'fyo/utils';
|
2022-08-29 12:35:33 +00:00
|
|
|
import { DuplicateEntryError } from 'fyo/utils/errors';
|
2022-04-01 09:35:51 +00:00
|
|
|
import { isEqual } from 'lodash';
|
2022-05-23 05:30:54 +00:00
|
|
|
import { Money } from 'pesa';
|
2022-04-01 09:35:51 +00:00
|
|
|
import { Field, FieldType, FieldTypeEnum } from 'schemas/types';
|
|
|
|
import { getIsNullOrUndef } from 'utils';
|
2022-04-24 06:48:44 +00:00
|
|
|
import { Doc } from './doc';
|
2022-04-01 09:35:51 +00:00
|
|
|
|
|
|
|
export function areDocValuesEqual(
|
|
|
|
dvOne: DocValue | Doc[],
|
|
|
|
dvTwo: DocValue | Doc[]
|
|
|
|
): boolean {
|
|
|
|
if (['string', 'number'].includes(typeof dvOne) || dvOne instanceof Date) {
|
|
|
|
return dvOne === dvTwo;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isPesa(dvOne)) {
|
|
|
|
try {
|
|
|
|
return (dvOne as Money).eq(dvTwo as string | number);
|
|
|
|
} catch {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return isEqual(dvOne, dvTwo);
|
|
|
|
}
|
|
|
|
|
2022-04-18 11:29:20 +00:00
|
|
|
export function getPreDefaultValues(
|
|
|
|
fieldtype: FieldType,
|
2022-04-19 05:59:36 +00:00
|
|
|
fyo: Fyo
|
2022-04-18 11:29:20 +00:00
|
|
|
): DocValue | Doc[] {
|
2022-04-01 09:35:51 +00:00
|
|
|
switch (fieldtype) {
|
|
|
|
case FieldTypeEnum.Table:
|
|
|
|
return [] as Doc[];
|
|
|
|
case FieldTypeEnum.Currency:
|
2022-04-19 05:59:36 +00:00
|
|
|
return fyo.pesa!(0.0);
|
2022-04-01 09:35:51 +00:00
|
|
|
case FieldTypeEnum.Int:
|
|
|
|
case FieldTypeEnum.Float:
|
|
|
|
return 0;
|
|
|
|
default:
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getMissingMandatoryMessage(doc: Doc) {
|
|
|
|
const mandatoryFields = getMandatory(doc);
|
|
|
|
const message = mandatoryFields
|
|
|
|
.filter((f) => {
|
|
|
|
const value = doc.get(f.fieldname);
|
|
|
|
const isNullOrUndef = getIsNullOrUndef(value);
|
|
|
|
|
|
|
|
if (f.fieldtype === FieldTypeEnum.Table) {
|
|
|
|
return isNullOrUndef || (value as Doc[])?.length === 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return isNullOrUndef || value === '';
|
|
|
|
})
|
2022-04-25 06:33:31 +00:00
|
|
|
.map((f) => f.label ?? f.fieldname)
|
2022-04-01 09:35:51 +00:00
|
|
|
.join(', ');
|
|
|
|
|
2022-04-25 06:33:31 +00:00
|
|
|
if (message && doc.schema.isChild && doc.parentdoc && doc.parentFieldname) {
|
|
|
|
const parentfield = doc.parentdoc.fieldMap[doc.parentFieldname];
|
2022-04-01 09:35:51 +00:00
|
|
|
return `${parentfield.label} Row ${(doc.idx ?? 0) + 1}: ${message}`;
|
|
|
|
}
|
|
|
|
|
|
|
|
return message;
|
|
|
|
}
|
|
|
|
|
|
|
|
function getMandatory(doc: Doc): Field[] {
|
|
|
|
const mandatoryFields: Field[] = [];
|
|
|
|
for (const field of doc.schema.fields) {
|
|
|
|
if (field.required) {
|
|
|
|
mandatoryFields.push(field);
|
|
|
|
}
|
|
|
|
|
|
|
|
const requiredFunction = doc.required[field.fieldname];
|
2022-04-25 06:33:31 +00:00
|
|
|
if (requiredFunction?.()) {
|
2022-04-01 09:35:51 +00:00
|
|
|
mandatoryFields.push(field);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return mandatoryFields;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function shouldApplyFormula(field: Field, doc: Doc, fieldname?: string) {
|
2022-04-23 09:23:44 +00:00
|
|
|
if (!doc.formulas[field.fieldname]) {
|
2022-04-01 09:35:51 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (field.readOnly) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2022-04-29 13:00:24 +00:00
|
|
|
const { dependsOn } = doc.formulas[field.fieldname] ?? {};
|
|
|
|
if (dependsOn === undefined) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dependsOn.length === 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-04-01 09:35:51 +00:00
|
|
|
if (fieldname && dependsOn.includes(fieldname)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
const value = doc.get(field.fieldname);
|
|
|
|
return getIsNullOrUndef(value);
|
|
|
|
}
|
2022-05-01 07:39:03 +00:00
|
|
|
|
|
|
|
export function setChildDocIdx(childDocs: Doc[]) {
|
|
|
|
for (const idx in childDocs) {
|
|
|
|
childDocs[idx].idx = +idx;
|
|
|
|
}
|
|
|
|
}
|
2022-08-29 12:35:33 +00:00
|
|
|
|
|
|
|
export function getInsertionError(err: Error, validDict: DocValueMap): Error {
|
|
|
|
if (err.message.includes('UNIQUE constraint failed:')) {
|
|
|
|
return getDuplicateEntryError(err, validDict);
|
|
|
|
}
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getDuplicateEntryError(
|
|
|
|
err: Error,
|
|
|
|
validDict: DocValueMap
|
|
|
|
): Error | DuplicateEntryError {
|
|
|
|
const matches = err.message.match(/UNIQUE constraint failed:\s(\w+)\.(\w+)$/);
|
|
|
|
if (!matches) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
const schemaName = matches[1];
|
|
|
|
const fieldname = matches[2];
|
|
|
|
if (!schemaName || !fieldname) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
const duplicateEntryError = new DuplicateEntryError(err.message, false);
|
|
|
|
duplicateEntryError.stack = err.stack;
|
|
|
|
duplicateEntryError.more = {
|
|
|
|
schemaName,
|
|
|
|
fieldname,
|
|
|
|
value: validDict[fieldname],
|
|
|
|
};
|
|
|
|
|
|
|
|
return duplicateEntryError;
|
|
|
|
}
|