ComfyUI/web/scripts/api.js

300 lines
7.5 KiB
JavaScript
Raw Normal View History

2023-03-02 21:33:50 +00:00
class ComfyApi extends EventTarget {
#registered = new Set();
2023-03-02 21:33:50 +00:00
constructor() {
2023-03-03 15:20:49 +00:00
super();
2023-03-02 21:33:50 +00:00
}
addEventListener(type, callback, options) {
super.addEventListener(type, callback, options);
this.#registered.add(type);
}
2023-03-03 15:47:33 +00:00
/**
* Poll status for colab and other things that don't support websockets.
*/
2023-03-02 21:33:50 +00:00
#pollQueue() {
setInterval(async () => {
try {
const resp = await fetch("./prompt");
2023-03-02 21:33:50 +00:00
const status = await resp.json();
this.dispatchEvent(new CustomEvent("status", { detail: status }));
} catch (error) {
this.dispatchEvent(new CustomEvent("status", { detail: null }));
}
}, 1000);
}
2023-03-03 15:47:33 +00:00
/**
* Creates and connects a WebSocket for realtime updates
* @param {boolean} isReconnect If the socket is connection is a reconnect attempt
*/
2023-03-02 21:33:50 +00:00
#createSocket(isReconnect) {
if (this.socket) {
return;
}
let opened = false;
let existingSession = window.name;
if (existingSession) {
existingSession = "?clientId=" + existingSession;
}
this.socket = new WebSocket(
`ws${window.location.protocol === "https:" ? "s" : ""}://${location.host}${location.pathname}ws${existingSession}`
);
2023-05-31 01:43:29 +00:00
this.socket.binaryType = "arraybuffer";
2023-03-02 21:33:50 +00:00
this.socket.addEventListener("open", () => {
opened = true;
if (isReconnect) {
this.dispatchEvent(new CustomEvent("reconnected"));
}
});
this.socket.addEventListener("error", () => {
if (this.socket) this.socket.close();
2023-03-06 16:47:02 +00:00
if (!isReconnect && !opened) {
2023-03-03 21:13:11 +00:00
this.#pollQueue();
}
2023-03-02 21:33:50 +00:00
});
this.socket.addEventListener("close", () => {
setTimeout(() => {
this.socket = null;
this.#createSocket(true);
}, 300);
if (opened) {
this.dispatchEvent(new CustomEvent("status", { detail: null }));
this.dispatchEvent(new CustomEvent("reconnecting"));
}
});
this.socket.addEventListener("message", (event) => {
try {
2023-05-31 01:43:29 +00:00
if (event.data instanceof ArrayBuffer) {
const view = new DataView(event.data);
const eventType = view.getUint32(0);
const buffer = event.data.slice(4);
switch (eventType) {
case 1:
const view2 = new DataView(event.data);
const imageType = view2.getUint32(0)
let imageMime
switch (imageType) {
case 1:
default:
imageMime = "image/jpeg";
break;
case 2:
imageMime = "image/png"
2023-03-02 21:33:50 +00:00
}
2023-05-31 02:15:06 +00:00
const imageBlob = new Blob([buffer.slice(4)], { type: imageMime });
this.dispatchEvent(new CustomEvent("b_preview", { detail: imageBlob }));
break;
2023-03-02 21:33:50 +00:00
default:
2023-05-31 01:43:29 +00:00
throw new Error(`Unknown binary websocket message of type ${eventType}`);
}
}
else {
const msg = JSON.parse(event.data);
switch (msg.type) {
case "status":
if (msg.data.sid) {
this.clientId = msg.data.sid;
window.name = this.clientId;
}
this.dispatchEvent(new CustomEvent("status", { detail: msg.data.status }));
break;
case "progress":
this.dispatchEvent(new CustomEvent("progress", { detail: msg.data }));
break;
case "executing":
this.dispatchEvent(new CustomEvent("executing", { detail: msg.data.node }));
break;
case "executed":
this.dispatchEvent(new CustomEvent("executed", { detail: msg.data }));
break;
case "execution_start":
this.dispatchEvent(new CustomEvent("execution_start", { detail: msg.data }));
break;
case "execution_error":
this.dispatchEvent(new CustomEvent("execution_error", { detail: msg.data }));
break;
case "execution_cached":
this.dispatchEvent(new CustomEvent("execution_cached", { detail: msg.data }));
break;
2023-05-31 01:43:29 +00:00
default:
if (this.#registered.has(msg.type)) {
this.dispatchEvent(new CustomEvent(msg.type, { detail: msg.data }));
} else {
throw new Error(`Unknown message type ${msg.type}`);
}
}
2023-03-02 21:33:50 +00:00
}
} catch (error) {
2023-05-31 01:43:29 +00:00
console.warn("Unhandled message:", event.data, error);
2023-03-02 21:33:50 +00:00
}
});
}
2023-03-03 15:47:33 +00:00
/**
* Initialises sockets and realtime updates
*/
2023-03-03 15:20:49 +00:00
init() {
this.#createSocket();
}
2023-03-02 21:33:50 +00:00
2023-03-03 19:05:39 +00:00
/**
* Gets a list of extension urls
* @returns An array of script urls to import
*/
async getExtensions() {
const resp = await fetch("./extensions", { cache: "no-store" });
2023-03-03 19:05:39 +00:00
return await resp.json();
}
/**
* Gets a list of embedding names
* @returns An array of script urls to import
*/
async getEmbeddings() {
const resp = await fetch("./embeddings", { cache: "no-store" });
return await resp.json();
}
2023-03-03 15:47:33 +00:00
/**
* Loads node object definitions for the graph
* @returns The node definitions
*/
async getNodeDefs() {
const resp = await fetch("./object_info", { cache: "no-store" });
return await resp.json();
}
2023-03-03 15:47:33 +00:00
/**
*
* @param {number} number The index at which to queue the prompt, passing -1 will insert the prompt at the front of the queue
* @param {object} prompt The prompt data to queue
*/
async queuePrompt(number, { output, workflow }) {
const body = {
client_id: this.clientId,
prompt: output,
extra_data: { extra_pnginfo: { workflow } },
};
if (number === -1) {
body.front = true;
} else if (number != 0) {
body.number = number;
}
const res = await fetch("./prompt", {
method: "POST",
headers: {
"Content-Type": "application/json",
},
body: JSON.stringify(body),
});
if (res.status !== 200) {
throw {
response: await res.json(),
};
}
}
2023-03-03 15:20:49 +00:00
2023-03-03 15:47:33 +00:00
/**
* Loads a list of items (queue or history)
* @param {string} type The type of items to load, queue or history
* @returns The items of the specified type grouped by their status
*/
2023-03-03 15:20:49 +00:00
async getItems(type) {
if (type === "queue") {
return this.getQueue();
}
return this.getHistory();
}
2023-03-03 15:47:33 +00:00
/**
* Gets the current state of the queue
* @returns The currently running and queued items
*/
2023-03-03 15:20:49 +00:00
async getQueue() {
try {
const res = await fetch("./queue");
2023-03-03 15:20:49 +00:00
const data = await res.json();
return {
// Running action uses a different endpoint for cancelling
2023-03-03 15:47:33 +00:00
Running: data.queue_running.map((prompt) => ({
prompt,
remove: { name: "Cancel", cb: () => api.interrupt() },
})),
2023-03-03 15:20:49 +00:00
Pending: data.queue_pending.map((prompt) => ({ prompt })),
};
} catch (error) {
console.error(error);
return { Running: [], Pending: [] };
}
}
2023-03-03 15:47:33 +00:00
/**
* Gets the prompt execution history
* @returns Prompt history including node outputs
*/
2023-03-03 15:20:49 +00:00
async getHistory() {
try {
const res = await fetch("./history");
2023-03-03 15:20:49 +00:00
return { History: Object.values(await res.json()) };
} catch (error) {
console.error(error);
return { History: [] };
}
}
2023-03-03 15:47:33 +00:00
/**
* Sends a POST request to the API
* @param {*} type The endpoint to post to
* @param {*} body Optional POST data
*/
2023-03-03 15:20:49 +00:00
async #postItem(type, body) {
try {
await fetch("./" + type, {
2023-03-03 15:20:49 +00:00
method: "POST",
headers: {
"Content-Type": "application/json",
},
body: body ? JSON.stringify(body) : undefined,
});
} catch (error) {
console.error(error);
}
}
2023-03-03 15:47:33 +00:00
/**
* Deletes an item from the specified list
* @param {string} type The type of item to delete, queue or history
* @param {number} id The id of the item to delete
*/
2023-03-03 15:20:49 +00:00
async deleteItem(type, id) {
await this.#postItem(type, { delete: [id] });
}
2023-03-03 15:47:33 +00:00
/**
* Clears the specified list
* @param {string} type The type of list to clear, queue or history
*/
2023-03-03 15:20:49 +00:00
async clearItems(type) {
await this.#postItem(type, { clear: true });
}
2023-03-03 15:47:33 +00:00
/**
* Interrupts the execution of the running prompt
*/
2023-03-03 15:20:49 +00:00
async interrupt() {
await this.#postItem("interrupt", null);
}
}
export const api = new ComfyApi();