| | 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> |
| | ); |
| | } |
| |
|