InvokeAI/invokeai/backend/install/migrate_to_3.py

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

604 lines
22 KiB
Python
Raw Normal View History

2023-07-28 13:46:44 +00:00
"""
2023-06-22 19:47:12 +00:00
Migrate the models directory and models.yaml file from an existing
InvokeAI 2.3 installation to 3.0.0.
2023-07-28 13:46:44 +00:00
"""
2023-06-22 19:47:12 +00:00
import os
import argparse
import shutil
import yaml
import transformers
import diffusers
import warnings
from dataclasses import dataclass
from pathlib import Path
2023-06-23 17:56:30 +00:00
from omegaconf import OmegaConf, DictConfig
from typing import Union
2023-06-22 19:47:12 +00:00
from diffusers import StableDiffusionPipeline, AutoencoderKL
from diffusers.pipelines.stable_diffusion.safety_checker import StableDiffusionSafetyChecker
from transformers import (
CLIPTextModel,
CLIPTokenizer,
AutoFeatureExtractor,
BertTokenizerFast,
)
import invokeai.backend.util.logging as logger
from invokeai.app.services.config import InvokeAIAppConfig
2023-06-22 19:47:12 +00:00
from invokeai.backend.model_management import ModelManager
2023-07-28 13:46:44 +00:00
from invokeai.backend.model_management.model_probe import ModelProbe, ModelType, BaseModelType, ModelProbeInfo
2023-06-22 19:47:12 +00:00
warnings.filterwarnings("ignore")
transformers.logging.set_verbosity_error()
diffusers.logging.set_verbosity_error()
2023-07-28 13:46:44 +00:00
2023-06-22 19:47:12 +00:00
# holder for paths that we will migrate
@dataclass
class ModelPaths:
models: Path
embeddings: Path
loras: Path
controlnets: Path
2023-07-28 13:46:44 +00:00
2023-06-22 19:47:12 +00:00
class MigrateTo3(object):
2023-07-28 13:46:44 +00:00
def __init__(
self,
from_root: Path,
to_models: Path,
model_manager: ModelManager,
src_paths: ModelPaths,
):
self.root_directory = from_root
self.dest_models = to_models
self.mgr = model_manager
2023-06-22 19:47:12 +00:00
self.src_paths = src_paths
2023-07-28 13:46:44 +00:00
@classmethod
def initialize_yaml(cls, yaml_file: Path):
2023-07-28 13:46:44 +00:00
with open(yaml_file, "w") as file:
file.write(yaml.dump({"__metadata__": {"version": "3.0.0"}}))
2023-06-22 19:47:12 +00:00
def create_directory_structure(self):
2023-07-28 13:46:44 +00:00
"""
2023-06-22 19:47:12 +00:00
Create the basic directory structure for the models folder.
2023-07-28 13:46:44 +00:00
"""
for model_base in [BaseModelType.StableDiffusion1, BaseModelType.StableDiffusion2]:
for model_type in [
ModelType.Main,
ModelType.Vae,
ModelType.Lora,
ModelType.ControlNet,
ModelType.TextualInversion,
]:
2023-06-22 19:47:12 +00:00
path = self.dest_models / model_base.value / model_type.value
path.mkdir(parents=True, exist_ok=True)
2023-07-28 13:46:44 +00:00
path = self.dest_models / "core"
2023-06-22 19:47:12 +00:00
path.mkdir(parents=True, exist_ok=True)
@staticmethod
2023-07-28 13:46:44 +00:00
def copy_file(src: Path, dest: Path):
"""
2023-06-22 19:47:12 +00:00
copy a single file with logging
2023-07-28 13:46:44 +00:00
"""
2023-06-23 17:56:30 +00:00
if dest.exists():
2023-07-28 13:46:44 +00:00
logger.info(f"Skipping existing {str(dest)}")
2023-06-23 17:56:30 +00:00
return
2023-07-28 13:46:44 +00:00
logger.info(f"Copying {str(src)} to {str(dest)}")
2023-06-22 19:47:12 +00:00
try:
shutil.copy(src, dest)
except Exception as e:
2023-07-28 13:46:44 +00:00
logger.error(f"COPY FAILED: {str(e)}")
2023-06-22 19:47:12 +00:00
@staticmethod
2023-07-28 13:46:44 +00:00
def copy_dir(src: Path, dest: Path):
"""
2023-06-22 19:47:12 +00:00
Recursively copy a directory with logging
2023-07-28 13:46:44 +00:00
"""
2023-06-23 17:56:30 +00:00
if dest.exists():
2023-07-28 13:46:44 +00:00
logger.info(f"Skipping existing {str(dest)}")
2023-06-23 17:56:30 +00:00
return
2023-07-28 13:46:44 +00:00
logger.info(f"Copying {str(src)} to {str(dest)}")
2023-06-22 19:47:12 +00:00
try:
shutil.copytree(src, dest)
except Exception as e:
2023-07-28 13:46:44 +00:00
logger.error(f"COPY FAILED: {str(e)}")
2023-06-22 19:47:12 +00:00
def migrate_models(self, src_dir: Path):
2023-07-28 13:46:44 +00:00
"""
2023-06-22 19:47:12 +00:00
Recursively walk through src directory, probe anything
that looks like a model, and copy the model into the
appropriate location within the destination models directory.
2023-07-28 13:46:44 +00:00
"""
directories_scanned = set()
2023-06-22 19:47:12 +00:00
for root, dirs, files in os.walk(src_dir):
for d in dirs:
2023-06-22 19:47:12 +00:00
try:
2023-07-28 13:46:44 +00:00
model = Path(root, d)
2023-06-22 19:47:12 +00:00
info = ModelProbe().heuristic_probe(model)
if not info:
continue
dest = self._model_probe_to_path(info) / model.name
self.copy_dir(model, dest)
directories_scanned.add(model)
except Exception as e:
logger.error(str(e))
2023-06-22 19:47:12 +00:00
except KeyboardInterrupt:
raise
except Exception as e:
logger.error(str(e))
for f in files:
# don't copy raw learned_embeds.bin or pytorch_lora_weights.bin
# let them be copied as part of a tree copy operation
2023-06-22 19:47:12 +00:00
try:
2023-07-28 13:46:44 +00:00
if f in {"learned_embeds.bin", "pytorch_lora_weights.bin"}:
continue
2023-07-28 13:46:44 +00:00
model = Path(root, f)
if model.parent in directories_scanned:
continue
2023-06-22 19:47:12 +00:00
info = ModelProbe().heuristic_probe(model)
if not info:
continue
dest = self._model_probe_to_path(info) / f
self.copy_file(model, dest)
except Exception as e:
logger.error(str(e))
2023-06-22 19:47:12 +00:00
except KeyboardInterrupt:
raise
except Exception as e:
logger.error(str(e))
def migrate_support_models(self):
2023-07-28 13:46:44 +00:00
"""
2023-06-22 19:47:12 +00:00
Copy the clipseg, upscaler, and restoration models to their new
locations.
2023-07-28 13:46:44 +00:00
"""
2023-06-22 19:47:12 +00:00
dest_directory = self.dest_models
2023-07-28 13:46:44 +00:00
if (self.root_directory / "models/clipseg").exists():
self.copy_dir(self.root_directory / "models/clipseg", dest_directory / "core/misc/clipseg")
if (self.root_directory / "models/realesrgan").exists():
self.copy_dir(self.root_directory / "models/realesrgan", dest_directory / "core/upscaling/realesrgan")
for d in ["codeformer", "gfpgan"]:
path = self.root_directory / "models" / d
2023-06-22 19:47:12 +00:00
if path.exists():
2023-07-28 13:46:44 +00:00
self.copy_dir(path, dest_directory / f"core/face_restoration/{d}")
2023-06-22 19:47:12 +00:00
def migrate_tuning_models(self):
2023-07-28 13:46:44 +00:00
"""
2023-06-22 19:47:12 +00:00
Migrate the embeddings, loras and controlnets directories to their new homes.
2023-07-28 13:46:44 +00:00
"""
2023-06-22 19:47:12 +00:00
for src in [self.src_paths.embeddings, self.src_paths.loras, self.src_paths.controlnets]:
if not src:
continue
if src.is_dir():
2023-07-28 13:46:44 +00:00
logger.info(f"Scanning {src}")
2023-06-22 19:47:12 +00:00
self.migrate_models(src)
else:
2023-07-28 13:46:44 +00:00
logger.info(f"{src} directory not found; skipping")
2023-06-22 19:47:12 +00:00
continue
def migrate_conversion_models(self):
2023-07-28 13:46:44 +00:00
"""
2023-06-22 19:47:12 +00:00
Migrate all the models that are needed by the ckpt_to_diffusers conversion
script.
2023-07-28 13:46:44 +00:00
"""
2023-06-22 19:47:12 +00:00
dest_directory = self.dest_models
kwargs = dict(
2023-07-28 13:46:44 +00:00
cache_dir=self.root_directory / "models/hub",
# local_files_only = True
2023-06-22 19:47:12 +00:00
)
try:
2023-07-28 13:46:44 +00:00
logger.info("Migrating core tokenizers and text encoders")
target_dir = dest_directory / "core" / "convert"
2023-06-22 19:47:12 +00:00
2023-07-28 13:46:44 +00:00
self._migrate_pretrained(
BertTokenizerFast, repo_id="bert-base-uncased", dest=target_dir / "bert-base-uncased", **kwargs
)
2023-06-22 19:47:12 +00:00
# sd-1
2023-07-28 13:46:44 +00:00
repo_id = "openai/clip-vit-large-patch14"
self._migrate_pretrained(
CLIPTokenizer, repo_id=repo_id, dest=target_dir / "clip-vit-large-patch14", **kwargs
)
self._migrate_pretrained(
CLIPTextModel, repo_id=repo_id, dest=target_dir / "clip-vit-large-patch14", force=True, **kwargs
)
2023-06-22 19:47:12 +00:00
# sd-2
repo_id = "stabilityai/stable-diffusion-2"
2023-07-28 13:46:44 +00:00
self._migrate_pretrained(
CLIPTokenizer,
repo_id=repo_id,
dest=target_dir / "stable-diffusion-2-clip" / "tokenizer",
**{"subfolder": "tokenizer", **kwargs},
)
self._migrate_pretrained(
CLIPTextModel,
repo_id=repo_id,
dest=target_dir / "stable-diffusion-2-clip" / "text_encoder",
**{"subfolder": "text_encoder", **kwargs},
)
2023-06-22 19:47:12 +00:00
# VAE
2023-07-28 13:46:44 +00:00
logger.info("Migrating stable diffusion VAE")
self._migrate_pretrained(
AutoencoderKL, repo_id="stabilityai/sd-vae-ft-mse", dest=target_dir / "sd-vae-ft-mse", **kwargs
)
2023-06-22 19:47:12 +00:00
# safety checking
2023-07-28 13:46:44 +00:00
logger.info("Migrating safety checker")
2023-06-22 19:47:12 +00:00
repo_id = "CompVis/stable-diffusion-safety-checker"
2023-07-28 13:46:44 +00:00
self._migrate_pretrained(
AutoFeatureExtractor, repo_id=repo_id, dest=target_dir / "stable-diffusion-safety-checker", **kwargs
)
self._migrate_pretrained(
StableDiffusionSafetyChecker,
repo_id=repo_id,
dest=target_dir / "stable-diffusion-safety-checker",
**kwargs,
)
2023-06-22 19:47:12 +00:00
except KeyboardInterrupt:
raise
except Exception as e:
logger.error(str(e))
2023-07-28 13:46:44 +00:00
def _model_probe_to_path(self, info: ModelProbeInfo) -> Path:
2023-06-23 17:56:30 +00:00
return Path(self.dest_models, info.base_type.value, info.model_type.value)
2023-07-28 13:46:44 +00:00
def _migrate_pretrained(self, model_class, repo_id: str, dest: Path, force: bool = False, **kwargs):
if dest.exists() and not force:
2023-07-28 13:46:44 +00:00
logger.info(f"Skipping existing {dest}")
2023-06-23 17:56:30 +00:00
return
model = model_class.from_pretrained(repo_id, **kwargs)
self._save_pretrained(model, dest, overwrite=force)
2023-06-22 19:47:12 +00:00
2023-07-28 13:46:44 +00:00
def _save_pretrained(self, model, dest: Path, overwrite: bool = False):
2023-06-23 17:56:30 +00:00
model_name = dest.name
if overwrite:
model.save_pretrained(dest, safe_serialization=True)
else:
2023-07-28 13:46:44 +00:00
download_path = dest.with_name(f"{model_name}.downloading")
model.save_pretrained(download_path, safe_serialization=True)
download_path.replace(dest)
2023-06-23 17:56:30 +00:00
2023-07-28 13:46:44 +00:00
def _download_vae(self, repo_id: str, subfolder: str = None) -> Path:
vae = AutoencoderKL.from_pretrained(repo_id, cache_dir=self.root_directory / "models/hub", subfolder=subfolder)
2023-06-23 17:56:30 +00:00
info = ModelProbe().heuristic_probe(vae)
2023-07-28 13:46:44 +00:00
_, model_name = repo_id.split("/")
2023-06-23 17:56:30 +00:00
dest = self._model_probe_to_path(info) / self.unique_name(model_name, info)
vae.save_pretrained(dest, safe_serialization=True)
return dest
2023-07-28 13:46:44 +00:00
def _vae_path(self, vae: Union[str, dict]) -> Path:
"""
2023-06-23 17:56:30 +00:00
Convert 2.3 VAE stanza to a straight path.
2023-07-28 13:46:44 +00:00
"""
2023-06-23 17:56:30 +00:00
vae_path = None
2023-07-28 13:46:44 +00:00
2023-06-23 17:56:30 +00:00
# First get a path
2023-07-28 13:46:44 +00:00
if isinstance(vae, str):
2023-06-23 17:56:30 +00:00
vae_path = vae
2023-07-28 13:46:44 +00:00
elif isinstance(vae, DictConfig):
if p := vae.get("path"):
2023-06-23 17:56:30 +00:00
vae_path = p
2023-07-28 13:46:44 +00:00
elif repo_id := vae.get("repo_id"):
if repo_id == "stabilityai/sd-vae-ft-mse": # this guy is already downloaded
vae_path = "models/core/convert/sd-vae-ft-mse"
return vae_path
2023-06-23 17:56:30 +00:00
else:
2023-07-28 13:46:44 +00:00
vae_path = self._download_vae(repo_id, vae.get("subfolder"))
2023-06-23 17:56:30 +00:00
assert vae_path is not None, "Couldn't find VAE for this model"
# if the VAE is in the old models directory, then we must move it into the new
# one. VAEs outside of this directory can stay where they are.
vae_path = Path(vae_path)
if vae_path.is_relative_to(self.src_paths.models):
info = ModelProbe().heuristic_probe(vae_path)
dest = self._model_probe_to_path(info) / vae_path.name
if not dest.exists():
if vae_path.is_dir():
2023-07-28 13:46:44 +00:00
self.copy_dir(vae_path, dest)
else:
2023-07-28 13:46:44 +00:00
self.copy_file(vae_path, dest)
2023-06-23 17:56:30 +00:00
vae_path = dest
if vae_path.is_relative_to(self.dest_models):
rel_path = vae_path.relative_to(self.dest_models)
2023-07-28 13:46:44 +00:00
return Path("models", rel_path)
2023-06-23 17:56:30 +00:00
else:
return vae_path
2023-07-28 13:46:44 +00:00
def migrate_repo_id(self, repo_id: str, model_name: str = None, **extra_config):
"""
2023-06-22 19:47:12 +00:00
Migrate a locally-cached diffusers pipeline identified with a repo_id
2023-07-28 13:46:44 +00:00
"""
2023-06-22 19:47:12 +00:00
dest_dir = self.dest_models
2023-07-28 13:46:44 +00:00
cache = self.root_directory / "models/hub"
2023-06-22 19:47:12 +00:00
kwargs = dict(
2023-07-28 13:46:44 +00:00
cache_dir=cache,
safety_checker=None,
2023-06-22 19:47:12 +00:00
# local_files_only = True,
)
2023-07-28 13:46:44 +00:00
owner, repo_name = repo_id.split("/")
2023-06-22 19:47:12 +00:00
model_name = model_name or repo_name
2023-07-28 13:46:44 +00:00
model = cache / "--".join(["models", owner, repo_name])
if len(list(model.glob("snapshots/**/model_index.json"))) == 0:
2023-06-22 19:47:12 +00:00
return
2023-07-28 13:46:44 +00:00
revisions = [x.name for x in model.glob("refs/*")]
2023-06-22 19:47:12 +00:00
# if an fp16 is available we use that
2023-07-28 13:46:44 +00:00
revision = "fp16" if len(revisions) > 1 and "fp16" in revisions else revisions[0]
pipeline = StableDiffusionPipeline.from_pretrained(repo_id, revision=revision, **kwargs)
2023-06-22 19:47:12 +00:00
info = ModelProbe().heuristic_probe(pipeline)
if not info:
return
if self.mgr.model_exists(model_name, info.base_type, info.model_type):
2023-07-28 13:46:44 +00:00
logger.warning(f"A model named {model_name} already exists at the destination. Skipping migration.")
return
dest = self._model_probe_to_path(info) / model_name
2023-06-23 17:56:30 +00:00
self._save_pretrained(pipeline, dest)
2023-07-28 13:46:44 +00:00
rel_path = Path("models", dest.relative_to(dest_dir))
self._add_model(model_name, info, rel_path, **extra_config)
2023-06-22 19:47:12 +00:00
2023-07-28 13:46:44 +00:00
def migrate_path(self, location: Path, model_name: str = None, **extra_config):
"""
2023-06-22 19:47:12 +00:00
Migrate a model referred to using 'weights' or 'path'
2023-07-28 13:46:44 +00:00
"""
2023-06-22 19:47:12 +00:00
# handle relative paths
dest_dir = self.dest_models
location = self.root_directory / location
model_name = model_name or location.stem
2023-07-28 13:46:44 +00:00
2023-06-22 19:47:12 +00:00
info = ModelProbe().heuristic_probe(location)
if not info:
return
2023-07-28 13:46:44 +00:00
if self.mgr.model_exists(model_name, info.base_type, info.model_type):
2023-07-28 13:46:44 +00:00
logger.warning(f"A model named {model_name} already exists at the destination. Skipping migration.")
return
2023-06-22 19:47:12 +00:00
# uh oh, weights is in the old models directory - move it into the new one
if Path(location).is_relative_to(self.src_paths.models):
dest = Path(dest_dir, info.base_type.value, info.model_type.value, location.name)
if location.is_dir():
2023-07-28 13:46:44 +00:00
self.copy_dir(location, dest)
else:
2023-07-28 13:46:44 +00:00
self.copy_file(location, dest)
location = Path("models", info.base_type.value, info.model_type.value, location.name)
2023-06-22 19:47:12 +00:00
self._add_model(model_name, info, location, **extra_config)
2023-07-28 13:46:44 +00:00
def _add_model(self, model_name: str, info: ModelProbeInfo, location: Path, **extra_config):
if info.model_type != ModelType.Main:
return
2023-07-28 13:46:44 +00:00
self.mgr.add_model(
2023-07-28 13:46:44 +00:00
model_name=model_name,
base_model=info.base_type,
model_type=info.model_type,
clobber=True,
model_attributes={
"path": str(location),
"description": f"A {info.base_type.value} {info.model_type.value} model",
"model_format": info.format,
"variant": info.variant_type.value,
**extra_config,
2023-07-28 13:46:44 +00:00
},
)
2023-07-28 13:46:44 +00:00
2023-06-22 19:47:12 +00:00
def migrate_defined_models(self):
2023-07-28 13:46:44 +00:00
"""
2023-06-22 19:47:12 +00:00
Migrate models defined in models.yaml
2023-07-28 13:46:44 +00:00
"""
2023-06-22 19:47:12 +00:00
# find any models referred to in old models.yaml
2023-07-28 13:46:44 +00:00
conf = OmegaConf.load(self.root_directory / "configs/models.yaml")
2023-06-22 19:47:12 +00:00
2023-07-28 13:46:44 +00:00
for model_name, stanza in conf.items():
2023-06-22 19:47:12 +00:00
try:
2023-06-23 17:56:30 +00:00
passthru_args = {}
2023-07-28 13:46:44 +00:00
if vae := stanza.get("vae"):
2023-06-23 17:56:30 +00:00
try:
2023-07-28 13:46:44 +00:00
passthru_args["vae"] = str(self._vae_path(vae))
2023-06-23 17:56:30 +00:00
except Exception as e:
logger.warning(f'Could not find a VAE matching "{vae}" for model "{model_name}"')
logger.warning(str(e))
2023-07-28 13:46:44 +00:00
if config := stanza.get("config"):
passthru_args["config"] = config
2023-07-28 13:46:44 +00:00
if description := stanza.get("description"):
passthru_args["description"] = description
if repo_id := stanza.get("repo_id"):
logger.info(f"Migrating diffusers model {model_name}")
2023-06-23 17:56:30 +00:00
self.migrate_repo_id(repo_id, model_name, **passthru_args)
2023-06-22 19:47:12 +00:00
2023-07-28 13:46:44 +00:00
elif location := stanza.get("weights"):
logger.info(f"Migrating checkpoint model {model_name}")
2023-06-23 17:56:30 +00:00
self.migrate_path(Path(location), model_name, **passthru_args)
2023-07-28 13:46:44 +00:00
elif location := stanza.get("path"):
logger.info(f"Migrating diffusers model {model_name}")
2023-06-23 17:56:30 +00:00
self.migrate_path(Path(location), model_name, **passthru_args)
2023-07-28 13:46:44 +00:00
2023-06-22 19:47:12 +00:00
except KeyboardInterrupt:
raise
except Exception as e:
logger.error(str(e))
2023-07-28 13:46:44 +00:00
2023-06-22 19:47:12 +00:00
def migrate(self):
self.create_directory_structure()
# the configure script is doing this
self.migrate_support_models()
self.migrate_conversion_models()
self.migrate_tuning_models()
self.migrate_defined_models()
2023-07-28 13:46:44 +00:00
def _parse_legacy_initfile(root: Path, initfile: Path) -> ModelPaths:
"""
2023-06-22 19:47:12 +00:00
Returns tuple of (embedding_path, lora_path, controlnet_path)
2023-07-28 13:46:44 +00:00
"""
parser = argparse.ArgumentParser(fromfile_prefix_chars="@")
2023-06-22 19:47:12 +00:00
parser.add_argument(
2023-07-28 13:46:44 +00:00
"--embedding_directory",
"--embedding_path",
2023-06-22 19:47:12 +00:00
type=Path,
2023-07-28 13:46:44 +00:00
dest="embedding_path",
default=Path("embeddings"),
2023-06-22 19:47:12 +00:00
)
parser.add_argument(
2023-07-28 13:46:44 +00:00
"--lora_directory",
dest="lora_path",
2023-06-22 19:47:12 +00:00
type=Path,
2023-07-28 13:46:44 +00:00
default=Path("loras"),
2023-06-22 19:47:12 +00:00
)
2023-07-28 13:46:44 +00:00
opt, _ = parser.parse_known_args([f"@{str(initfile)}"])
2023-06-22 19:47:12 +00:00
return ModelPaths(
2023-07-28 13:46:44 +00:00
models=root / "models",
embeddings=root / str(opt.embedding_path).strip('"'),
loras=root / str(opt.lora_path).strip('"'),
controlnets=root / "controlnets",
2023-06-22 19:47:12 +00:00
)
2023-07-28 13:46:44 +00:00
def _parse_legacy_yamlfile(root: Path, initfile: Path) -> ModelPaths:
"""
2023-06-22 19:47:12 +00:00
Returns tuple of (embedding_path, lora_path, controlnet_path)
2023-07-28 13:46:44 +00:00
"""
2023-06-22 19:47:12 +00:00
# Don't use the config object because it is unforgiving of version updates
# Just use omegaconf directly
opt = OmegaConf.load(initfile)
paths = opt.InvokeAI.Paths
2023-07-28 13:46:44 +00:00
models = paths.get("models_dir", "models")
embeddings = paths.get("embedding_dir", "embeddings")
loras = paths.get("lora_dir", "loras")
controlnets = paths.get("controlnet_dir", "controlnets")
2023-06-22 19:47:12 +00:00
return ModelPaths(
2023-07-28 13:46:44 +00:00
models=root / models,
embeddings=root / embeddings,
loras=root / loras,
controlnets=root / controlnets,
2023-06-22 19:47:12 +00:00
)
2023-07-28 13:46:44 +00:00
2023-06-22 19:47:12 +00:00
def get_legacy_embeddings(root: Path) -> ModelPaths:
2023-07-28 13:46:44 +00:00
path = root / "invokeai.init"
2023-06-22 19:47:12 +00:00
if path.exists():
return _parse_legacy_initfile(root, path)
2023-07-28 13:46:44 +00:00
path = root / "invokeai.yaml"
2023-06-22 19:47:12 +00:00
if path.exists():
return _parse_legacy_yamlfile(root, path)
2023-07-28 13:46:44 +00:00
def do_migrate(src_directory: Path, dest_directory: Path):
"""
Migrate models from src to dest InvokeAI root directories
"""
2023-07-28 13:46:44 +00:00
config_file = dest_directory / "configs" / "models.yaml.3"
dest_models = dest_directory / "models.3"
version_3 = (dest_directory / "models" / "core").exists()
# Here we create the destination models.yaml file.
# If we are writing into a version 3 directory and the
# file already exists, then we write into a copy of it to
# avoid deleting its previous customizations. Otherwise we
# create a new empty one.
if version_3: # write into the dest directory
try:
2023-07-28 13:46:44 +00:00
shutil.copy(dest_directory / "configs" / "models.yaml", config_file)
except:
MigrateTo3.initialize_yaml(config_file)
2023-07-28 13:46:44 +00:00
mgr = ModelManager(config_file) # important to initialize BEFORE moving the models directory
(dest_directory / "models").replace(dest_models)
else:
MigrateTo3.initialize_yaml(config_file)
mgr = ModelManager(config_file)
2023-07-28 13:46:44 +00:00
paths = get_legacy_embeddings(src_directory)
2023-07-28 13:46:44 +00:00
migrator = MigrateTo3(from_root=src_directory, to_models=dest_models, model_manager=mgr, src_paths=paths)
migrator.migrate()
print("Migration successful.")
if not version_3:
2023-07-28 13:46:44 +00:00
(dest_directory / "models").replace(src_directory / "models.orig")
print(f"Original models directory moved to {dest_directory}/models.orig")
(dest_directory / "configs" / "models.yaml").replace(src_directory / "configs" / "models.yaml.orig")
print(f"Original models.yaml file moved to {dest_directory}/configs/models.yaml.orig")
config_file.replace(config_file.with_suffix(""))
dest_models.replace(dest_models.with_suffix(""))
2023-06-22 19:47:12 +00:00
def main():
2023-07-28 13:46:44 +00:00
parser = argparse.ArgumentParser(
prog="invokeai-migrate3",
description="""
2023-06-22 19:47:12 +00:00
This will copy and convert the models directory and the configs/models.yaml from the InvokeAI 2.3 format
'--from-directory' root to the InvokeAI 3.0 '--to-directory' root. These may be abbreviated '--from' and '--to'.a
The old models directory and config file will be renamed 'models.orig' and 'models.yaml.orig' respectively.
It is safe to provide the same directory for both arguments, but it is better to use the invokeai_configure
2023-07-28 13:46:44 +00:00
script, which will perform a full upgrade in place.""",
)
parser.add_argument(
"--from-directory",
dest="src_root",
type=Path,
required=True,
help='Source InvokeAI 2.3 root directory (containing "invokeai.init" or "invokeai.yaml")',
)
parser.add_argument(
"--to-directory",
dest="dest_root",
type=Path,
required=True,
help='Destination InvokeAI 3.0 directory (containing "invokeai.yaml")',
)
2023-06-22 19:47:12 +00:00
args = parser.parse_args()
src_root = args.src_root
assert src_root.is_dir(), f"{src_root} is not a valid directory"
2023-07-28 13:46:44 +00:00
assert (src_root / "models").is_dir(), f"{src_root} does not contain a 'models' subdirectory"
assert (src_root / "models" / "hub").exists(), f"{src_root} does not contain a version 2.3 models directory"
assert (src_root / "invokeai.init").exists() or (
src_root / "invokeai.yaml"
).exists(), f"{src_root} does not contain an InvokeAI init file."
2023-06-22 19:47:12 +00:00
dest_root = args.dest_root
assert dest_root.is_dir(), f"{dest_root} is not a valid directory"
config = InvokeAIAppConfig.get_config()
2023-07-28 13:46:44 +00:00
config.parse_args(["--root", str(dest_root)])
# TODO: revisit - don't rely on invokeai.yaml to exist yet!
2023-07-28 13:46:44 +00:00
dest_is_setup = (dest_root / "models/core").exists() and (dest_root / "databases").exists()
if not dest_is_setup:
import invokeai.frontend.install.invokeai_configure
from invokeai.backend.install.invokeai_configure import initialize_rootdir
2023-06-22 19:47:12 +00:00
2023-07-28 13:46:44 +00:00
initialize_rootdir(dest_root, True)
2023-06-22 19:47:12 +00:00
2023-07-28 13:46:44 +00:00
do_migrate(src_root, dest_root)
2023-06-23 17:56:30 +00:00
2023-07-28 13:46:44 +00:00
if __name__ == "__main__":
main()