|
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";
|
|
|
|
|
|
|
|
|
|
export class PromptExecution {
|
|
id: string;
|
|
promptApi: ComfyApiFormat | null = null;
|
|
executedNodeIds: string[] = [];
|
|
totalNodes: number = 0;
|
|
currentlyExecuting: {
|
|
nodeId: string;
|
|
nodeLabel?: string;
|
|
step?: number;
|
|
maxSteps?: number;
|
|
|
|
pass: number;
|
|
|
|
|
|
|
|
|
|
maxPasses?: number;
|
|
} | null = null;
|
|
errorDetails: any | null = null;
|
|
|
|
apiPrompt: Resolver<null> = getResolver();
|
|
|
|
constructor(id: string) {
|
|
this.id = id;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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;
|
|
}
|
|
|
|
|
|
|
|
|
|
executing(nodeId: string | null, step?: number, maxSteps?: number) {
|
|
if (nodeId == null) {
|
|
|
|
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 };
|
|
|
|
|
|
|
|
|
|
this.apiPrompt.promise.then(() => {
|
|
|
|
|
|
|
|
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") {
|
|
|
|
|
|
|
|
|
|
|
|
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 (!this.currentlyExecuting!.step || step < this.currentlyExecuting!.step) {
|
|
this.currentlyExecuting!.pass!++;
|
|
}
|
|
this.currentlyExecuting!.step = step;
|
|
this.currentlyExecuting!.maxSteps = maxSteps;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
error(details: any) {
|
|
this.errorDetails = details;
|
|
}
|
|
|
|
private maybeGetComfyGraph(): TLGraph | null {
|
|
return ((window as any)?.app?.graph as TLGraph) || null;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
class PromptService extends EventTarget {
|
|
promptsMap: Map<string, PromptExecution> = new Map();
|
|
currentExecution: PromptExecution | null = null;
|
|
lastQueueRemaining = 0;
|
|
|
|
constructor(api: any) {
|
|
super();
|
|
const that = this;
|
|
|
|
|
|
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." });
|
|
|
|
throw e;
|
|
}
|
|
|
|
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>) => {
|
|
|
|
|
|
if (!e.detail?.exec_info) return;
|
|
this.lastQueueRemaining = e.detail.exec_info.queue_remaining;
|
|
this.dispatchProgressUpdate();
|
|
});
|
|
|
|
api.addEventListener("execution_start", (e: CustomEvent<ComfyApiEventDetailExecutionStart>) => {
|
|
|
|
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>) => {
|
|
|
|
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>) => {
|
|
|
|
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>) => {
|
|
|
|
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>) => {
|
|
|
|
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>) => {
|
|
|
|
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();
|
|
});
|
|
}
|
|
|
|
|
|
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);
|
|
|