multimodalart's picture
Squashing commit
4450790 verified
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);