mirror of
https://github.com/Comfy-Org/ComfyUI_frontend.git
synced 2026-02-04 07:00:23 +00:00
## Summary Reorganized custom nodes manager functionality from scattered technical layers into a cohesive domain-focused module following [domain-driven design](https://en.wikipedia.org/wiki/Domain-driven_design) principles. ## Changes - **What**: Migrated all manager code from technical layers (`src/components/`, `src/stores/`, etc.) to unified domain structure at `src/workbench/extensions/manager/` - **Breaking**: Import paths changed for all manager-related modules (40+ files updated) ## Review Focus Verify all import path updates are correct and no circular dependencies introduced. Check that [Vue 3 composition API](https://vuejs.org/guide/reusability/composables.html) patterns remain consistent across relocated composables. ┆Issue is synchronized with this [Notion page](https://www.notion.so/PR-5662-refactor-Migrate-manager-code-to-DDD-structure-2736d73d3650812c87faf6ed0fffb196) by [Unito](https://www.unito.io)
171 lines
5.5 KiB
TypeScript
171 lines
5.5 KiB
TypeScript
import { useEventListener } from '@vueuse/core'
|
|
import { pickBy } from 'es-toolkit/compat'
|
|
import type { Ref } from 'vue'
|
|
import { computed, ref } from 'vue'
|
|
|
|
import { app } from '@/scripts/app'
|
|
import { useDialogService } from '@/services/dialogService'
|
|
import { normalizePackKeys } from '@/utils/packUtils'
|
|
import type { components } from '@/workbench/extensions/manager/types/generatedManagerTypes'
|
|
|
|
type ManagerTaskHistory = Record<
|
|
string,
|
|
components['schemas']['TaskHistoryItem']
|
|
>
|
|
type ManagerTaskQueue = components['schemas']['TaskStateMessage']
|
|
type ManagerWsTaskDoneMsg = components['schemas']['MessageTaskDone']
|
|
type ManagerWsTaskStartedMsg = components['schemas']['MessageTaskStarted']
|
|
type QueueTaskItem = components['schemas']['QueueTaskItem']
|
|
type HistoryTaskItem = components['schemas']['TaskHistoryItem']
|
|
type Task = QueueTaskItem | HistoryTaskItem
|
|
|
|
const MANAGER_WS_TASK_DONE_NAME = 'cm-task-completed'
|
|
const MANAGER_WS_TASK_STARTED_NAME = 'cm-task-started'
|
|
|
|
export const useManagerQueue = (
|
|
taskHistory: Ref<ManagerTaskHistory>,
|
|
taskQueue: Ref<ManagerTaskQueue>,
|
|
installedPacks: Ref<Record<string, any>>
|
|
) => {
|
|
const { showManagerProgressDialog } = useDialogService()
|
|
|
|
// Task queue state (read-only from server)
|
|
const maxHistoryItems = ref(64)
|
|
const isLoading = ref(false)
|
|
const isProcessing = ref(false)
|
|
|
|
// Computed values
|
|
const currentQueueLength = computed(
|
|
() =>
|
|
taskQueue.value.running_queue.length +
|
|
taskQueue.value.pending_queue.length
|
|
)
|
|
|
|
/**
|
|
* Update the processing state based on the current queue length.
|
|
* If the queue is empty, or all tasks in the queue are associated
|
|
* with different clients, then this client is not processing any tasks.
|
|
*/
|
|
const updateProcessingState = (): void => {
|
|
isProcessing.value = currentQueueLength.value > 0
|
|
}
|
|
|
|
const allTasksDone = computed(() => currentQueueLength.value === 0)
|
|
const historyCount = computed(() => Object.keys(taskHistory.value).length)
|
|
|
|
/**
|
|
* Check if a task is associated with this client.
|
|
* Task can be from running queue, pending queue, or history.
|
|
* @param task - The task to check
|
|
* @returns True if the task belongs to this client
|
|
*/
|
|
const isTaskFromThisClient = (task: Task): boolean =>
|
|
task.client_id === app.api.clientId
|
|
|
|
/**
|
|
* Check if a history task is associated with this client.
|
|
* @param task - The history task to check
|
|
* @returns True if the task belongs to this client
|
|
*/
|
|
const isHistoryTaskFromThisClient = (task: HistoryTaskItem): boolean =>
|
|
task.client_id === app.api.clientId
|
|
|
|
/**
|
|
* Filter queue tasks by client id.
|
|
* Ensures that only tasks associated with this client are processed and
|
|
* added to client state.
|
|
* @param tasks - Array of queue tasks to filter
|
|
* @returns Filtered array containing only tasks from this client
|
|
*/
|
|
const filterQueueByClientId = (tasks: QueueTaskItem[]): QueueTaskItem[] =>
|
|
tasks.filter(isTaskFromThisClient)
|
|
|
|
/**
|
|
* Filter history tasks by client id using pickBy for optimal performance.
|
|
* Returns a new object containing only tasks associated with this client.
|
|
* @param history - The history object to filter
|
|
* @returns Filtered history object containing only tasks from this client
|
|
*/
|
|
const filterHistoryByClientId = (history: ManagerTaskHistory) =>
|
|
pickBy(history, isHistoryTaskFromThisClient)
|
|
|
|
/**
|
|
* Update task queue and history state with filtered data from server.
|
|
* Ensures only tasks from this client are stored in local state.
|
|
* @param state - The task state message from the server
|
|
*/
|
|
const updateTaskState = (state: ManagerTaskQueue) => {
|
|
taskQueue.value.running_queue = filterQueueByClientId(state.running_queue)
|
|
taskQueue.value.pending_queue = filterQueueByClientId(state.pending_queue)
|
|
taskHistory.value = filterHistoryByClientId(state.history)
|
|
|
|
if (state.installed_packs) {
|
|
// Normalize pack keys to ensure consistent access
|
|
installedPacks.value = normalizePackKeys(state.installed_packs)
|
|
}
|
|
updateProcessingState()
|
|
}
|
|
|
|
// WebSocket event listener for task done
|
|
const cleanupTaskDoneListener = useEventListener(
|
|
app.api,
|
|
MANAGER_WS_TASK_DONE_NAME,
|
|
(event: CustomEvent<ManagerWsTaskDoneMsg>) => {
|
|
if (event?.type === MANAGER_WS_TASK_DONE_NAME && event.detail?.state) {
|
|
updateTaskState(event.detail.state)
|
|
|
|
// If no more tasks are running/pending, hide the progress dialog
|
|
if (allTasksDone.value) {
|
|
setTimeout(() => {
|
|
if (allTasksDone.value) {
|
|
showManagerProgressDialog()
|
|
}
|
|
}, 1000) // Small delay to let users see completion
|
|
}
|
|
}
|
|
}
|
|
)
|
|
|
|
// WebSocket event listener for task started
|
|
const cleanupTaskStartedListener = useEventListener(
|
|
app.api,
|
|
MANAGER_WS_TASK_STARTED_NAME,
|
|
(event: CustomEvent<ManagerWsTaskStartedMsg>) => {
|
|
if (event?.type === MANAGER_WS_TASK_STARTED_NAME && event.detail?.state) {
|
|
updateTaskState(event.detail.state)
|
|
|
|
// Show progress dialog when a task starts
|
|
showManagerProgressDialog()
|
|
}
|
|
}
|
|
)
|
|
|
|
/**
|
|
* Cleanup function to remove event listeners and reset state
|
|
*/
|
|
const cleanup = () => {
|
|
cleanupTaskDoneListener()
|
|
cleanupTaskStartedListener()
|
|
|
|
// Reset state
|
|
isProcessing.value = false
|
|
isLoading.value = false
|
|
}
|
|
|
|
return {
|
|
// State
|
|
isLoading,
|
|
isProcessing,
|
|
maxHistoryItems,
|
|
|
|
// Computed
|
|
allTasksDone,
|
|
historyCount,
|
|
currentQueueLength,
|
|
|
|
// Actions
|
|
updateTaskState,
|
|
cleanup
|
|
}
|
|
}
|