mirror of
https://github.com/invoke-ai/InvokeAI
synced 2024-08-30 20:32:17 +00:00
Add installer support for ip-adapters (#4677)
## What type of PR is this? (check all applicable) - [X] Feature ## Have you discussed this change with the InvokeAI team? - [X] Yes ## Have you updated all relevant documentation? - [X] Yes ## Description This PR adds support for selecting and installing IP-Adapters at configure time. The user is offered the four existing InvokeAI IP Adapters in the UI as shown below. The matching image encoders are selected and installed behind the scenes. That is, if the user selects one of the three sd15 adapters, then the SD encoder will be installed. If they select the sdxl adapter, then the SDXL encoder will be installed. ![image](https://github.com/invoke-ai/InvokeAI/assets/111189/19f46401-99fb-4f7b-9a5e-8f2efd0a5b77) Note that the automatic selection of the encoder does not work when the installer is run in headless mode. I may be able to fix that soon, but I'm out of time today.
This commit is contained in:
commit
ce5122f87c
@ -75,14 +75,6 @@ LEGACY_CONFIGS = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
|
||||||
class ModelInstallList:
|
|
||||||
"""Class for listing models to be installed/removed"""
|
|
||||||
|
|
||||||
install_models: List[str] = field(default_factory=list)
|
|
||||||
remove_models: List[str] = field(default_factory=list)
|
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
@dataclass
|
||||||
class InstallSelections:
|
class InstallSelections:
|
||||||
install_models: List[str] = field(default_factory=list)
|
install_models: List[str] = field(default_factory=list)
|
||||||
@ -100,6 +92,7 @@ class ModelLoadInfo:
|
|||||||
installed: bool = False
|
installed: bool = False
|
||||||
recommended: bool = False
|
recommended: bool = False
|
||||||
default: bool = False
|
default: bool = False
|
||||||
|
requires: Optional[List[str]] = field(default_factory=list)
|
||||||
|
|
||||||
|
|
||||||
class ModelInstall(object):
|
class ModelInstall(object):
|
||||||
@ -137,8 +130,6 @@ class ModelInstall(object):
|
|||||||
|
|
||||||
# supplement with entries in models.yaml
|
# supplement with entries in models.yaml
|
||||||
installed_models = [x for x in self.mgr.list_models()]
|
installed_models = [x for x in self.mgr.list_models()]
|
||||||
# suppresses autoloaded models
|
|
||||||
# installed_models = [x for x in self.mgr.list_models() if not self._is_autoloaded(x)]
|
|
||||||
|
|
||||||
for md in installed_models:
|
for md in installed_models:
|
||||||
base = md["base_model"]
|
base = md["base_model"]
|
||||||
@ -170,9 +161,12 @@ class ModelInstall(object):
|
|||||||
|
|
||||||
def list_models(self, model_type):
|
def list_models(self, model_type):
|
||||||
installed = self.mgr.list_models(model_type=model_type)
|
installed = self.mgr.list_models(model_type=model_type)
|
||||||
|
print()
|
||||||
print(f"Installed models of type `{model_type}`:")
|
print(f"Installed models of type `{model_type}`:")
|
||||||
|
print(f"{'Model Key':50} Model Path")
|
||||||
for i in installed:
|
for i in installed:
|
||||||
print(f"{i['model_name']}\t{i['base_model']}\t{i['path']}")
|
print(f"{'/'.join([i['base_model'],i['model_type'],i['model_name']]):50} {i['path']}")
|
||||||
|
print()
|
||||||
|
|
||||||
# logic here a little reversed to maintain backward compatibility
|
# logic here a little reversed to maintain backward compatibility
|
||||||
def starter_models(self, all_models: bool = False) -> Set[str]:
|
def starter_models(self, all_models: bool = False) -> Set[str]:
|
||||||
@ -210,6 +204,8 @@ class ModelInstall(object):
|
|||||||
job += 1
|
job += 1
|
||||||
|
|
||||||
# add requested models
|
# add requested models
|
||||||
|
self._remove_installed(selections.install_models)
|
||||||
|
self._add_required_models(selections.install_models)
|
||||||
for path in selections.install_models:
|
for path in selections.install_models:
|
||||||
logger.info(f"Installing {path} [{job}/{jobs}]")
|
logger.info(f"Installing {path} [{job}/{jobs}]")
|
||||||
try:
|
try:
|
||||||
@ -269,6 +265,26 @@ class ModelInstall(object):
|
|||||||
|
|
||||||
return models_installed
|
return models_installed
|
||||||
|
|
||||||
|
def _remove_installed(self, model_list: List[str]):
|
||||||
|
all_models = self.all_models()
|
||||||
|
for path in model_list:
|
||||||
|
key = self.reverse_paths.get(path)
|
||||||
|
if key and all_models[key].installed:
|
||||||
|
logger.warning(f"{path} already installed. Skipping.")
|
||||||
|
model_list.remove(path)
|
||||||
|
|
||||||
|
def _add_required_models(self, model_list: List[str]):
|
||||||
|
additional_models = []
|
||||||
|
all_models = self.all_models()
|
||||||
|
for path in model_list:
|
||||||
|
if not (key := self.reverse_paths.get(path)):
|
||||||
|
continue
|
||||||
|
for requirement in all_models[key].requires:
|
||||||
|
requirement_key = self.reverse_paths.get(requirement)
|
||||||
|
if not all_models[requirement_key].installed:
|
||||||
|
additional_models.append(requirement)
|
||||||
|
model_list.extend(additional_models)
|
||||||
|
|
||||||
# install a model from a local path. The optional info parameter is there to prevent
|
# install a model from a local path. The optional info parameter is there to prevent
|
||||||
# the model from being probed twice in the event that it has already been probed.
|
# the model from being probed twice in the event that it has already been probed.
|
||||||
def _install_path(self, path: Path, info: ModelProbeInfo = None) -> AddModelResult:
|
def _install_path(self, path: Path, info: ModelProbeInfo = None) -> AddModelResult:
|
||||||
|
@ -103,3 +103,35 @@ sd-1/lora/LowRA:
|
|||||||
recommended: True
|
recommended: True
|
||||||
sd-1/lora/Ink scenery:
|
sd-1/lora/Ink scenery:
|
||||||
path: https://civitai.com/api/download/models/83390
|
path: https://civitai.com/api/download/models/83390
|
||||||
|
sd-1/ip_adapter/ip_adapter_sd15:
|
||||||
|
repo_id: InvokeAI/ip_adapter_sd15
|
||||||
|
recommended: True
|
||||||
|
requires:
|
||||||
|
- InvokeAI/ip_adapter_sd_image_encoder
|
||||||
|
description: IP-Adapter for SD 1.5 models
|
||||||
|
sd-1/ip_adapter/ip_adapter_plus_sd15:
|
||||||
|
repo_id: InvokeAI/ip_adapter_plus_sd15
|
||||||
|
recommended: False
|
||||||
|
requires:
|
||||||
|
- InvokeAI/ip_adapter_sd_image_encoder
|
||||||
|
description: Refined IP-Adapter for SD 1.5 models
|
||||||
|
sd-1/ip_adapter/ip_adapter_plus_face_sd15:
|
||||||
|
repo_id: InvokeAI/ip_adapter_plus_face_sd15
|
||||||
|
recommended: False
|
||||||
|
requires:
|
||||||
|
- InvokeAI/ip_adapter_sd_image_encoder
|
||||||
|
description: Refined IP-Adapter for SD 1.5 models, adapted for faces
|
||||||
|
sdxl/ip_adapter/ip_adapter_sdxl:
|
||||||
|
repo_id: InvokeAI/ip_adapter_sdxl
|
||||||
|
recommended: False
|
||||||
|
requires:
|
||||||
|
- InvokeAI/ip_adapter_sdxl_image_encoder
|
||||||
|
description: IP-Adapter for SDXL models
|
||||||
|
any/clip_vision/ip_adapter_sd_image_encoder:
|
||||||
|
repo_id: InvokeAI/ip_adapter_sd_image_encoder
|
||||||
|
recommended: False
|
||||||
|
description: Required model for using IP-Adapters with SD-1/2 models
|
||||||
|
any/clip_vision/ip_adapter_sdxl_image_encoder:
|
||||||
|
repo_id: InvokeAI/ip_adapter_sdxl_image_encoder
|
||||||
|
recommended: False
|
||||||
|
description: Required model for using IP-Adapters with SDXL models
|
||||||
|
@ -101,11 +101,12 @@ class addModelsForm(CyclingForm, npyscreen.FormMultiPage):
|
|||||||
"STARTER MODELS",
|
"STARTER MODELS",
|
||||||
"MAIN MODELS",
|
"MAIN MODELS",
|
||||||
"CONTROLNETS",
|
"CONTROLNETS",
|
||||||
|
"IP-ADAPTERS",
|
||||||
"LORA/LYCORIS",
|
"LORA/LYCORIS",
|
||||||
"TEXTUAL INVERSION",
|
"TEXTUAL INVERSION",
|
||||||
],
|
],
|
||||||
value=[self.current_tab],
|
value=[self.current_tab],
|
||||||
columns=5,
|
columns=6,
|
||||||
max_height=2,
|
max_height=2,
|
||||||
relx=8,
|
relx=8,
|
||||||
scroll_exit=True,
|
scroll_exit=True,
|
||||||
@ -130,6 +131,13 @@ class addModelsForm(CyclingForm, npyscreen.FormMultiPage):
|
|||||||
)
|
)
|
||||||
bottom_of_table = max(bottom_of_table, self.nextrely)
|
bottom_of_table = max(bottom_of_table, self.nextrely)
|
||||||
|
|
||||||
|
self.nextrely = top_of_table
|
||||||
|
self.ipadapter_models = self.add_model_widgets(
|
||||||
|
model_type=ModelType.IPAdapter,
|
||||||
|
window_width=window_width,
|
||||||
|
)
|
||||||
|
bottom_of_table = max(bottom_of_table, self.nextrely)
|
||||||
|
|
||||||
self.nextrely = top_of_table
|
self.nextrely = top_of_table
|
||||||
self.lora_models = self.add_model_widgets(
|
self.lora_models = self.add_model_widgets(
|
||||||
model_type=ModelType.Lora,
|
model_type=ModelType.Lora,
|
||||||
@ -343,6 +351,7 @@ class addModelsForm(CyclingForm, npyscreen.FormMultiPage):
|
|||||||
self.starter_pipelines,
|
self.starter_pipelines,
|
||||||
self.pipeline_models,
|
self.pipeline_models,
|
||||||
self.controlnet_models,
|
self.controlnet_models,
|
||||||
|
self.ipadapter_models,
|
||||||
self.lora_models,
|
self.lora_models,
|
||||||
self.ti_models,
|
self.ti_models,
|
||||||
]
|
]
|
||||||
@ -532,6 +541,7 @@ class addModelsForm(CyclingForm, npyscreen.FormMultiPage):
|
|||||||
self.starter_pipelines,
|
self.starter_pipelines,
|
||||||
self.pipeline_models,
|
self.pipeline_models,
|
||||||
self.controlnet_models,
|
self.controlnet_models,
|
||||||
|
self.ipadapter_models,
|
||||||
self.lora_models,
|
self.lora_models,
|
||||||
self.ti_models,
|
self.ti_models,
|
||||||
]
|
]
|
||||||
@ -553,6 +563,25 @@ class addModelsForm(CyclingForm, npyscreen.FormMultiPage):
|
|||||||
if downloads := section.get("download_ids"):
|
if downloads := section.get("download_ids"):
|
||||||
selections.install_models.extend(downloads.value.split())
|
selections.install_models.extend(downloads.value.split())
|
||||||
|
|
||||||
|
# NOT NEEDED - DONE IN BACKEND NOW
|
||||||
|
# # special case for the ipadapter_models. If any of the adapters are
|
||||||
|
# # chosen, then we add the corresponding encoder(s) to the install list.
|
||||||
|
# section = self.ipadapter_models
|
||||||
|
# if section.get("models_selected"):
|
||||||
|
# selected_adapters = [
|
||||||
|
# self.all_models[section["models"][x]].name for x in section.get("models_selected").value
|
||||||
|
# ]
|
||||||
|
# encoders = []
|
||||||
|
# if any(["sdxl" in x for x in selected_adapters]):
|
||||||
|
# encoders.append("ip_adapter_sdxl_image_encoder")
|
||||||
|
# if any(["sd15" in x for x in selected_adapters]):
|
||||||
|
# encoders.append("ip_adapter_sd_image_encoder")
|
||||||
|
# for encoder in encoders:
|
||||||
|
# key = f"any/clip_vision/{encoder}"
|
||||||
|
# repo_id = f"InvokeAI/{encoder}"
|
||||||
|
# if key not in self.all_models:
|
||||||
|
# selections.install_models.append(repo_id)
|
||||||
|
|
||||||
|
|
||||||
class AddModelApplication(npyscreen.NPSAppManaged):
|
class AddModelApplication(npyscreen.NPSAppManaged):
|
||||||
def __init__(self, opt):
|
def __init__(self, opt):
|
||||||
|
Loading…
Reference in New Issue
Block a user