|  | class ComfyApi extends EventTarget { | 
					
						
						|  | #registered = new Set(); | 
					
						
						|  |  | 
					
						
						|  | constructor() { | 
					
						
						|  | super(); | 
					
						
						|  | this.api_host = location.host; | 
					
						
						|  | this.api_base = location.pathname.split('/').slice(0, -1).join('/'); | 
					
						
						|  | this.initialClientId = sessionStorage.getItem("clientId"); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | apiURL(route) { | 
					
						
						|  | return this.api_base + route; | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | fetchApi(route, options) { | 
					
						
						|  | if (!options) { | 
					
						
						|  | options = {}; | 
					
						
						|  | } | 
					
						
						|  | if (!options.headers) { | 
					
						
						|  | options.headers = {}; | 
					
						
						|  | } | 
					
						
						|  | options.headers["Comfy-User"] = this.user; | 
					
						
						|  | return fetch(this.apiURL(route), options); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | addEventListener(type, callback, options) { | 
					
						
						|  | super.addEventListener(type, callback, options); | 
					
						
						|  | this.#registered.add(type); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | #pollQueue() { | 
					
						
						|  | setInterval(async () => { | 
					
						
						|  | try { | 
					
						
						|  | const resp = await this.fetchApi("/prompt"); | 
					
						
						|  | const status = await resp.json(); | 
					
						
						|  | this.dispatchEvent(new CustomEvent("status", { detail: status })); | 
					
						
						|  | } catch (error) { | 
					
						
						|  | this.dispatchEvent(new CustomEvent("status", { detail: null })); | 
					
						
						|  | } | 
					
						
						|  | }, 1000); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | #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" : ""}://${this.api_host}${this.api_base}/ws${existingSession}` | 
					
						
						|  | ); | 
					
						
						|  | this.socket.binaryType = "arraybuffer"; | 
					
						
						|  |  | 
					
						
						|  | this.socket.addEventListener("open", () => { | 
					
						
						|  | opened = true; | 
					
						
						|  | if (isReconnect) { | 
					
						
						|  | this.dispatchEvent(new CustomEvent("reconnected")); | 
					
						
						|  | } | 
					
						
						|  | }); | 
					
						
						|  |  | 
					
						
						|  | this.socket.addEventListener("error", () => { | 
					
						
						|  | if (this.socket) this.socket.close(); | 
					
						
						|  | if (!isReconnect && !opened) { | 
					
						
						|  | this.#pollQueue(); | 
					
						
						|  | } | 
					
						
						|  | }); | 
					
						
						|  |  | 
					
						
						|  | 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 { | 
					
						
						|  | 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" | 
					
						
						|  | } | 
					
						
						|  | const imageBlob = new Blob([buffer.slice(4)], { type: imageMime }); | 
					
						
						|  | this.dispatchEvent(new CustomEvent("b_preview", { detail: imageBlob })); | 
					
						
						|  | break; | 
					
						
						|  | default: | 
					
						
						|  | 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; | 
					
						
						|  | sessionStorage.setItem("clientId", 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_success": | 
					
						
						|  | this.dispatchEvent(new CustomEvent("execution_success", { 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; | 
					
						
						|  | 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}`); | 
					
						
						|  | } | 
					
						
						|  | } | 
					
						
						|  | } | 
					
						
						|  | } catch (error) { | 
					
						
						|  | console.warn("Unhandled message:", event.data, error); | 
					
						
						|  | } | 
					
						
						|  | }); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | init() { | 
					
						
						|  | this.#createSocket(); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | async getExtensions() { | 
					
						
						|  | const resp = await this.fetchApi("/extensions", { cache: "no-store" }); | 
					
						
						|  | return await resp.json(); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | async getEmbeddings() { | 
					
						
						|  | const resp = await this.fetchApi("/embeddings", { cache: "no-store" }); | 
					
						
						|  | return await resp.json(); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | async getNodeDefs() { | 
					
						
						|  | const resp = await this.fetchApi("/object_info", { cache: "no-store" }); | 
					
						
						|  | return await resp.json(); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | 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 this.fetchApi("/prompt", { | 
					
						
						|  | method: "POST", | 
					
						
						|  | headers: { | 
					
						
						|  | "Content-Type": "application/json", | 
					
						
						|  | }, | 
					
						
						|  | body: JSON.stringify(body), | 
					
						
						|  | }); | 
					
						
						|  |  | 
					
						
						|  | if (res.status !== 200) { | 
					
						
						|  | throw { | 
					
						
						|  | response: await res.json(), | 
					
						
						|  | }; | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | return await res.json(); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | async getItems(type) { | 
					
						
						|  | if (type === "queue") { | 
					
						
						|  | return this.getQueue(); | 
					
						
						|  | } | 
					
						
						|  | return this.getHistory(); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | async getQueue() { | 
					
						
						|  | try { | 
					
						
						|  | const res = await this.fetchApi("/queue"); | 
					
						
						|  | const data = await res.json(); | 
					
						
						|  | return { | 
					
						
						|  |  | 
					
						
						|  | Running: data.queue_running.map((prompt) => ({ | 
					
						
						|  | prompt, | 
					
						
						|  | remove: { name: "Cancel", cb: () => api.interrupt() }, | 
					
						
						|  | })), | 
					
						
						|  | Pending: data.queue_pending.map((prompt) => ({ prompt })), | 
					
						
						|  | }; | 
					
						
						|  | } catch (error) { | 
					
						
						|  | console.error(error); | 
					
						
						|  | return { Running: [], Pending: [] }; | 
					
						
						|  | } | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | async getHistory(max_items=200) { | 
					
						
						|  | try { | 
					
						
						|  | const res = await this.fetchApi(`/history?max_items=${max_items}`); | 
					
						
						|  | return { History: Object.values(await res.json()) }; | 
					
						
						|  | } catch (error) { | 
					
						
						|  | console.error(error); | 
					
						
						|  | return { History: [] }; | 
					
						
						|  | } | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | async getSystemStats() { | 
					
						
						|  | const res = await this.fetchApi("/system_stats"); | 
					
						
						|  | return await res.json(); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | async #postItem(type, body) { | 
					
						
						|  | try { | 
					
						
						|  | await this.fetchApi("/" + type, { | 
					
						
						|  | method: "POST", | 
					
						
						|  | headers: { | 
					
						
						|  | "Content-Type": "application/json", | 
					
						
						|  | }, | 
					
						
						|  | body: body ? JSON.stringify(body) : undefined, | 
					
						
						|  | }); | 
					
						
						|  | } catch (error) { | 
					
						
						|  | console.error(error); | 
					
						
						|  | } | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | async deleteItem(type, id) { | 
					
						
						|  | await this.#postItem(type, { delete: [id] }); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | async clearItems(type) { | 
					
						
						|  | await this.#postItem(type, { clear: true }); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | async interrupt() { | 
					
						
						|  | await this.#postItem("interrupt", null); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | async getUserConfig() { | 
					
						
						|  | return (await this.fetchApi("/users")).json(); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | createUser(username) { | 
					
						
						|  | return this.fetchApi("/users", { | 
					
						
						|  | method: "POST", | 
					
						
						|  | headers: { | 
					
						
						|  | "Content-Type": "application/json", | 
					
						
						|  | }, | 
					
						
						|  | body: JSON.stringify({ username }), | 
					
						
						|  | }); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | async getSettings() { | 
					
						
						|  | return (await this.fetchApi("/settings")).json(); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | async getSetting(id) { | 
					
						
						|  | return (await this.fetchApi(`/settings/${encodeURIComponent(id)}`)).json(); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | async storeSettings(settings) { | 
					
						
						|  | return this.fetchApi(`/settings`, { | 
					
						
						|  | method: "POST", | 
					
						
						|  | body: JSON.stringify(settings) | 
					
						
						|  | }); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | async storeSetting(id, value) { | 
					
						
						|  | return this.fetchApi(`/settings/${encodeURIComponent(id)}`, { | 
					
						
						|  | method: "POST", | 
					
						
						|  | body: JSON.stringify(value) | 
					
						
						|  | }); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | async getUserData(file, options) { | 
					
						
						|  | return this.fetchApi(`/userdata/${encodeURIComponent(file)}`, options); | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | async storeUserData(file, data, options = { overwrite: true, stringify: true, throwOnError: true }) { | 
					
						
						|  | const resp = await this.fetchApi(`/userdata/${encodeURIComponent(file)}?overwrite=${options?.overwrite}`, { | 
					
						
						|  | method: "POST", | 
					
						
						|  | body: options?.stringify ? JSON.stringify(data) : data, | 
					
						
						|  | ...options, | 
					
						
						|  | }); | 
					
						
						|  | if (resp.status !== 200 && options?.throwOnError !== false) { | 
					
						
						|  | throw new Error(`Error storing user data file '${file}': ${resp.status} ${(await resp).statusText}`); | 
					
						
						|  | } | 
					
						
						|  | return resp; | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | async deleteUserData(file) { | 
					
						
						|  | const resp = await this.fetchApi(`/userdata/${encodeURIComponent(file)}`, { | 
					
						
						|  | method: "DELETE", | 
					
						
						|  | }); | 
					
						
						|  | if (resp.status !== 204) { | 
					
						
						|  | throw new Error(`Error removing user data file '${file}': ${resp.status} ${(resp).statusText}`); | 
					
						
						|  | } | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | async moveUserData(source, dest, options = { overwrite: false }) { | 
					
						
						|  | const resp = await this.fetchApi(`/userdata/${encodeURIComponent(source)}/move/${encodeURIComponent(dest)}?overwrite=${options?.overwrite}`, { | 
					
						
						|  | method: "POST", | 
					
						
						|  | }); | 
					
						
						|  | return resp; | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  |  | 
					
						
						|  | async listUserData(dir, recurse, split) { | 
					
						
						|  | const resp = await this.fetchApi( | 
					
						
						|  | `/userdata?${new URLSearchParams({ | 
					
						
						|  | recurse, | 
					
						
						|  | dir, | 
					
						
						|  | split, | 
					
						
						|  | })}` | 
					
						
						|  | ); | 
					
						
						|  | if (resp.status === 404) return []; | 
					
						
						|  | if (resp.status !== 200) { | 
					
						
						|  | throw new Error(`Error getting user data list '${dir}': ${resp.status} ${resp.statusText}`); | 
					
						
						|  | } | 
					
						
						|  | return resp.json(); | 
					
						
						|  | } | 
					
						
						|  | } | 
					
						
						|  |  | 
					
						
						|  | export const api = new ComfyApi(); | 
					
						
						|  |  |