2024-07-17 10:47:17 -04:00
|
|
|
// errorHandler.js
|
2025-06-25 17:16:26 -04:00
|
|
|
const { logger } = require('@librechat/data-schemas');
|
2024-07-17 10:47:17 -04:00
|
|
|
const { CacheKeys, ViolationTypes, ContentTypes } = require('librechat-data-provider');
|
|
|
|
|
const { recordUsage, checkMessageGaps } = require('~/server/services/Threads');
|
2025-06-25 17:16:26 -04:00
|
|
|
const { sendResponse } = require('~/server/middleware/error');
|
|
|
|
|
const getLogStores = require('~/cache/getLogStores');
|
📦 refactor: Consolidate DB models, encapsulating Mongoose usage in `data-schemas` (#11830)
* chore: move database model methods to /packages/data-schemas
* chore: add TypeScript ESLint rule to warn on unused variables
* refactor: model imports to streamline access
- Consolidated model imports across various files to improve code organization and reduce redundancy.
- Updated imports for models such as Assistant, Message, Conversation, and others to a unified import path.
- Adjusted middleware and service files to reflect the new import structure, ensuring functionality remains intact.
- Enhanced test files to align with the new import paths, maintaining test coverage and integrity.
* chore: migrate database models to packages/data-schemas and refactor all direct Mongoose Model usage outside of data-schemas
* test: update agent model mocks in unit tests
- Added `getAgent` mock to `client.test.js` to enhance test coverage for agent-related functionality.
- Removed redundant `getAgent` and `getAgents` mocks from `openai.spec.js` and `responses.unit.spec.js` to streamline test setup and reduce duplication.
- Ensured consistency in agent mock implementations across test files.
* fix: update types in data-schemas
* refactor: enhance type definitions in transaction and spending methods
- Updated type definitions in `checkBalance.ts` to use specific request and response types.
- Refined `spendTokens.ts` to utilize a new `SpendTxData` interface for better clarity and type safety.
- Improved transaction handling in `transaction.ts` by introducing `TransactionResult` and `TxData` interfaces, ensuring consistent data structures across methods.
- Adjusted unit tests in `transaction.spec.ts` to accommodate new type definitions and enhance robustness.
* refactor: streamline model imports and enhance code organization
- Consolidated model imports across various controllers and services to a unified import path, improving code clarity and reducing redundancy.
- Updated multiple files to reflect the new import structure, ensuring all functionalities remain intact.
- Enhanced overall code organization by removing duplicate import statements and optimizing the usage of model methods.
* feat: implement loadAddedAgent and refactor agent loading logic
- Introduced `loadAddedAgent` function to handle loading agents from added conversations, supporting multi-convo parallel execution.
- Created a new `load.ts` file to encapsulate agent loading functionalities, including `loadEphemeralAgent` and `loadAgent`.
- Updated the `index.ts` file to export the new `load` module instead of the deprecated `loadAgent`.
- Enhanced type definitions and improved error handling in the agent loading process.
- Adjusted unit tests to reflect changes in the agent loading structure and ensure comprehensive coverage.
* refactor: enhance balance handling with new update interface
- Introduced `IBalanceUpdate` interface to streamline balance update operations across the codebase.
- Updated `upsertBalanceFields` method signatures in `balance.ts`, `transaction.ts`, and related tests to utilize the new interface for improved type safety.
- Adjusted type imports in `balance.spec.ts` to include `IBalanceUpdate`, ensuring consistency in balance management functionalities.
- Enhanced overall code clarity and maintainability by refining type definitions related to balance operations.
* feat: add unit tests for loadAgent functionality and enhance agent loading logic
- Introduced comprehensive unit tests for the `loadAgent` function, covering various scenarios including null and empty agent IDs, loading of ephemeral agents, and permission checks.
- Enhanced the `initializeClient` function by moving `getConvoFiles` to the correct position in the database method exports, ensuring proper functionality.
- Improved test coverage for agent loading, including handling of non-existent agents and user permissions.
* chore: reorder memory method exports for consistency
- Moved `deleteAllUserMemories` to the correct position in the exported memory methods, ensuring a consistent and logical order of method exports in `memory.ts`.
2026-02-17 18:23:44 -05:00
|
|
|
const { getConvo } = require('~/models');
|
2024-07-17 10:47:17 -04:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @typedef {Object} ErrorHandlerContext
|
|
|
|
|
* @property {OpenAIClient} openai - The OpenAI client
|
|
|
|
|
* @property {string} thread_id - The thread ID
|
|
|
|
|
* @property {string} run_id - The run ID
|
|
|
|
|
* @property {boolean} completedRun - Whether the run has completed
|
|
|
|
|
* @property {string} assistant_id - The assistant ID
|
|
|
|
|
* @property {string} conversationId - The conversation ID
|
|
|
|
|
* @property {string} parentMessageId - The parent message ID
|
|
|
|
|
* @property {string} responseMessageId - The response message ID
|
|
|
|
|
* @property {string} endpoint - The endpoint being used
|
|
|
|
|
* @property {string} cacheKey - The cache key for the current request
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @typedef {Object} ErrorHandlerDependencies
|
2025-08-26 12:10:18 -04:00
|
|
|
* @property {ServerRequest} req - The Express request object
|
2024-07-17 10:47:17 -04:00
|
|
|
* @property {Express.Response} res - The Express response object
|
|
|
|
|
* @property {() => ErrorHandlerContext} getContext - Function to get the current context
|
|
|
|
|
* @property {string} [originPath] - The origin path for the error handler
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Creates an error handler function with the given dependencies
|
|
|
|
|
* @param {ErrorHandlerDependencies} dependencies - The dependencies for the error handler
|
|
|
|
|
* @returns {(error: Error) => Promise<void>} The error handler function
|
|
|
|
|
*/
|
|
|
|
|
const createErrorHandler = ({ req, res, getContext, originPath = '/assistants/chat/' }) => {
|
|
|
|
|
const cache = getLogStores(CacheKeys.ABORT_KEYS);
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Handles errors that occur during the chat process
|
|
|
|
|
* @param {Error} error - The error that occurred
|
|
|
|
|
* @returns {Promise<void>}
|
|
|
|
|
*/
|
|
|
|
|
return async (error) => {
|
|
|
|
|
const {
|
|
|
|
|
openai,
|
|
|
|
|
run_id,
|
|
|
|
|
endpoint,
|
|
|
|
|
cacheKey,
|
|
|
|
|
thread_id,
|
|
|
|
|
completedRun,
|
|
|
|
|
assistant_id,
|
|
|
|
|
conversationId,
|
|
|
|
|
parentMessageId,
|
|
|
|
|
responseMessageId,
|
|
|
|
|
} = getContext();
|
|
|
|
|
|
|
|
|
|
const defaultErrorMessage =
|
|
|
|
|
'The Assistant run failed to initialize. Try sending a message in a new conversation.';
|
|
|
|
|
const messageData = {
|
|
|
|
|
thread_id,
|
|
|
|
|
assistant_id,
|
|
|
|
|
conversationId,
|
|
|
|
|
parentMessageId,
|
|
|
|
|
sender: 'System',
|
|
|
|
|
user: req.user.id,
|
|
|
|
|
shouldSaveMessage: false,
|
|
|
|
|
messageId: responseMessageId,
|
|
|
|
|
endpoint,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if (error.message === 'Run cancelled') {
|
|
|
|
|
return res.end();
|
|
|
|
|
} else if (error.message === 'Request closed' && completedRun) {
|
|
|
|
|
return;
|
|
|
|
|
} else if (error.message === 'Request closed') {
|
|
|
|
|
logger.debug(`[${originPath}] Request aborted on close`);
|
|
|
|
|
} else if (/Files.*are invalid/.test(error.message)) {
|
|
|
|
|
const errorMessage = `Files are invalid, or may not have uploaded yet.${
|
|
|
|
|
endpoint === 'azureAssistants'
|
2025-06-25 17:16:26 -04:00
|
|
|
? " If using Azure OpenAI, files are only available in the region of the assistant's model at the time of upload."
|
2024-07-17 10:47:17 -04:00
|
|
|
: ''
|
|
|
|
|
}`;
|
|
|
|
|
return sendResponse(req, res, messageData, errorMessage);
|
|
|
|
|
} else if (error?.message?.includes('string too long')) {
|
|
|
|
|
return sendResponse(
|
|
|
|
|
req,
|
|
|
|
|
res,
|
|
|
|
|
messageData,
|
|
|
|
|
'Message too long. The Assistants API has a limit of 32,768 characters per message. Please shorten it and try again.',
|
|
|
|
|
);
|
|
|
|
|
} else if (error?.message?.includes(ViolationTypes.TOKEN_BALANCE)) {
|
|
|
|
|
return sendResponse(req, res, messageData, error.message);
|
|
|
|
|
} else {
|
|
|
|
|
logger.error(`[${originPath}]`, error);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!openai || !thread_id || !run_id) {
|
|
|
|
|
return sendResponse(req, res, messageData, defaultErrorMessage);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
await new Promise((resolve) => setTimeout(resolve, 2000));
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
const status = await cache.get(cacheKey);
|
|
|
|
|
if (status === 'cancelled') {
|
|
|
|
|
logger.debug(`[${originPath}] Run already cancelled`);
|
|
|
|
|
return res.end();
|
|
|
|
|
}
|
|
|
|
|
await cache.delete(cacheKey);
|
2025-08-02 12:19:58 -04:00
|
|
|
const cancelledRun = await openai.beta.threads.runs.cancel(run_id, { thread_id });
|
2024-07-17 10:47:17 -04:00
|
|
|
logger.debug(`[${originPath}] Cancelled run:`, cancelledRun);
|
|
|
|
|
} catch (error) {
|
|
|
|
|
logger.error(`[${originPath}] Error cancelling run`, error);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
await new Promise((resolve) => setTimeout(resolve, 2000));
|
|
|
|
|
|
|
|
|
|
let run;
|
|
|
|
|
try {
|
2025-08-02 12:19:58 -04:00
|
|
|
run = await openai.beta.threads.runs.retrieve(run_id, { thread_id });
|
2024-07-17 10:47:17 -04:00
|
|
|
await recordUsage({
|
|
|
|
|
...run.usage,
|
|
|
|
|
model: run.model,
|
|
|
|
|
user: req.user.id,
|
|
|
|
|
conversationId,
|
|
|
|
|
});
|
|
|
|
|
} catch (error) {
|
|
|
|
|
logger.error(`[${originPath}] Error fetching or processing run`, error);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let finalEvent;
|
|
|
|
|
try {
|
|
|
|
|
const runMessages = await checkMessageGaps({
|
|
|
|
|
openai,
|
|
|
|
|
run_id,
|
|
|
|
|
endpoint,
|
|
|
|
|
thread_id,
|
|
|
|
|
conversationId,
|
|
|
|
|
latestMessageId: responseMessageId,
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
const errorContentPart = {
|
|
|
|
|
text: {
|
|
|
|
|
value:
|
|
|
|
|
error?.message ?? 'There was an error processing your request. Please try again later.',
|
|
|
|
|
},
|
|
|
|
|
type: ContentTypes.ERROR,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if (!Array.isArray(runMessages[runMessages.length - 1]?.content)) {
|
|
|
|
|
runMessages[runMessages.length - 1].content = [errorContentPart];
|
|
|
|
|
} else {
|
|
|
|
|
const contentParts = runMessages[runMessages.length - 1].content;
|
|
|
|
|
for (let i = 0; i < contentParts.length; i++) {
|
|
|
|
|
const currentPart = contentParts[i];
|
|
|
|
|
/** @type {CodeToolCall | RetrievalToolCall | FunctionToolCall | undefined} */
|
|
|
|
|
const toolCall = currentPart?.[ContentTypes.TOOL_CALL];
|
|
|
|
|
if (
|
|
|
|
|
toolCall &&
|
|
|
|
|
toolCall?.function &&
|
|
|
|
|
!(toolCall?.function?.output || toolCall?.function?.output?.length)
|
|
|
|
|
) {
|
|
|
|
|
contentParts[i] = {
|
|
|
|
|
...currentPart,
|
|
|
|
|
[ContentTypes.TOOL_CALL]: {
|
|
|
|
|
...toolCall,
|
|
|
|
|
function: {
|
|
|
|
|
...toolCall.function,
|
|
|
|
|
output: 'error processing tool',
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
runMessages[runMessages.length - 1].content.push(errorContentPart);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
finalEvent = {
|
|
|
|
|
final: true,
|
|
|
|
|
conversation: await getConvo(req.user.id, conversationId),
|
|
|
|
|
runMessages,
|
|
|
|
|
};
|
|
|
|
|
} catch (error) {
|
|
|
|
|
logger.error(`[${originPath}] Error finalizing error process`, error);
|
|
|
|
|
return sendResponse(req, res, messageData, 'The Assistant run failed');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return sendResponse(req, res, finalEvent);
|
|
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
module.exports = { createErrorHandler };
|