2025-08-16 20:45:55 -04:00
|
|
|
const { logger } = require('@librechat/data-schemas');
|
2025-04-12 18:46:36 -04:00
|
|
|
|
2025-08-16 20:45:55 -04:00
|
|
|
/** WeakMap to hold temporary data associated with requests */
|
2025-04-12 18:46:36 -04:00
|
|
|
const requestDataMap = new WeakMap();
|
|
|
|
|
|
|
|
|
|
const FinalizationRegistry = global.FinalizationRegistry || null;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* FinalizationRegistry to clean up client objects when they are garbage collected.
|
|
|
|
|
* This is used to prevent memory leaks and ensure that client objects are
|
|
|
|
|
* properly disposed of when they are no longer needed.
|
|
|
|
|
* The registry holds a weak reference to the client object and a cleanup
|
|
|
|
|
* callback that is called when the client object is garbage collected.
|
|
|
|
|
* The callback can be used to perform any necessary cleanup operations,
|
|
|
|
|
* such as removing event listeners or freeing up resources.
|
|
|
|
|
*/
|
|
|
|
|
const clientRegistry = FinalizationRegistry
|
|
|
|
|
? new FinalizationRegistry((heldValue) => {
|
2025-05-22 15:00:44 -04:00
|
|
|
try {
|
|
|
|
|
// This will run when the client is garbage collected
|
|
|
|
|
if (heldValue && heldValue.userId) {
|
|
|
|
|
logger.debug(`[FinalizationRegistry] Cleaning up client for user ${heldValue.userId}`);
|
|
|
|
|
} else {
|
|
|
|
|
logger.debug('[FinalizationRegistry] Cleaning up client');
|
|
|
|
|
}
|
2025-08-16 20:45:55 -04:00
|
|
|
} catch {
|
2025-05-22 15:00:44 -04:00
|
|
|
// Ignore errors
|
2025-04-12 18:46:36 -04:00
|
|
|
}
|
2025-05-22 15:00:44 -04:00
|
|
|
})
|
2025-04-12 18:46:36 -04:00
|
|
|
: null;
|
|
|
|
|
|
2025-11-10 13:50:17 -05:00
|
|
|
const graphPropsToClean = [
|
|
|
|
|
'handlerRegistry',
|
|
|
|
|
'runId',
|
|
|
|
|
'tools',
|
|
|
|
|
'signal',
|
|
|
|
|
'config',
|
|
|
|
|
'messages',
|
|
|
|
|
'contentData',
|
|
|
|
|
'stepKeyIds',
|
|
|
|
|
'contentIndexMap',
|
|
|
|
|
'toolCallStepIds',
|
|
|
|
|
'messageIdsByStepKey',
|
|
|
|
|
'messageStepHasToolCalls',
|
|
|
|
|
'prelimMessageIdsByStepKey',
|
|
|
|
|
'startIndex',
|
|
|
|
|
'defaultAgentId',
|
|
|
|
|
'dispatchReasoningDelta',
|
|
|
|
|
'compileOptions',
|
|
|
|
|
'invokedToolIds',
|
|
|
|
|
'overrideModel',
|
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
const graphRunnablePropsToClean = [
|
|
|
|
|
'lc_serializable',
|
|
|
|
|
'lc_kwargs',
|
|
|
|
|
'lc_runnable',
|
|
|
|
|
'name',
|
|
|
|
|
'lc_namespace',
|
|
|
|
|
'lg_is_pregel',
|
|
|
|
|
'nodes',
|
|
|
|
|
'channels',
|
|
|
|
|
'inputChannels',
|
|
|
|
|
'outputChannels',
|
|
|
|
|
'autoValidate',
|
|
|
|
|
'streamMode',
|
|
|
|
|
'streamChannels',
|
|
|
|
|
'interruptAfter',
|
|
|
|
|
'interruptBefore',
|
|
|
|
|
'stepTimeout',
|
|
|
|
|
'debug',
|
|
|
|
|
'checkpointer',
|
|
|
|
|
'retryPolicy',
|
|
|
|
|
'config',
|
|
|
|
|
'store',
|
|
|
|
|
'triggerToNodes',
|
|
|
|
|
'cache',
|
|
|
|
|
'description',
|
|
|
|
|
'metaRegistry',
|
|
|
|
|
];
|
|
|
|
|
|
2025-04-12 18:46:36 -04:00
|
|
|
/**
|
2025-11-10 13:50:17 -05:00
|
|
|
* Cleans up the client object by removing potential circular references to its properties.
|
2025-04-12 18:46:36 -04:00
|
|
|
* This is useful for preventing memory leaks and ensuring that the client
|
|
|
|
|
* and its properties can be garbage collected when it is no longer needed.
|
|
|
|
|
*/
|
|
|
|
|
function disposeClient(client) {
|
|
|
|
|
if (!client) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
if (client.user) {
|
|
|
|
|
client.user = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.apiKey) {
|
|
|
|
|
client.apiKey = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.azure) {
|
|
|
|
|
client.azure = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.conversationId) {
|
|
|
|
|
client.conversationId = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.responseMessageId) {
|
|
|
|
|
client.responseMessageId = null;
|
|
|
|
|
}
|
2025-08-16 20:45:55 -04:00
|
|
|
if (client.parentMessageId) {
|
|
|
|
|
client.parentMessageId = null;
|
|
|
|
|
}
|
2025-04-12 20:53:38 -04:00
|
|
|
if (client.message_file_map) {
|
|
|
|
|
client.message_file_map = null;
|
|
|
|
|
}
|
2025-04-12 18:46:36 -04:00
|
|
|
if (client.clientName) {
|
|
|
|
|
client.clientName = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.sender) {
|
|
|
|
|
client.sender = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.model) {
|
|
|
|
|
client.model = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.maxContextTokens) {
|
|
|
|
|
client.maxContextTokens = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.contextStrategy) {
|
|
|
|
|
client.contextStrategy = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.currentDateString) {
|
|
|
|
|
client.currentDateString = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.inputTokensKey) {
|
|
|
|
|
client.inputTokensKey = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.outputTokensKey) {
|
|
|
|
|
client.outputTokensKey = null;
|
|
|
|
|
}
|
2025-04-12 20:53:38 -04:00
|
|
|
if (client.skipSaveUserMessage !== undefined) {
|
|
|
|
|
client.skipSaveUserMessage = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.visionMode) {
|
|
|
|
|
client.visionMode = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.continued !== undefined) {
|
|
|
|
|
client.continued = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.fetchedConvo !== undefined) {
|
|
|
|
|
client.fetchedConvo = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.previous_summary) {
|
|
|
|
|
client.previous_summary = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.metadata) {
|
|
|
|
|
client.metadata = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.isVisionModel) {
|
|
|
|
|
client.isVisionModel = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.isChatCompletion !== undefined) {
|
|
|
|
|
client.isChatCompletion = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.contextHandlers) {
|
|
|
|
|
client.contextHandlers = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.augmentedPrompt) {
|
|
|
|
|
client.augmentedPrompt = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.systemMessage) {
|
|
|
|
|
client.systemMessage = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.azureEndpoint) {
|
|
|
|
|
client.azureEndpoint = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.langchainProxy) {
|
|
|
|
|
client.langchainProxy = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.isOmni !== undefined) {
|
|
|
|
|
client.isOmni = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.runManager) {
|
|
|
|
|
client.runManager = null;
|
|
|
|
|
}
|
|
|
|
|
// Properties specific to AnthropicClient
|
|
|
|
|
if (client.message_start) {
|
|
|
|
|
client.message_start = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.message_delta) {
|
|
|
|
|
client.message_delta = null;
|
|
|
|
|
}
|
2025-05-22 15:00:44 -04:00
|
|
|
if (client.isClaudeLatest !== undefined) {
|
|
|
|
|
client.isClaudeLatest = null;
|
2025-04-12 20:53:38 -04:00
|
|
|
}
|
|
|
|
|
if (client.useMessages !== undefined) {
|
|
|
|
|
client.useMessages = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.supportsCacheControl !== undefined) {
|
|
|
|
|
client.supportsCacheControl = null;
|
|
|
|
|
}
|
|
|
|
|
// Properties specific to GoogleClient
|
|
|
|
|
if (client.serviceKey) {
|
|
|
|
|
client.serviceKey = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.project_id) {
|
|
|
|
|
client.project_id = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.client_email) {
|
|
|
|
|
client.client_email = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.private_key) {
|
|
|
|
|
client.private_key = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.access_token) {
|
|
|
|
|
client.access_token = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.reverseProxyUrl) {
|
|
|
|
|
client.reverseProxyUrl = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.authHeader) {
|
|
|
|
|
client.authHeader = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.isGenerativeModel !== undefined) {
|
|
|
|
|
client.isGenerativeModel = null;
|
|
|
|
|
}
|
|
|
|
|
// Properties specific to OpenAIClient
|
|
|
|
|
if (client.completionsUrl) {
|
|
|
|
|
client.completionsUrl = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.shouldSummarize !== undefined) {
|
|
|
|
|
client.shouldSummarize = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.isOllama !== undefined) {
|
|
|
|
|
client.isOllama = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.FORCE_PROMPT !== undefined) {
|
|
|
|
|
client.FORCE_PROMPT = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.isChatGptModel !== undefined) {
|
|
|
|
|
client.isChatGptModel = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.isUnofficialChatGptModel !== undefined) {
|
|
|
|
|
client.isUnofficialChatGptModel = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.useOpenRouter !== undefined) {
|
|
|
|
|
client.useOpenRouter = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.startToken) {
|
|
|
|
|
client.startToken = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.endToken) {
|
|
|
|
|
client.endToken = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.userLabel) {
|
|
|
|
|
client.userLabel = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.chatGptLabel) {
|
|
|
|
|
client.chatGptLabel = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.modelLabel) {
|
|
|
|
|
client.modelLabel = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.modelOptions) {
|
|
|
|
|
client.modelOptions = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.defaultVisionModel) {
|
|
|
|
|
client.defaultVisionModel = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.maxPromptTokens) {
|
|
|
|
|
client.maxPromptTokens = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.maxResponseTokens) {
|
|
|
|
|
client.maxResponseTokens = null;
|
|
|
|
|
}
|
🧠 feat: User Memories for Conversational Context (#7760)
* 🧠 feat: User Memories for Conversational Context
chore: mcp typing, use `t`
WIP: first pass, Memories UI
- Added MemoryViewer component for displaying, editing, and deleting user memories.
- Integrated data provider hooks for fetching, updating, and deleting memories.
- Implemented pagination and loading states for better user experience.
- Created unit tests for MemoryViewer to ensure functionality and interaction with data provider.
- Updated translation files to include new UI strings related to memories.
chore: move mcp-related files to own directory
chore: rename librechat-mcp to librechat-api
WIP: first pass, memory processing and data schemas
chore: linting in fileSearch.js query description
chore: rename librechat-api to @librechat/api across the project
WIP: first pass, functional memory agent
feat: add MemoryEditDialog and MemoryViewer components for managing user memories
- Introduced MemoryEditDialog for editing memory entries with validation and toast notifications.
- Updated MemoryViewer to support editing and deleting memories, including pagination and loading states.
- Enhanced data provider to handle memory updates with optional original key for better management.
- Added new localization strings for memory-related UI elements.
feat: add memory permissions management
- Implemented memory permissions in the backend, allowing roles to have specific permissions for using, creating, updating, and reading memories.
- Added new API endpoints for updating memory permissions associated with roles.
- Created a new AdminSettings component for managing memory permissions in the frontend.
- Integrated memory permissions into the existing roles and permissions schemas.
- Updated the interface to include memory settings and permissions.
- Enhanced the MemoryViewer component to conditionally render admin settings based on user roles.
- Added localization support for memory permissions in the translation files.
feat: move AdminSettings component to a new position in MemoryViewer for better visibility
refactor: clean up commented code in MemoryViewer component
feat: enhance MemoryViewer with search functionality and improve MemoryEditDialog integration
- Added a search input to filter memories in the MemoryViewer component.
- Refactored MemoryEditDialog to accept children for better customization.
- Updated MemoryViewer to utilize the new EditMemoryButton and DeleteMemoryButton components for editing and deleting memories.
- Improved localization support by adding new strings for memory filtering and deletion confirmation.
refactor: optimize memory filtering in MemoryViewer using match-sorter
- Replaced manual filtering logic with match-sorter for improved search functionality.
- Enhanced performance and readability of the filteredMemories computation.
feat: enhance MemoryEditDialog with triggerRef and improve updateMemory mutation handling
feat: implement access control for MemoryEditDialog and MemoryViewer components
refactor: remove commented out code and create runMemory method
refactor: rename role based files
feat: implement access control for memory usage in AgentClient
refactor: simplify checkVisionRequest method in AgentClient by removing commented-out code
refactor: make `agents` dir in api package
refactor: migrate Azure utilities to TypeScript and consolidate imports
refactor: move sanitizeFilename function to a new file and update imports, add related tests
refactor: update LLM configuration types and consolidate Azure options in the API package
chore: linting
chore: import order
refactor: replace getLLMConfig with getOpenAIConfig and remove unused LLM configuration file
chore: update winston-daily-rotate-file to version 5.0.0 and add object-hash dependency in package-lock.json
refactor: move primeResources and optionalChainWithEmptyCheck functions to resources.ts and update imports
refactor: move createRun function to a new run.ts file and update related imports
fix: ensure safeAttachments is correctly typed as an array of TFile
chore: add node-fetch dependency and refactor fetch-related functions into packages/api/utils, removing the old generators file
refactor: enhance TEndpointOption type by using Pick to streamline endpoint fields and add new properties for model parameters and client options
feat: implement initializeOpenAIOptions function and update OpenAI types for enhanced configuration handling
fix: update types due to new TEndpointOption typing
fix: ensure safe access to group parameters in initializeOpenAIOptions function
fix: remove redundant API key validation comment in initializeOpenAIOptions function
refactor: rename initializeOpenAIOptions to initializeOpenAI for consistency and update related documentation
refactor: decouple req.body fields and tool loading from initializeAgentOptions
chore: linting
refactor: adjust column widths in MemoryViewer for improved layout
refactor: simplify agent initialization by creating loadAgent function and removing unused code
feat: add memory configuration loading and validation functions
WIP: first pass, memory processing with config
feat: implement memory callback and artifact handling
feat: implement memory artifacts display and processing updates
feat: add memory configuration options and schema validation for validKeys
fix: update MemoryEditDialog and MemoryViewer to handle memory state and display improvements
refactor: remove padding from BookmarkTable and MemoryViewer headers for consistent styling
WIP: initial tokenLimit config and move Tokenizer to @librechat/api
refactor: update mongoMeili plugin methods to use callback for better error handling
feat: enhance memory management with token tracking and usage metrics
- Added token counting for memory entries to enforce limits and provide usage statistics.
- Updated memory retrieval and update routes to include total token usage and limit.
- Enhanced MemoryEditDialog and MemoryViewer components to display memory usage and token information.
- Refactored memory processing functions to handle token limits and provide feedback on memory capacity.
feat: implement memory artifact handling in attachment handler
- Enhanced useAttachmentHandler to process memory artifacts when receiving updates.
- Introduced handleMemoryArtifact utility to manage memory updates and deletions.
- Updated query client to reflect changes in memory state based on incoming data.
refactor: restructure web search key extraction logic
- Moved the logic for extracting API keys from the webSearchAuth configuration into a dedicated function, getWebSearchKeys.
- Updated webSearchKeys to utilize the new function for improved clarity and maintainability.
- Prevents build time errors
feat: add personalization settings and memory preferences management
- Introduced a new Personalization tab in settings to manage user memory preferences.
- Implemented API endpoints and client-side logic for updating memory preferences.
- Enhanced user interface components to reflect personalization options and memory usage.
- Updated permissions to allow users to opt out of memory features.
- Added localization support for new settings and messages related to personalization.
style: personalization switch class
feat: add PersonalizationIcon and align Side Panel UI
feat: implement memory creation functionality
- Added a new API endpoint for creating memory entries, including validation for key and value.
- Introduced MemoryCreateDialog component for user interface to facilitate memory creation.
- Integrated token limit checks to prevent exceeding user memory capacity.
- Updated MemoryViewer to include a button for opening the memory creation dialog.
- Enhanced localization support for new messages related to memory creation.
feat: enhance message processing with configurable window size
- Updated AgentClient to use a configurable message window size for processing messages.
- Introduced messageWindowSize option in memory configuration schema with a default value of 5.
- Improved logic for selecting messages to process based on the configured window size.
chore: update librechat-data-provider version to 0.7.87 in package.json and package-lock.json
chore: remove OpenAPIPlugin and its associated tests
chore: remove MIGRATION_README.md as migration tasks are completed
ci: fix backend tests
chore: remove unused translation keys from localization file
chore: remove problematic test file and unused var in AgentClient
chore: remove unused import and import directly for JSDoc
* feat: add api package build stage in Dockerfile for improved modularity
* docs: reorder build steps in contributing guide for clarity
2025-06-07 18:52:22 -04:00
|
|
|
if (client.processMemory) {
|
|
|
|
|
client.processMemory = null;
|
|
|
|
|
}
|
2025-11-10 13:50:17 -05:00
|
|
|
|
2025-04-12 18:46:36 -04:00
|
|
|
if (client.run) {
|
|
|
|
|
if (client.run.Graph) {
|
🪣 fix: Prevent Memory Retention from AsyncLocalStorage Context Propagation (#11942)
* fix: store hide_sequential_outputs before processStream clears config
processStream now clears config.configurable after completion to break
memory retention chains. Save hide_sequential_outputs to a local
variable before calling runAgents so the post-stream filter still works.
* feat: memory diagnostics
* chore: expose garbage collection in backend inspect command
Updated the backend inspect command in package.json to include the --expose-gc flag, enabling garbage collection diagnostics for improved memory management during development.
* chore: update @librechat/agents dependency to version 3.1.52
Bumped the version of @librechat/agents in package.json and package-lock.json to ensure compatibility and access to the latest features and fixes.
* fix: clear heavy config state after processStream to prevent memory leaks
Break the reference chain from LangGraph's internal __pregel_scratchpad
through @langchain/core RunTree.extra[lc:child_config] into the
AsyncLocalStorage context captured by timers and I/O handles.
After stream completion, null out symbol-keyed scratchpad properties
(currentTaskInput), config.configurable, and callbacks. Also call
Graph.clearHeavyState() to release config, signal, content maps,
handler registry, and tool sessions.
* chore: fix imports for memory utils
* chore: add circular dependency check in API build step
Enhanced the backend review workflow to include a check for circular dependencies during the API build process. If a circular dependency is detected, an error message is displayed, and the process exits with a failure status.
* chore: update API build step to include circular dependency detection
Modified the backend review workflow to rename the API package installation step to reflect its new functionality, which now includes detection of circular dependencies during the build process.
* chore: add memory diagnostics option to .env.example
Included a commented-out configuration option for enabling memory diagnostics in the .env.example file, which logs heap and RSS snapshots every 60 seconds when activated.
* chore: remove redundant agentContexts cleanup in disposeClient function
Streamlined the disposeClient function by eliminating duplicate cleanup logic for agentContexts, ensuring efficient memory management during client disposal.
* refactor: move runOutsideTracing utility to utils and update its usage
Refactored the runOutsideTracing function by relocating it to the utils module for better organization. Updated the tool execution handler to utilize the new import, ensuring consistent tracing behavior during tool execution.
* refactor: enhance connection management and diagnostics
Added a method to ConnectionsRepository for retrieving the active connection count. Updated UserConnectionManager to utilize this new method for app connection count reporting. Refined the OAuthReconnectionTracker's getStats method to improve clarity in diagnostics. Introduced a new tracing utility in the utils module to streamline tracing context management. Additionally, added a safeguard in memory diagnostics to prevent unnecessary snapshot collection for very short intervals.
* refactor: enhance tracing utility and add memory diagnostics tests
Refactored the runOutsideTracing function to improve warning logic when the AsyncLocalStorage context is missing. Added tests for memory diagnostics and tracing utilities to ensure proper functionality and error handling. Introduced a new test suite for memory diagnostics, covering snapshot collection and garbage collection behavior.
2026-02-25 17:41:23 -05:00
|
|
|
if (typeof client.run.Graph.clearHeavyState === 'function') {
|
|
|
|
|
client.run.Graph.clearHeavyState();
|
|
|
|
|
} else {
|
|
|
|
|
client.run.Graph.resetValues();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (client.run.Graph.agentContexts) {
|
|
|
|
|
client.run.Graph.agentContexts.clear();
|
|
|
|
|
client.run.Graph.agentContexts = null;
|
|
|
|
|
}
|
2025-11-10 13:50:17 -05:00
|
|
|
|
|
|
|
|
graphPropsToClean.forEach((prop) => {
|
|
|
|
|
if (client.run.Graph[prop] !== undefined) {
|
|
|
|
|
client.run.Graph[prop] = null;
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
2025-04-12 18:46:36 -04:00
|
|
|
client.run.Graph = null;
|
|
|
|
|
}
|
2025-11-10 13:50:17 -05:00
|
|
|
|
2025-04-12 18:46:36 -04:00
|
|
|
if (client.run.graphRunnable) {
|
2025-11-10 13:50:17 -05:00
|
|
|
graphRunnablePropsToClean.forEach((prop) => {
|
|
|
|
|
if (client.run.graphRunnable[prop] !== undefined) {
|
|
|
|
|
client.run.graphRunnable[prop] = null;
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
if (client.run.graphRunnable.builder) {
|
|
|
|
|
if (client.run.graphRunnable.builder.nodes !== undefined) {
|
|
|
|
|
client.run.graphRunnable.builder.nodes = null;
|
|
|
|
|
}
|
2025-04-12 18:46:36 -04:00
|
|
|
client.run.graphRunnable.builder = null;
|
|
|
|
|
}
|
2025-11-10 13:50:17 -05:00
|
|
|
|
2025-04-12 18:46:36 -04:00
|
|
|
client.run.graphRunnable = null;
|
|
|
|
|
}
|
2025-11-10 13:50:17 -05:00
|
|
|
|
|
|
|
|
const runPropsToClean = [
|
|
|
|
|
'handlerRegistry',
|
|
|
|
|
'id',
|
|
|
|
|
'indexTokenCountMap',
|
|
|
|
|
'returnContent',
|
|
|
|
|
'tokenCounter',
|
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
runPropsToClean.forEach((prop) => {
|
|
|
|
|
if (client.run[prop] !== undefined) {
|
|
|
|
|
client.run[prop] = null;
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
2025-04-12 18:46:36 -04:00
|
|
|
client.run = null;
|
|
|
|
|
}
|
2025-11-10 13:50:17 -05:00
|
|
|
|
2025-04-12 18:46:36 -04:00
|
|
|
if (client.sendMessage) {
|
|
|
|
|
client.sendMessage = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.savedMessageIds) {
|
|
|
|
|
client.savedMessageIds.clear();
|
|
|
|
|
client.savedMessageIds = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.currentMessages) {
|
|
|
|
|
client.currentMessages = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.streamHandler) {
|
|
|
|
|
client.streamHandler = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.contentParts) {
|
|
|
|
|
client.contentParts = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.abortController) {
|
|
|
|
|
client.abortController = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.collectedUsage) {
|
|
|
|
|
client.collectedUsage = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.indexTokenCountMap) {
|
|
|
|
|
client.indexTokenCountMap = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.agentConfigs) {
|
|
|
|
|
client.agentConfigs = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.artifactPromises) {
|
|
|
|
|
client.artifactPromises = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.usage) {
|
|
|
|
|
client.usage = null;
|
|
|
|
|
}
|
|
|
|
|
if (typeof client.dispose === 'function') {
|
|
|
|
|
client.dispose();
|
|
|
|
|
}
|
|
|
|
|
if (client.options) {
|
|
|
|
|
if (client.options.req) {
|
|
|
|
|
client.options.req = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.options.res) {
|
|
|
|
|
client.options.res = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.options.attachments) {
|
|
|
|
|
client.options.attachments = null;
|
|
|
|
|
}
|
|
|
|
|
if (client.options.agent) {
|
|
|
|
|
client.options.agent = null;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
client.options = null;
|
2025-08-16 20:45:55 -04:00
|
|
|
} catch {
|
2025-04-12 18:46:36 -04:00
|
|
|
// Ignore errors during disposal
|
2025-11-10 17:12:06 -05:00
|
|
|
} finally {
|
|
|
|
|
logger.debug('[disposeClient] Client disposed');
|
2025-04-12 18:46:36 -04:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function processReqData(data = {}, context) {
|
|
|
|
|
let {
|
|
|
|
|
abortKey,
|
|
|
|
|
userMessage,
|
|
|
|
|
userMessagePromise,
|
|
|
|
|
responseMessageId,
|
|
|
|
|
promptTokens,
|
|
|
|
|
conversationId,
|
|
|
|
|
userMessageId,
|
|
|
|
|
} = context;
|
|
|
|
|
for (const key in data) {
|
|
|
|
|
if (key === 'userMessage') {
|
|
|
|
|
userMessage = data[key];
|
|
|
|
|
userMessageId = data[key].messageId;
|
|
|
|
|
} else if (key === 'userMessagePromise') {
|
|
|
|
|
userMessagePromise = data[key];
|
|
|
|
|
} else if (key === 'responseMessageId') {
|
|
|
|
|
responseMessageId = data[key];
|
|
|
|
|
} else if (key === 'promptTokens') {
|
|
|
|
|
promptTokens = data[key];
|
|
|
|
|
} else if (key === 'abortKey') {
|
|
|
|
|
abortKey = data[key];
|
|
|
|
|
} else if (!conversationId && key === 'conversationId') {
|
|
|
|
|
conversationId = data[key];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return {
|
|
|
|
|
abortKey,
|
|
|
|
|
userMessage,
|
|
|
|
|
userMessagePromise,
|
|
|
|
|
responseMessageId,
|
|
|
|
|
promptTokens,
|
|
|
|
|
conversationId,
|
|
|
|
|
userMessageId,
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
module.exports = {
|
|
|
|
|
disposeClient,
|
|
|
|
|
requestDataMap,
|
|
|
|
|
clientRegistry,
|
|
|
|
|
processReqData,
|
|
|
|
|
};
|