| import { useState, useRef } from 'react'; |
| import { Folder } from 'lucide-react'; |
| import * as Ariakit from '@ariakit/react'; |
| import { useFormContext } from 'react-hook-form'; |
| import { SharePointIcon, AttachmentIcon, DropdownPopup } from '@librechat/client'; |
| import { |
| EModelEndpoint, |
| EToolResources, |
| mergeFileConfig, |
| AgentCapabilities, |
| getEndpointFileConfig, |
| } from 'librechat-data-provider'; |
| import type { ExtendedFile, AgentForm } from '~/common'; |
| import useSharePointFileHandling from '~/hooks/Files/useSharePointFileHandling'; |
| import { useGetFileConfig, useGetStartupConfig } from '~/data-provider'; |
| import { useFileHandling, useLocalize, useLazyEffect } from '~/hooks'; |
| import { SharePointPickerDialog } from '~/components/SharePoint'; |
| import FileRow from '~/components/Chat/Input/Files/FileRow'; |
| import FileSearchCheckbox from './FileSearchCheckbox'; |
| import { useChatContext } from '~/Providers'; |
| import { isEphemeralAgent } from '~/common'; |
|
|
| export default function FileSearch({ |
| agent_id, |
| files: _files, |
| }: { |
| agent_id: string; |
| files?: [string, ExtendedFile][]; |
| }) { |
| const localize = useLocalize(); |
| const { setFilesLoading } = useChatContext(); |
| const { watch } = useFormContext<AgentForm>(); |
| const fileInputRef = useRef<HTMLInputElement>(null); |
| const [files, setFiles] = useState<Map<string, ExtendedFile>>(new Map()); |
| const [isPopoverActive, setIsPopoverActive] = useState(false); |
| const [isSharePointDialogOpen, setIsSharePointDialogOpen] = useState(false); |
|
|
| |
| const { data: startupConfig } = useGetStartupConfig(); |
|
|
| const { data: fileConfig = null } = useGetFileConfig({ |
| select: (data) => mergeFileConfig(data), |
| }); |
|
|
| const { handleFileChange } = useFileHandling({ |
| additionalMetadata: { agent_id, tool_resource: EToolResources.file_search }, |
| fileSetter: setFiles, |
| }); |
|
|
| const { handleSharePointFiles, isProcessing, downloadProgress } = useSharePointFileHandling({ |
| additionalMetadata: { agent_id, tool_resource: EToolResources.file_search }, |
| fileSetter: setFiles, |
| }); |
|
|
| useLazyEffect( |
| () => { |
| if (_files) { |
| setFiles(new Map(_files)); |
| } |
| }, |
| [_files], |
| 750, |
| ); |
|
|
| const fileSearchChecked = watch(AgentCapabilities.file_search); |
|
|
| const endpointFileConfig = getEndpointFileConfig({ |
| fileConfig, |
| endpoint: EModelEndpoint.agents, |
| endpointType: EModelEndpoint.agents, |
| }); |
| const isUploadDisabled = endpointFileConfig?.disabled ?? false; |
|
|
| const sharePointEnabled = startupConfig?.sharePointFilePickerEnabled; |
| const disabledUploadButton = isEphemeralAgent(agent_id) || fileSearchChecked === false; |
|
|
| const handleSharePointFilesSelected = async (sharePointFiles: any[]) => { |
| try { |
| await handleSharePointFiles(sharePointFiles); |
| setIsSharePointDialogOpen(false); |
| } catch (error) { |
| console.error('SharePoint file processing error:', error); |
| } |
| }; |
| if (isUploadDisabled) { |
| return null; |
| } |
|
|
| const handleButtonClick = () => { |
| |
| if (fileInputRef.current) { |
| fileInputRef.current.value = ''; |
| } |
| fileInputRef.current?.click(); |
| }; |
|
|
| const handleLocalFileClick = () => { |
| if (fileInputRef.current) { |
| fileInputRef.current.value = ''; |
| } |
| fileInputRef.current?.click(); |
| }; |
|
|
| const dropdownItems = [ |
| { |
| label: localize('com_files_upload_local_machine'), |
| onClick: handleLocalFileClick, |
| icon: <Folder className="icon-md" />, |
| }, |
| { |
| label: localize('com_files_upload_sharepoint'), |
| onClick: () => setIsSharePointDialogOpen(true), |
| icon: <SharePointIcon className="icon-md" />, |
| }, |
| ]; |
|
|
| const menuTrigger = ( |
| <Ariakit.MenuButton |
| disabled={disabledUploadButton} |
| className="btn btn-neutral border-token-border-light relative h-9 w-full rounded-lg font-medium" |
| > |
| <div className="flex w-full items-center justify-center gap-1"> |
| <AttachmentIcon className="text-token-text-primary h-4 w-4" /> |
| {localize('com_ui_upload_file_search')} |
| </div> |
| </Ariakit.MenuButton> |
| ); |
|
|
| return ( |
| <div className="w-full"> |
| <div className="mb-1.5 flex items-center gap-2"> |
| <span> |
| <label className="text-token-text-primary block font-medium"> |
| {localize('com_assistants_file_search')} |
| </label> |
| </span> |
| </div> |
| <FileSearchCheckbox /> |
| <div className="flex flex-col gap-3"> |
| {/* File Search (RAG API) Files */} |
| <FileRow |
| files={files} |
| setFiles={setFiles} |
| setFilesLoading={setFilesLoading} |
| agent_id={agent_id} |
| tool_resource={EToolResources.file_search} |
| Wrapper={({ children }) => <div className="flex flex-wrap gap-2">{children}</div>} |
| /> |
| <div> |
| {sharePointEnabled ? ( |
| <DropdownPopup |
| gutter={2} |
| menuId="file-search-upload-menu" |
| isOpen={isPopoverActive} |
| setIsOpen={setIsPopoverActive} |
| trigger={menuTrigger} |
| items={dropdownItems} |
| modal={true} |
| unmountOnHide={true} |
| /> |
| ) : ( |
| <button |
| type="button" |
| disabled={disabledUploadButton} |
| className="btn btn-neutral border-token-border-light relative h-9 w-full rounded-lg font-medium" |
| onClick={handleButtonClick} |
| > |
| <div className="flex w-full items-center justify-center gap-1"> |
| <AttachmentIcon className="text-token-text-primary h-4 w-4" /> |
| {localize('com_ui_upload_file_search')} |
| </div> |
| </button> |
| )} |
| <input |
| multiple={true} |
| type="file" |
| style={{ display: 'none' }} |
| tabIndex={-1} |
| ref={fileInputRef} |
| disabled={disabledUploadButton} |
| onChange={handleFileChange} |
| /> |
| </div> |
| {/* Disabled Message */} |
| {agent_id ? null : ( |
| <div className="text-xs text-text-secondary"> |
| {localize('com_agents_file_search_disabled')} |
| </div> |
| )} |
| </div> |
| |
| <SharePointPickerDialog |
| isOpen={isSharePointDialogOpen} |
| onOpenChange={setIsSharePointDialogOpen} |
| onFilesSelected={handleSharePointFilesSelected} |
| disabled={disabledUploadButton} |
| isDownloading={isProcessing} |
| downloadProgress={downloadProgress} |
| maxSelectionCount={endpointFileConfig?.fileLimit} |
| /> |
| </div> |
| ); |
| } |
|
|