2024-06-09 23:34:34 +00:00
|
|
|
<script lang="ts">
|
2025-07-12 23:23:05 +00:00
|
|
|
import { getContext, onMount } from 'svelte';
|
2025-07-19 10:50:36 +00:00
|
|
|
import { config, knowledge, settings, user } from '$lib/stores';
|
2025-07-12 23:23:05 +00:00
|
|
|
|
2024-06-10 00:17:35 +00:00
|
|
|
import Selector from './Knowledge/Selector.svelte';
|
2024-10-02 06:21:33 +00:00
|
|
|
import FileItem from '$lib/components/common/FileItem.svelte';
|
2025-07-19 10:50:36 +00:00
|
|
|
|
2025-07-12 23:23:05 +00:00
|
|
|
import { getKnowledgeBases } from '$lib/apis/knowledge';
|
2025-07-19 10:50:36 +00:00
|
|
|
import { uploadFile } from '$lib/apis/files';
|
|
|
|
|
|
|
|
|
|
import { toast } from 'svelte-sonner';
|
|
|
|
|
import { v4 as uuidv4 } from 'uuid';
|
|
|
|
|
import { WEBUI_API_BASE_URL } from '$lib/constants';
|
2024-06-09 23:34:34 +00:00
|
|
|
|
2025-07-10 21:34:24 +00:00
|
|
|
export let selectedItems = [];
|
2024-06-09 23:34:34 +00:00
|
|
|
const i18n = getContext('i18n');
|
2025-07-12 23:23:05 +00:00
|
|
|
|
|
|
|
|
let loaded = false;
|
|
|
|
|
|
2025-07-19 10:50:36 +00:00
|
|
|
let filesInputElement = null;
|
|
|
|
|
let inputFiles = [];
|
|
|
|
|
|
|
|
|
|
const uploadFileHandler = async (file, fullContext: boolean = false) => {
|
|
|
|
|
if ($user?.role !== 'admin' && !($user?.permissions?.chat?.file_upload ?? true)) {
|
|
|
|
|
toast.error($i18n.t('You do not have permission to upload files.'));
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const tempItemId = uuidv4();
|
|
|
|
|
const fileItem = {
|
|
|
|
|
type: 'file',
|
|
|
|
|
file: '',
|
|
|
|
|
id: null,
|
|
|
|
|
url: '',
|
|
|
|
|
name: file.name,
|
|
|
|
|
collection_name: '',
|
|
|
|
|
status: 'uploading',
|
|
|
|
|
size: file.size,
|
|
|
|
|
error: '',
|
|
|
|
|
itemId: tempItemId,
|
|
|
|
|
...(fullContext ? { context: 'full' } : {})
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if (fileItem.size == 0) {
|
|
|
|
|
toast.error($i18n.t('You cannot upload an empty file.'));
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
selectedItems = [...selectedItems, fileItem];
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
// If the file is an audio file, provide the language for STT.
|
|
|
|
|
let metadata = null;
|
|
|
|
|
if (
|
|
|
|
|
(file.type.startsWith('audio/') || file.type.startsWith('video/')) &&
|
|
|
|
|
$settings?.audio?.stt?.language
|
|
|
|
|
) {
|
|
|
|
|
metadata = {
|
|
|
|
|
language: $settings?.audio?.stt?.language
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// During the file upload, file content is automatically extracted.
|
|
|
|
|
const uploadedFile = await uploadFile(localStorage.token, file, metadata);
|
|
|
|
|
|
|
|
|
|
if (uploadedFile) {
|
|
|
|
|
console.log('File upload completed:', {
|
|
|
|
|
id: uploadedFile.id,
|
|
|
|
|
name: fileItem.name,
|
|
|
|
|
collection: uploadedFile?.meta?.collection_name
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
if (uploadedFile.error) {
|
|
|
|
|
console.warn('File upload warning:', uploadedFile.error);
|
|
|
|
|
toast.warning(uploadedFile.error);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fileItem.status = 'uploaded';
|
|
|
|
|
fileItem.file = uploadedFile;
|
|
|
|
|
fileItem.id = uploadedFile.id;
|
|
|
|
|
fileItem.collection_name =
|
|
|
|
|
uploadedFile?.meta?.collection_name || uploadedFile?.collection_name;
|
|
|
|
|
fileItem.url = `${WEBUI_API_BASE_URL}/files/${uploadedFile.id}`;
|
|
|
|
|
|
|
|
|
|
selectedItems = selectedItems;
|
|
|
|
|
} else {
|
|
|
|
|
selectedItems = selectedItems.filter((item) => item?.itemId !== tempItemId);
|
|
|
|
|
}
|
|
|
|
|
} catch (e) {
|
|
|
|
|
toast.error(`${e}`);
|
|
|
|
|
selectedItems = selectedItems.filter((item) => item?.itemId !== tempItemId);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const inputFilesHandler = async (inputFiles) => {
|
|
|
|
|
console.log('Input files handler called with:', inputFiles);
|
|
|
|
|
|
|
|
|
|
if (
|
|
|
|
|
($config?.file?.max_count ?? null) !== null &&
|
|
|
|
|
files.length + inputFiles.length > $config?.file?.max_count
|
|
|
|
|
) {
|
|
|
|
|
toast.error(
|
|
|
|
|
$i18n.t(`You can only chat with a maximum of {{maxCount}} file(s) at a time.`, {
|
|
|
|
|
maxCount: $config?.file?.max_count
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
inputFiles.forEach(async (file) => {
|
|
|
|
|
console.log('Processing file:', {
|
|
|
|
|
name: file.name,
|
|
|
|
|
type: file.type,
|
|
|
|
|
size: file.size,
|
|
|
|
|
extension: file.name.split('.').at(-1)
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
if (
|
|
|
|
|
($config?.file?.max_size ?? null) !== null &&
|
|
|
|
|
file.size > ($config?.file?.max_size ?? 0) * 1024 * 1024
|
|
|
|
|
) {
|
|
|
|
|
console.log('File exceeds max size limit:', {
|
|
|
|
|
fileSize: file.size,
|
|
|
|
|
maxSize: ($config?.file?.max_size ?? 0) * 1024 * 1024
|
|
|
|
|
});
|
|
|
|
|
toast.error(
|
|
|
|
|
$i18n.t(`File size should not exceed {{maxSize}} MB.`, {
|
|
|
|
|
maxSize: $config?.file?.max_size
|
|
|
|
|
})
|
|
|
|
|
);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!file['type'].startsWith('image/')) {
|
|
|
|
|
uploadFileHandler(file);
|
|
|
|
|
} else {
|
|
|
|
|
toast.error($i18n.t(`Unsupported file type.`));
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
};
|
|
|
|
|
|
2025-07-12 23:23:05 +00:00
|
|
|
onMount(async () => {
|
|
|
|
|
if (!$knowledge) {
|
|
|
|
|
knowledge.set(await getKnowledgeBases(localStorage.token));
|
|
|
|
|
}
|
|
|
|
|
loaded = true;
|
|
|
|
|
});
|
2024-06-09 23:34:34 +00:00
|
|
|
</script>
|
|
|
|
|
|
2025-07-19 10:50:36 +00:00
|
|
|
<input
|
|
|
|
|
bind:this={filesInputElement}
|
|
|
|
|
bind:files={inputFiles}
|
|
|
|
|
type="file"
|
|
|
|
|
hidden
|
|
|
|
|
multiple
|
|
|
|
|
on:change={async () => {
|
|
|
|
|
if (inputFiles && inputFiles.length > 0) {
|
|
|
|
|
const _inputFiles = Array.from(inputFiles);
|
|
|
|
|
inputFilesHandler(_inputFiles);
|
|
|
|
|
} else {
|
|
|
|
|
toast.error($i18n.t(`File not found.`));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
filesInputElement.value = '';
|
|
|
|
|
}}
|
|
|
|
|
/>
|
|
|
|
|
|
2024-06-09 23:34:34 +00:00
|
|
|
<div>
|
2025-07-12 23:23:05 +00:00
|
|
|
<slot name="label">
|
|
|
|
|
<div class="mb-2">
|
|
|
|
|
<div class="flex w-full justify-between mb-1">
|
|
|
|
|
<div class=" self-center text-sm font-semibold">
|
|
|
|
|
{$i18n.t('Knowledge')}
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
2024-06-09 23:34:34 +00:00
|
|
|
|
2025-07-12 23:23:05 +00:00
|
|
|
<div class=" text-xs dark:text-gray-500">
|
|
|
|
|
{$i18n.t('To attach knowledge base here, add them to the "Knowledge" workspace first.')}
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</slot>
|
2024-06-09 23:34:34 +00:00
|
|
|
|
|
|
|
|
<div class="flex flex-col">
|
2025-07-10 21:34:24 +00:00
|
|
|
{#if selectedItems?.length > 0}
|
2025-07-12 23:23:05 +00:00
|
|
|
<div class=" flex flex-wrap items-center gap-2 mb-2.5">
|
2025-07-10 21:34:24 +00:00
|
|
|
{#each selectedItems as file, fileIdx}
|
2024-10-02 06:21:33 +00:00
|
|
|
<FileItem
|
|
|
|
|
{file}
|
2025-07-08 23:22:24 +00:00
|
|
|
item={file}
|
2024-10-04 23:53:25 +00:00
|
|
|
name={file.name}
|
2025-07-08 23:22:24 +00:00
|
|
|
modal={true}
|
|
|
|
|
edit={true}
|
2024-10-04 23:53:25 +00:00
|
|
|
type={file?.legacy
|
|
|
|
|
? `Legacy${file.type ? ` ${file.type}` : ''}`
|
|
|
|
|
: (file?.type ?? 'Collection')}
|
2024-10-02 06:21:33 +00:00
|
|
|
dismissible
|
|
|
|
|
on:dismiss={(e) => {
|
2025-07-10 21:34:24 +00:00
|
|
|
selectedItems = selectedItems.filter((_, idx) => idx !== fileIdx);
|
2024-10-02 06:21:33 +00:00
|
|
|
}}
|
|
|
|
|
/>
|
2024-06-10 00:17:35 +00:00
|
|
|
{/each}
|
|
|
|
|
</div>
|
2024-06-09 23:34:34 +00:00
|
|
|
{/if}
|
|
|
|
|
|
2025-07-12 23:23:05 +00:00
|
|
|
{#if loaded}
|
2025-07-19 10:50:36 +00:00
|
|
|
<div class="flex flex-wrap flex-row text-sm gap-1">
|
2025-07-12 23:23:05 +00:00
|
|
|
<Selector
|
|
|
|
|
knowledgeItems={$knowledge || []}
|
|
|
|
|
on:select={(e) => {
|
|
|
|
|
const item = e.detail;
|
2024-10-02 06:21:33 +00:00
|
|
|
|
2025-07-12 23:23:05 +00:00
|
|
|
if (!selectedItems.find((k) => k.id === item.id)) {
|
|
|
|
|
selectedItems = [
|
|
|
|
|
...selectedItems,
|
|
|
|
|
{
|
|
|
|
|
...item
|
|
|
|
|
}
|
|
|
|
|
];
|
|
|
|
|
}
|
|
|
|
|
}}
|
2024-06-10 00:17:35 +00:00
|
|
|
>
|
2025-07-19 10:50:36 +00:00
|
|
|
<div
|
2025-07-12 23:23:05 +00:00
|
|
|
class=" px-3.5 py-1.5 font-medium hover:bg-black/5 dark:hover:bg-white/5 outline outline-1 outline-gray-100 dark:outline-gray-850 rounded-3xl"
|
|
|
|
|
>
|
2025-07-19 10:50:36 +00:00
|
|
|
{$i18n.t('Select Knowledge')}
|
|
|
|
|
</div>
|
2025-07-12 23:23:05 +00:00
|
|
|
</Selector>
|
2025-07-19 10:50:36 +00:00
|
|
|
|
2025-07-19 14:12:16 +00:00
|
|
|
{#if $user?.role === 'admin' || $user?.permissions?.chat?.file_upload}
|
|
|
|
|
<button
|
|
|
|
|
class=" px-3.5 py-1.5 font-medium hover:bg-black/5 dark:hover:bg-white/5 outline outline-1 outline-gray-100 dark:outline-gray-850 rounded-3xl"
|
|
|
|
|
type="button"
|
|
|
|
|
on:click={() => {
|
|
|
|
|
filesInputElement.click();
|
|
|
|
|
}}>{$i18n.t('Upload Files')}</button
|
|
|
|
|
>
|
|
|
|
|
{/if}
|
2025-07-12 23:23:05 +00:00
|
|
|
</div>
|
|
|
|
|
{/if}
|
2024-06-09 23:34:34 +00:00
|
|
|
<!-- {knowledge} -->
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|