import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query'; import type { UseQueryOptions, UseMutationResult, QueryObserverResult, } from '@tanstack/react-query'; import { Constants, initialModelsConfig } from '../config'; import { defaultOrderQuery } from '../types/assistants'; import { MCPServerConnectionStatusResponse } from '../types/queries'; import * as dataService from '../data-service'; import * as m from '../types/mutations'; import * as q from '../types/queries'; import { QueryKeys } from '../keys'; import * as s from '../schemas'; import * as t from '../types'; import * as permissions from '../accessPermissions'; import { ResourceType } from '../accessPermissions'; export { hasPermissions } from '../accessPermissions'; export const useGetSharedMessages = ( shareId: string, config?: UseQueryOptions, ): QueryObserverResult => { return useQuery( [QueryKeys.sharedMessages, shareId], () => dataService.getSharedMessages(shareId), { refetchOnWindowFocus: false, refetchOnReconnect: false, refetchOnMount: false, ...config, }, ); }; export const useGetSharedLinkQuery = ( conversationId: string, config?: UseQueryOptions, ): QueryObserverResult => { const queryClient = useQueryClient(); return useQuery( [QueryKeys.sharedLinks, conversationId], () => dataService.getSharedLink(conversationId), { enabled: !!conversationId && conversationId !== Constants.NEW_CONVO && conversationId !== Constants.PENDING_CONVO, refetchOnWindowFocus: false, refetchOnReconnect: false, refetchOnMount: false, onSuccess: (data) => { queryClient.setQueryData([QueryKeys.sharedLinks, conversationId], { conversationId: data.conversationId, shareId: data.shareId, }); }, ...config, }, ); }; export const useGetConversationByIdQuery = ( id: string, config?: UseQueryOptions, ): QueryObserverResult => { return useQuery( [QueryKeys.conversation, id], () => dataService.getConversationById(id), { refetchOnWindowFocus: false, refetchOnReconnect: false, refetchOnMount: false, ...config, }, ); }; //This isn't ideal because its just a query and we're using mutation, but it was the only way //to make it work with how the Chat component is structured export const useGetConversationByIdMutation = (id: string): UseMutationResult => { const queryClient = useQueryClient(); return useMutation(() => dataService.getConversationById(id), { // onSuccess: (res: s.TConversation) => { onSuccess: () => { queryClient.invalidateQueries([QueryKeys.conversation, id]); }, }); }; export const useUpdateMessageMutation = ( id: string, ): UseMutationResult => { const queryClient = useQueryClient(); return useMutation((payload: t.TUpdateMessageRequest) => dataService.updateMessage(payload), { onSuccess: () => { queryClient.invalidateQueries([QueryKeys.messages, id]); }, }); }; export const useUpdateMessageContentMutation = ( conversationId: string, ): UseMutationResult => { const queryClient = useQueryClient(); return useMutation( (payload: t.TUpdateMessageContent) => dataService.updateMessageContent(payload), { onSuccess: () => { queryClient.invalidateQueries([QueryKeys.messages, conversationId]); }, }, ); }; export const useUpdateUserKeysMutation = (): UseMutationResult< t.TUser, unknown, t.TUpdateUserKeyRequest, unknown > => { const queryClient = useQueryClient(); return useMutation((payload: t.TUpdateUserKeyRequest) => dataService.updateUserKey(payload), { onSuccess: (data, variables) => { queryClient.invalidateQueries([QueryKeys.name, variables.name]); }, }); }; export const useClearConversationsMutation = (): UseMutationResult => { const queryClient = useQueryClient(); return useMutation(() => dataService.clearAllConversations(), { onSuccess: () => { queryClient.invalidateQueries([QueryKeys.allConversations]); }, }); }; export const useRevokeUserKeyMutation = (name: string): UseMutationResult => { const queryClient = useQueryClient(); return useMutation(() => dataService.revokeUserKey(name), { onSuccess: () => { queryClient.invalidateQueries([QueryKeys.name, name]); if (s.isAssistantsEndpoint(name)) { queryClient.invalidateQueries([QueryKeys.assistants, name, defaultOrderQuery]); queryClient.invalidateQueries([QueryKeys.assistantDocs]); queryClient.invalidateQueries([QueryKeys.assistants]); queryClient.invalidateQueries([QueryKeys.assistant]); queryClient.invalidateQueries([QueryKeys.mcpTools]); queryClient.invalidateQueries([QueryKeys.actions]); queryClient.invalidateQueries([QueryKeys.tools]); } }, }); }; export const useRevokeAllUserKeysMutation = (): UseMutationResult => { const queryClient = useQueryClient(); return useMutation(() => dataService.revokeAllUserKeys(), { onSuccess: () => { queryClient.invalidateQueries([QueryKeys.name]); queryClient.invalidateQueries([ QueryKeys.assistants, s.EModelEndpoint.assistants, defaultOrderQuery, ]); queryClient.invalidateQueries([ QueryKeys.assistants, s.EModelEndpoint.azureAssistants, defaultOrderQuery, ]); queryClient.invalidateQueries([QueryKeys.assistantDocs]); queryClient.invalidateQueries([QueryKeys.assistants]); queryClient.invalidateQueries([QueryKeys.assistant]); queryClient.invalidateQueries([QueryKeys.mcpTools]); queryClient.invalidateQueries([QueryKeys.actions]); queryClient.invalidateQueries([QueryKeys.tools]); }, }); }; export const useGetModelsQuery = ( config?: UseQueryOptions, ): QueryObserverResult => { return useQuery([QueryKeys.models], () => dataService.getModels(), { initialData: initialModelsConfig, refetchOnWindowFocus: false, refetchOnReconnect: false, refetchOnMount: false, staleTime: Infinity, ...config, }); }; export const useCreatePresetMutation = (): UseMutationResult< s.TPreset, unknown, s.TPreset, unknown > => { const queryClient = useQueryClient(); return useMutation((payload: s.TPreset) => dataService.createPreset(payload), { onSuccess: () => { queryClient.invalidateQueries([QueryKeys.presets]); }, }); }; export const useDeletePresetMutation = (): UseMutationResult< m.PresetDeleteResponse, unknown, s.TPreset | undefined, unknown > => { const queryClient = useQueryClient(); return useMutation((payload: s.TPreset | undefined) => dataService.deletePreset(payload), { onSuccess: () => { queryClient.invalidateQueries([QueryKeys.presets]); }, }); }; export const useUpdateTokenCountMutation = (): UseMutationResult< t.TUpdateTokenCountResponse, unknown, { text: string }, unknown > => { const queryClient = useQueryClient(); return useMutation(({ text }: { text: string }) => dataService.updateTokenCount(text), { onSuccess: () => { queryClient.invalidateQueries([QueryKeys.tokenCount]); }, }); }; export const useRegisterUserMutation = ( options?: m.RegistrationOptions, ): UseMutationResult => { const queryClient = useQueryClient(); return useMutation( (payload: t.TRegisterUser) => dataService.register(payload), { ...options, onSuccess: (...args) => { queryClient.invalidateQueries([QueryKeys.user]); if (options?.onSuccess) { options.onSuccess(...args); } }, }, ); }; export const useUserKeyQuery = ( name: string, config?: UseQueryOptions, ): QueryObserverResult => { return useQuery( [QueryKeys.name, name], () => { if (!name) { return Promise.resolve({ expiresAt: '' }); } return dataService.userKeyQuery(name); }, { refetchOnWindowFocus: false, refetchOnReconnect: false, refetchOnMount: false, retry: false, ...config, }, ); }; export const useRequestPasswordResetMutation = (): UseMutationResult< t.TRequestPasswordResetResponse, unknown, t.TRequestPasswordReset, unknown > => { return useMutation((payload: t.TRequestPasswordReset) => dataService.requestPasswordReset(payload), ); }; export const useResetPasswordMutation = (): UseMutationResult< unknown, unknown, t.TResetPassword, unknown > => { return useMutation((payload: t.TResetPassword) => dataService.resetPassword(payload)); }; export const useAvailablePluginsQuery = ( config?: UseQueryOptions, ): QueryObserverResult => { return useQuery( [QueryKeys.availablePlugins], () => dataService.getAvailablePlugins(), { refetchOnWindowFocus: false, refetchOnReconnect: false, refetchOnMount: false, ...config, }, ); }; export const useUpdateUserPluginsMutation = ( _options?: m.UpdatePluginAuthOptions, ): UseMutationResult => { const queryClient = useQueryClient(); const { onSuccess, ...options } = _options ?? {}; return useMutation((payload: t.TUpdateUserPlugins) => dataService.updateUserPlugins(payload), { ...options, onSuccess: (...args) => { queryClient.invalidateQueries([QueryKeys.user]); onSuccess?.(...args); if (args[1]?.action === 'uninstall' && args[1]?.pluginKey?.startsWith(Constants.mcp_prefix)) { const serverName = args[1]?.pluginKey?.substring(Constants.mcp_prefix.length); queryClient.invalidateQueries([QueryKeys.mcpAuthValues, serverName]); } }, }); }; export const useReinitializeMCPServerMutation = (): UseMutationResult< { success: boolean; message: string; serverName: string; oauthRequired?: boolean; oauthUrl?: string; }, unknown, string, unknown > => { const queryClient = useQueryClient(); return useMutation((serverName: string) => dataService.reinitializeMCPServer(serverName), { onSuccess: () => { queryClient.invalidateQueries([QueryKeys.mcpTools]); }, }); }; export const useCancelMCPOAuthMutation = (): UseMutationResult< m.CancelMCPOAuthResponse, unknown, string, unknown > => { const queryClient = useQueryClient(); return useMutation((serverName: string) => dataService.cancelMCPOAuth(serverName), { onSuccess: () => { queryClient.invalidateQueries([QueryKeys.mcpConnectionStatus]); }, }); }; export const useGetCustomConfigSpeechQuery = ( config?: UseQueryOptions, ): QueryObserverResult => { return useQuery( [QueryKeys.customConfigSpeech], () => dataService.getCustomConfigSpeech(), { refetchOnWindowFocus: false, refetchOnReconnect: false, refetchOnMount: false, ...config, }, ); }; export const useUpdateFeedbackMutation = ( conversationId: string, messageId: string, ): UseMutationResult => { const queryClient = useQueryClient(); return useMutation( (payload: t.TUpdateFeedbackRequest) => dataService.updateFeedback(conversationId, messageId, payload), { onSuccess: () => { queryClient.invalidateQueries([QueryKeys.messages, messageId]); }, }, ); }; export const useSearchPrincipalsQuery = ( params: q.PrincipalSearchParams, config?: UseQueryOptions, ): QueryObserverResult => { return useQuery( [QueryKeys.principalSearch, params], () => dataService.searchPrincipals(params), { enabled: !!params.q && params.q.length >= 2, refetchOnWindowFocus: false, refetchOnReconnect: false, refetchOnMount: false, staleTime: 30000, ...config, }, ); }; export const useGetAccessRolesQuery = ( resourceType: ResourceType, config?: UseQueryOptions, ): QueryObserverResult => { return useQuery( [QueryKeys.accessRoles, resourceType], () => dataService.getAccessRoles(resourceType), { enabled: !!resourceType, refetchOnWindowFocus: false, refetchOnReconnect: false, refetchOnMount: false, staleTime: 5 * 60 * 1000, // Cache for 5 minutes ...config, }, ); }; export const useGetResourcePermissionsQuery = ( resourceType: ResourceType, resourceId: string, config?: UseQueryOptions, ): QueryObserverResult => { return useQuery( [QueryKeys.resourcePermissions, resourceType, resourceId], () => dataService.getResourcePermissions(resourceType, resourceId), { enabled: !!resourceType && !!resourceId, refetchOnWindowFocus: false, refetchOnReconnect: false, refetchOnMount: false, staleTime: 2 * 60 * 1000, // Cache for 2 minutes ...config, }, ); }; export const useUpdateResourcePermissionsMutation = (): UseMutationResult< permissions.TUpdateResourcePermissionsResponse, Error, { resourceType: ResourceType; resourceId: string; data: permissions.TUpdateResourcePermissionsRequest; } > => { const queryClient = useQueryClient(); return useMutation({ mutationFn: ({ resourceType, resourceId, data }) => dataService.updateResourcePermissions(resourceType, resourceId, data), onSuccess: (_, variables) => { queryClient.invalidateQueries({ queryKey: [QueryKeys.accessRoles, variables.resourceType], }); queryClient.invalidateQueries({ queryKey: [QueryKeys.resourcePermissions, variables.resourceType, variables.resourceId], }); queryClient.invalidateQueries({ queryKey: [QueryKeys.effectivePermissions, variables.resourceType, variables.resourceId], }); }, }); }; export const useGetEffectivePermissionsQuery = ( resourceType: ResourceType, resourceId: string, config?: UseQueryOptions, ): QueryObserverResult => { return useQuery({ queryKey: [QueryKeys.effectivePermissions, resourceType, resourceId], queryFn: () => dataService.getEffectivePermissions(resourceType, resourceId), enabled: !!resourceType && !!resourceId, refetchOnWindowFocus: false, staleTime: 30000, ...config, }); }; export const useMCPServerConnectionStatusQuery = ( serverName: string, config?: UseQueryOptions, ): QueryObserverResult => { return useQuery( [QueryKeys.mcpConnectionStatus, serverName], () => dataService.getMCPServerConnectionStatus(serverName), { refetchOnWindowFocus: false, refetchOnReconnect: false, refetchOnMount: false, staleTime: 10000, // 10 seconds enabled: !!serverName, ...config, }, ); };