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