🔧 fix: Error Handling Improvements (#1518)

* style(Icon): remove error bubble from message icon

* fix(custom): `initializeClient` now throws error if apiKey or baseURL are admin provided but no env var was found

* refactor(tPresetSchema): match `conversationId` type to `tConversationSchema` but optional, use `extendedModelEndpointSchema` for `endpoint`

* fix(useSSE): minor improvements
- use `completed` set to avoid submitting unecessary abort request
- set preset with `newConversation` calls using initial conversation settings to prevent default Preset override as well as default settings
- return if there is a parsing error within `onerror` as expected errors from server are properly formatted
This commit is contained in:
Danny Avila 2024-01-08 09:30:38 -05:00 committed by GitHub
parent c9aaf502af
commit ead1c3c797
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
5 changed files with 57 additions and 14 deletions

View file

@ -4,6 +4,8 @@ const { isUserProvided, extractEnvVariable } = require('~/server/utils');
const getCustomConfig = require('~/cache/getCustomConfig'); const getCustomConfig = require('~/cache/getCustomConfig');
const { OpenAIClient } = require('~/app'); const { OpenAIClient } = require('~/app');
const envVarRegex = /^\${(.+)}$/;
const { PROXY } = process.env; const { PROXY } = process.env;
const initializeClient = async ({ req, res, endpointOption }) => { const initializeClient = async ({ req, res, endpointOption }) => {
@ -20,6 +22,14 @@ const initializeClient = async ({ req, res, endpointOption }) => {
const CUSTOM_API_KEY = extractEnvVariable(endpointConfig.apiKey); const CUSTOM_API_KEY = extractEnvVariable(endpointConfig.apiKey);
const CUSTOM_BASE_URL = extractEnvVariable(endpointConfig.baseURL); const CUSTOM_BASE_URL = extractEnvVariable(endpointConfig.baseURL);
if (CUSTOM_API_KEY.match(envVarRegex)) {
throw new Error(`Missing API Key for ${endpoint}.`);
}
if (CUSTOM_BASE_URL.match(envVarRegex)) {
throw new Error(`Missing Base URL for ${endpoint}.`);
}
const customOptions = { const customOptions = {
addParams: endpointConfig.addParams, addParams: endpointConfig.addParams,
dropParams: endpointConfig.dropParams, dropParams: endpointConfig.dropParams,

View file

@ -16,7 +16,7 @@ import { cn } from '~/utils';
const Icon: React.FC<IconProps> = (props) => { const Icon: React.FC<IconProps> = (props) => {
const { user } = useAuthContext(); const { user } = useAuthContext();
const { size = 30, isCreatedByUser, button, model = '', endpoint, error, jailbreak } = props; const { size = 30, isCreatedByUser, button, model = '', endpoint, jailbreak } = props;
if (isCreatedByUser) { if (isCreatedByUser) {
const username = user?.name || 'User'; const username = user?.name || 'User';
@ -130,11 +130,11 @@ const Icon: React.FC<IconProps> = (props) => {
)} )}
> >
{icon} {icon}
{error && ( {/* {error && (
<span className="absolute right-0 top-[20px] -mr-2 flex h-4 w-4 items-center justify-center rounded-full border border-white bg-red-500 text-[10px] text-white"> <span className="absolute right-0 top-[20px] -mr-2 flex h-4 w-4 items-center justify-center rounded-full border border-white bg-red-500 text-[10px] text-white">
! !
</span> </span>
)} )} */}
</div> </div>
); );
} }

View file

@ -1,14 +1,15 @@
import { v4 } from 'uuid'; import { v4 } from 'uuid';
import { useEffect } from 'react'; import { useEffect, useState } from 'react';
import { useParams } from 'react-router-dom'; import { useParams } from 'react-router-dom';
import { import {
/* @ts-ignore */ /* @ts-ignore */
SSE, SSE,
EndpointURLs, EndpointURLs,
createPayload, createPayload,
tPresetSchema,
tMessageSchema, tMessageSchema,
tConvoUpdateSchema,
EModelEndpoint, EModelEndpoint,
tConvoUpdateSchema,
removeNullishValues, removeNullishValues,
} from 'librechat-data-provider'; } from 'librechat-data-provider';
import { useGetUserBalance, useGetStartupConfig } from 'librechat-data-provider/react-query'; import { useGetUserBalance, useGetStartupConfig } from 'librechat-data-provider/react-query';
@ -31,6 +32,7 @@ export default function useSSE(submission: TSubmission | null, index = 0) {
const setStorage = useSetStorage(); const setStorage = useSetStorage();
const { conversationId: paramId } = useParams(); const { conversationId: paramId } = useParams();
const { token, isAuthenticated } = useAuthContext(); const { token, isAuthenticated } = useAuthContext();
const [completed, setCompleted] = useState(new Set());
const { const {
addConvo, addConvo,
setMessages, setMessages,
@ -172,6 +174,8 @@ export default function useSSE(submission: TSubmission | null, index = 0) {
const { requestMessage, responseMessage, conversation } = data; const { requestMessage, responseMessage, conversation } = data;
const { messages, conversation: submissionConvo, isRegenerate = false } = submission; const { messages, conversation: submissionConvo, isRegenerate = false } = submission;
setCompleted((prev) => new Set(prev.add(submission?.initialResponse?.messageId)));
// update the messages // update the messages
if (isRegenerate) { if (isRegenerate) {
setMessages([...messages, responseMessage]); setMessages([...messages, responseMessage]);
@ -212,7 +216,10 @@ export default function useSSE(submission: TSubmission | null, index = 0) {
const errorHandler = ({ data, submission }: { data?: TResData; submission: TSubmission }) => { const errorHandler = ({ data, submission }: { data?: TResData; submission: TSubmission }) => {
const { messages, message, initialResponse } = submission; const { messages, message, initialResponse } = submission;
setCompleted((prev) => new Set(prev.add(initialResponse.messageId)));
const conversationId = message?.conversationId ?? submission?.conversationId; const conversationId = message?.conversationId ?? submission?.conversationId;
const parseErrorResponse = (data: TResData | Partial<TMessage>) => { const parseErrorResponse = (data: TResData | Partial<TMessage>) => {
const metadata = data['responseMessage'] ?? data; const metadata = data['responseMessage'] ?? data;
const errorMessage = { const errorMessage = {
@ -237,7 +244,10 @@ export default function useSSE(submission: TSubmission | null, index = 0) {
conversationId: convoId, conversationId: convoId,
}); });
setMessages([...messages, message, errorResponse]); setMessages([...messages, message, errorResponse]);
newConversation({ template: { conversationId: convoId } }); newConversation({
template: { conversationId: convoId },
preset: tPresetSchema.parse(submission?.conversation),
});
setIsSubmitting(false); setIsSubmitting(false);
return; return;
} }
@ -246,7 +256,10 @@ export default function useSSE(submission: TSubmission | null, index = 0) {
const convoId = v4(); const convoId = v4();
const errorResponse = parseErrorResponse(data); const errorResponse = parseErrorResponse(data);
setMessages([...messages, message, errorResponse]); setMessages([...messages, message, errorResponse]);
newConversation({ template: { conversationId: convoId } }); newConversation({
template: { conversationId: convoId },
preset: tPresetSchema.parse(submission?.conversation),
});
setIsSubmitting(false); setIsSubmitting(false);
return; return;
} }
@ -260,7 +273,10 @@ export default function useSSE(submission: TSubmission | null, index = 0) {
setMessages([...messages, message, errorResponse]); setMessages([...messages, message, errorResponse]);
if (data.conversationId && paramId === 'new') { if (data.conversationId && paramId === 'new') {
newConversation({ template: { conversationId: data.conversationId } }); newConversation({
template: { conversationId: data.conversationId },
preset: tPresetSchema.parse(submission?.conversation),
});
} }
setIsSubmitting(false); setIsSubmitting(false);
@ -334,7 +350,10 @@ export default function useSSE(submission: TSubmission | null, index = 0) {
const errorResponse = tMessageSchema.parse(errorMessage); const errorResponse = tMessageSchema.parse(errorMessage);
setMessages([...submission.messages, submission.message, errorResponse]); setMessages([...submission.messages, submission.message, errorResponse]);
newConversation({ template: { conversationId: convoId } }); newConversation({
template: { conversationId: convoId },
preset: tPresetSchema.parse(submission?.conversation),
});
setIsSubmitting(false); setIsSubmitting(false);
}); });
return; return;
@ -389,8 +408,20 @@ export default function useSSE(submission: TSubmission | null, index = 0) {
events.onopen = () => console.log('connection is opened'); events.onopen = () => console.log('connection is opened');
events.oncancel = () => events.oncancel = () => {
abortConversation(message?.conversationId ?? submission?.conversationId, submission); const streamKey = submission?.initialResponse?.messageId;
if (completed.has(streamKey)) {
setIsSubmitting(false);
setCompleted((prev) => {
prev.delete(streamKey);
return new Set(prev);
});
return;
}
setCompleted((prev) => new Set(prev.add(streamKey)));
return abortConversation(message?.conversationId ?? submission?.conversationId, submission);
};
events.onerror = function (e: MessageEvent) { events.onerror = function (e: MessageEvent) {
console.log('error in server stream.'); console.log('error in server stream.');
@ -403,10 +434,11 @@ export default function useSSE(submission: TSubmission | null, index = 0) {
} catch (error) { } catch (error) {
console.error(error); console.error(error);
console.log(e); console.log(e);
setIsSubmitting(false);
return;
} }
errorHandler({ data, submission: { ...submission, message } }); errorHandler({ data, submission: { ...submission, message } });
events.oncancel();
}; };
setIsSubmitting(true); setIsSubmitting(true);

View file

@ -1,6 +1,6 @@
{ {
"name": "librechat-data-provider", "name": "librechat-data-provider",
"version": "0.3.4", "version": "0.3.5",
"description": "data services for librechat apps", "description": "data services for librechat apps",
"main": "dist/index.js", "main": "dist/index.js",
"module": "dist/index.es.js", "module": "dist/index.es.js",

View file

@ -200,11 +200,12 @@ export const tPresetSchema = tConversationSchema
}) })
.merge( .merge(
z.object({ z.object({
conversationId: z.string().optional(), conversationId: z.string().nullable().optional(),
presetId: z.string().nullable().optional(), presetId: z.string().nullable().optional(),
title: z.string().nullable().optional(), title: z.string().nullable().optional(),
defaultPreset: z.boolean().optional(), defaultPreset: z.boolean().optional(),
order: z.number().optional(), order: z.number().optional(),
endpoint: extendedModelEndpointSchema.nullable(),
}), }),
); );