From b0350e9bc887450e8e04cd11ff4b22132a743754 Mon Sep 17 00:00:00 2001
From: psychedelicious <4822129+psychedelicious@users.noreply.github.com>
Date: Fri, 1 Dec 2023 19:36:38 +1100
Subject: [PATCH] feat: workflow library - system graphs - wip
---
invokeai/app/api/dependencies.py | 2 +
invokeai/app/api/routers/workflows.py | 6 +-
.../assets/workflows/Text_to_Image_SD15.json | 787 ++++++++++++++++++
.../app/services/shared/system_workflows.py | 33 +
.../workflow_records/workflow_records_base.py | 7 +
.../workflow_records_common.py | 46 +-
.../workflow_records_sqlite.py | 55 +-
invokeai/frontend/web/public/locales/en.json | 7 +-
.../panels/TopCenterPanel/TopCenterPanel.tsx | 4 +-
.../src/features/nodes/store/nodesSlice.ts | 60 +-
.../web/src/features/nodes/types/workflow.ts | 4 +
.../nodes/util/workflow/buildWorkflow.ts | 16 +-
.../nodes/util/workflow/migrations.ts | 1 +
.../nodes/util/workflow/validateWorkflow.ts | 7 +
.../components/DuplicateWorkflowButton.tsx | 21 -
.../components/SaveWorkflowAsButton.tsx | 83 ++
.../components/WorkflowLibraryContent.tsx | 26 +-
...ryUserList.tsx => WorkflowLibraryList.tsx} | 39 +-
.../components/WorkflowLibraryListItem.tsx | 18 +-
.../hooks/useDuplicateWorkflow.ts | 50 +-
.../frontend/web/src/services/api/schema.d.ts | 79 +-
pyproject.toml | 1 +
22 files changed, 1167 insertions(+), 185 deletions(-)
create mode 100644 invokeai/app/assets/workflows/Text_to_Image_SD15.json
create mode 100644 invokeai/app/services/shared/system_workflows.py
delete mode 100644 invokeai/frontend/web/src/features/workflowLibrary/components/DuplicateWorkflowButton.tsx
create mode 100644 invokeai/frontend/web/src/features/workflowLibrary/components/SaveWorkflowAsButton.tsx
rename invokeai/frontend/web/src/features/workflowLibrary/components/{WorkflowLibraryUserList.tsx => WorkflowLibraryList.tsx} (82%)
diff --git a/invokeai/app/api/dependencies.py b/invokeai/app/api/dependencies.py
index d2ca877a45..bceaf43644 100644
--- a/invokeai/app/api/dependencies.py
+++ b/invokeai/app/api/dependencies.py
@@ -30,6 +30,7 @@ from ..services.session_queue.session_queue_sqlite import SqliteSessionQueue
from ..services.shared.default_graphs import create_system_graphs
from ..services.shared.graph import GraphExecutionState, LibraryGraph
from ..services.shared.sqlite.sqlite_database import SqliteDatabase
+from ..services.shared.system_workflows import create_system_workflows
from ..services.urls.urls_default import LocalUrlService
from ..services.workflow_records.workflow_records_sqlite import SqliteWorkflowRecordsStorage
from .events import FastAPIEventService
@@ -123,6 +124,7 @@ class ApiDependencies:
)
create_system_graphs(services.graph_library)
+ create_system_workflows(workflow_records=services.workflow_records, logger=logger)
db.clean()
ApiDependencies.invoker = Invoker(services)
diff --git a/invokeai/app/api/routers/workflows.py b/invokeai/app/api/routers/workflows.py
index 38fb659525..e757d3f317 100644
--- a/invokeai/app/api/routers/workflows.py
+++ b/invokeai/app/api/routers/workflows.py
@@ -7,6 +7,7 @@ from invokeai.app.services.shared.pagination import PaginatedResults
from invokeai.app.services.shared.sqlite.sqlite_common import SQLiteDirection
from invokeai.app.services.workflow_records.workflow_records_common import (
Workflow,
+ WorkflowCategory,
WorkflowNotFoundError,
WorkflowRecordDTO,
WorkflowRecordListItemDTO,
@@ -70,7 +71,7 @@ async def create_workflow(
workflow: WorkflowWithoutID = Body(description="The workflow to create", embed=True),
) -> WorkflowRecordDTO:
"""Creates a workflow"""
- return ApiDependencies.invoker.services.workflow_records.create(workflow)
+ return ApiDependencies.invoker.services.workflow_records.create(workflow=workflow)
@workflows_router.get(
@@ -85,9 +86,10 @@ async def list_workflows(
per_page: int = Query(default=10, description="The number of workflows per page"),
order_by: WorkflowRecordOrderBy = Query(default=WorkflowRecordOrderBy.Name, description="The order by"),
direction: SQLiteDirection = Query(default=SQLiteDirection.Ascending, description="The order by"),
+ category: WorkflowCategory = Query(default=WorkflowCategory.User, description="The category to get"),
filter_text: Optional[str] = Query(default=None, description="The name to filter by"),
) -> PaginatedResults[WorkflowRecordListItemDTO]:
"""Gets a page of workflows"""
return ApiDependencies.invoker.services.workflow_records.get_many(
- page=page, per_page=per_page, order_by=order_by, direction=direction, filter_text=filter_text
+ page=page, per_page=per_page, order_by=order_by, direction=direction, filter_text=filter_text, category=category
)
diff --git a/invokeai/app/assets/workflows/Text_to_Image_SD15.json b/invokeai/app/assets/workflows/Text_to_Image_SD15.json
new file mode 100644
index 0000000000..d660c576e9
--- /dev/null
+++ b/invokeai/app/assets/workflows/Text_to_Image_SD15.json
@@ -0,0 +1,787 @@
+{
+ "id": "af7030e2-c64f-4f03-b387-9634bb54ae5f",
+ "name": "Text to Image - SD1.5",
+ "author": "InvokeAI",
+ "description": "Sample text to image workflow for Stable Diffusion 1.5/2",
+ "version": "1.1.0",
+ "contact": "invoke@invoke.ai",
+ "tags": "text2image, SD1.5, SD2, default",
+ "notes": "",
+ "exposedFields": [
+ {
+ "nodeId": "c8d55139-f380-4695-b7f2-8b3d1e1e3db8",
+ "fieldName": "model"
+ },
+ {
+ "nodeId": "7d8bf987-284f-413a-b2fd-d825445a5d6c",
+ "fieldName": "prompt"
+ },
+ {
+ "nodeId": "93dc02a4-d05b-48ed-b99c-c9b616af3402",
+ "fieldName": "prompt"
+ },
+ {
+ "nodeId": "55705012-79b9-4aac-9f26-c0b10309785b",
+ "fieldName": "width"
+ },
+ {
+ "nodeId": "55705012-79b9-4aac-9f26-c0b10309785b",
+ "fieldName": "height"
+ }
+ ],
+ "meta": {
+ "category": "system",
+ "version": "2.0.0"
+ },
+ "nodes": [
+ {
+ "id": "93dc02a4-d05b-48ed-b99c-c9b616af3402",
+ "type": "invocation",
+ "data": {
+ "id": "93dc02a4-d05b-48ed-b99c-c9b616af3402",
+ "type": "compel",
+ "label": "Negative Compel Prompt",
+ "isOpen": true,
+ "notes": "",
+ "isIntermediate": true,
+ "useCache": true,
+ "version": "1.0.0",
+ "nodePack": "invokeai",
+ "inputs": {
+ "prompt": {
+ "id": "7739aff6-26cb-4016-8897-5a1fb2305e4e",
+ "name": "prompt",
+ "fieldKind": "input",
+ "label": "Negative Prompt",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "StringField"
+ },
+ "value": ""
+ },
+ "clip": {
+ "id": "48d23dce-a6ae-472a-9f8c-22a714ea5ce0",
+ "name": "clip",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "ClipField"
+ }
+ }
+ },
+ "outputs": {
+ "conditioning": {
+ "id": "37cf3a9d-f6b7-4b64-8ff6-2558c5ecc447",
+ "name": "conditioning",
+ "fieldKind": "output",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "ConditioningField"
+ }
+ }
+ }
+ },
+ "width": 320,
+ "height": 259,
+ "position": {
+ "x": 1000,
+ "y": 350
+ }
+ },
+ {
+ "id": "55705012-79b9-4aac-9f26-c0b10309785b",
+ "type": "invocation",
+ "data": {
+ "id": "55705012-79b9-4aac-9f26-c0b10309785b",
+ "type": "noise",
+ "label": "",
+ "isOpen": true,
+ "notes": "",
+ "isIntermediate": true,
+ "useCache": true,
+ "version": "1.0.1",
+ "nodePack": "invokeai",
+ "inputs": {
+ "seed": {
+ "id": "6431737c-918a-425d-a3b4-5d57e2f35d4d",
+ "name": "seed",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "IntegerField"
+ },
+ "value": 0
+ },
+ "width": {
+ "id": "38fc5b66-fe6e-47c8-bba9-daf58e454ed7",
+ "name": "width",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "IntegerField"
+ },
+ "value": 512
+ },
+ "height": {
+ "id": "16298330-e2bf-4872-a514-d6923df53cbb",
+ "name": "height",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "IntegerField"
+ },
+ "value": 512
+ },
+ "use_cpu": {
+ "id": "c7c436d3-7a7a-4e76-91e4-c6deb271623c",
+ "name": "use_cpu",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "BooleanField"
+ },
+ "value": true
+ }
+ },
+ "outputs": {
+ "noise": {
+ "id": "50f650dc-0184-4e23-a927-0497a96fe954",
+ "name": "noise",
+ "fieldKind": "output",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "LatentsField"
+ }
+ },
+ "width": {
+ "id": "bb8a452b-133d-42d1-ae4a-3843d7e4109a",
+ "name": "width",
+ "fieldKind": "output",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "IntegerField"
+ }
+ },
+ "height": {
+ "id": "35cfaa12-3b8b-4b7a-a884-327ff3abddd9",
+ "name": "height",
+ "fieldKind": "output",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "IntegerField"
+ }
+ }
+ }
+ },
+ "width": 320,
+ "height": 388,
+ "position": {
+ "x": 600,
+ "y": 325
+ }
+ },
+ {
+ "id": "c8d55139-f380-4695-b7f2-8b3d1e1e3db8",
+ "type": "invocation",
+ "data": {
+ "id": "c8d55139-f380-4695-b7f2-8b3d1e1e3db8",
+ "type": "main_model_loader",
+ "label": "",
+ "isOpen": true,
+ "notes": "",
+ "isIntermediate": true,
+ "useCache": true,
+ "version": "1.0.0",
+ "nodePack": "invokeai",
+ "inputs": {
+ "model": {
+ "id": "993eabd2-40fd-44fe-bce7-5d0c7075ddab",
+ "name": "model",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "MainModelField"
+ },
+ "value": {
+ "model_name": "stable-diffusion-v1-5",
+ "base_model": "sd-1",
+ "model_type": "main"
+ }
+ }
+ },
+ "outputs": {
+ "unet": {
+ "id": "5c18c9db-328d-46d0-8cb9-143391c410be",
+ "name": "unet",
+ "fieldKind": "output",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "UNetField"
+ }
+ },
+ "clip": {
+ "id": "6effcac0-ec2f-4bf5-a49e-a2c29cf921f4",
+ "name": "clip",
+ "fieldKind": "output",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "ClipField"
+ }
+ },
+ "vae": {
+ "id": "57683ba3-f5f5-4f58-b9a2-4b83dacad4a1",
+ "name": "vae",
+ "fieldKind": "output",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "VaeField"
+ }
+ }
+ }
+ },
+ "width": 320,
+ "height": 226,
+ "position": {
+ "x": 600,
+ "y": 25
+ }
+ },
+ {
+ "id": "7d8bf987-284f-413a-b2fd-d825445a5d6c",
+ "type": "invocation",
+ "data": {
+ "id": "7d8bf987-284f-413a-b2fd-d825445a5d6c",
+ "type": "compel",
+ "label": "Positive Compel Prompt",
+ "isOpen": true,
+ "notes": "",
+ "isIntermediate": true,
+ "useCache": true,
+ "version": "1.0.0",
+ "nodePack": "invokeai",
+ "inputs": {
+ "prompt": {
+ "id": "7739aff6-26cb-4016-8897-5a1fb2305e4e",
+ "name": "prompt",
+ "fieldKind": "input",
+ "label": "Positive Prompt",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "StringField"
+ },
+ "value": "Super cute tiger cub, national geographic award-winning photograph"
+ },
+ "clip": {
+ "id": "48d23dce-a6ae-472a-9f8c-22a714ea5ce0",
+ "name": "clip",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "ClipField"
+ }
+ }
+ },
+ "outputs": {
+ "conditioning": {
+ "id": "37cf3a9d-f6b7-4b64-8ff6-2558c5ecc447",
+ "name": "conditioning",
+ "fieldKind": "output",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "ConditioningField"
+ }
+ }
+ }
+ },
+ "width": 320,
+ "height": 259,
+ "position": {
+ "x": 1000,
+ "y": 25
+ }
+ },
+ {
+ "id": "ea94bc37-d995-4a83-aa99-4af42479f2f2",
+ "type": "invocation",
+ "data": {
+ "id": "ea94bc37-d995-4a83-aa99-4af42479f2f2",
+ "type": "rand_int",
+ "label": "Random Seed",
+ "isOpen": false,
+ "notes": "",
+ "isIntermediate": true,
+ "useCache": false,
+ "version": "1.0.0",
+ "nodePack": "invokeai",
+ "inputs": {
+ "low": {
+ "id": "3ec65a37-60ba-4b6c-a0b2-553dd7a84b84",
+ "name": "low",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "IntegerField"
+ },
+ "value": 0
+ },
+ "high": {
+ "id": "085f853a-1a5f-494d-8bec-e4ba29a3f2d1",
+ "name": "high",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "IntegerField"
+ },
+ "value": 2147483647
+ }
+ },
+ "outputs": {
+ "value": {
+ "id": "812ade4d-7699-4261-b9fc-a6c9d2ab55ee",
+ "name": "value",
+ "fieldKind": "output",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "IntegerField"
+ }
+ }
+ }
+ },
+ "width": 320,
+ "height": 32,
+ "position": {
+ "x": 600,
+ "y": 275
+ }
+ },
+ {
+ "id": "eea2702a-19fb-45b5-9d75-56b4211ec03c",
+ "type": "invocation",
+ "data": {
+ "id": "eea2702a-19fb-45b5-9d75-56b4211ec03c",
+ "type": "denoise_latents",
+ "label": "",
+ "isOpen": true,
+ "notes": "",
+ "isIntermediate": true,
+ "useCache": true,
+ "version": "1.4.0",
+ "nodePack": "invokeai",
+ "inputs": {
+ "positive_conditioning": {
+ "id": "90b7f4f8-ada7-4028-8100-d2e54f192052",
+ "name": "positive_conditioning",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "ConditioningField"
+ }
+ },
+ "negative_conditioning": {
+ "id": "9393779e-796c-4f64-b740-902a1177bf53",
+ "name": "negative_conditioning",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "ConditioningField"
+ }
+ },
+ "noise": {
+ "id": "8e17f1e5-4f98-40b1-b7f4-86aeeb4554c1",
+ "name": "noise",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "LatentsField"
+ }
+ },
+ "steps": {
+ "id": "9b63302d-6bd2-42c9-ac13-9b1afb51af88",
+ "name": "steps",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "IntegerField"
+ },
+ "value": 50
+ },
+ "cfg_scale": {
+ "id": "87dd04d3-870e-49e1-98bf-af003a810109",
+ "name": "cfg_scale",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": true,
+ "name": "FloatField"
+ },
+ "value": 7.5
+ },
+ "denoising_start": {
+ "id": "f369d80f-4931-4740-9bcd-9f0620719fab",
+ "name": "denoising_start",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "FloatField"
+ },
+ "value": 0
+ },
+ "denoising_end": {
+ "id": "747d10e5-6f02-445c-994c-0604d814de8c",
+ "name": "denoising_end",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "FloatField"
+ },
+ "value": 1
+ },
+ "scheduler": {
+ "id": "1de84a4e-3a24-4ec8-862b-16ce49633b9b",
+ "name": "scheduler",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "SchedulerField"
+ },
+ "value": "unipc"
+ },
+ "unet": {
+ "id": "ffa6fef4-3ce2-4bdb-9296-9a834849489b",
+ "name": "unet",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "UNetField"
+ }
+ },
+ "control": {
+ "id": "077b64cb-34be-4fcc-83f2-e399807a02bd",
+ "name": "control",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": true,
+ "name": "ControlField"
+ }
+ },
+ "ip_adapter": {
+ "id": "1d6948f7-3a65-4a65-a20c-768b287251aa",
+ "name": "ip_adapter",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": true,
+ "name": "IPAdapterField"
+ }
+ },
+ "t2i_adapter": {
+ "id": "75e67b09-952f-4083-aaf4-6b804d690412",
+ "name": "t2i_adapter",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": true,
+ "name": "T2IAdapterField"
+ }
+ },
+ "latents": {
+ "id": "334d4ba3-5a99-4195-82c5-86fb3f4f7d43",
+ "name": "latents",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "LatentsField"
+ }
+ },
+ "denoise_mask": {
+ "id": "0d3dbdbf-b014-4e95-8b18-ff2ff9cb0bfa",
+ "name": "denoise_mask",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "DenoiseMaskField"
+ }
+ }
+ },
+ "outputs": {
+ "latents": {
+ "id": "70fa5bbc-0c38-41bb-861a-74d6d78d2f38",
+ "name": "latents",
+ "fieldKind": "output",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "LatentsField"
+ }
+ },
+ "width": {
+ "id": "98ee0e6c-82aa-4e8f-8be5-dc5f00ee47f0",
+ "name": "width",
+ "fieldKind": "output",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "IntegerField"
+ }
+ },
+ "height": {
+ "id": "e8cb184a-5e1a-47c8-9695-4b8979564f5d",
+ "name": "height",
+ "fieldKind": "output",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "IntegerField"
+ }
+ }
+ }
+ },
+ "width": 320,
+ "height": 646,
+ "position": {
+ "x": 1400,
+ "y": 25
+ }
+ },
+ {
+ "id": "58c957f5-0d01-41fc-a803-b2bbf0413d4f",
+ "type": "invocation",
+ "data": {
+ "id": "58c957f5-0d01-41fc-a803-b2bbf0413d4f",
+ "type": "l2i",
+ "label": "",
+ "isOpen": true,
+ "notes": "",
+ "isIntermediate": false,
+ "useCache": true,
+ "version": "1.2.0",
+ "nodePack": "invokeai",
+ "inputs": {
+ "metadata": {
+ "id": "ab375f12-0042-4410-9182-29e30db82c85",
+ "name": "metadata",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "MetadataField"
+ }
+ },
+ "latents": {
+ "id": "3a7e7efd-bff5-47d7-9d48-615127afee78",
+ "name": "latents",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "LatentsField"
+ }
+ },
+ "vae": {
+ "id": "a1f5f7a1-0795-4d58-b036-7820c0b0ef2b",
+ "name": "vae",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "VaeField"
+ }
+ },
+ "tiled": {
+ "id": "da52059a-0cee-4668-942f-519aa794d739",
+ "name": "tiled",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "BooleanField"
+ },
+ "value": false
+ },
+ "fp32": {
+ "id": "c4841df3-b24e-4140-be3b-ccd454c2522c",
+ "name": "fp32",
+ "fieldKind": "input",
+ "label": "",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "BooleanField"
+ },
+ "value": true
+ }
+ },
+ "outputs": {
+ "image": {
+ "id": "72d667d0-cf85-459d-abf2-28bd8b823fe7",
+ "name": "image",
+ "fieldKind": "output",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "ImageField"
+ }
+ },
+ "width": {
+ "id": "c8c907d8-1066-49d1-b9a6-83bdcd53addc",
+ "name": "width",
+ "fieldKind": "output",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "IntegerField"
+ }
+ },
+ "height": {
+ "id": "230f359c-b4ea-436c-b372-332d7dcdca85",
+ "name": "height",
+ "fieldKind": "output",
+ "type": {
+ "isCollection": false,
+ "isCollectionOrScalar": false,
+ "name": "IntegerField"
+ }
+ }
+ }
+ },
+ "width": 320,
+ "height": 266,
+ "position": {
+ "x": 1800,
+ "y": 25
+ }
+ }
+ ],
+ "edges": [
+ {
+ "id": "reactflow__edge-ea94bc37-d995-4a83-aa99-4af42479f2f2value-55705012-79b9-4aac-9f26-c0b10309785bseed",
+ "source": "ea94bc37-d995-4a83-aa99-4af42479f2f2",
+ "target": "55705012-79b9-4aac-9f26-c0b10309785b",
+ "type": "default",
+ "sourceHandle": "value",
+ "targetHandle": "seed"
+ },
+ {
+ "id": "reactflow__edge-c8d55139-f380-4695-b7f2-8b3d1e1e3db8clip-7d8bf987-284f-413a-b2fd-d825445a5d6cclip",
+ "source": "c8d55139-f380-4695-b7f2-8b3d1e1e3db8",
+ "target": "7d8bf987-284f-413a-b2fd-d825445a5d6c",
+ "type": "default",
+ "sourceHandle": "clip",
+ "targetHandle": "clip"
+ },
+ {
+ "id": "reactflow__edge-c8d55139-f380-4695-b7f2-8b3d1e1e3db8clip-93dc02a4-d05b-48ed-b99c-c9b616af3402clip",
+ "source": "c8d55139-f380-4695-b7f2-8b3d1e1e3db8",
+ "target": "93dc02a4-d05b-48ed-b99c-c9b616af3402",
+ "type": "default",
+ "sourceHandle": "clip",
+ "targetHandle": "clip"
+ },
+ {
+ "id": "reactflow__edge-55705012-79b9-4aac-9f26-c0b10309785bnoise-eea2702a-19fb-45b5-9d75-56b4211ec03cnoise",
+ "source": "55705012-79b9-4aac-9f26-c0b10309785b",
+ "target": "eea2702a-19fb-45b5-9d75-56b4211ec03c",
+ "type": "default",
+ "sourceHandle": "noise",
+ "targetHandle": "noise"
+ },
+ {
+ "id": "reactflow__edge-7d8bf987-284f-413a-b2fd-d825445a5d6cconditioning-eea2702a-19fb-45b5-9d75-56b4211ec03cpositive_conditioning",
+ "source": "7d8bf987-284f-413a-b2fd-d825445a5d6c",
+ "target": "eea2702a-19fb-45b5-9d75-56b4211ec03c",
+ "type": "default",
+ "sourceHandle": "conditioning",
+ "targetHandle": "positive_conditioning"
+ },
+ {
+ "id": "reactflow__edge-93dc02a4-d05b-48ed-b99c-c9b616af3402conditioning-eea2702a-19fb-45b5-9d75-56b4211ec03cnegative_conditioning",
+ "source": "93dc02a4-d05b-48ed-b99c-c9b616af3402",
+ "target": "eea2702a-19fb-45b5-9d75-56b4211ec03c",
+ "type": "default",
+ "sourceHandle": "conditioning",
+ "targetHandle": "negative_conditioning"
+ },
+ {
+ "id": "reactflow__edge-c8d55139-f380-4695-b7f2-8b3d1e1e3db8unet-eea2702a-19fb-45b5-9d75-56b4211ec03cunet",
+ "source": "c8d55139-f380-4695-b7f2-8b3d1e1e3db8",
+ "target": "eea2702a-19fb-45b5-9d75-56b4211ec03c",
+ "type": "default",
+ "sourceHandle": "unet",
+ "targetHandle": "unet"
+ },
+ {
+ "id": "reactflow__edge-eea2702a-19fb-45b5-9d75-56b4211ec03clatents-58c957f5-0d01-41fc-a803-b2bbf0413d4flatents",
+ "source": "eea2702a-19fb-45b5-9d75-56b4211ec03c",
+ "target": "58c957f5-0d01-41fc-a803-b2bbf0413d4f",
+ "type": "default",
+ "sourceHandle": "latents",
+ "targetHandle": "latents"
+ },
+ {
+ "id": "reactflow__edge-c8d55139-f380-4695-b7f2-8b3d1e1e3db8vae-58c957f5-0d01-41fc-a803-b2bbf0413d4fvae",
+ "source": "c8d55139-f380-4695-b7f2-8b3d1e1e3db8",
+ "target": "58c957f5-0d01-41fc-a803-b2bbf0413d4f",
+ "type": "default",
+ "sourceHandle": "vae",
+ "targetHandle": "vae"
+ }
+ ]
+}
diff --git a/invokeai/app/services/shared/system_workflows.py b/invokeai/app/services/shared/system_workflows.py
new file mode 100644
index 0000000000..e47a5e58cc
--- /dev/null
+++ b/invokeai/app/services/shared/system_workflows.py
@@ -0,0 +1,33 @@
+from logging import Logger
+from pathlib import Path
+
+import semver
+
+import invokeai.app.assets.workflows as system_workflows_dir
+from invokeai.app.services.workflow_records.workflow_records_base import WorkflowRecordsStorageBase
+from invokeai.app.services.workflow_records.workflow_records_common import WorkflowNotFoundError, WorkflowValidator
+
+system_workflows = Path(system_workflows_dir.__path__[0]).glob("*.json")
+
+
+def create_system_workflows(workflow_records: WorkflowRecordsStorageBase, logger: Logger) -> None:
+ """Creates the system workflows."""
+ for workflow_filename in system_workflows:
+ with open(workflow_filename, "rb") as f:
+ workflow_bytes = f.read()
+ if workflow_bytes is None:
+ raise ValueError(f"Could not find system workflow: {workflow_filename}")
+
+ new_workflow = WorkflowValidator.validate_json(workflow_bytes)
+
+ try:
+ installed_workflow = workflow_records.get(new_workflow.id).workflow
+ installed_version = semver.Version.parse(installed_workflow.version)
+ new_version = semver.Version.parse(new_workflow.version)
+
+ if new_version.compare(installed_version) > 0:
+ logger.info(f"Updating system workflow: {new_workflow.name}")
+ workflow_records._add_system_workflow(new_workflow)
+ except WorkflowNotFoundError:
+ logger.info(f"Installing system workflow: {new_workflow.name}")
+ workflow_records._add_system_workflow(new_workflow)
diff --git a/invokeai/app/services/workflow_records/workflow_records_base.py b/invokeai/app/services/workflow_records/workflow_records_base.py
index c9e033718d..ae4b6b37fa 100644
--- a/invokeai/app/services/workflow_records/workflow_records_base.py
+++ b/invokeai/app/services/workflow_records/workflow_records_base.py
@@ -5,6 +5,7 @@ from invokeai.app.services.shared.pagination import PaginatedResults
from invokeai.app.services.shared.sqlite.sqlite_common import SQLiteDirection
from invokeai.app.services.workflow_records.workflow_records_common import (
Workflow,
+ WorkflowCategory,
WorkflowRecordDTO,
WorkflowRecordListItemDTO,
WorkflowRecordOrderBy,
@@ -42,7 +43,13 @@ class WorkflowRecordsStorageBase(ABC):
per_page: int,
order_by: WorkflowRecordOrderBy,
direction: SQLiteDirection,
+ category: WorkflowCategory,
filter_text: Optional[str],
) -> PaginatedResults[WorkflowRecordListItemDTO]:
"""Gets many workflows."""
pass
+
+ @abstractmethod
+ def _add_system_workflow(self, workflow: Workflow) -> None:
+ """Adds a system workflow. Internal use only."""
+ pass
diff --git a/invokeai/app/services/workflow_records/workflow_records_common.py b/invokeai/app/services/workflow_records/workflow_records_common.py
index 4fd34f18b2..c29350794b 100644
--- a/invokeai/app/services/workflow_records/workflow_records_common.py
+++ b/invokeai/app/services/workflow_records/workflow_records_common.py
@@ -16,8 +16,27 @@ class ExposedField(BaseModel):
fieldName: str
+class WorkflowNotFoundError(Exception):
+ """Raised when a workflow is not found"""
+
+
+class WorkflowRecordOrderBy(str, Enum, metaclass=MetaEnum):
+ """The order by options for workflow records"""
+
+ CreatedAt = "created_at"
+ UpdatedAt = "updated_at"
+ OpenedAt = "opened_at"
+ Name = "name"
+
+
+class WorkflowCategory(str, Enum, metaclass=MetaEnum):
+ User = "user"
+ System = "system"
+
+
class WorkflowMeta(BaseModel):
version: str = Field(description="The version of the workflow schema.")
+ category: WorkflowCategory = Field(description="The category of the workflow (user or system).")
@field_validator("version")
def validate_version(cls, version: str):
@@ -56,14 +75,17 @@ class Workflow(WorkflowWithoutID):
WorkflowValidator = TypeAdapter(Workflow)
-class WorkflowRecordDTO(BaseModel):
+class WorkflowRecordDTOBase(BaseModel):
workflow_id: str = Field(description="The id of the workflow.")
- workflow: Workflow = Field(description="The workflow.")
name: str = Field(description="The name of the workflow.")
created_at: Union[datetime.datetime, str] = Field(description="The created timestamp of the workflow.")
updated_at: Union[datetime.datetime, str] = Field(description="The updated timestamp of the workflow.")
opened_at: Union[datetime.datetime, str] = Field(description="The opened timestamp of the workflow.")
+
+class WorkflowRecordDTO(WorkflowRecordDTOBase):
+ workflow: Workflow = Field(description="The workflow.")
+
@classmethod
def from_dict(cls, data: dict[str, Any]) -> "WorkflowRecordDTO":
data["workflow"] = WorkflowValidator.validate_json(data.get("workflow", ""))
@@ -73,26 +95,8 @@ class WorkflowRecordDTO(BaseModel):
WorkflowRecordDTOValidator = TypeAdapter(WorkflowRecordDTO)
-class WorkflowRecordListItemDTO(BaseModel):
- workflow_id: str = Field(description="The id of the workflow.")
- name: str = Field(description="The name of the workflow.")
+class WorkflowRecordListItemDTO(WorkflowRecordDTOBase):
description: str = Field(description="The description of the workflow.")
- created_at: Union[datetime.datetime, str] = Field(description="The created timestamp of the workflow.")
- updated_at: Union[datetime.datetime, str] = Field(description="The updated timestamp of the workflow.")
- opened_at: Union[datetime.datetime, str] = Field(description="The opened timestamp of the workflow.")
WorkflowRecordListItemDTOValidator = TypeAdapter(WorkflowRecordListItemDTO)
-
-
-class WorkflowNotFoundError(Exception):
- """Raised when a workflow is not found"""
-
-
-class WorkflowRecordOrderBy(str, Enum, metaclass=MetaEnum):
- """The order by options for workflow records"""
-
- CreatedAt = "created_at"
- UpdatedAt = "updated_at"
- OpenedAt = "opened_at"
- Name = "name"
diff --git a/invokeai/app/services/workflow_records/workflow_records_sqlite.py b/invokeai/app/services/workflow_records/workflow_records_sqlite.py
index 3499a35026..8b2c4dfee6 100644
--- a/invokeai/app/services/workflow_records/workflow_records_sqlite.py
+++ b/invokeai/app/services/workflow_records/workflow_records_sqlite.py
@@ -7,6 +7,7 @@ from invokeai.app.services.shared.sqlite.sqlite_database import SqliteDatabase
from invokeai.app.services.workflow_records.workflow_records_base import WorkflowRecordsStorageBase
from invokeai.app.services.workflow_records.workflow_records_common import (
Workflow,
+ WorkflowCategory,
WorkflowNotFoundError,
WorkflowRecordDTO,
WorkflowRecordListItemDTO,
@@ -40,6 +41,7 @@ class SqliteWorkflowRecordsStorage(WorkflowRecordsStorageBase):
""",
(workflow_id,),
)
+ self._conn.commit()
self._cursor.execute(
"""--sql
SELECT workflow_id, workflow, name, created_at, updated_at, opened_at
@@ -60,6 +62,8 @@ class SqliteWorkflowRecordsStorage(WorkflowRecordsStorageBase):
def create(self, workflow: WorkflowWithoutID) -> WorkflowRecordDTO:
try:
+ # Only user workflows may be created by this method
+ assert workflow.meta.category is WorkflowCategory.User
workflow_with_id = WorkflowValidator.validate_python(workflow.model_dump())
self._lock.acquire()
self._cursor.execute(
@@ -70,10 +74,7 @@ class SqliteWorkflowRecordsStorage(WorkflowRecordsStorageBase):
)
VALUES (?, ?);
""",
- (
- workflow_with_id.id,
- workflow_with_id.model_dump_json(),
- ),
+ (workflow_with_id.id, workflow_with_id.model_dump_json()),
)
self._conn.commit()
except Exception:
@@ -90,7 +91,7 @@ class SqliteWorkflowRecordsStorage(WorkflowRecordsStorageBase):
"""--sql
UPDATE workflow_library
SET workflow = ?
- WHERE workflow_id = ?;
+ WHERE workflow_id = ? AND category = "user";
""",
(workflow.model_dump_json(), workflow.id),
)
@@ -108,7 +109,7 @@ class SqliteWorkflowRecordsStorage(WorkflowRecordsStorageBase):
self._cursor.execute(
"""--sql
DELETE from workflow_library
- WHERE workflow_id = ?;
+ WHERE workflow_id = ? AND category = "user";
""",
(workflow_id,),
)
@@ -126,6 +127,7 @@ class SqliteWorkflowRecordsStorage(WorkflowRecordsStorageBase):
per_page: int,
order_by: WorkflowRecordOrderBy,
direction: SQLiteDirection,
+ category: WorkflowCategory,
filter_text: Optional[str] = None,
) -> PaginatedResults[WorkflowRecordListItemDTO]:
try:
@@ -133,24 +135,27 @@ class SqliteWorkflowRecordsStorage(WorkflowRecordsStorageBase):
# sanitize!
assert order_by in WorkflowRecordOrderBy
assert direction in SQLiteDirection
- count_query = "SELECT COUNT(*) FROM workflow_library"
+ assert category in WorkflowCategory
+ count_query = "SELECT COUNT(*) FROM workflow_library WHERE category = ?"
main_query = """
SELECT
workflow_id,
+ category,
name,
description,
created_at,
updated_at,
opened_at
FROM workflow_library
+ WHERE category = ?
"""
- main_params = []
- count_params = []
+ main_params = [category.value]
+ count_params = [category.value]
stripped_filter_name = filter_text.strip() if filter_text else None
if stripped_filter_name:
filter_string = "%" + stripped_filter_name + "%"
- main_query += " WHERE name LIKE ? OR description LIKE ? "
- count_query += " WHERE name LIKE ? OR description LIKE ?;"
+ main_query += " AND name LIKE ? OR description LIKE ? "
+ count_query += " AND name LIKE ? OR description LIKE ?;"
main_params.extend([filter_string, filter_string])
count_params.extend([filter_string, filter_string])
@@ -177,6 +182,28 @@ class SqliteWorkflowRecordsStorage(WorkflowRecordsStorageBase):
finally:
self._lock.release()
+ def _add_system_workflow(self, workflow: Workflow) -> None:
+ try:
+ self._lock.acquire()
+ # Only system workflows may be created by this method
+ assert workflow.meta.category is WorkflowCategory.System
+ self._cursor.execute(
+ """--sql
+ INSERT OR REPLACE INTO workflow_library (
+ workflow_id,
+ workflow
+ )
+ VALUES (?, ?);
+ """,
+ (workflow.id, workflow.model_dump_json()),
+ )
+ self._conn.commit()
+ except Exception:
+ self._conn.rollback()
+ raise
+ finally:
+ self._lock.release()
+
def _create_tables(self) -> None:
try:
self._lock.acquire()
@@ -191,6 +218,7 @@ class SqliteWorkflowRecordsStorage(WorkflowRecordsStorageBase):
-- updated manually when retrieving workflow
opened_at DATETIME NOT NULL DEFAULT(STRFTIME('%Y-%m-%d %H:%M:%f', 'NOW')),
-- Generated columns, needed for indexing and searching
+ category TEXT GENERATED ALWAYS as (json_extract(workflow, '$.meta.category')) VIRTUAL NOT NULL,
name TEXT GENERATED ALWAYS as (json_extract(workflow, '$.name')) VIRTUAL NOT NULL,
description TEXT GENERATED ALWAYS as (json_extract(workflow, '$.description')) VIRTUAL NOT NULL
);
@@ -225,6 +253,11 @@ class SqliteWorkflowRecordsStorage(WorkflowRecordsStorageBase):
CREATE INDEX IF NOT EXISTS idx_workflow_library_opened_at ON workflow_library(opened_at);
"""
)
+ self._cursor.execute(
+ """--sql
+ CREATE INDEX IF NOT EXISTS idx_workflow_library_category ON workflow_library(category);
+ """
+ )
self._cursor.execute(
"""--sql
CREATE INDEX IF NOT EXISTS idx_workflow_library_name ON workflow_library(name);
diff --git a/invokeai/frontend/web/public/locales/en.json b/invokeai/frontend/web/public/locales/en.json
index 269e3010b5..e734ae9e08 100644
--- a/invokeai/frontend/web/public/locales/en.json
+++ b/invokeai/frontend/web/public/locales/en.json
@@ -128,6 +128,8 @@
"random": "Random",
"reportBugLabel": "Report Bug",
"safetensors": "Safetensors",
+ "save": "Save",
+ "saveAs": "Save As",
"settingsLabel": "Settings",
"simple": "Simple",
"somethingWentWrong": "Something went wrong",
@@ -1625,7 +1627,7 @@
"unnamedWorkflow": "Unnamed Workflow",
"downloadWorkflow": "Download Workflow",
"saveWorkflow": "Save Workflow",
- "duplicateWorkflow": "Duplicate Workflow",
+ "saveWorkflowAs": "Save Workflow As",
"problemSavingWorkflow": "Problem Saving Workflow",
"workflowSaved": "Workflow Saved",
"noRecentWorkflows": "No Recent Workflows",
@@ -1635,6 +1637,7 @@
"loading": "Loading Workflows",
"noDescription": "No description",
"searchWorkflows": "Search Workflows",
- "clearWorkflowSearchFilter": "Clear Workflow Search Filter"
+ "clearWorkflowSearchFilter": "Clear Workflow Search Filter",
+ "workflowName": "Workflow Name"
}
}
diff --git a/invokeai/frontend/web/src/features/nodes/components/flow/panels/TopCenterPanel/TopCenterPanel.tsx b/invokeai/frontend/web/src/features/nodes/components/flow/panels/TopCenterPanel/TopCenterPanel.tsx
index 3481a1d9eb..e4c08da69b 100644
--- a/invokeai/frontend/web/src/features/nodes/components/flow/panels/TopCenterPanel/TopCenterPanel.tsx
+++ b/invokeai/frontend/web/src/features/nodes/components/flow/panels/TopCenterPanel/TopCenterPanel.tsx
@@ -4,7 +4,7 @@ import DownloadWorkflowButton from 'features/workflowLibrary/components/Download
import UploadWorkflowButton from 'features/workflowLibrary/components/LoadWorkflowFromFileButton';
import ResetWorkflowButton from 'features/workflowLibrary/components/ResetWorkflowButton';
import SaveWorkflowButton from 'features/workflowLibrary/components/SaveWorkflowButton';
-import DuplicateWorkflowButton from 'features/workflowLibrary/components/DuplicateWorkflowButton';
+import SaveWorkflowAsButton from 'features/workflowLibrary/components/SaveWorkflowAsButton';
const TopCenterPanel = () => {
return (
@@ -20,7 +20,7 @@ const TopCenterPanel = () => {
-
+
);
diff --git a/invokeai/frontend/web/src/features/nodes/store/nodesSlice.ts b/invokeai/frontend/web/src/features/nodes/store/nodesSlice.ts
index 11dbc53e97..562e32108f 100644
--- a/invokeai/frontend/web/src/features/nodes/store/nodesSlice.ts
+++ b/invokeai/frontend/web/src/features/nodes/store/nodesSlice.ts
@@ -1,33 +1,4 @@
import { createSlice, PayloadAction } from '@reduxjs/toolkit';
-import { cloneDeep, forEach, isEqual, uniqBy } from 'lodash-es';
-import {
- addEdge,
- applyEdgeChanges,
- applyNodeChanges,
- Connection,
- Edge,
- EdgeChange,
- EdgeRemoveChange,
- getConnectedEdges,
- getIncomers,
- getOutgoers,
- Node,
- NodeChange,
- OnConnectStartParams,
- SelectionMode,
- updateEdge,
- Viewport,
- XYPosition,
-} from 'reactflow';
-import { receivedOpenAPISchema } from 'services/api/thunks/schema';
-import {
- appSocketGeneratorProgress,
- appSocketInvocationComplete,
- appSocketInvocationError,
- appSocketInvocationStarted,
- appSocketQueueItemStatusChanged,
-} from 'services/events/actions';
-import { v4 as uuidv4 } from 'uuid';
import { SHARED_NODE_PROPERTIES } from 'features/nodes/types/constants';
import {
BoardFieldValue,
@@ -58,6 +29,35 @@ import {
zNodeStatus,
} from 'features/nodes/types/invocation';
import { WorkflowV2 } from 'features/nodes/types/workflow';
+import { cloneDeep, forEach, isEqual, uniqBy } from 'lodash-es';
+import {
+ addEdge,
+ applyEdgeChanges,
+ applyNodeChanges,
+ Connection,
+ Edge,
+ EdgeChange,
+ EdgeRemoveChange,
+ getConnectedEdges,
+ getIncomers,
+ getOutgoers,
+ Node,
+ NodeChange,
+ OnConnectStartParams,
+ SelectionMode,
+ updateEdge,
+ Viewport,
+ XYPosition,
+} from 'reactflow';
+import { receivedOpenAPISchema } from 'services/api/thunks/schema';
+import {
+ appSocketGeneratorProgress,
+ appSocketInvocationComplete,
+ appSocketInvocationError,
+ appSocketInvocationStarted,
+ appSocketQueueItemStatusChanged,
+} from 'services/events/actions';
+import { v4 as uuidv4 } from 'uuid';
import { NodesState } from './types';
import { findConnectionToValidHandle } from './util/findConnectionToValidHandle';
import { findUnoccupiedPosition } from './util/findUnoccupiedPosition';
@@ -81,7 +81,7 @@ const INITIAL_WORKFLOW: WorkflowV2 = {
nodes: [],
edges: [],
exposedFields: [],
- meta: { version: '2.0.0' },
+ meta: { version: '2.0.0', category: 'user' },
};
export const initialNodesState: NodesState = {
diff --git a/invokeai/frontend/web/src/features/nodes/types/workflow.ts b/invokeai/frontend/web/src/features/nodes/types/workflow.ts
index 05cf539291..d4ba400e97 100644
--- a/invokeai/frontend/web/src/features/nodes/types/workflow.ts
+++ b/invokeai/frontend/web/src/features/nodes/types/workflow.ts
@@ -13,6 +13,9 @@ export type XYPosition = z.infer;
export const zDimension = z.number().gt(0).nullish();
export type Dimension = z.infer;
+
+export const zWorkflowCategory = z.enum(['user', 'system']);
+export type WorkflowCategory = z.infer;
// #endregion
// #region Workflow Nodes
@@ -85,6 +88,7 @@ export const zWorkflowV2 = z.object({
edges: z.array(zWorkflowEdge),
exposedFields: z.array(zFieldIdentifier),
meta: z.object({
+ category: zWorkflowCategory.default('user'),
version: z.literal('2.0.0'),
}),
});
diff --git a/invokeai/frontend/web/src/features/nodes/util/workflow/buildWorkflow.ts b/invokeai/frontend/web/src/features/nodes/util/workflow/buildWorkflow.ts
index 6b2d148706..b5c0e918c6 100644
--- a/invokeai/frontend/web/src/features/nodes/util/workflow/buildWorkflow.ts
+++ b/invokeai/frontend/web/src/features/nodes/util/workflow/buildWorkflow.ts
@@ -8,11 +8,15 @@ import {
import { fromZodError } from 'zod-validation-error';
import { parseify } from 'common/util/serialize';
import i18n from 'i18next';
+import { cloneDeep } from 'lodash-es';
export const buildWorkflow = (nodesState: NodesState): WorkflowV2 => {
- const { workflow: workflowMeta, nodes, edges } = nodesState;
- const workflow: WorkflowV2 = {
- ...workflowMeta,
+ const workflow = cloneDeep(nodesState.workflow);
+ const nodes = cloneDeep(nodesState.nodes);
+ const edges = cloneDeep(nodesState.edges);
+
+ const newWorkflow: WorkflowV2 = {
+ ...workflow,
nodes: [],
edges: [],
};
@@ -30,7 +34,7 @@ export const buildWorkflow = (nodesState: NodesState): WorkflowV2 => {
logger('nodes').warn({ node: parseify(node) }, message);
return;
}
- workflow.nodes.push(result.data);
+ newWorkflow.nodes.push(result.data);
});
edges.forEach((edge) => {
@@ -42,8 +46,8 @@ export const buildWorkflow = (nodesState: NodesState): WorkflowV2 => {
logger('nodes').warn({ edge: parseify(edge) }, message);
return;
}
- workflow.edges.push(result.data);
+ newWorkflow.edges.push(result.data);
});
- return workflow;
+ return newWorkflow;
};
diff --git a/invokeai/frontend/web/src/features/nodes/util/workflow/migrations.ts b/invokeai/frontend/web/src/features/nodes/util/workflow/migrations.ts
index 61ddbca6dd..80458ac8af 100644
--- a/invokeai/frontend/web/src/features/nodes/util/workflow/migrations.ts
+++ b/invokeai/frontend/web/src/features/nodes/util/workflow/migrations.ts
@@ -66,6 +66,7 @@ const migrateV1toV2 = (workflowToMigrate: WorkflowV1): WorkflowV2 => {
});
// Bump version
(workflowToMigrate as unknown as WorkflowV2).meta.version = '2.0.0';
+ (workflowToMigrate as unknown as WorkflowV2).meta.category = 'user';
// Parsing strips out any extra properties not in the latest version
return zWorkflowV2.parse(workflowToMigrate);
};
diff --git a/invokeai/frontend/web/src/features/nodes/util/workflow/validateWorkflow.ts b/invokeai/frontend/web/src/features/nodes/util/workflow/validateWorkflow.ts
index 002a0feff4..eb665b78fe 100644
--- a/invokeai/frontend/web/src/features/nodes/util/workflow/validateWorkflow.ts
+++ b/invokeai/frontend/web/src/features/nodes/util/workflow/validateWorkflow.ts
@@ -39,6 +39,13 @@ export const validateWorkflow = (
// Parse the raw workflow data & migrate it to the latest version
const _workflow = parseAndMigrateWorkflow(workflow);
+ // System workflows are only allowed to be used as templates.
+ // If a system workflow is loaded, change its category to user and remove its ID so that we can save it as a user workflow.
+ if (_workflow.meta.category === 'system') {
+ _workflow.meta.category = 'user';
+ _workflow.id = undefined;
+ }
+
// Now we can validate the graph
const { nodes, edges } = _workflow;
const warnings: WorkflowWarning[] = [];
diff --git a/invokeai/frontend/web/src/features/workflowLibrary/components/DuplicateWorkflowButton.tsx b/invokeai/frontend/web/src/features/workflowLibrary/components/DuplicateWorkflowButton.tsx
deleted file mode 100644
index c3beb9a8e0..0000000000
--- a/invokeai/frontend/web/src/features/workflowLibrary/components/DuplicateWorkflowButton.tsx
+++ /dev/null
@@ -1,21 +0,0 @@
-import IAIIconButton from 'common/components/IAIIconButton';
-import { useDuplicateLibraryWorkflow } from 'features/workflowLibrary/hooks/useDuplicateWorkflow';
-import { memo } from 'react';
-import { useTranslation } from 'react-i18next';
-import { FaClone } from 'react-icons/fa';
-
-const DuplicateLibraryWorkflowButton = () => {
- const { t } = useTranslation();
- const { duplicateWorkflow, isLoading } = useDuplicateLibraryWorkflow();
- return (
- }
- onClick={duplicateWorkflow}
- isLoading={isLoading}
- tooltip={t('workflows.duplicateWorkflow')}
- aria-label={t('workflows.duplicateWorkflow')}
- />
- );
-};
-
-export default memo(DuplicateLibraryWorkflowButton);
diff --git a/invokeai/frontend/web/src/features/workflowLibrary/components/SaveWorkflowAsButton.tsx b/invokeai/frontend/web/src/features/workflowLibrary/components/SaveWorkflowAsButton.tsx
new file mode 100644
index 0000000000..6bff13a887
--- /dev/null
+++ b/invokeai/frontend/web/src/features/workflowLibrary/components/SaveWorkflowAsButton.tsx
@@ -0,0 +1,83 @@
+import {
+ AlertDialog,
+ AlertDialogBody,
+ AlertDialogContent,
+ AlertDialogFooter,
+ AlertDialogHeader,
+ AlertDialogOverlay,
+ FormControl,
+ FormLabel,
+ Input,
+ useDisclosure,
+} from '@chakra-ui/react';
+import { useAppSelector } from 'app/store/storeHooks';
+import IAIButton from 'common/components/IAIButton';
+import IAIIconButton from 'common/components/IAIIconButton';
+import { useSaveWorkflowAs } from 'features/workflowLibrary/hooks/useDuplicateWorkflow';
+import { ChangeEvent, memo, useCallback, useRef, useState } from 'react';
+import { useTranslation } from 'react-i18next';
+import { FaClone } from 'react-icons/fa';
+
+const SaveWorkflowAsButton = () => {
+ const currentName = useAppSelector((state) => state.nodes.workflow.name);
+ const { t } = useTranslation();
+ const { saveWorkflowAs, isLoading } = useSaveWorkflowAs();
+ const [name, setName] = useState(currentName.trim());
+ const { isOpen, onOpen, onClose } = useDisclosure();
+ const inputRef = useRef(null);
+
+ const onSave = useCallback(async () => {
+ saveWorkflowAs({ name, onSuccess: onClose, onError: onClose });
+ }, [name, onClose, saveWorkflowAs]);
+
+ const onChange = useCallback((e: ChangeEvent) => {
+ setName(e.target.value);
+ }, []);
+
+ return (
+ <>
+ }
+ onClick={onOpen}
+ isLoading={isLoading}
+ tooltip={t('workflows.saveWorkflowAs')}
+ aria-label={t('workflows.saveWorkflowAs')}
+ />
+
+
+
+
+ {t('workflows.saveWorkflowAs')}
+
+
+
+
+ {t('workflows.workflowName')}
+
+
+
+
+
+ {t('common.cancel')}
+
+ {t('common.saveAs')}
+
+
+
+
+
+ >
+ );
+};
+
+export default memo(SaveWorkflowAsButton);
diff --git a/invokeai/frontend/web/src/features/workflowLibrary/components/WorkflowLibraryContent.tsx b/invokeai/frontend/web/src/features/workflowLibrary/components/WorkflowLibraryContent.tsx
index b1183892f8..04e51397b0 100644
--- a/invokeai/frontend/web/src/features/workflowLibrary/components/WorkflowLibraryContent.tsx
+++ b/invokeai/frontend/web/src/features/workflowLibrary/components/WorkflowLibraryContent.tsx
@@ -1,33 +1,15 @@
-import { Tab, TabList, TabPanel, TabPanels, Tabs } from '@chakra-ui/react';
+import WorkflowLibraryList from 'features/workflowLibrary/components/WorkflowLibraryList';
import WorkflowLibraryListWrapper from 'features/workflowLibrary/components/WorkflowLibraryListWrapper';
-import WorkflowLibrarySystemList from 'features/workflowLibrary/components/WorkflowLibrarySystemList';
import { memo } from 'react';
import { useTranslation } from 'react-i18next';
-import WorkflowLibraryUserList from './WorkflowLibraryUserList';
const WorkflowLibraryContent = () => {
const { t } = useTranslation();
return (
-
-
- {t('workflows.user')}
- {t('workflows.system')}
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
+
+
+
);
};
diff --git a/invokeai/frontend/web/src/features/workflowLibrary/components/WorkflowLibraryUserList.tsx b/invokeai/frontend/web/src/features/workflowLibrary/components/WorkflowLibraryList.tsx
similarity index 82%
rename from invokeai/frontend/web/src/features/workflowLibrary/components/WorkflowLibraryUserList.tsx
rename to invokeai/frontend/web/src/features/workflowLibrary/components/WorkflowLibraryList.tsx
index e67cb83543..831ebddc36 100644
--- a/invokeai/frontend/web/src/features/workflowLibrary/components/WorkflowLibraryUserList.tsx
+++ b/invokeai/frontend/web/src/features/workflowLibrary/components/WorkflowLibraryList.tsx
@@ -1,8 +1,8 @@
import { CloseIcon } from '@chakra-ui/icons';
import {
+ ButtonGroup,
Divider,
Flex,
- Heading,
IconButton,
Input,
InputGroup,
@@ -10,12 +10,14 @@ import {
Spacer,
} from '@chakra-ui/react';
import { SelectItem } from '@mantine/core';
+import IAIButton from 'common/components/IAIButton';
import {
IAINoContentFallback,
IAINoContentFallbackWithSpinner,
} from 'common/components/IAIImageFallback';
import IAIMantineSelect from 'common/components/IAIMantineSelect';
import ScrollableContent from 'features/nodes/components/sidePanel/ScrollableContent';
+import { WorkflowCategory } from 'features/nodes/types/workflow';
import WorkflowLibraryListItem from 'features/workflowLibrary/components/WorkflowLibraryListItem';
import WorkflowLibraryPagination from 'features/workflowLibrary/components/WorkflowLibraryPagination';
import { ChangeEvent, KeyboardEvent, memo, useCallback, useState } from 'react';
@@ -40,19 +42,19 @@ const DIRECTION_DATA: SelectItem[] = [
const WorkflowLibraryList = () => {
const { t } = useTranslation();
+ const [category, setCategory] = useState('user');
const [page, setPage] = useState(0);
const [filter_text, setFilterText] = useState('');
const [order_by, setOrderBy] = useState('opened_at');
const [direction, setDirection] = useState('ASC');
- const [debouncedFilterText] = useDebounce(filter_text, 500, {
- leading: true,
- });
+ const [debouncedFilterText] = useDebounce(filter_text, 500);
const { data, isLoading, isError, isFetching } = useListWorkflowsQuery({
page,
per_page: PER_PAGE,
order_by,
direction,
- filter_name: debouncedFilterText,
+ category,
+ filter_text: debouncedFilterText,
});
const handleChangeOrderBy = useCallback(
@@ -102,6 +104,14 @@ const WorkflowLibraryList = () => {
[]
);
+ const handleSetUserCategory = useCallback(() => {
+ setCategory('user');
+ }, []);
+
+ const handleSetSystemCategory = useCallback(() => {
+ setCategory('system');
+ }, []);
+
if (isLoading) {
return ;
}
@@ -113,7 +123,22 @@ const WorkflowLibraryList = () => {
return (
<>
- {t('workflows.userWorkflows')}
+
+
+ {t('workflows.userWorkflows')}
+
+
+ {t('workflows.systemWorkflows')}
+
+
{
{data.items.length ? (
-
+
{data.items.map((w) => (
))}
diff --git a/invokeai/frontend/web/src/features/workflowLibrary/components/WorkflowLibraryListItem.tsx b/invokeai/frontend/web/src/features/workflowLibrary/components/WorkflowLibraryListItem.tsx
index 7de0c2dac2..f400a7a338 100644
--- a/invokeai/frontend/web/src/features/workflowLibrary/components/WorkflowLibraryListItem.tsx
+++ b/invokeai/frontend/web/src/features/workflowLibrary/components/WorkflowLibraryListItem.tsx
@@ -72,14 +72,16 @@ const WorkflowLibraryListItem = ({ workflowDTO }: Props) => {
>
{t('common.load')}
-
- {t('common.delete')}
-
+ {workflowDTO.category === 'user' && (
+
+ {t('common.delete')}
+
+ )}
);
diff --git a/invokeai/frontend/web/src/features/workflowLibrary/hooks/useDuplicateWorkflow.ts b/invokeai/frontend/web/src/features/workflowLibrary/hooks/useDuplicateWorkflow.ts
index decef66920..20b3fc980a 100644
--- a/invokeai/frontend/web/src/features/workflowLibrary/hooks/useDuplicateWorkflow.ts
+++ b/invokeai/frontend/web/src/features/workflowLibrary/hooks/useDuplicateWorkflow.ts
@@ -3,35 +3,47 @@ import { useAppDispatch } from 'app/store/storeHooks';
import { useWorkflow } from 'features/nodes/hooks/useWorkflow';
import { workflowLoaded } from 'features/nodes/store/nodesSlice';
import { zWorkflowV2 } from 'features/nodes/types/workflow';
-import { omit } from 'lodash-es';
import { useCallback } from 'react';
import { useTranslation } from 'react-i18next';
import { useCreateWorkflowMutation } from 'services/api/endpoints/workflows';
-export const useDuplicateLibraryWorkflow = () => {
+type Arg = {
+ name: string;
+ onSuccess?: () => void;
+ onError?: () => void;
+};
+
+export const useSaveWorkflowAs = () => {
const { t } = useTranslation();
const dispatch = useAppDispatch();
const workflow = useWorkflow();
const [createWorkflow, createWorkflowResult] = useCreateWorkflowMutation();
const toaster = useAppToaster();
- const duplicateWorkflow = useCallback(async () => {
- try {
- const data = await createWorkflow(omit(workflow, 'id')).unwrap();
- const createdWorkflow = zWorkflowV2.parse(data.workflow);
- dispatch(workflowLoaded(createdWorkflow));
- toaster({
- title: t('workflows.workflowSaved'),
- status: 'success',
- });
- } catch (e) {
- toaster({
- title: t('workflows.problemSavingWorkflow'),
- status: 'error',
- });
- }
- }, [workflow, dispatch, toaster, t, createWorkflow]);
+ const saveWorkflowAs = useCallback(
+ async ({ name: newName, onSuccess, onError }: Arg) => {
+ try {
+ workflow.id = undefined;
+ workflow.name = newName;
+ const data = await createWorkflow(workflow).unwrap();
+ const createdWorkflow = zWorkflowV2.parse(data.workflow);
+ dispatch(workflowLoaded(createdWorkflow));
+ onSuccess && onSuccess();
+ toaster({
+ title: t('workflows.workflowSaved'),
+ status: 'success',
+ });
+ } catch (e) {
+ onError && onError();
+ toaster({
+ title: t('workflows.problemSavingWorkflow'),
+ status: 'error',
+ });
+ }
+ },
+ [workflow, dispatch, toaster, t, createWorkflow]
+ );
return {
- duplicateWorkflow,
+ saveWorkflowAs,
isLoading: createWorkflowResult.isLoading,
isError: createWorkflowResult.isError,
};
diff --git a/invokeai/frontend/web/src/services/api/schema.d.ts b/invokeai/frontend/web/src/services/api/schema.d.ts
index f3cf56c4c6..9d323c456f 100644
--- a/invokeai/frontend/web/src/services/api/schema.d.ts
+++ b/invokeai/frontend/web/src/services/api/schema.d.ts
@@ -3243,7 +3243,7 @@ export type components = {
* @description The nodes in this graph
*/
nodes?: {
- [key: string]: components["schemas"]["MediapipeFaceProcessorInvocation"] | components["schemas"]["CreateDenoiseMaskInvocation"] | components["schemas"]["StringCollectionInvocation"] | components["schemas"]["CollectInvocation"] | components["schemas"]["ZoeDepthImageProcessorInvocation"] | components["schemas"]["MaskEdgeInvocation"] | components["schemas"]["ImageToLatentsInvocation"] | components["schemas"]["ConditioningInvocation"] | components["schemas"]["FloatToIntegerInvocation"] | components["schemas"]["AddInvocation"] | components["schemas"]["ColorInvocation"] | components["schemas"]["ColorCorrectInvocation"] | components["schemas"]["SDXLRefinerCompelPromptInvocation"] | components["schemas"]["PromptsFromFileInvocation"] | components["schemas"]["StringInvocation"] | components["schemas"]["RoundInvocation"] | components["schemas"]["SDXLLoraLoaderInvocation"] | components["schemas"]["SchedulerInvocation"] | components["schemas"]["MergeMetadataInvocation"] | components["schemas"]["InfillColorInvocation"] | components["schemas"]["ImageMultiplyInvocation"] | components["schemas"]["PidiImageProcessorInvocation"] | components["schemas"]["MaskFromAlphaInvocation"] | components["schemas"]["ImageCollectionInvocation"] | components["schemas"]["FloatCollectionInvocation"] | components["schemas"]["ESRGANInvocation"] | components["schemas"]["FaceIdentifierInvocation"] | components["schemas"]["ONNXLatentsToImageInvocation"] | components["schemas"]["BlendLatentsInvocation"] | components["schemas"]["ImageChannelInvocation"] | components["schemas"]["IPAdapterInvocation"] | components["schemas"]["SDXLCompelPromptInvocation"] | components["schemas"]["NormalbaeImageProcessorInvocation"] | components["schemas"]["HedImageProcessorInvocation"] | components["schemas"]["RandomIntInvocation"] | components["schemas"]["FreeUInvocation"] | components["schemas"]["OnnxModelLoaderInvocation"] | components["schemas"]["StringSplitNegInvocation"] | components["schemas"]["ImageHueAdjustmentInvocation"] | components["schemas"]["NoiseInvocation"] | components["schemas"]["SegmentAnythingProcessorInvocation"] | components["schemas"]["InfillPatchMatchInvocation"] | components["schemas"]["CoreMetadataInvocation"] | components["schemas"]["ScaleLatentsInvocation"] | components["schemas"]["ImageChannelMultiplyInvocation"] | components["schemas"]["ImageConvertInvocation"] | components["schemas"]["FaceOffInvocation"] | components["schemas"]["DivideInvocation"] | components["schemas"]["StringJoinInvocation"] | components["schemas"]["LatentsToImageInvocation"] | components["schemas"]["ImageLerpInvocation"] | components["schemas"]["ImageWatermarkInvocation"] | components["schemas"]["RandomFloatInvocation"] | components["schemas"]["LinearUIOutputInvocation"] | components["schemas"]["RandomRangeInvocation"] | components["schemas"]["ImageResizeInvocation"] | components["schemas"]["CannyImageProcessorInvocation"] | components["schemas"]["MultiplyInvocation"] | components["schemas"]["GraphInvocation"] | components["schemas"]["MidasDepthImageProcessorInvocation"] | components["schemas"]["StringJoinThreeInvocation"] | components["schemas"]["OpenposeImageProcessorInvocation"] | components["schemas"]["ResizeLatentsInvocation"] | components["schemas"]["LeresImageProcessorInvocation"] | components["schemas"]["ColorMapImageProcessorInvocation"] | components["schemas"]["FaceMaskInvocation"] | components["schemas"]["LineartImageProcessorInvocation"] | components["schemas"]["FloatLinearRangeInvocation"] | components["schemas"]["IntegerCollectionInvocation"] | components["schemas"]["ControlNetInvocation"] | components["schemas"]["LoraLoaderInvocation"] | components["schemas"]["CV2InfillInvocation"] | components["schemas"]["FloatInvocation"] | components["schemas"]["ConditioningCollectionInvocation"] | components["schemas"]["VaeLoaderInvocation"] | components["schemas"]["BooleanCollectionInvocation"] | components["schemas"]["ImageInverseLerpInvocation"] | components["schemas"]["RangeInvocation"] | components["schemas"]["BooleanInvocation"] | components["schemas"]["StepParamEasingInvocation"] | components["schemas"]["CompelInvocation"] | components["schemas"]["LaMaInfillInvocation"] | components["schemas"]["MetadataInvocation"] | components["schemas"]["SeamlessModeInvocation"] | components["schemas"]["InfillTileInvocation"] | components["schemas"]["StringSplitInvocation"] | components["schemas"]["ImageScaleInvocation"] | components["schemas"]["IntegerMathInvocation"] | components["schemas"]["IterateInvocation"] | components["schemas"]["DenoiseLatentsInvocation"] | components["schemas"]["MlsdImageProcessorInvocation"] | components["schemas"]["MainModelLoaderInvocation"] | components["schemas"]["ONNXPromptInvocation"] | components["schemas"]["T2IAdapterInvocation"] | components["schemas"]["ShowImageInvocation"] | components["schemas"]["FloatMathInvocation"] | components["schemas"]["ImageInvocation"] | components["schemas"]["MaskCombineInvocation"] | components["schemas"]["ImageNSFWBlurInvocation"] | components["schemas"]["StringReplaceInvocation"] | components["schemas"]["SDXLModelLoaderInvocation"] | components["schemas"]["ImageChannelOffsetInvocation"] | components["schemas"]["SubtractInvocation"] | components["schemas"]["ONNXTextToLatentsInvocation"] | components["schemas"]["ImageCropInvocation"] | components["schemas"]["CvInpaintInvocation"] | components["schemas"]["DynamicPromptInvocation"] | components["schemas"]["TileResamplerProcessorInvocation"] | components["schemas"]["ClipSkipInvocation"] | components["schemas"]["SDXLRefinerModelLoaderInvocation"] | components["schemas"]["IntegerInvocation"] | components["schemas"]["ImagePasteInvocation"] | components["schemas"]["RangeOfSizeInvocation"] | components["schemas"]["SaveImageInvocation"] | components["schemas"]["ContentShuffleImageProcessorInvocation"] | components["schemas"]["ImageBlurInvocation"] | components["schemas"]["MetadataItemInvocation"] | components["schemas"]["LatentsCollectionInvocation"] | components["schemas"]["LineartAnimeImageProcessorInvocation"] | components["schemas"]["LatentsInvocation"] | components["schemas"]["BlankImageInvocation"];
+ [key: string]: components["schemas"]["ConditioningCollectionInvocation"] | components["schemas"]["IPAdapterInvocation"] | components["schemas"]["BooleanInvocation"] | components["schemas"]["SDXLRefinerModelLoaderInvocation"] | components["schemas"]["NormalbaeImageProcessorInvocation"] | components["schemas"]["StepParamEasingInvocation"] | components["schemas"]["RangeInvocation"] | components["schemas"]["StringSplitNegInvocation"] | components["schemas"]["ImageMultiplyInvocation"] | components["schemas"]["ImageBlurInvocation"] | components["schemas"]["MaskFromAlphaInvocation"] | components["schemas"]["CoreMetadataInvocation"] | components["schemas"]["ImageLerpInvocation"] | components["schemas"]["DivideInvocation"] | components["schemas"]["CompelInvocation"] | components["schemas"]["ClipSkipInvocation"] | components["schemas"]["OnnxModelLoaderInvocation"] | components["schemas"]["LoraLoaderInvocation"] | components["schemas"]["ImageChannelInvocation"] | components["schemas"]["MetadataInvocation"] | components["schemas"]["ImageChannelMultiplyInvocation"] | components["schemas"]["MultiplyInvocation"] | components["schemas"]["VaeLoaderInvocation"] | components["schemas"]["GraphInvocation"] | components["schemas"]["ImageInvocation"] | components["schemas"]["MidasDepthImageProcessorInvocation"] | components["schemas"]["SeamlessModeInvocation"] | components["schemas"]["ShowImageInvocation"] | components["schemas"]["LatentsInvocation"] | components["schemas"]["LineartImageProcessorInvocation"] | components["schemas"]["BooleanCollectionInvocation"] | components["schemas"]["DynamicPromptInvocation"] | components["schemas"]["ONNXTextToLatentsInvocation"] | components["schemas"]["MainModelLoaderInvocation"] | components["schemas"]["StringJoinThreeInvocation"] | components["schemas"]["FloatCollectionInvocation"] | components["schemas"]["LaMaInfillInvocation"] | components["schemas"]["OpenposeImageProcessorInvocation"] | components["schemas"]["StringJoinInvocation"] | components["schemas"]["CV2InfillInvocation"] | components["schemas"]["LatentsCollectionInvocation"] | components["schemas"]["IntegerMathInvocation"] | components["schemas"]["RangeOfSizeInvocation"] | components["schemas"]["CvInpaintInvocation"] | components["schemas"]["ScaleLatentsInvocation"] | components["schemas"]["ImageConvertInvocation"] | components["schemas"]["FloatMathInvocation"] | components["schemas"]["StringSplitInvocation"] | components["schemas"]["SegmentAnythingProcessorInvocation"] | components["schemas"]["InfillTileInvocation"] | components["schemas"]["ImageHueAdjustmentInvocation"] | components["schemas"]["ImageCropInvocation"] | components["schemas"]["ColorInvocation"] | components["schemas"]["ImageWatermarkInvocation"] | components["schemas"]["ESRGANInvocation"] | components["schemas"]["InfillColorInvocation"] | components["schemas"]["StringCollectionInvocation"] | components["schemas"]["MetadataItemInvocation"] | components["schemas"]["IterateInvocation"] | components["schemas"]["MlsdImageProcessorInvocation"] | components["schemas"]["BlankImageInvocation"] | components["schemas"]["SubtractInvocation"] | components["schemas"]["FloatToIntegerInvocation"] | components["schemas"]["CannyImageProcessorInvocation"] | components["schemas"]["ConditioningInvocation"] | components["schemas"]["MaskCombineInvocation"] | components["schemas"]["IntegerCollectionInvocation"] | components["schemas"]["ControlNetInvocation"] | components["schemas"]["SDXLRefinerCompelPromptInvocation"] | components["schemas"]["AddInvocation"] | components["schemas"]["ONNXLatentsToImageInvocation"] | components["schemas"]["StringReplaceInvocation"] | components["schemas"]["LeresImageProcessorInvocation"] | components["schemas"]["SDXLModelLoaderInvocation"] | components["schemas"]["T2IAdapterInvocation"] | components["schemas"]["ImageResizeInvocation"] | components["schemas"]["ColorMapImageProcessorInvocation"] | components["schemas"]["ImageChannelOffsetInvocation"] | components["schemas"]["ImageCollectionInvocation"] | components["schemas"]["ImagePasteInvocation"] | components["schemas"]["ImageToLatentsInvocation"] | components["schemas"]["CreateDenoiseMaskInvocation"] | components["schemas"]["FloatInvocation"] | components["schemas"]["ImageInverseLerpInvocation"] | components["schemas"]["FaceIdentifierInvocation"] | components["schemas"]["SchedulerInvocation"] | components["schemas"]["SDXLLoraLoaderInvocation"] | components["schemas"]["HedImageProcessorInvocation"] | components["schemas"]["StringInvocation"] | components["schemas"]["SaveImageInvocation"] | components["schemas"]["FaceOffInvocation"] | components["schemas"]["PromptsFromFileInvocation"] | components["schemas"]["MaskEdgeInvocation"] | components["schemas"]["ImageScaleInvocation"] | components["schemas"]["ContentShuffleImageProcessorInvocation"] | components["schemas"]["ONNXPromptInvocation"] | components["schemas"]["ImageNSFWBlurInvocation"] | components["schemas"]["BlendLatentsInvocation"] | components["schemas"]["ColorCorrectInvocation"] | components["schemas"]["CollectInvocation"] | components["schemas"]["LatentsToImageInvocation"] | components["schemas"]["FreeUInvocation"] | components["schemas"]["IntegerInvocation"] | components["schemas"]["LinearUIOutputInvocation"] | components["schemas"]["FaceMaskInvocation"] | components["schemas"]["MergeMetadataInvocation"] | components["schemas"]["ZoeDepthImageProcessorInvocation"] | components["schemas"]["MediapipeFaceProcessorInvocation"] | components["schemas"]["ResizeLatentsInvocation"] | components["schemas"]["TileResamplerProcessorInvocation"] | components["schemas"]["DenoiseLatentsInvocation"] | components["schemas"]["LineartAnimeImageProcessorInvocation"] | components["schemas"]["NoiseInvocation"] | components["schemas"]["RoundInvocation"] | components["schemas"]["RandomRangeInvocation"] | components["schemas"]["SDXLCompelPromptInvocation"] | components["schemas"]["FloatLinearRangeInvocation"] | components["schemas"]["RandomFloatInvocation"] | components["schemas"]["RandomIntInvocation"] | components["schemas"]["PidiImageProcessorInvocation"] | components["schemas"]["InfillPatchMatchInvocation"];
};
/**
* Edges
@@ -3280,7 +3280,7 @@ export type components = {
* @description The results of node executions
*/
results: {
- [key: string]: components["schemas"]["ColorOutput"] | components["schemas"]["FaceOffOutput"] | components["schemas"]["ControlOutput"] | components["schemas"]["NoiseOutput"] | components["schemas"]["IntegerCollectionOutput"] | components["schemas"]["LatentsCollectionOutput"] | components["schemas"]["BooleanCollectionOutput"] | components["schemas"]["FloatCollectionOutput"] | components["schemas"]["IPAdapterOutput"] | components["schemas"]["StringOutput"] | components["schemas"]["ModelLoaderOutput"] | components["schemas"]["SDXLRefinerModelLoaderOutput"] | components["schemas"]["FloatOutput"] | components["schemas"]["MetadataItemOutput"] | components["schemas"]["String2Output"] | components["schemas"]["SeamlessModeOutput"] | components["schemas"]["BooleanOutput"] | components["schemas"]["StringCollectionOutput"] | components["schemas"]["CLIPOutput"] | components["schemas"]["T2IAdapterOutput"] | components["schemas"]["ConditioningOutput"] | components["schemas"]["FaceMaskOutput"] | components["schemas"]["ClipSkipInvocationOutput"] | components["schemas"]["StringPosNegOutput"] | components["schemas"]["ONNXModelLoaderOutput"] | components["schemas"]["ConditioningCollectionOutput"] | components["schemas"]["CollectInvocationOutput"] | components["schemas"]["SDXLLoraLoaderOutput"] | components["schemas"]["IterateInvocationOutput"] | components["schemas"]["IntegerOutput"] | components["schemas"]["MetadataOutput"] | components["schemas"]["GraphInvocationOutput"] | components["schemas"]["ImageCollectionOutput"] | components["schemas"]["SchedulerOutput"] | components["schemas"]["UNetOutput"] | components["schemas"]["LoraLoaderOutput"] | components["schemas"]["DenoiseMaskOutput"] | components["schemas"]["LatentsOutput"] | components["schemas"]["ImageOutput"] | components["schemas"]["ColorCollectionOutput"] | components["schemas"]["VAEOutput"] | components["schemas"]["SDXLModelLoaderOutput"];
+ [key: string]: components["schemas"]["LoraLoaderOutput"] | components["schemas"]["BooleanCollectionOutput"] | components["schemas"]["ConditioningOutput"] | components["schemas"]["FloatCollectionOutput"] | components["schemas"]["DenoiseMaskOutput"] | components["schemas"]["ClipSkipInvocationOutput"] | components["schemas"]["ConditioningCollectionOutput"] | components["schemas"]["FloatOutput"] | components["schemas"]["String2Output"] | components["schemas"]["SDXLModelLoaderOutput"] | components["schemas"]["BooleanOutput"] | components["schemas"]["MetadataOutput"] | components["schemas"]["StringPosNegOutput"] | components["schemas"]["ONNXModelLoaderOutput"] | components["schemas"]["ColorCollectionOutput"] | components["schemas"]["GraphInvocationOutput"] | components["schemas"]["StringOutput"] | components["schemas"]["ControlOutput"] | components["schemas"]["VAEOutput"] | components["schemas"]["SchedulerOutput"] | components["schemas"]["ImageCollectionOutput"] | components["schemas"]["ColorOutput"] | components["schemas"]["SDXLRefinerModelLoaderOutput"] | components["schemas"]["FaceOffOutput"] | components["schemas"]["NoiseOutput"] | components["schemas"]["FaceMaskOutput"] | components["schemas"]["IntegerOutput"] | components["schemas"]["LatentsCollectionOutput"] | components["schemas"]["ImageOutput"] | components["schemas"]["SeamlessModeOutput"] | components["schemas"]["LatentsOutput"] | components["schemas"]["UNetOutput"] | components["schemas"]["MetadataItemOutput"] | components["schemas"]["CollectInvocationOutput"] | components["schemas"]["ModelLoaderOutput"] | components["schemas"]["StringCollectionOutput"] | components["schemas"]["IntegerCollectionOutput"] | components["schemas"]["CLIPOutput"] | components["schemas"]["SDXLLoraLoaderOutput"] | components["schemas"]["IPAdapterOutput"] | components["schemas"]["IterateInvocationOutput"] | components["schemas"]["T2IAdapterOutput"];
};
/**
* Errors
@@ -9488,6 +9488,11 @@ export type components = {
*/
id?: string;
};
+ /**
+ * WorkflowCategory
+ * @enum {string}
+ */
+ WorkflowCategory: "user" | "system";
/** WorkflowMeta */
WorkflowMeta: {
/**
@@ -9503,8 +9508,6 @@ export type components = {
* @description The id of the workflow.
*/
workflow_id: string;
- /** @description The workflow. */
- workflow: components["schemas"]["Workflow"];
/**
* Name
* @description The name of the workflow.
@@ -9525,6 +9528,10 @@ export type components = {
* @description The opened timestamp of the workflow.
*/
opened_at: string;
+ /** @description The category of the workflow (user or system). */
+ category: components["schemas"]["WorkflowCategory"];
+ /** @description The workflow. */
+ workflow: components["schemas"]["Workflow"];
};
/** WorkflowRecordListItemDTO */
WorkflowRecordListItemDTO: {
@@ -9538,11 +9545,6 @@ export type components = {
* @description The name of the workflow.
*/
name: string;
- /**
- * Description
- * @description The description of the workflow.
- */
- description: string;
/**
* Created At
* @description The created timestamp of the workflow.
@@ -9558,6 +9560,13 @@ export type components = {
* @description The opened timestamp of the workflow.
*/
opened_at: string;
+ /** @description The category of the workflow (user or system). */
+ category: components["schemas"]["WorkflowCategory"];
+ /**
+ * Description
+ * @description The description of the workflow.
+ */
+ description: string;
};
/**
* WorkflowRecordOrderBy
@@ -9866,6 +9875,18 @@ export type components = {
* @enum {string}
*/
UIType: "SDXLMainModelField" | "SDXLRefinerModelField" | "ONNXModelField" | "VAEModelField" | "LoRAModelField" | "ControlNetModelField" | "IPAdapterModelField" | "SchedulerField" | "AnyField" | "CollectionField" | "CollectionItemField" | "DEPRECATED_Boolean" | "DEPRECATED_Color" | "DEPRECATED_Conditioning" | "DEPRECATED_Control" | "DEPRECATED_Float" | "DEPRECATED_Image" | "DEPRECATED_Integer" | "DEPRECATED_Latents" | "DEPRECATED_String" | "DEPRECATED_BooleanCollection" | "DEPRECATED_ColorCollection" | "DEPRECATED_ConditioningCollection" | "DEPRECATED_ControlCollection" | "DEPRECATED_FloatCollection" | "DEPRECATED_ImageCollection" | "DEPRECATED_IntegerCollection" | "DEPRECATED_LatentsCollection" | "DEPRECATED_StringCollection" | "DEPRECATED_BooleanPolymorphic" | "DEPRECATED_ColorPolymorphic" | "DEPRECATED_ConditioningPolymorphic" | "DEPRECATED_ControlPolymorphic" | "DEPRECATED_FloatPolymorphic" | "DEPRECATED_ImagePolymorphic" | "DEPRECATED_IntegerPolymorphic" | "DEPRECATED_LatentsPolymorphic" | "DEPRECATED_StringPolymorphic" | "DEPRECATED_MainModel" | "DEPRECATED_UNet" | "DEPRECATED_Vae" | "DEPRECATED_CLIP" | "DEPRECATED_Collection" | "DEPRECATED_CollectionItem" | "DEPRECATED_Enum" | "DEPRECATED_WorkflowField" | "DEPRECATED_IsIntermediate" | "DEPRECATED_BoardField" | "DEPRECATED_MetadataItem" | "DEPRECATED_MetadataItemCollection" | "DEPRECATED_MetadataItemPolymorphic" | "DEPRECATED_MetadataDict";
+ /**
+ * IPAdapterModelFormat
+ * @description An enumeration.
+ * @enum {string}
+ */
+ IPAdapterModelFormat: "invokeai";
+ /**
+ * ControlNetModelFormat
+ * @description An enumeration.
+ * @enum {string}
+ */
+ ControlNetModelFormat: "checkpoint" | "diffusers";
/**
* StableDiffusionOnnxModelFormat
* @description An enumeration.
@@ -9878,30 +9899,6 @@ export type components = {
* @enum {string}
*/
StableDiffusion1ModelFormat: "checkpoint" | "diffusers";
- /**
- * ControlNetModelFormat
- * @description An enumeration.
- * @enum {string}
- */
- ControlNetModelFormat: "checkpoint" | "diffusers";
- /**
- * CLIPVisionModelFormat
- * @description An enumeration.
- * @enum {string}
- */
- CLIPVisionModelFormat: "diffusers";
- /**
- * IPAdapterModelFormat
- * @description An enumeration.
- * @enum {string}
- */
- IPAdapterModelFormat: "invokeai";
- /**
- * T2IAdapterModelFormat
- * @description An enumeration.
- * @enum {string}
- */
- T2IAdapterModelFormat: "diffusers";
/**
* StableDiffusionXLModelFormat
* @description An enumeration.
@@ -9914,6 +9911,18 @@ export type components = {
* @enum {string}
*/
StableDiffusion2ModelFormat: "checkpoint" | "diffusers";
+ /**
+ * CLIPVisionModelFormat
+ * @description An enumeration.
+ * @enum {string}
+ */
+ CLIPVisionModelFormat: "diffusers";
+ /**
+ * T2IAdapterModelFormat
+ * @description An enumeration.
+ * @enum {string}
+ */
+ T2IAdapterModelFormat: "diffusers";
};
responses: never;
parameters: never;
@@ -11732,8 +11741,10 @@ export type operations = {
order_by?: components["schemas"]["WorkflowRecordOrderBy"];
/** @description The order by */
direction?: components["schemas"]["SQLiteDirection"];
+ /** @description The category to get */
+ category?: components["schemas"]["WorkflowCategory"];
/** @description The name to filter by */
- filter_name?: string | null;
+ filter_text?: string | null;
};
};
responses: {
diff --git a/pyproject.toml b/pyproject.toml
index 20c25f6449..d28788b31d 100644
--- a/pyproject.toml
+++ b/pyproject.toml
@@ -165,6 +165,7 @@ version = { attr = "invokeai.version.__version__" }
[tool.setuptools.package-data]
"invokeai.app.assets" = ["**/*.png"]
+"invokeai.app.assets.workflows" = ["**/*.json"]
"invokeai.assets.fonts" = ["**/*.ttf"]
"invokeai.backend" = ["**.png"]
"invokeai.configs" = ["*.example", "**/*.yaml", "*.txt"]