Spaces:
Running
on
Zero
Running
on
Zero
import type { | |
ComfyApiEventDetailCached, | |
ComfyApiEventDetailError, | |
ComfyApiEventDetailExecuted, | |
ComfyApiEventDetailExecuting, | |
ComfyApiEventDetailExecutionStart, | |
ComfyApiEventDetailProgress, | |
ComfyApiEventDetailStatus, | |
ComfyApiFormat, | |
ComfyApiPrompt, | |
} from "typings/comfy.js"; | |
import { api } from "scripts/api.js"; | |
import type { LGraph as TLGraph, LGraphCanvas as TLGraphCanvas } from "typings/litegraph.js"; | |
import { Resolver, getResolver } from "./shared_utils.js"; | |
/** | |
* Wraps general data of a prompt's execution. | |
*/ | |
export class PromptExecution { | |
id: string; | |
promptApi: ComfyApiFormat | null = null; | |
executedNodeIds: string[] = []; | |
totalNodes: number = 0; | |
currentlyExecuting: { | |
nodeId: string; | |
nodeLabel?: string; | |
step?: number; | |
maxSteps?: number; | |
/** The current pass, for nodes with multiple progress passes. */ | |
pass: number; | |
/** | |
* The max num of passes. Can be calculated for some nodes, or set to -1 when known there will | |
* be multiple passes, but the number cannot be calculated. | |
*/ | |
maxPasses?: number; | |
} | null = null; | |
errorDetails: any | null = null; | |
apiPrompt: Resolver<null> = getResolver(); | |
constructor(id: string) { | |
this.id = id; | |
} | |
/** | |
* Sets the prompt and prompt-related data. This can technically come in lazily, like if the web | |
* socket fires the 'execution-start' event before we actually get a response back from the | |
* initial prompt call. | |
*/ | |
setPrompt(prompt: ComfyApiPrompt) { | |
this.promptApi = prompt.output; | |
this.totalNodes = Object.keys(this.promptApi).length; | |
this.apiPrompt.resolve(null); | |
} | |
getApiNode(nodeId: string | number) { | |
return this.promptApi?.[String(nodeId)] || null; | |
} | |
private getNodeLabel(nodeId: string | number) { | |
const apiNode = this.getApiNode(nodeId); | |
let label = apiNode?._meta?.title || apiNode?.class_type || undefined; | |
if (!label) { | |
const graphNode = this.maybeGetComfyGraph()?.getNodeById(Number(nodeId)); | |
label = graphNode?.title || graphNode?.type || undefined; | |
} | |
return label; | |
} | |
/** | |
* Updates the execution data depending on the passed data, fed from api events. | |
*/ | |
executing(nodeId: string | null, step?: number, maxSteps?: number) { | |
if (nodeId == null) { | |
// We're done, any left over nodes must be skipped... | |
this.currentlyExecuting = null; | |
return; | |
} | |
if (this.currentlyExecuting?.nodeId !== nodeId) { | |
if (this.currentlyExecuting != null) { | |
this.executedNodeIds.push(nodeId); | |
} | |
this.currentlyExecuting = { nodeId, nodeLabel: this.getNodeLabel(nodeId), pass: 0 }; | |
// We'll see if we're known node for multiple passes, that will come in as generic 'progress' | |
// updates from the api. If we're known to have multiple passes, then we'll pre-set data to | |
// allow the progress bar to handle intial rendering. If we're not, that's OK, the data will | |
// be shown with the second pass. | |
this.apiPrompt.promise.then(() => { | |
// If we execute with a null node id and clear the currently executing, then we can just | |
// move on. This seems to only happen with a super-fast execution (like, just seed node | |
// and display any for testing). | |
if (this.currentlyExecuting == null) { | |
return; | |
} | |
const apiNode = this.getApiNode(nodeId); | |
if (!this.currentlyExecuting.nodeLabel) { | |
this.currentlyExecuting.nodeLabel = this.getNodeLabel(nodeId); | |
} | |
if (apiNode?.class_type === "UltimateSDUpscale") { | |
// From what I can tell, UltimateSDUpscale, does an initial pass that isn't actually a | |
// tile. It seems to always be 4 steps... We'll start our pass at -1, so this prepass is | |
// "0" and "1" will start with the first tile. This way, a user knows they have 4 tiles, | |
// know this pass counter will go to 4 (and not 5). Also, we cannot calculate maxPasses | |
// for 'UltimateSDUpscale' :( | |
this.currentlyExecuting.pass--; | |
this.currentlyExecuting.maxPasses = -1; | |
} else if (apiNode?.class_type === "IterativeImageUpscale") { | |
this.currentlyExecuting.maxPasses = (apiNode?.inputs["steps"] as number) ?? -1; | |
} | |
}); | |
} | |
if (step != null) { | |
// If we haven't had any stpes before, or the passes step is lower than the previous, then | |
// increase the passes. | |
if (!this.currentlyExecuting!.step || step < this.currentlyExecuting!.step) { | |
this.currentlyExecuting!.pass!++; | |
} | |
this.currentlyExecuting!.step = step; | |
this.currentlyExecuting!.maxSteps = maxSteps; | |
} | |
} | |
/** | |
* If there's an error, we add the details. | |
*/ | |
error(details: any) { | |
this.errorDetails = details; | |
} | |
private maybeGetComfyGraph(): TLGraph | null { | |
return ((window as any)?.app?.graph as TLGraph) || null; | |
} | |
} | |
/** | |
* A singleton service that wraps the Comfy API and simplifies the event data being fired. | |
*/ | |
class PromptService extends EventTarget { | |
promptsMap: Map<string, PromptExecution> = new Map(); | |
currentExecution: PromptExecution | null = null; | |
lastQueueRemaining = 0; | |
constructor(api: any) { | |
super(); | |
const that = this; | |
// Patch the queuePrompt method so we can capture new data going through. | |
const queuePrompt = api.queuePrompt; | |
api.queuePrompt = async function (num: number, prompt: ComfyApiPrompt) { | |
let response; | |
try { | |
response = await queuePrompt.apply(api, [...arguments]); | |
} catch (e) { | |
const promptExecution = that.getOrMakePrompt("error"); | |
promptExecution.error({ exception_type: "Unknown." }); | |
// console.log("ERROR QUEUE PROMPT", response, arguments); | |
throw e; | |
} | |
// console.log("QUEUE PROMPT", response, arguments); | |
const promptExecution = that.getOrMakePrompt(response.prompt_id); | |
promptExecution.setPrompt(prompt); | |
if (!that.currentExecution) { | |
that.currentExecution = promptExecution; | |
} | |
that.promptsMap.set(response.prompt_id, promptExecution); | |
that.dispatchEvent( | |
new CustomEvent("queue-prompt", { | |
detail: { | |
prompt: promptExecution, | |
}, | |
}), | |
); | |
return response; | |
}; | |
api.addEventListener("status", (e: CustomEvent<ComfyApiEventDetailStatus>) => { | |
// console.log("status", JSON.stringify(e.detail)); | |
// Sometimes a status message is fired when the app loades w/o any details. | |
if (!e.detail?.exec_info) return; | |
this.lastQueueRemaining = e.detail.exec_info.queue_remaining; | |
this.dispatchProgressUpdate(); | |
}); | |
api.addEventListener("execution_start", (e: CustomEvent<ComfyApiEventDetailExecutionStart>) => { | |
// console.log("execution_start", JSON.stringify(e.detail)); | |
if (!this.promptsMap.has(e.detail.prompt_id)) { | |
console.warn("'execution_start' fired before prompt was made."); | |
} | |
const prompt = this.getOrMakePrompt(e.detail.prompt_id); | |
this.currentExecution = prompt; | |
this.dispatchProgressUpdate(); | |
}); | |
api.addEventListener("executing", (e: CustomEvent<ComfyApiEventDetailExecuting>) => { | |
// console.log("executing", JSON.stringify(e.detail)); | |
if (!this.currentExecution) { | |
this.currentExecution = this.getOrMakePrompt("unknown"); | |
console.warn("'executing' fired before prompt was made."); | |
} | |
this.currentExecution.executing(e.detail); | |
this.dispatchProgressUpdate(); | |
if (e.detail == null) { | |
this.currentExecution = null; | |
} | |
}); | |
api.addEventListener("progress", (e: CustomEvent<ComfyApiEventDetailProgress>) => { | |
// console.log("progress", JSON.stringify(e.detail)); | |
if (!this.currentExecution) { | |
this.currentExecution = this.getOrMakePrompt(e.detail.prompt_id); | |
console.warn("'progress' fired before prompt was made."); | |
} | |
this.currentExecution.executing(e.detail.node, e.detail.value, e.detail.max); | |
this.dispatchProgressUpdate(); | |
}); | |
api.addEventListener("execution_cached", (e: CustomEvent<ComfyApiEventDetailCached>) => { | |
// console.log("execution_cached", JSON.stringify(e.detail)); | |
if (!this.currentExecution) { | |
this.currentExecution = this.getOrMakePrompt(e.detail.prompt_id); | |
console.warn("'execution_cached' fired before prompt was made."); | |
} | |
for (const cached of e.detail.nodes) { | |
this.currentExecution.executing(cached); | |
} | |
this.dispatchProgressUpdate(); | |
}); | |
api.addEventListener("executed", (e: CustomEvent<ComfyApiEventDetailExecuted>) => { | |
// console.log("executed", JSON.stringify(e.detail)); | |
if (!this.currentExecution) { | |
this.currentExecution = this.getOrMakePrompt(e.detail.prompt_id); | |
console.warn("'executed' fired before prompt was made."); | |
} | |
}); | |
api.addEventListener("execution_error", (e: CustomEvent<ComfyApiEventDetailError>) => { | |
// console.log("execution_error", e.detail); | |
if (!this.currentExecution) { | |
this.currentExecution = this.getOrMakePrompt(e.detail.prompt_id); | |
console.warn("'execution_error' fired before prompt was made."); | |
} | |
this.currentExecution?.error(e.detail); | |
this.dispatchProgressUpdate(); | |
}); | |
} | |
/** A helper method, since we extend/override api.queuePrompt above anyway. */ | |
async queuePrompt(prompt: ComfyApiPrompt) { | |
return await api.queuePrompt(-1, prompt); | |
} | |
dispatchProgressUpdate() { | |
this.dispatchEvent( | |
new CustomEvent("progress-update", { | |
detail: { | |
queue: this.lastQueueRemaining, | |
prompt: this.currentExecution, | |
}, | |
}), | |
); | |
} | |
getOrMakePrompt(id: string) { | |
let prompt = this.promptsMap.get(id); | |
if (!prompt) { | |
prompt = new PromptExecution(id); | |
this.promptsMap.set(id, prompt); | |
} | |
return prompt; | |
} | |
} | |
export const SERVICE = new PromptService(api); | |