mirror of
https://github.com/arc53/DocsGPT.git
synced 2025-11-29 16:43:16 +00:00
897 lines
27 KiB
TypeScript
897 lines
27 KiB
TypeScript
import { useCallback, useState } from 'react';
|
|
import { nanoid } from '@reduxjs/toolkit';
|
|
import { useDropzone } from 'react-dropzone';
|
|
import { useTranslation } from 'react-i18next';
|
|
import { useDispatch, useSelector } from 'react-redux';
|
|
|
|
import userService from '../api/services/userService';
|
|
import { getSessionToken } from '../utils/providerUtils';
|
|
import Dropdown from '../components/Dropdown';
|
|
import Input from '../components/Input';
|
|
import ToggleSwitch from '../components/ToggleSwitch';
|
|
import WrapperModal from '../modals/WrapperModal';
|
|
import { ActiveState, Doc } from '../models/misc';
|
|
|
|
import { getDocs } from '../preferences/preferenceApi';
|
|
import {
|
|
selectSourceDocs,
|
|
selectToken,
|
|
setSelectedDocs,
|
|
setSourceDocs,
|
|
} from '../preferences/preferenceSlice';
|
|
import {
|
|
IngestorDefaultConfigs,
|
|
IngestorFormSchemas,
|
|
getIngestorSchema,
|
|
IngestorOption,
|
|
} from '../upload/types/ingestor';
|
|
import { addUploadTask, updateUploadTask } from './uploadSlice';
|
|
|
|
import { FormField, IngestorConfig, IngestorType } from './types/ingestor';
|
|
|
|
import { FilePicker } from '../components/FilePicker';
|
|
import GoogleDrivePicker from '../components/GoogleDrivePicker';
|
|
|
|
import ChevronRight from '../assets/chevron-right.svg';
|
|
|
|
function Upload({
|
|
receivedFile = [],
|
|
setModalState,
|
|
isOnboarding,
|
|
renderTab = null,
|
|
close,
|
|
onSuccessfulUpload = () => undefined,
|
|
}: {
|
|
receivedFile: File[];
|
|
setModalState: (state: ActiveState) => void;
|
|
isOnboarding: boolean;
|
|
renderTab: string | null;
|
|
close: () => void;
|
|
onSuccessfulUpload?: () => void;
|
|
}) {
|
|
const token = useSelector(selectToken);
|
|
|
|
const [files, setfiles] = useState<File[]>(receivedFile);
|
|
const [activeTab, setActiveTab] = useState<boolean>(true);
|
|
const [showAdvancedOptions, setShowAdvancedOptions] = useState(false);
|
|
|
|
// File picker state
|
|
const [selectedFiles, setSelectedFiles] = useState<string[]>([]);
|
|
const [selectedFolders, setSelectedFolders] = useState<string[]>([]);
|
|
|
|
const renderFormFields = () => {
|
|
if (!ingestor.type) return null;
|
|
const ingestorSchema = getIngestorSchema(ingestor.type as IngestorType);
|
|
if (!ingestorSchema) return null;
|
|
const schema: FormField[] = ingestorSchema.fields;
|
|
|
|
const generalFields = schema.filter((field: FormField) => !field.advanced);
|
|
const advancedFields = schema.filter((field: FormField) => field.advanced);
|
|
|
|
return (
|
|
<div className="flex flex-col gap-4">
|
|
<div className="flex flex-col gap-4">
|
|
{generalFields.map((field: FormField) => renderField(field))}
|
|
</div>
|
|
|
|
{advancedFields.length > 0 && (
|
|
<div
|
|
className={`grid transition-all duration-300 ease-in-out ${
|
|
showAdvancedOptions
|
|
? 'grid-rows-[1fr] opacity-100'
|
|
: 'grid-rows-[0fr] opacity-0'
|
|
}`}
|
|
>
|
|
<div className="flex flex-col gap-4 overflow-hidden">
|
|
<hr className="my-4 border border-[#C4C4C4]/40" />
|
|
<div className="flex flex-col gap-4">
|
|
{advancedFields.map((field: FormField) => renderField(field))}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
)}
|
|
</div>
|
|
);
|
|
};
|
|
|
|
const renderField = (field: FormField) => {
|
|
const isRequired = field.required ?? false;
|
|
switch (field.type) {
|
|
case 'string':
|
|
return (
|
|
<Input
|
|
key={field.name}
|
|
placeholder={field.label}
|
|
type="text"
|
|
name={field.name}
|
|
value={String(
|
|
ingestor.config[field.name as keyof typeof ingestor.config],
|
|
)}
|
|
onChange={(e) =>
|
|
handleIngestorChange(
|
|
field.name as keyof IngestorConfig['config'],
|
|
e.target.value,
|
|
)
|
|
}
|
|
borderVariant="thin"
|
|
required={isRequired}
|
|
colorVariant="silver"
|
|
labelBgClassName="bg-white dark:bg-charleston-green-2"
|
|
/>
|
|
);
|
|
case 'number':
|
|
return (
|
|
<Input
|
|
key={field.name}
|
|
placeholder={field.label}
|
|
type="number"
|
|
name={field.name}
|
|
value={String(
|
|
ingestor.config[field.name as keyof typeof ingestor.config],
|
|
)}
|
|
onChange={(e) =>
|
|
handleIngestorChange(
|
|
field.name as keyof IngestorConfig['config'],
|
|
Number(e.target.value),
|
|
)
|
|
}
|
|
borderVariant="thin"
|
|
required={isRequired}
|
|
colorVariant="silver"
|
|
labelBgClassName="bg-white dark:bg-charleston-green-2"
|
|
/>
|
|
);
|
|
case 'enum':
|
|
return (
|
|
<Dropdown
|
|
key={field.name}
|
|
options={field.options || []}
|
|
selectedValue={
|
|
field.options?.find(
|
|
(opt) =>
|
|
opt.value ===
|
|
ingestor.config[field.name as keyof typeof ingestor.config],
|
|
) || null
|
|
}
|
|
onSelect={(selected: { label: string; value: string }) => {
|
|
handleIngestorChange(
|
|
field.name as keyof IngestorConfig['config'],
|
|
selected.value,
|
|
);
|
|
}}
|
|
size="w-full"
|
|
rounded="3xl"
|
|
placeholder={field.label}
|
|
border="border"
|
|
buttonClassName="border-silver bg-white dark:border-dim-gray dark:bg-[#222327]"
|
|
optionsClassName="border-silver bg-white dark:border-dim-gray dark:bg-[#383838]"
|
|
placeholderClassName="text-gray-400 dark:text-silver"
|
|
contentSize="text-sm"
|
|
/>
|
|
);
|
|
case 'boolean':
|
|
return (
|
|
<ToggleSwitch
|
|
key={field.name}
|
|
label={field.label}
|
|
checked={Boolean(
|
|
ingestor.config[field.name as keyof typeof ingestor.config],
|
|
)}
|
|
onChange={(checked: boolean) => {
|
|
handleIngestorChange(
|
|
field.name as keyof IngestorConfig['config'],
|
|
checked,
|
|
);
|
|
}}
|
|
size="small"
|
|
className={`mt-2 text-base`}
|
|
/>
|
|
);
|
|
case 'local_file_picker':
|
|
return (
|
|
<div key={field.name}>
|
|
<div className="mb-3" {...getRootProps()}>
|
|
<span className="text-purple-30 dark:text-silver inline-block rounded-3xl border border-[#7F7F82] bg-transparent px-4 py-2 font-medium hover:cursor-pointer">
|
|
<input type="button" {...getInputProps()} />
|
|
{t('modals.uploadDoc.choose')}
|
|
</span>
|
|
</div>
|
|
<div className="mt-4 max-w-full">
|
|
<p className="text-eerie-black dark:text-light-gray mb-[14px] text-[14px] font-medium">
|
|
{t('modals.uploadDoc.selectedFiles')}
|
|
</p>
|
|
<div className="max-w-full overflow-hidden">
|
|
{files.map((file) => (
|
|
<p
|
|
key={file.name}
|
|
className="text-gray-6000 truncate overflow-hidden text-ellipsis dark:text-[#ececf1]"
|
|
title={file.name}
|
|
>
|
|
{file.name}
|
|
</p>
|
|
))}
|
|
{files.length === 0 && (
|
|
<p className="text-gray-6000 dark:text-light-gray text-[14px]">
|
|
{t('modals.uploadDoc.noFilesSelected')}
|
|
</p>
|
|
)}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
case 'remote_file_picker':
|
|
return (
|
|
<FilePicker
|
|
key={field.name}
|
|
onSelectionChange={(
|
|
selectedFileIds: string[],
|
|
selectedFolderIds: string[] = [],
|
|
) => {
|
|
setSelectedFiles(selectedFileIds);
|
|
setSelectedFolders(selectedFolderIds);
|
|
}}
|
|
provider={ingestor.type as unknown as string}
|
|
token={token}
|
|
initialSelectedFiles={selectedFiles}
|
|
initialSelectedFolders={selectedFolders}
|
|
/>
|
|
);
|
|
case 'google_drive_picker':
|
|
return (
|
|
<GoogleDrivePicker
|
|
key={field.name}
|
|
onSelectionChange={(
|
|
selectedFileIds: string[],
|
|
selectedFolderIds: string[] = [],
|
|
) => {
|
|
setSelectedFiles(selectedFileIds);
|
|
setSelectedFolders(selectedFolderIds);
|
|
}}
|
|
token={token}
|
|
/>
|
|
);
|
|
default:
|
|
return null;
|
|
}
|
|
};
|
|
|
|
// New unified ingestor state
|
|
const [ingestor, setIngestor] = useState<IngestorConfig>(() => ({
|
|
type: null,
|
|
name: '',
|
|
config: {},
|
|
}));
|
|
|
|
const { t } = useTranslation();
|
|
const dispatch = useDispatch();
|
|
|
|
const ingestorOptions: IngestorOption[] = IngestorFormSchemas.filter(
|
|
(schema) => (schema.validate ? schema.validate() : true),
|
|
).map((schema) => ({
|
|
label: schema.label,
|
|
value: schema.key,
|
|
icon: schema.icon,
|
|
heading: schema.heading,
|
|
}));
|
|
|
|
const sourceDocs = useSelector(selectSourceDocs);
|
|
|
|
const resetUploaderState = useCallback(() => {
|
|
setIngestor({ type: null, name: '', config: {} });
|
|
setfiles([]);
|
|
setSelectedFiles([]);
|
|
setSelectedFolders([]);
|
|
setShowAdvancedOptions(false);
|
|
}, []);
|
|
|
|
const handleTaskFailure = useCallback(
|
|
(clientTaskId: string, errorMessage?: string) => {
|
|
dispatch(
|
|
updateUploadTask({
|
|
id: clientTaskId,
|
|
updates: {
|
|
status: 'failed',
|
|
errorMessage: errorMessage || t('attachments.uploadFailed'),
|
|
},
|
|
}),
|
|
);
|
|
},
|
|
[dispatch, t],
|
|
);
|
|
|
|
const trackTraining = useCallback(
|
|
(backendTaskId: string, clientTaskId: string) => {
|
|
let timeoutId: number | null = null;
|
|
|
|
const poll = () => {
|
|
userService
|
|
.getTaskStatus(backendTaskId, null)
|
|
.then((response) => response.json())
|
|
.then(async (data) => {
|
|
if (data.status === 'SUCCESS') {
|
|
if (timeoutId !== null) {
|
|
clearTimeout(timeoutId);
|
|
timeoutId = null;
|
|
}
|
|
|
|
const docs = await getDocs(token);
|
|
dispatch(setSourceDocs(docs));
|
|
|
|
if (Array.isArray(docs)) {
|
|
const existingDocIds = new Set(
|
|
(Array.isArray(sourceDocs) ? sourceDocs : [])
|
|
.map((doc: Doc) => doc?.id)
|
|
.filter((id): id is string => Boolean(id)),
|
|
);
|
|
const newDoc = docs.find(
|
|
(doc: Doc) => doc.id && !existingDocIds.has(doc.id),
|
|
);
|
|
if (newDoc) {
|
|
dispatch(setSelectedDocs([newDoc]));
|
|
}
|
|
}
|
|
|
|
if (data.result?.limited) {
|
|
dispatch(
|
|
updateUploadTask({
|
|
id: clientTaskId,
|
|
updates: {
|
|
status: 'failed',
|
|
progress: 100,
|
|
errorMessage: t('modals.uploadDoc.progress.tokenLimit'),
|
|
},
|
|
}),
|
|
);
|
|
} else {
|
|
dispatch(
|
|
updateUploadTask({
|
|
id: clientTaskId,
|
|
updates: {
|
|
status: 'completed',
|
|
progress: 100,
|
|
errorMessage: undefined,
|
|
},
|
|
}),
|
|
);
|
|
onSuccessfulUpload?.();
|
|
}
|
|
} else if (data.status === 'FAILURE') {
|
|
if (timeoutId !== null) {
|
|
clearTimeout(timeoutId);
|
|
timeoutId = null;
|
|
}
|
|
handleTaskFailure(clientTaskId, data.result?.message);
|
|
} else if (data.status === 'PROGRESS') {
|
|
dispatch(
|
|
updateUploadTask({
|
|
id: clientTaskId,
|
|
updates: {
|
|
status: 'training',
|
|
progress: Math.min(100, data.result?.current ?? 0),
|
|
},
|
|
}),
|
|
);
|
|
timeoutId = window.setTimeout(poll, 5000);
|
|
} else {
|
|
timeoutId = window.setTimeout(poll, 5000);
|
|
}
|
|
})
|
|
.catch(() => {
|
|
if (timeoutId !== null) {
|
|
clearTimeout(timeoutId);
|
|
timeoutId = null;
|
|
}
|
|
handleTaskFailure(clientTaskId);
|
|
});
|
|
};
|
|
|
|
timeoutId = window.setTimeout(poll, 3000);
|
|
},
|
|
[dispatch, handleTaskFailure, onSuccessfulUpload, sourceDocs, t, token],
|
|
);
|
|
|
|
const onDrop = useCallback(
|
|
(acceptedFiles: File[]) => {
|
|
setfiles(acceptedFiles);
|
|
setIngestor((prev) => ({ ...prev, name: acceptedFiles[0]?.name || '' }));
|
|
|
|
// If we're in local_file mode, update the ingestor config
|
|
if (ingestor.type === 'local_file') {
|
|
setIngestor((prevState) => ({
|
|
...prevState,
|
|
config: {
|
|
...prevState.config,
|
|
files: acceptedFiles,
|
|
},
|
|
}));
|
|
}
|
|
},
|
|
[ingestor.type],
|
|
);
|
|
|
|
const doNothing = () => undefined;
|
|
|
|
const uploadFile = (clientTaskId: string) => {
|
|
const formData = new FormData();
|
|
files.forEach((file) => {
|
|
formData.append('file', file);
|
|
});
|
|
|
|
formData.append('name', ingestor.name);
|
|
formData.append('user', 'local');
|
|
|
|
const apiHost = import.meta.env.VITE_API_HOST;
|
|
const xhr = new XMLHttpRequest();
|
|
|
|
dispatch(
|
|
updateUploadTask({
|
|
id: clientTaskId,
|
|
updates: { status: 'uploading', progress: 0 },
|
|
}),
|
|
);
|
|
|
|
xhr.upload.addEventListener('progress', (event) => {
|
|
if (!event.lengthComputable) return;
|
|
const progressPercentage = Number(
|
|
((event.loaded / event.total) * 100).toFixed(2),
|
|
);
|
|
dispatch(
|
|
updateUploadTask({
|
|
id: clientTaskId,
|
|
updates: { progress: progressPercentage },
|
|
}),
|
|
);
|
|
});
|
|
|
|
xhr.onload = () => {
|
|
if (xhr.status >= 200 && xhr.status < 300) {
|
|
try {
|
|
const parsed = JSON.parse(xhr.responseText) as { task_id?: string };
|
|
if (parsed.task_id) {
|
|
dispatch(
|
|
updateUploadTask({
|
|
id: clientTaskId,
|
|
updates: {
|
|
taskId: parsed.task_id,
|
|
status: 'training',
|
|
progress: 0,
|
|
},
|
|
}),
|
|
);
|
|
trackTraining(parsed.task_id, clientTaskId);
|
|
} else {
|
|
dispatch(
|
|
updateUploadTask({
|
|
id: clientTaskId,
|
|
updates: { status: 'completed', progress: 100 },
|
|
}),
|
|
);
|
|
onSuccessfulUpload?.();
|
|
}
|
|
} catch (error) {
|
|
handleTaskFailure(clientTaskId);
|
|
}
|
|
} else {
|
|
handleTaskFailure(clientTaskId, xhr.statusText || undefined);
|
|
}
|
|
};
|
|
|
|
xhr.onerror = () => {
|
|
handleTaskFailure(clientTaskId);
|
|
};
|
|
|
|
xhr.open('POST', `${apiHost}/api/upload`);
|
|
xhr.setRequestHeader('Authorization', `Bearer ${token}`);
|
|
xhr.send(formData);
|
|
};
|
|
|
|
const uploadRemote = (clientTaskId: string) => {
|
|
if (!ingestor.type) {
|
|
handleTaskFailure(clientTaskId);
|
|
return;
|
|
}
|
|
|
|
const formData = new FormData();
|
|
formData.append('name', ingestor.name);
|
|
formData.append('user', 'local');
|
|
formData.append('source', ingestor.type as string);
|
|
|
|
const ingestorSchema = getIngestorSchema(ingestor.type as IngestorType);
|
|
if (!ingestorSchema) {
|
|
handleTaskFailure(clientTaskId);
|
|
return;
|
|
}
|
|
|
|
const schema: FormField[] = ingestorSchema.fields;
|
|
const hasLocalFilePicker = schema.some(
|
|
(field: FormField) => field.type === 'local_file_picker',
|
|
);
|
|
const hasRemoteFilePicker = schema.some(
|
|
(field: FormField) => field.type === 'remote_file_picker',
|
|
);
|
|
const hasGoogleDrivePicker = schema.some(
|
|
(field: FormField) => field.type === 'google_drive_picker',
|
|
);
|
|
|
|
let configData: Record<string, unknown> = { ...ingestor.config };
|
|
|
|
if (hasLocalFilePicker) {
|
|
files.forEach((file) => {
|
|
formData.append('file', file);
|
|
});
|
|
} else if (hasRemoteFilePicker || hasGoogleDrivePicker) {
|
|
const sessionToken = getSessionToken(ingestor.type as string);
|
|
configData = {
|
|
provider: ingestor.type as string,
|
|
session_token: sessionToken,
|
|
file_ids: selectedFiles,
|
|
folder_ids: selectedFolders,
|
|
};
|
|
}
|
|
|
|
formData.append('data', JSON.stringify(configData));
|
|
|
|
const apiHost: string = import.meta.env.VITE_API_HOST;
|
|
const endpoint =
|
|
ingestor.type === 'local_file'
|
|
? `${apiHost}/api/upload`
|
|
: `${apiHost}/api/remote`;
|
|
|
|
const xhr = new XMLHttpRequest();
|
|
|
|
dispatch(
|
|
updateUploadTask({
|
|
id: clientTaskId,
|
|
updates: { status: 'uploading', progress: 0 },
|
|
}),
|
|
);
|
|
|
|
xhr.upload.addEventListener('progress', (event: ProgressEvent) => {
|
|
if (!event.lengthComputable) return;
|
|
const progressPercentage = Number(
|
|
((event.loaded / event.total) * 100).toFixed(2),
|
|
);
|
|
dispatch(
|
|
updateUploadTask({
|
|
id: clientTaskId,
|
|
updates: { progress: progressPercentage },
|
|
}),
|
|
);
|
|
});
|
|
|
|
xhr.onload = () => {
|
|
if (xhr.status >= 200 && xhr.status < 300) {
|
|
try {
|
|
const response = JSON.parse(xhr.responseText) as { task_id?: string };
|
|
if (response.task_id) {
|
|
dispatch(
|
|
updateUploadTask({
|
|
id: clientTaskId,
|
|
updates: {
|
|
taskId: response.task_id,
|
|
status: 'training',
|
|
progress: 0,
|
|
},
|
|
}),
|
|
);
|
|
trackTraining(response.task_id, clientTaskId);
|
|
} else {
|
|
dispatch(
|
|
updateUploadTask({
|
|
id: clientTaskId,
|
|
updates: { status: 'completed', progress: 100 },
|
|
}),
|
|
);
|
|
onSuccessfulUpload?.();
|
|
}
|
|
} catch (error) {
|
|
handleTaskFailure(clientTaskId);
|
|
}
|
|
} else {
|
|
handleTaskFailure(clientTaskId, xhr.statusText || undefined);
|
|
}
|
|
};
|
|
|
|
xhr.onerror = () => {
|
|
handleTaskFailure(clientTaskId);
|
|
};
|
|
|
|
xhr.open('POST', endpoint);
|
|
xhr.setRequestHeader('Authorization', `Bearer ${token}`);
|
|
xhr.send(formData);
|
|
};
|
|
|
|
const handleClose = useCallback(() => {
|
|
resetUploaderState();
|
|
setModalState('INACTIVE');
|
|
close();
|
|
}, [close, resetUploaderState, setModalState]);
|
|
|
|
const handleUpload = () => {
|
|
if (!ingestor.type) return;
|
|
|
|
const ingestorSchemaForUpload = getIngestorSchema(
|
|
ingestor.type as IngestorType,
|
|
);
|
|
if (!ingestorSchemaForUpload) return;
|
|
|
|
const schema: FormField[] = ingestorSchemaForUpload.fields;
|
|
const hasLocalFilePicker = schema.some(
|
|
(field: FormField) => field.type === 'local_file_picker',
|
|
);
|
|
|
|
const displayName =
|
|
ingestor.name?.trim() || files[0]?.name || t('modals.uploadDoc.label');
|
|
|
|
const clientTaskId = nanoid();
|
|
|
|
dispatch(
|
|
addUploadTask({
|
|
id: clientTaskId,
|
|
fileName: displayName,
|
|
progress: 0,
|
|
status: 'preparing',
|
|
}),
|
|
);
|
|
|
|
if (hasLocalFilePicker) {
|
|
uploadFile(clientTaskId);
|
|
} else {
|
|
uploadRemote(clientTaskId);
|
|
}
|
|
|
|
handleClose();
|
|
};
|
|
|
|
const { getRootProps, getInputProps } = useDropzone({
|
|
onDrop,
|
|
multiple: true,
|
|
onDragEnter: doNothing,
|
|
onDragOver: doNothing,
|
|
onDragLeave: doNothing,
|
|
maxSize: 25000000,
|
|
accept: {
|
|
'application/pdf': ['.pdf'],
|
|
'text/plain': ['.txt'],
|
|
'text/x-rst': ['.rst'],
|
|
'text/x-markdown': ['.md'],
|
|
'application/zip': ['.zip'],
|
|
'application/vnd.openxmlformats-officedocument.wordprocessingml.document':
|
|
['.docx'],
|
|
'application/json': ['.json'],
|
|
'text/csv': ['.csv'],
|
|
'text/html': ['.html'],
|
|
'application/epub+zip': ['.epub'],
|
|
'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet': [
|
|
'.xlsx',
|
|
],
|
|
'application/vnd.openxmlformats-officedocument.presentationml.presentation':
|
|
['.pptx'],
|
|
'image/png': ['.png'],
|
|
'image/jpeg': ['.jpeg'],
|
|
'image/jpg': ['.jpg'],
|
|
},
|
|
});
|
|
|
|
const isUploadDisabled = (): boolean => {
|
|
if (!activeTab) return true;
|
|
|
|
if (!ingestor.name?.trim()) {
|
|
return true;
|
|
}
|
|
|
|
if (!ingestor.type) return true;
|
|
const ingestorSchemaForValidation = getIngestorSchema(
|
|
ingestor.type as IngestorType,
|
|
);
|
|
if (!ingestorSchemaForValidation) return true;
|
|
const schema: FormField[] = ingestorSchemaForValidation.fields;
|
|
const hasLocalFilePicker = schema.some(
|
|
(field: FormField) => field.type === 'local_file_picker',
|
|
);
|
|
const hasRemoteFilePicker = schema.some(
|
|
(field: FormField) => field.type === 'remote_file_picker',
|
|
);
|
|
const hasGoogleDrivePicker = schema.some(
|
|
(field: FormField) => field.type === 'google_drive_picker',
|
|
);
|
|
|
|
if (hasLocalFilePicker) {
|
|
if (files.length === 0) {
|
|
return true;
|
|
}
|
|
} else if (hasRemoteFilePicker || hasGoogleDrivePicker) {
|
|
if (selectedFiles.length === 0 && selectedFolders.length === 0) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
const ingestorSchemaForFields = getIngestorSchema(
|
|
ingestor.type as IngestorType,
|
|
);
|
|
if (!ingestorSchemaForFields) return false;
|
|
const formFields: FormField[] = ingestorSchemaForFields.fields;
|
|
for (const field of formFields) {
|
|
if (field.required) {
|
|
// Validate only required fields
|
|
const value =
|
|
ingestor.config[field.name as keyof typeof ingestor.config];
|
|
|
|
if (typeof value === 'string' && !value.trim()) {
|
|
return true;
|
|
}
|
|
|
|
if (
|
|
typeof value === 'number' &&
|
|
(value === null || value === undefined || value <= 0)
|
|
) {
|
|
return true;
|
|
}
|
|
|
|
if (typeof value === 'boolean' && value === undefined) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
};
|
|
const handleIngestorChange = (
|
|
key: keyof IngestorConfig['config'],
|
|
value: string | number | boolean,
|
|
) => {
|
|
setIngestor((prevState) => ({
|
|
...prevState,
|
|
config: {
|
|
...prevState.config,
|
|
[key]: value,
|
|
},
|
|
}));
|
|
};
|
|
const handleIngestorTypeChange = (type: IngestorType | null) => {
|
|
if (type === null) {
|
|
setIngestor({
|
|
type: null,
|
|
name: '',
|
|
config: {},
|
|
});
|
|
setfiles([]);
|
|
return;
|
|
}
|
|
|
|
const defaultConfig = IngestorDefaultConfigs[type];
|
|
setIngestor({
|
|
type,
|
|
name: defaultConfig.name,
|
|
config: defaultConfig.config,
|
|
});
|
|
|
|
// Clear files if switching away from local_file
|
|
if (type !== 'local_file') {
|
|
setfiles([]);
|
|
}
|
|
};
|
|
|
|
const renderIngestorSelection = () => {
|
|
return (
|
|
<div className="grid w-full grid-cols-1 gap-4 sm:grid-cols-2 md:grid-cols-3">
|
|
{ingestorOptions.map((option) => (
|
|
<div
|
|
key={option.value}
|
|
className={`relative mx-auto flex h-[91.2px] w-full cursor-pointer flex-col justify-between gap-2 rounded-2xl border border-solid pt-[21.1px] pr-[21px] pb-[15px] pl-[21px] transition-colors duration-300 ease-out ${
|
|
ingestor.type === option.value
|
|
? 'border-[#7D54D1] bg-[#7D54D1] text-white'
|
|
: 'border-[#D7D7D7] bg-transparent transition-shadow duration-300 hover:bg-[#ECECEC]/30 hover:shadow-[0_0_15px_0_#00000026] dark:border-[#4A4A4A] dark:hover:bg-[#383838]/30'
|
|
}`}
|
|
onClick={() =>
|
|
handleIngestorTypeChange(option.value as IngestorType)
|
|
}
|
|
>
|
|
<div className="flex h-full flex-col justify-between">
|
|
<div className="h-6 w-6">
|
|
<img
|
|
src={option.icon}
|
|
alt={option.label}
|
|
className={`${ingestor.type === option.value ? 'invert filter' : ''} dark:invert dark:filter`}
|
|
/>
|
|
</div>
|
|
<p className="font-inter self-start text-[13px] leading-[18px] font-semibold">
|
|
{t(`modals.uploadDoc.ingestors.${option.value}.label`)}
|
|
</p>
|
|
</div>
|
|
</div>
|
|
))}
|
|
</div>
|
|
);
|
|
};
|
|
return (
|
|
<WrapperModal
|
|
close={handleClose}
|
|
className="max-h-[90vh] w-11/12 sm:max-h-none sm:w-auto sm:min-w-[600px] md:min-w-[700px]"
|
|
contentClassName="max-h-[80vh] sm:max-h-none"
|
|
>
|
|
<div className="flex w-full flex-col gap-6">
|
|
{!ingestor.type && (
|
|
<p className="font-inter text-left text-[20px] leading-[28px] font-semibold tracking-[0.15px] text-[#18181B] dark:text-[#ECECF1]">
|
|
{t('modals.uploadDoc.selectSource')}
|
|
</p>
|
|
)}
|
|
|
|
{activeTab && (
|
|
<>
|
|
{!ingestor.type && renderIngestorSelection()}
|
|
{ingestor.type && (
|
|
<div className="flex flex-col gap-4">
|
|
<button
|
|
onClick={() => handleIngestorTypeChange(null)}
|
|
className="flex w-fit items-center gap-2 text-[#777777] hover:text-[#555555]"
|
|
>
|
|
<img
|
|
src={ChevronRight}
|
|
alt="back"
|
|
className="h-3 w-3 rotate-180 transform"
|
|
/>
|
|
<span>{t('modals.uploadDoc.back')}</span>
|
|
</button>
|
|
|
|
<h2 className="font-inter text-[22px] leading-[28px] font-semibold tracking-[0.15px] text-black dark:text-[#E0E0E0]">
|
|
{ingestor.type &&
|
|
t(`modals.uploadDoc.ingestors.${ingestor.type}.heading`)}
|
|
</h2>
|
|
|
|
<Input
|
|
type="text"
|
|
colorVariant="silver"
|
|
value={ingestor.name}
|
|
onChange={(e) => {
|
|
setIngestor((prevState) => ({
|
|
...prevState,
|
|
name: e.target.value,
|
|
}));
|
|
}}
|
|
borderVariant="thin"
|
|
placeholder={t('modals.uploadDoc.name')}
|
|
required={true}
|
|
labelBgClassName="bg-white dark:bg-charleston-green-2"
|
|
className="w-full"
|
|
/>
|
|
{renderFormFields()}
|
|
</div>
|
|
)}
|
|
|
|
{ingestor.type &&
|
|
getIngestorSchema(ingestor.type as IngestorType)?.fields.some(
|
|
(field: FormField) => field.advanced,
|
|
) && (
|
|
<button
|
|
onClick={() => setShowAdvancedOptions(!showAdvancedOptions)}
|
|
className="text-purple-30 bg-transparent py-2 pl-0 text-left text-sm font-normal hover:cursor-pointer"
|
|
>
|
|
{showAdvancedOptions
|
|
? t('modals.uploadDoc.hideAdvanced')
|
|
: t('modals.uploadDoc.showAdvanced')}
|
|
</button>
|
|
)}
|
|
</>
|
|
)}
|
|
<div className="flex justify-end gap-4">
|
|
{activeTab && ingestor.type && (
|
|
<button
|
|
onClick={handleUpload}
|
|
disabled={isUploadDisabled()}
|
|
className={`rounded-3xl px-4 py-2 text-[14px] font-medium ${
|
|
isUploadDisabled()
|
|
? 'cursor-not-allowed bg-gray-300 text-gray-500'
|
|
: 'bg-purple-30 hover:bg-violets-are-blue cursor-pointer text-white'
|
|
}`}
|
|
>
|
|
{t('modals.uploadDoc.train')}
|
|
</button>
|
|
)}
|
|
</div>
|
|
</div>
|
|
</WrapperModal>
|
|
);
|
|
}
|
|
|
|
export default Upload;
|