2023-11-05 03:03:26 +00:00
|
|
|
# Copyright (c) 2023 Lincoln D. Stein and the InvokeAI Development Team
|
|
|
|
"""
|
|
|
|
Configuration definitions for image generation models.
|
|
|
|
|
|
|
|
Typical usage:
|
|
|
|
|
|
|
|
from invokeai.backend.model_manager import ModelConfigFactory
|
|
|
|
raw = dict(path='models/sd-1/main/foo.ckpt',
|
|
|
|
name='foo',
|
2023-11-06 23:08:57 +00:00
|
|
|
base='sd-1',
|
|
|
|
type='main',
|
2023-11-05 03:03:26 +00:00
|
|
|
config='configs/stable-diffusion/v1-inference.yaml',
|
|
|
|
variant='normal',
|
|
|
|
format='checkpoint'
|
|
|
|
)
|
|
|
|
config = ModelConfigFactory.make_config(raw)
|
|
|
|
print(config.name)
|
|
|
|
|
|
|
|
Validation errors will raise an InvalidModelConfigException error.
|
|
|
|
|
|
|
|
"""
|
2024-02-29 23:04:59 +00:00
|
|
|
|
2024-02-04 03:55:09 +00:00
|
|
|
import time
|
2023-11-05 03:03:26 +00:00
|
|
|
from enum import Enum
|
2024-02-10 23:09:45 +00:00
|
|
|
from typing import Literal, Optional, Type, Union
|
2023-11-05 03:03:26 +00:00
|
|
|
|
2024-02-04 22:23:10 +00:00
|
|
|
import torch
|
2024-03-01 02:05:16 +00:00
|
|
|
from diffusers.models.modeling_utils import ModelMixin
|
2024-03-04 08:17:01 +00:00
|
|
|
from pydantic import BaseModel, ConfigDict, Discriminator, Field, Tag, TypeAdapter
|
2023-11-24 04:15:32 +00:00
|
|
|
from typing_extensions import Annotated, Any, Dict
|
2024-02-04 22:23:10 +00:00
|
|
|
|
2024-03-04 10:38:21 +00:00
|
|
|
from invokeai.app.util.misc import uuid_string
|
|
|
|
|
2024-02-17 16:45:32 +00:00
|
|
|
from ..raw_model import RawModel
|
2024-02-05 04:18:00 +00:00
|
|
|
|
2024-02-17 16:45:32 +00:00
|
|
|
# ModelMixin is the base class for all diffusers and transformers models
|
|
|
|
# RawModel is the InvokeAI wrapper class for ip_adapters, loras, textual_inversion and onnx runtime
|
|
|
|
AnyModel = Union[ModelMixin, RawModel, torch.nn.Module]
|
2024-02-06 03:56:32 +00:00
|
|
|
|
2024-02-05 04:18:00 +00:00
|
|
|
|
2023-11-05 03:03:26 +00:00
|
|
|
class InvalidModelConfigException(Exception):
|
|
|
|
"""Exception for when config parser doesn't recognized this combination of model type and format."""
|
|
|
|
|
|
|
|
|
|
|
|
class BaseModelType(str, Enum):
|
|
|
|
"""Base model type."""
|
|
|
|
|
|
|
|
Any = "any"
|
|
|
|
StableDiffusion1 = "sd-1"
|
|
|
|
StableDiffusion2 = "sd-2"
|
|
|
|
StableDiffusionXL = "sdxl"
|
|
|
|
StableDiffusionXLRefiner = "sdxl-refiner"
|
|
|
|
# Kandinsky2_1 = "kandinsky-2.1"
|
|
|
|
|
|
|
|
|
|
|
|
class ModelType(str, Enum):
|
|
|
|
"""Model type."""
|
|
|
|
|
|
|
|
ONNX = "onnx"
|
|
|
|
Main = "main"
|
|
|
|
Vae = "vae"
|
|
|
|
Lora = "lora"
|
|
|
|
ControlNet = "controlnet" # used by model_probe
|
|
|
|
TextualInversion = "embedding"
|
|
|
|
IPAdapter = "ip_adapter"
|
|
|
|
CLIPVision = "clip_vision"
|
|
|
|
T2IAdapter = "t2i_adapter"
|
|
|
|
|
|
|
|
|
|
|
|
class SubModelType(str, Enum):
|
|
|
|
"""Submodel type."""
|
|
|
|
|
|
|
|
UNet = "unet"
|
|
|
|
TextEncoder = "text_encoder"
|
|
|
|
TextEncoder2 = "text_encoder_2"
|
|
|
|
Tokenizer = "tokenizer"
|
|
|
|
Tokenizer2 = "tokenizer_2"
|
|
|
|
Vae = "vae"
|
|
|
|
VaeDecoder = "vae_decoder"
|
|
|
|
VaeEncoder = "vae_encoder"
|
|
|
|
Scheduler = "scheduler"
|
|
|
|
SafetyChecker = "safety_checker"
|
|
|
|
|
|
|
|
|
|
|
|
class ModelVariantType(str, Enum):
|
|
|
|
"""Variant type."""
|
|
|
|
|
|
|
|
Normal = "normal"
|
|
|
|
Inpaint = "inpaint"
|
|
|
|
Depth = "depth"
|
|
|
|
|
|
|
|
|
|
|
|
class ModelFormat(str, Enum):
|
|
|
|
"""Storage format of model."""
|
|
|
|
|
|
|
|
Diffusers = "diffusers"
|
|
|
|
Checkpoint = "checkpoint"
|
|
|
|
Lycoris = "lycoris"
|
|
|
|
Onnx = "onnx"
|
|
|
|
Olive = "olive"
|
|
|
|
EmbeddingFile = "embedding_file"
|
|
|
|
EmbeddingFolder = "embedding_folder"
|
|
|
|
InvokeAI = "invokeai"
|
|
|
|
|
|
|
|
|
|
|
|
class SchedulerPredictionType(str, Enum):
|
|
|
|
"""Scheduler prediction type."""
|
|
|
|
|
|
|
|
Epsilon = "epsilon"
|
|
|
|
VPrediction = "v_prediction"
|
|
|
|
Sample = "sample"
|
|
|
|
|
|
|
|
|
2024-01-14 19:54:53 +00:00
|
|
|
class ModelRepoVariant(str, Enum):
|
|
|
|
"""Various hugging face variants on the diffusers format."""
|
|
|
|
|
2024-02-06 02:55:11 +00:00
|
|
|
DEFAULT = "" # model files without "fp16" or other qualifier - empty str
|
2024-01-14 19:54:53 +00:00
|
|
|
FP16 = "fp16"
|
|
|
|
FP32 = "fp32"
|
|
|
|
ONNX = "onnx"
|
|
|
|
OPENVINO = "openvino"
|
|
|
|
FLAX = "flax"
|
|
|
|
|
|
|
|
|
2024-03-01 11:12:48 +00:00
|
|
|
class ModelSourceType(str, Enum):
|
|
|
|
"""Model source type."""
|
|
|
|
|
|
|
|
Path = "path"
|
|
|
|
Url = "url"
|
|
|
|
HFRepoID = "hf_repo_id"
|
|
|
|
CivitAI = "civitai"
|
|
|
|
|
|
|
|
|
2023-11-05 03:03:26 +00:00
|
|
|
class ModelConfigBase(BaseModel):
|
|
|
|
"""Base class for model configuration information."""
|
|
|
|
|
2024-03-04 10:38:21 +00:00
|
|
|
key: str = Field(description="A unique key for this model.", default_factory=uuid_string)
|
2024-03-01 12:04:33 +00:00
|
|
|
hash: str = Field(description="The hash of the model file(s).")
|
|
|
|
path: str = Field(
|
|
|
|
description="Path to the model on the filesystem. Relative paths are relative to the Invoke root directory."
|
|
|
|
)
|
|
|
|
name: str = Field(description="Name of the model.")
|
|
|
|
base: BaseModelType = Field(description="The base model.")
|
2024-03-01 11:12:48 +00:00
|
|
|
description: Optional[str] = Field(description="Model description", default=None)
|
2024-03-01 12:04:33 +00:00
|
|
|
source: str = Field(description="The original source of the model (path, URL or repo_id).")
|
2024-03-01 11:12:48 +00:00
|
|
|
source_type: ModelSourceType = Field(description="The type of source")
|
2024-03-04 10:38:21 +00:00
|
|
|
source_api_response: Optional[str] = Field(
|
|
|
|
description="The original API response from the source, as stringified JSON.", default=None
|
|
|
|
)
|
2024-03-04 08:16:25 +00:00
|
|
|
trigger_words: Optional[set[str]] = Field(description="Set of trigger words for this model", default=None)
|
2023-11-05 03:03:26 +00:00
|
|
|
|
2024-03-01 12:43:07 +00:00
|
|
|
model_config = ConfigDict(use_enum_values=False, validate_assignment=True)
|
2023-11-05 03:03:26 +00:00
|
|
|
|
|
|
|
|
2024-03-01 02:18:31 +00:00
|
|
|
class CheckpointConfigBase(ModelConfigBase):
|
2023-11-05 03:03:26 +00:00
|
|
|
"""Model config for checkpoint-style models."""
|
|
|
|
|
|
|
|
format: Literal[ModelFormat.Checkpoint] = ModelFormat.Checkpoint
|
2024-03-01 04:25:21 +00:00
|
|
|
config_path: str = Field(description="path to the checkpoint model config file")
|
2024-03-01 04:27:41 +00:00
|
|
|
converted_at: Optional[float] = Field(
|
2024-03-01 04:21:35 +00:00
|
|
|
description="When this model was last converted to diffusers", default_factory=time.time
|
|
|
|
)
|
2023-11-05 03:03:26 +00:00
|
|
|
|
|
|
|
|
2024-03-01 02:18:31 +00:00
|
|
|
class DiffusersConfigBase(ModelConfigBase):
|
2023-11-05 03:03:26 +00:00
|
|
|
"""Model config for diffusers-style models."""
|
|
|
|
|
|
|
|
format: Literal[ModelFormat.Diffusers] = ModelFormat.Diffusers
|
2024-01-22 19:37:23 +00:00
|
|
|
repo_variant: Optional[ModelRepoVariant] = ModelRepoVariant.DEFAULT
|
2023-11-05 03:03:26 +00:00
|
|
|
|
2024-02-01 04:37:59 +00:00
|
|
|
|
2024-03-01 01:57:46 +00:00
|
|
|
class LoRALycorisConfig(ModelConfigBase):
|
2023-11-05 03:03:26 +00:00
|
|
|
"""Model config for LoRA/Lycoris models."""
|
|
|
|
|
2023-11-11 00:14:15 +00:00
|
|
|
type: Literal[ModelType.Lora] = ModelType.Lora
|
2024-03-01 01:57:46 +00:00
|
|
|
format: Literal[ModelFormat.Lycoris] = ModelFormat.Lycoris
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def get_tag() -> Tag:
|
|
|
|
return Tag(f"{ModelType.Lora}.{ModelFormat.Lycoris}")
|
|
|
|
|
|
|
|
|
|
|
|
class LoRADiffusersConfig(ModelConfigBase):
|
|
|
|
"""Model config for LoRA/Diffusers models."""
|
|
|
|
|
|
|
|
type: Literal[ModelType.Lora] = ModelType.Lora
|
|
|
|
format: Literal[ModelFormat.Diffusers] = ModelFormat.Diffusers
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def get_tag() -> Tag:
|
|
|
|
return Tag(f"{ModelType.Lora}.{ModelFormat.Diffusers}")
|
2023-11-05 03:03:26 +00:00
|
|
|
|
|
|
|
|
2024-03-01 04:21:35 +00:00
|
|
|
class VaeCheckpointConfig(CheckpointConfigBase):
|
2023-11-05 03:03:26 +00:00
|
|
|
"""Model config for standalone VAE models."""
|
|
|
|
|
2023-11-11 00:14:15 +00:00
|
|
|
type: Literal[ModelType.Vae] = ModelType.Vae
|
2023-11-05 03:03:26 +00:00
|
|
|
format: Literal[ModelFormat.Checkpoint] = ModelFormat.Checkpoint
|
|
|
|
|
2024-03-01 01:57:46 +00:00
|
|
|
@staticmethod
|
|
|
|
def get_tag() -> Tag:
|
|
|
|
return Tag(f"{ModelType.Vae}.{ModelFormat.Checkpoint}")
|
|
|
|
|
2023-11-05 03:03:26 +00:00
|
|
|
|
|
|
|
class VaeDiffusersConfig(ModelConfigBase):
|
|
|
|
"""Model config for standalone VAE models (diffusers version)."""
|
|
|
|
|
2023-11-11 00:14:15 +00:00
|
|
|
type: Literal[ModelType.Vae] = ModelType.Vae
|
2023-11-05 03:03:26 +00:00
|
|
|
format: Literal[ModelFormat.Diffusers] = ModelFormat.Diffusers
|
|
|
|
|
2024-03-01 01:57:46 +00:00
|
|
|
@staticmethod
|
|
|
|
def get_tag() -> Tag:
|
|
|
|
return Tag(f"{ModelType.Vae}.{ModelFormat.Diffusers}")
|
|
|
|
|
2023-11-05 03:03:26 +00:00
|
|
|
|
2024-03-01 02:18:31 +00:00
|
|
|
class ControlNetDiffusersConfig(DiffusersConfigBase):
|
2023-11-05 03:03:26 +00:00
|
|
|
"""Model config for ControlNet models (diffusers version)."""
|
|
|
|
|
2023-11-11 00:14:15 +00:00
|
|
|
type: Literal[ModelType.ControlNet] = ModelType.ControlNet
|
2023-11-05 03:03:26 +00:00
|
|
|
format: Literal[ModelFormat.Diffusers] = ModelFormat.Diffusers
|
|
|
|
|
2024-03-01 01:57:46 +00:00
|
|
|
@staticmethod
|
|
|
|
def get_tag() -> Tag:
|
|
|
|
return Tag(f"{ModelType.ControlNet}.{ModelFormat.Diffusers}")
|
|
|
|
|
2024-02-01 04:37:59 +00:00
|
|
|
|
2024-03-01 02:18:31 +00:00
|
|
|
class ControlNetCheckpointConfig(CheckpointConfigBase):
|
2023-11-05 03:03:26 +00:00
|
|
|
"""Model config for ControlNet models (diffusers version)."""
|
|
|
|
|
2023-11-11 00:14:15 +00:00
|
|
|
type: Literal[ModelType.ControlNet] = ModelType.ControlNet
|
2023-11-05 03:03:26 +00:00
|
|
|
format: Literal[ModelFormat.Checkpoint] = ModelFormat.Checkpoint
|
|
|
|
|
2024-03-01 01:57:46 +00:00
|
|
|
@staticmethod
|
|
|
|
def get_tag() -> Tag:
|
|
|
|
return Tag(f"{ModelType.ControlNet}.{ModelFormat.Checkpoint}")
|
|
|
|
|
|
|
|
|
|
|
|
class TextualInversionFileConfig(ModelConfigBase):
|
|
|
|
"""Model config for textual inversion embeddings."""
|
|
|
|
|
|
|
|
type: Literal[ModelType.TextualInversion] = ModelType.TextualInversion
|
|
|
|
format: Literal[ModelFormat.EmbeddingFile] = ModelFormat.EmbeddingFile
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def get_tag() -> Tag:
|
|
|
|
return Tag(f"{ModelType.TextualInversion}.{ModelFormat.EmbeddingFile}")
|
|
|
|
|
2023-11-05 03:03:26 +00:00
|
|
|
|
2024-03-01 01:57:46 +00:00
|
|
|
class TextualInversionFolderConfig(ModelConfigBase):
|
2023-11-05 03:03:26 +00:00
|
|
|
"""Model config for textual inversion embeddings."""
|
|
|
|
|
2023-11-11 00:14:15 +00:00
|
|
|
type: Literal[ModelType.TextualInversion] = ModelType.TextualInversion
|
2024-03-01 01:57:46 +00:00
|
|
|
format: Literal[ModelFormat.EmbeddingFolder] = ModelFormat.EmbeddingFolder
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def get_tag() -> Tag:
|
|
|
|
return Tag(f"{ModelType.TextualInversion}.{ModelFormat.EmbeddingFolder}")
|
2023-11-05 03:03:26 +00:00
|
|
|
|
|
|
|
|
2024-03-01 04:21:35 +00:00
|
|
|
class MainCheckpointConfig(CheckpointConfigBase):
|
|
|
|
"""Model config for main checkpoint models."""
|
2023-11-05 03:03:26 +00:00
|
|
|
|
2024-03-01 04:21:35 +00:00
|
|
|
type: Literal[ModelType.Main] = ModelType.Main
|
2023-11-05 03:03:26 +00:00
|
|
|
variant: ModelVariantType = ModelVariantType.Normal
|
2024-02-04 22:23:10 +00:00
|
|
|
prediction_type: SchedulerPredictionType = SchedulerPredictionType.Epsilon
|
|
|
|
upcast_attention: bool = False
|
2023-11-05 03:03:26 +00:00
|
|
|
|
2024-03-01 01:57:46 +00:00
|
|
|
@staticmethod
|
|
|
|
def get_tag() -> Tag:
|
|
|
|
return Tag(f"{ModelType.Main}.{ModelFormat.Checkpoint}")
|
|
|
|
|
2023-11-05 03:03:26 +00:00
|
|
|
|
2024-03-01 04:21:35 +00:00
|
|
|
class MainDiffusersConfig(DiffusersConfigBase):
|
2023-11-05 03:03:26 +00:00
|
|
|
"""Model config for main diffusers models."""
|
|
|
|
|
2023-11-11 00:14:15 +00:00
|
|
|
type: Literal[ModelType.Main] = ModelType.Main
|
2023-11-06 23:08:57 +00:00
|
|
|
|
2024-03-01 01:57:46 +00:00
|
|
|
@staticmethod
|
|
|
|
def get_tag() -> Tag:
|
|
|
|
return Tag(f"{ModelType.Main}.{ModelFormat.Diffusers}")
|
|
|
|
|
2024-02-01 04:37:59 +00:00
|
|
|
|
2023-11-05 03:03:26 +00:00
|
|
|
class IPAdapterConfig(ModelConfigBase):
|
|
|
|
"""Model config for IP Adaptor format models."""
|
|
|
|
|
2023-11-11 00:14:15 +00:00
|
|
|
type: Literal[ModelType.IPAdapter] = ModelType.IPAdapter
|
2024-02-10 01:46:47 +00:00
|
|
|
image_encoder_model_id: str
|
2023-11-05 03:03:26 +00:00
|
|
|
format: Literal[ModelFormat.InvokeAI]
|
|
|
|
|
2024-03-01 01:57:46 +00:00
|
|
|
@staticmethod
|
|
|
|
def get_tag() -> Tag:
|
|
|
|
return Tag(f"{ModelType.IPAdapter}.{ModelFormat.InvokeAI}")
|
|
|
|
|
2023-11-05 03:03:26 +00:00
|
|
|
|
|
|
|
class CLIPVisionDiffusersConfig(ModelConfigBase):
|
|
|
|
"""Model config for ClipVision."""
|
|
|
|
|
2023-11-11 00:14:15 +00:00
|
|
|
type: Literal[ModelType.CLIPVision] = ModelType.CLIPVision
|
2023-11-05 03:03:26 +00:00
|
|
|
format: Literal[ModelFormat.Diffusers]
|
|
|
|
|
2024-03-01 01:57:46 +00:00
|
|
|
@staticmethod
|
|
|
|
def get_tag() -> Tag:
|
|
|
|
return Tag(f"{ModelType.CLIPVision}.{ModelFormat.Diffusers}")
|
|
|
|
|
2023-11-05 03:03:26 +00:00
|
|
|
|
2024-03-01 01:57:46 +00:00
|
|
|
class T2IAdapterConfig(ModelConfigBase):
|
2023-11-05 03:03:26 +00:00
|
|
|
"""Model config for T2I."""
|
|
|
|
|
2023-11-11 00:14:15 +00:00
|
|
|
type: Literal[ModelType.T2IAdapter] = ModelType.T2IAdapter
|
2023-11-05 03:03:26 +00:00
|
|
|
format: Literal[ModelFormat.Diffusers]
|
|
|
|
|
2024-03-01 01:57:46 +00:00
|
|
|
@staticmethod
|
|
|
|
def get_tag() -> Tag:
|
|
|
|
return Tag(f"{ModelType.T2IAdapter}.{ModelFormat.Diffusers}")
|
|
|
|
|
|
|
|
|
|
|
|
def get_model_discriminator_value(v: Any) -> str:
|
|
|
|
"""
|
|
|
|
Computes the discriminator value for a model config.
|
|
|
|
https://docs.pydantic.dev/latest/concepts/unions/#discriminated-unions-with-callable-discriminator
|
|
|
|
"""
|
|
|
|
if isinstance(v, dict):
|
|
|
|
return f"{v.get('type')}.{v.get('format')}" # pyright: ignore [reportUnknownMemberType]
|
2024-03-01 04:19:31 +00:00
|
|
|
return f"{v.type}.{v.format}"
|
2024-03-01 01:57:46 +00:00
|
|
|
|
|
|
|
|
|
|
|
AnyModelConfig = Annotated[
|
|
|
|
Union[
|
|
|
|
Annotated[MainDiffusersConfig, MainDiffusersConfig.get_tag()],
|
|
|
|
Annotated[MainCheckpointConfig, MainCheckpointConfig.get_tag()],
|
|
|
|
Annotated[VaeDiffusersConfig, VaeDiffusersConfig.get_tag()],
|
|
|
|
Annotated[VaeCheckpointConfig, VaeCheckpointConfig.get_tag()],
|
|
|
|
Annotated[ControlNetDiffusersConfig, ControlNetDiffusersConfig.get_tag()],
|
|
|
|
Annotated[ControlNetCheckpointConfig, ControlNetCheckpointConfig.get_tag()],
|
|
|
|
Annotated[LoRALycorisConfig, LoRALycorisConfig.get_tag()],
|
|
|
|
Annotated[LoRADiffusersConfig, LoRADiffusersConfig.get_tag()],
|
|
|
|
Annotated[TextualInversionFileConfig, TextualInversionFileConfig.get_tag()],
|
|
|
|
Annotated[TextualInversionFolderConfig, TextualInversionFolderConfig.get_tag()],
|
|
|
|
Annotated[IPAdapterConfig, IPAdapterConfig.get_tag()],
|
|
|
|
Annotated[T2IAdapterConfig, T2IAdapterConfig.get_tag()],
|
|
|
|
Annotated[CLIPVisionDiffusersConfig, CLIPVisionDiffusersConfig.get_tag()],
|
|
|
|
],
|
|
|
|
Discriminator(get_model_discriminator_value),
|
2023-11-05 03:03:26 +00:00
|
|
|
]
|
|
|
|
|
2023-11-12 21:50:05 +00:00
|
|
|
AnyModelConfigValidator = TypeAdapter(AnyModelConfig)
|
2024-02-06 03:56:32 +00:00
|
|
|
|
2023-11-11 00:14:29 +00:00
|
|
|
|
2023-11-05 03:03:26 +00:00
|
|
|
class ModelConfigFactory(object):
|
|
|
|
"""Class for parsing config dicts into StableDiffusion Config obects."""
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def make_config(
|
|
|
|
cls,
|
2024-02-10 04:08:38 +00:00
|
|
|
model_data: Union[Dict[str, Any], AnyModelConfig],
|
2023-11-05 03:03:26 +00:00
|
|
|
key: Optional[str] = None,
|
2024-02-10 23:09:45 +00:00
|
|
|
dest_class: Optional[Type[ModelConfigBase]] = None,
|
2024-02-04 22:23:10 +00:00
|
|
|
timestamp: Optional[float] = None,
|
2023-11-05 03:03:26 +00:00
|
|
|
) -> AnyModelConfig:
|
|
|
|
"""
|
|
|
|
Return the appropriate config object from raw dict values.
|
|
|
|
|
|
|
|
:param model_data: A raw dict corresponding the obect fields to be
|
|
|
|
parsed into a ModelConfigBase obect (or descendent), or a ModelConfigBase
|
|
|
|
object, which will be passed through unchanged.
|
|
|
|
:param dest_class: The config class to be returned. If not provided, will
|
|
|
|
be selected automatically.
|
|
|
|
"""
|
2024-02-10 23:09:45 +00:00
|
|
|
model: Optional[ModelConfigBase] = None
|
2023-11-05 03:03:26 +00:00
|
|
|
if isinstance(model_data, ModelConfigBase):
|
2023-11-11 17:22:38 +00:00
|
|
|
model = model_data
|
|
|
|
elif dest_class:
|
2024-02-10 23:09:45 +00:00
|
|
|
model = dest_class.model_validate(model_data)
|
2023-11-11 00:14:15 +00:00
|
|
|
else:
|
2024-02-10 23:09:45 +00:00
|
|
|
# mypy doesn't typecheck TypeAdapters well?
|
|
|
|
model = AnyModelConfigValidator.validate_python(model_data) # type: ignore
|
|
|
|
assert model is not None
|
2023-11-11 17:22:38 +00:00
|
|
|
if key:
|
|
|
|
model.key = key
|
2024-03-01 04:21:35 +00:00
|
|
|
if isinstance(model, CheckpointConfigBase) and timestamp is not None:
|
2024-03-01 04:27:41 +00:00
|
|
|
model.converted_at = timestamp
|
2024-02-10 23:09:45 +00:00
|
|
|
return model # type: ignore
|