2
0
mirror of https://github.com/frappe/books.git synced 2025-01-26 00:28:25 +00:00
books/src/dataImport.ts

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

450 lines
10 KiB
TypeScript
Raw Normal View History

2022-04-20 12:08:47 +05:30
import { t } from 'fyo';
import { DocValueMap } from 'fyo/core/types';
import Doc from 'fyo/model/doc';
import { isNameAutoSet } from 'fyo/model/naming';
2022-04-20 12:08:47 +05:30
import { Noun, Verb } from 'fyo/telemetry/types';
import { FieldType, FieldTypeEnum } from 'schemas/types';
import { parseCSV } from '../utils/csvParser';
2022-04-20 12:08:47 +05:30
import { fyo } from './initFyo';
2022-02-21 17:57:01 +05:30
2022-02-21 16:26:57 +05:30
export const importable = [
'SalesInvoice',
'PurchaseInvoice',
'Payment',
'JournalEntry',
'Customer',
'Supplier',
'Item',
];
2022-02-21 17:57:01 +05:30
type Status = {
success: boolean;
message: string;
names: string[];
};
2022-02-22 14:13:56 +05:30
type Exclusion = {
[key: string]: string[];
};
type Map = Record<string, unknown>;
type ObjectMap = Record<string, Map>;
type LabelTemplateFieldMap = {
[key: string]: TemplateField;
2022-02-22 14:13:56 +05:30
};
2022-03-01 14:16:12 +05:30
type LoadingStatusCallback = (
isMakingEntries: boolean,
entriesMade: number,
totalEntries: number
) => void;
2022-02-21 17:57:01 +05:30
interface TemplateField {
label: string;
fieldname: string;
required: boolean;
2022-02-23 12:52:14 +05:30
doctype: string;
options?: string[];
fieldtype: FieldType;
parentField: string;
}
function formatValue(value: string, fieldtype: FieldType): unknown {
switch (fieldtype) {
case FieldTypeEnum.Date:
if (value === '') {
return '';
}
return new Date(value);
case FieldTypeEnum.Currency:
// @ts-ignore
2022-02-25 14:14:25 +05:30
return frappe.pesa(value || 0);
case FieldTypeEnum.Int:
case FieldTypeEnum.Float: {
const n = parseFloat(value);
if (!Number.isNaN(n)) {
return n;
}
return 0;
}
default:
return value;
}
2022-02-21 17:57:01 +05:30
}
2022-02-22 14:13:56 +05:30
const exclusion: Exclusion = {
Item: ['image'],
2022-02-23 12:52:14 +05:30
Supplier: ['address', 'outstandingAmount', 'supplier', 'image', 'customer'],
Customer: ['address', 'outstandingAmount', 'supplier', 'image', 'customer'],
2022-02-21 17:57:01 +05:30
};
function getFilteredDocFields(
df: string | string[]
): [TemplateField[], string[][]] {
let doctype = df[0];
let parentField = df[1] ?? '';
if (typeof df === 'string') {
doctype = df;
parentField = '';
}
2022-02-21 17:57:01 +05:30
// @ts-ignore
const primaryFields: Field[] = frappe.models[doctype].fields;
const fields: TemplateField[] = [];
const tableTypes: string[][] = [];
2022-02-23 12:52:14 +05:30
const exclusionFields: string[] = exclusion[doctype] ?? [];
2022-02-21 17:57:01 +05:30
primaryFields.forEach(
2022-02-23 12:52:14 +05:30
({
label,
fieldtype,
childtype,
fieldname,
readOnly,
required,
hidden,
options,
}) => {
if (
!(fieldname === 'name' && !parentField) &&
(readOnly ||
(hidden && typeof hidden === 'number') ||
exclusionFields.includes(fieldname))
2022-02-23 12:52:14 +05:30
) {
2022-02-22 14:13:56 +05:30
return;
}
if (fieldtype === FieldTypeEnum.Table && childtype) {
tableTypes.push([childtype, fieldname]);
2022-02-23 12:52:14 +05:30
return;
2022-02-21 17:57:01 +05:30
}
fields.push({
label,
fieldname,
2022-02-23 12:52:14 +05:30
doctype,
options,
fieldtype,
parentField,
2022-02-21 17:57:01 +05:30
required: Boolean(required ?? false),
});
}
);
2022-02-23 12:52:14 +05:30
return [fields, tableTypes];
}
2022-02-21 17:57:01 +05:30
2022-02-23 12:52:14 +05:30
function getTemplateFields(doctype: string): TemplateField[] {
const fields: TemplateField[] = [];
if (!doctype) {
return [];
}
const doctypes: string[][] = [[doctype]];
2022-02-23 12:52:14 +05:30
while (doctypes.length > 0) {
const dt = doctypes.pop();
if (!dt) {
break;
}
2022-02-21 17:57:01 +05:30
2022-02-23 12:52:14 +05:30
const [templateFields, tableTypes] = getFilteredDocFields(dt);
fields.push(...templateFields);
doctypes.push(...tableTypes);
}
2022-02-21 17:57:01 +05:30
return fields;
}
2022-02-22 14:13:56 +05:30
function getLabelFieldMap(templateFields: TemplateField[]): Map {
const map: Map = {};
2022-02-21 17:57:01 +05:30
templateFields.reduce((acc, tf) => {
const key = tf.label as string;
acc[key] = tf.fieldname;
return acc;
}, map);
return map;
}
function getTemplate(templateFields: TemplateField[]): string {
const labels = templateFields.map(({ label }) => `"${label}"`).join(',');
return [labels, ''].join('\n');
}
export class Importer {
doctype: string;
templateFields: TemplateField[];
2022-02-22 14:13:56 +05:30
map: Map;
template: string;
2022-02-22 18:33:21 +05:30
indices: number[] = [];
2022-02-22 14:13:56 +05:30
parsedLabels: string[] = [];
2022-02-22 18:33:21 +05:30
parsedValues: string[][] = [];
2022-02-22 14:13:56 +05:30
assignedMap: Map = {}; // target: import
2022-02-23 11:33:15 +05:30
requiredMap: Map = {};
labelTemplateFieldMap: LabelTemplateFieldMap = {};
shouldSubmit: boolean = false;
2022-02-23 17:15:20 +05:30
labelIndex: number = -1;
csv: string[][] = [];
2022-02-21 17:57:01 +05:30
constructor(doctype: string) {
this.doctype = doctype;
this.templateFields = getTemplateFields(doctype);
2022-02-22 14:13:56 +05:30
this.map = getLabelFieldMap(this.templateFields);
this.template = getTemplate(this.templateFields);
this.assignedMap = this.assignableLabels.reduce((acc: Map, k) => {
acc[k] = '';
return acc;
}, {});
2022-02-23 11:33:15 +05:30
this.requiredMap = this.templateFields.reduce((acc: Map, k) => {
acc[k.label] = k.required;
return acc;
}, {});
this.labelTemplateFieldMap = this.templateFields.reduce(
(acc: LabelTemplateFieldMap, k) => {
acc[k.label] = k;
return acc;
},
{}
);
2022-02-22 14:13:56 +05:30
}
get assignableLabels() {
2022-02-23 11:33:15 +05:30
const req: string[] = [];
const nreq: string[] = [];
Object.keys(this.map).forEach((k) => {
if (this.requiredMap[k]) {
req.push(k);
return;
}
nreq.push(k);
});
return [...req, ...nreq];
2022-02-22 14:13:56 +05:30
}
get unassignedLabels() {
const assigned = Object.keys(this.assignedMap).map(
(k) => this.assignedMap[k]
);
return this.parsedLabels.filter((l) => !assigned.includes(l));
2022-02-21 17:57:01 +05:30
}
2022-02-22 18:33:21 +05:30
get columnLabels() {
2022-02-23 11:33:15 +05:30
const req: string[] = [];
const nreq: string[] = [];
2022-02-22 14:13:56 +05:30
2022-02-22 18:33:21 +05:30
this.assignableLabels.forEach((k) => {
2022-02-23 11:33:15 +05:30
if (!this.assignedMap[k]) {
2022-02-22 14:13:56 +05:30
return;
}
2022-02-23 11:33:15 +05:30
if (this.requiredMap[k]) {
req.push(k);
return;
}
nreq.push(k);
2022-02-22 14:13:56 +05:30
});
2022-02-23 11:33:15 +05:30
return [...req, ...nreq];
2022-02-22 18:33:21 +05:30
}
get assignedMatrix() {
this.indices = this.columnLabels
.map((k) => this.assignedMap[k])
.filter(Boolean)
2022-02-23 11:33:15 +05:30
.map((k) => this.parsedLabels.indexOf(k as string));
2022-02-22 18:33:21 +05:30
const rows = this.parsedValues.length;
const cols = this.columnLabels.length;
const matrix = [];
for (let i = 0; i < rows; i++) {
const row = [];
for (let j = 0; j < cols; j++) {
const ix = this.indices[j];
const value = this.parsedValues[i][ix] ?? '';
row.push(value);
}
matrix.push(row);
}
return matrix;
}
dropRow(i: number) {
this.parsedValues = this.parsedValues.filter((_, ix) => i !== ix);
}
updateValue(value: string, i: number, j: number) {
this.parsedValues[i][this.indices[j]] = value ?? '';
2022-02-21 17:57:01 +05:30
}
2022-02-22 14:13:56 +05:30
selectFile(text: string): boolean {
2022-02-23 17:15:20 +05:30
this.csv = parseCSV(text);
try {
this.initialize(0, true);
} catch (err) {
return false;
}
2022-02-23 17:15:20 +05:30
return true;
}
2022-02-22 14:13:56 +05:30
2022-02-23 17:15:20 +05:30
initialize(labelIndex: number, force: boolean) {
if (
(typeof labelIndex !== 'number' && !labelIndex) ||
(labelIndex === this.labelIndex && !force)
) {
return;
2022-02-22 14:13:56 +05:30
}
2022-02-23 17:15:20 +05:30
const source = this.csv.map((row) => [...row]);
2022-02-23 17:15:20 +05:30
this.labelIndex = labelIndex;
this.parsedLabels = source[labelIndex];
this.parsedValues = source.slice(labelIndex + 1);
2022-02-23 17:15:20 +05:30
this.setAssigned();
2022-02-22 14:13:56 +05:30
}
2022-02-23 17:15:20 +05:30
setAssigned() {
2022-02-22 14:13:56 +05:30
const labels = [...this.parsedLabels];
2022-02-23 17:15:20 +05:30
for (const k of Object.keys(this.assignedMap)) {
const l = this.assignedMap[k] as string;
if (!labels.includes(l)) {
this.assignedMap[k] = '';
}
}
2022-02-22 14:13:56 +05:30
labels.forEach((l) => {
if (this.assignedMap[l] !== '') {
return;
}
this.assignedMap[l] = l;
});
2022-02-21 17:57:01 +05:30
}
getDocs(): Map[] {
const fields = this.columnLabels.map((k) => this.labelTemplateFieldMap[k]);
const nameIndex = fields.findIndex(({ fieldname }) => fieldname === 'name');
const docMap: ObjectMap = {};
2022-02-28 18:43:29 +05:30
const assignedMatrix = this.assignedMatrix;
for (let r = 0; r < assignedMatrix.length; r++) {
const row = assignedMatrix[r];
const cts: ObjectMap = {};
const name = row[nameIndex];
docMap[name] ??= {};
for (let f = 0; f < fields.length; f++) {
const field = fields[f];
const value = formatValue(row[f], field.fieldtype);
if (field.parentField) {
cts[field.parentField] ??= {};
cts[field.parentField][field.fieldname] = value;
continue;
}
docMap[name][field.fieldname] = value;
}
for (const k of Object.keys(cts)) {
docMap[name][k] ??= [];
(docMap[name][k] as Map[]).push(cts[k]);
}
}
return Object.keys(docMap).map((k) => docMap[k]);
}
2022-03-01 14:16:12 +05:30
async importData(setLoadingStatus: LoadingStatusCallback): Promise<Status> {
const status: Status = { success: false, names: [], message: '' };
2022-04-20 12:08:47 +05:30
const shouldDeleteName = isNameAutoSet(this.doctype, fyo);
2022-03-01 14:16:12 +05:30
const docObjs = this.getDocs();
let entriesMade = 0;
setLoadingStatus(true, 0, docObjs.length);
2022-03-01 14:16:12 +05:30
for (const docObj of docObjs) {
if (shouldDeleteName) {
delete docObj.name;
}
2022-02-25 15:22:28 +05:30
for (const key in docObj) {
if (docObj[key] !== '') {
continue;
}
2022-02-25 15:22:28 +05:30
delete docObj[key];
}
2022-04-20 12:08:47 +05:30
const doc: Doc = fyo.doc.getEmptyDoc(this.doctype, false);
try {
await this.makeEntry(doc, docObj);
2022-03-01 14:16:12 +05:30
entriesMade += 1;
setLoadingStatus(true, entriesMade, docObjs.length);
} catch (err) {
2022-03-01 14:16:12 +05:30
setLoadingStatus(false, entriesMade, docObjs.length);
2022-03-09 16:18:26 +05:30
2022-04-20 12:08:47 +05:30
fyo.telemetry.log(Verb.Imported, this.doctype as Noun, {
2022-03-09 16:18:26 +05:30
success: false,
count: entriesMade,
});
return this.handleError(doc, err as Error, status);
}
status.names.push(doc.name!);
}
2022-03-01 14:16:12 +05:30
setLoadingStatus(false, entriesMade, docObjs.length);
status.success = true;
2022-03-09 16:18:26 +05:30
2022-04-20 12:08:47 +05:30
fyo.telemetry.log(Verb.Imported, this.doctype as Noun, {
2022-03-09 16:18:26 +05:30
success: true,
count: entriesMade,
});
return status;
}
addRow() {
const emptyRow = Array(this.columnLabels.length).fill('');
this.parsedValues.push(emptyRow);
}
async makeEntry(doc: Doc, docObj: Map) {
await doc.setMultiple(docObj as DocValueMap);
await doc.insert();
if (this.shouldSubmit) {
await doc.submit();
}
}
handleError(doc: Doc, err: Error, status: Status): Status {
2022-04-20 12:08:47 +05:30
const messages = [t`Could not import ${this.doctype} ${doc.name!}.`];
const message = err.message;
if (message?.includes('UNIQUE constraint failed')) {
2022-04-20 12:08:47 +05:30
messages.push(t`${doc.name!} already exists.`);
} else if (message) {
messages.push(message);
}
if (status.names.length) {
messages.push(
2022-04-20 12:08:47 +05:30
t`The following ${
status.names.length
} entries were created: ${status.names.join(', ')}`
);
}
status.message = messages.join(' ');
return status;
}
2022-02-21 17:57:01 +05:30
}