2023-06-24 22:18:09 -05:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2023-06-27 12:21:50 -05:00
|
|
|
from abc import ABC, abstractmethod
|
2023-06-24 22:18:09 -05:00
|
|
|
from pathlib import Path
|
2023-06-27 16:01:24 -05:00
|
|
|
from shutil import rmtree
|
2024-06-06 22:09:47 -05:00
|
|
|
from typing import Any, ClassVar
|
2023-06-24 22:18:09 -05:00
|
|
|
|
2023-11-11 20:04:49 -05:00
|
|
|
from huggingface_hub import snapshot_download
|
2023-06-27 16:01:24 -05:00
|
|
|
|
2024-01-11 12:26:46 -05:00
|
|
|
import ann.ann
|
2024-06-25 11:00:24 -05:00
|
|
|
from app.sessions.ort import OrtSession
|
2024-01-11 12:26:46 -05:00
|
|
|
|
2024-06-06 22:09:47 -05:00
|
|
|
from ..config import clean_name, log, settings
|
|
|
|
from ..schemas import ModelFormat, ModelIdentity, ModelSession, ModelTask, ModelType
|
2024-06-25 11:00:24 -05:00
|
|
|
from ..sessions.ann import AnnSession
|
2023-06-24 22:18:09 -05:00
|
|
|
|
|
|
|
|
|
|
|
class InferenceModel(ABC):
|
2024-06-06 22:09:47 -05:00
|
|
|
depends: ClassVar[list[ModelIdentity]]
|
|
|
|
identity: ClassVar[ModelIdentity]
|
2023-06-24 22:18:09 -05:00
|
|
|
|
2023-08-05 21:45:13 -05:00
|
|
|
def __init__(
|
2023-08-24 23:28:51 -05:00
|
|
|
self,
|
|
|
|
model_name: str,
|
|
|
|
cache_dir: Path | str | None = None,
|
2024-07-10 09:20:43 -05:00
|
|
|
model_format: ModelFormat | None = None,
|
2024-06-25 11:00:24 -05:00
|
|
|
session: ModelSession | None = None,
|
2023-08-24 23:28:51 -05:00
|
|
|
**model_kwargs: Any,
|
2023-08-05 21:45:13 -05:00
|
|
|
) -> None:
|
2024-06-25 11:00:24 -05:00
|
|
|
self.loaded = session is not None
|
2024-06-20 13:13:18 -05:00
|
|
|
self.load_attempts = 0
|
2024-06-06 22:09:47 -05:00
|
|
|
self.model_name = clean_name(model_name)
|
2024-06-25 11:00:24 -05:00
|
|
|
self.cache_dir = Path(cache_dir) if cache_dir is not None else self._cache_dir_default
|
2024-07-10 09:20:43 -05:00
|
|
|
self.model_format = model_format if model_format is not None else self._model_format_default
|
2024-06-25 11:00:24 -05:00
|
|
|
if session is not None:
|
|
|
|
self.session = session
|
2023-08-24 23:28:51 -05:00
|
|
|
|
2023-09-09 04:02:44 -05:00
|
|
|
def download(self) -> None:
|
2023-08-05 21:45:13 -05:00
|
|
|
if not self.cached:
|
2023-08-30 03:22:01 -05:00
|
|
|
log.info(
|
2023-12-20 20:47:56 -05:00
|
|
|
f"Downloading {self.model_type.replace('-', ' ')} model '{self.model_name}'. This may take a while."
|
2023-08-30 03:22:01 -05:00
|
|
|
)
|
2023-09-09 04:02:44 -05:00
|
|
|
self._download()
|
2023-06-27 16:01:24 -05:00
|
|
|
|
2023-09-09 04:02:44 -05:00
|
|
|
def load(self) -> None:
|
|
|
|
if self.loaded:
|
|
|
|
return
|
2024-06-20 13:13:18 -05:00
|
|
|
self.load_attempts += 1
|
2024-06-06 22:09:47 -05:00
|
|
|
|
2023-09-09 04:02:44 -05:00
|
|
|
self.download()
|
2024-07-10 09:20:43 -05:00
|
|
|
attempt = f"Attempt #{self.load_attempts} to load" if self.load_attempts > 1 else "Loading"
|
2024-06-20 13:13:18 -05:00
|
|
|
log.info(f"{attempt} {self.model_type.replace('-', ' ')} model '{self.model_name}' to memory")
|
2024-06-06 22:09:47 -05:00
|
|
|
self.session = self._load()
|
2023-09-09 04:02:44 -05:00
|
|
|
self.loaded = True
|
2023-08-05 21:45:13 -05:00
|
|
|
|
2024-06-06 22:09:47 -05:00
|
|
|
def predict(self, *inputs: Any, **model_kwargs: Any) -> Any:
|
2023-09-09 04:02:44 -05:00
|
|
|
self.load()
|
2023-08-29 08:58:00 -05:00
|
|
|
if model_kwargs:
|
|
|
|
self.configure(**model_kwargs)
|
2024-06-06 22:09:47 -05:00
|
|
|
return self._predict(*inputs, **model_kwargs)
|
2023-08-05 21:45:13 -05:00
|
|
|
|
|
|
|
@abstractmethod
|
2024-06-06 22:09:47 -05:00
|
|
|
def _predict(self, *inputs: Any, **model_kwargs: Any) -> Any: ...
|
2023-06-27 16:01:24 -05:00
|
|
|
|
2024-06-06 22:09:47 -05:00
|
|
|
def configure(self, **kwargs: Any) -> None:
|
2023-08-29 08:58:00 -05:00
|
|
|
pass
|
|
|
|
|
2023-09-09 04:02:44 -05:00
|
|
|
def _download(self) -> None:
|
2024-06-25 11:00:24 -05:00
|
|
|
ignore_patterns = [] if self.model_format == ModelFormat.ARMNN else ["*.armnn"]
|
2023-11-11 20:04:49 -05:00
|
|
|
snapshot_download(
|
2024-06-06 22:09:47 -05:00
|
|
|
f"immich-app/{clean_name(self.model_name)}",
|
2023-11-11 20:04:49 -05:00
|
|
|
cache_dir=self.cache_dir,
|
|
|
|
local_dir=self.cache_dir,
|
|
|
|
local_dir_use_symlinks=False,
|
2024-01-28 10:31:59 -05:00
|
|
|
ignore_patterns=ignore_patterns,
|
2023-11-11 20:04:49 -05:00
|
|
|
)
|
2023-08-05 21:45:13 -05:00
|
|
|
|
2024-06-06 22:09:47 -05:00
|
|
|
def _load(self) -> ModelSession:
|
|
|
|
return self._make_session(self.model_path)
|
2023-06-24 22:18:09 -05:00
|
|
|
|
2023-06-27 16:01:24 -05:00
|
|
|
def clear_cache(self) -> None:
|
|
|
|
if not self.cache_dir.exists():
|
2024-01-21 18:22:39 -05:00
|
|
|
log.warning(
|
2023-12-20 20:47:56 -05:00
|
|
|
f"Attempted to clear cache for model '{self.model_name}', but cache directory does not exist",
|
2023-08-30 03:22:01 -05:00
|
|
|
)
|
2023-06-27 16:01:24 -05:00
|
|
|
return
|
2023-08-05 21:45:13 -05:00
|
|
|
if not rmtree.avoids_symlink_attacks:
|
2023-12-20 20:47:56 -05:00
|
|
|
raise RuntimeError("Attempted to clear cache, but rmtree is not safe on this platform")
|
2023-06-27 16:01:24 -05:00
|
|
|
|
2023-08-05 21:45:13 -05:00
|
|
|
if self.cache_dir.is_dir():
|
2023-08-30 03:22:01 -05:00
|
|
|
log.info(f"Cleared cache directory for model '{self.model_name}'.")
|
2023-08-05 21:45:13 -05:00
|
|
|
rmtree(self.cache_dir)
|
|
|
|
else:
|
2024-01-21 18:22:39 -05:00
|
|
|
log.warning(
|
2023-08-30 03:22:01 -05:00
|
|
|
(
|
|
|
|
f"Encountered file instead of directory at cache path "
|
|
|
|
f"for '{self.model_name}'. Removing file and replacing with a directory."
|
|
|
|
),
|
|
|
|
)
|
2023-08-05 21:45:13 -05:00
|
|
|
self.cache_dir.unlink()
|
|
|
|
self.cache_dir.mkdir(parents=True, exist_ok=True)
|
2023-08-24 23:28:51 -05:00
|
|
|
|
2024-06-06 22:09:47 -05:00
|
|
|
def _make_session(self, model_path: Path) -> ModelSession:
|
2024-07-10 09:20:43 -05:00
|
|
|
if not model_path.is_file():
|
|
|
|
raise FileNotFoundError(f"Model file not found: {model_path}")
|
|
|
|
|
2024-01-28 10:31:59 -05:00
|
|
|
match model_path.suffix:
|
|
|
|
case ".armnn":
|
2024-06-25 11:00:24 -05:00
|
|
|
session: ModelSession = AnnSession(model_path)
|
2024-01-28 10:31:59 -05:00
|
|
|
case ".onnx":
|
2024-06-25 11:00:24 -05:00
|
|
|
session = OrtSession(model_path)
|
2024-01-28 10:31:59 -05:00
|
|
|
case _:
|
|
|
|
raise ValueError(f"Unsupported model file type: {model_path.suffix}")
|
2024-01-11 12:26:46 -05:00
|
|
|
return session
|
|
|
|
|
2024-06-06 22:09:47 -05:00
|
|
|
@property
|
|
|
|
def model_dir(self) -> Path:
|
|
|
|
return self.cache_dir / self.model_type.value
|
|
|
|
|
|
|
|
@property
|
|
|
|
def model_path(self) -> Path:
|
2024-06-25 11:00:24 -05:00
|
|
|
return self.model_dir / f"model.{self.model_format}"
|
2024-06-06 22:09:47 -05:00
|
|
|
|
|
|
|
@property
|
|
|
|
def model_task(self) -> ModelTask:
|
|
|
|
return self.identity[1]
|
|
|
|
|
2024-01-21 18:22:39 -05:00
|
|
|
@property
|
|
|
|
def model_type(self) -> ModelType:
|
2024-06-06 22:09:47 -05:00
|
|
|
return self.identity[0]
|
2024-01-21 18:22:39 -05:00
|
|
|
|
|
|
|
@property
|
|
|
|
def cache_dir(self) -> Path:
|
|
|
|
return self._cache_dir
|
|
|
|
|
|
|
|
@cache_dir.setter
|
|
|
|
def cache_dir(self, cache_dir: Path) -> None:
|
|
|
|
self._cache_dir = cache_dir
|
|
|
|
|
|
|
|
@property
|
2024-06-25 11:00:24 -05:00
|
|
|
def _cache_dir_default(self) -> Path:
|
2024-06-06 22:09:47 -05:00
|
|
|
return settings.cache_folder / self.model_task.value / self.model_name
|
2024-01-21 18:22:39 -05:00
|
|
|
|
|
|
|
@property
|
|
|
|
def cached(self) -> bool:
|
2024-06-06 22:09:47 -05:00
|
|
|
return self.model_path.is_file()
|
2024-01-21 18:22:39 -05:00
|
|
|
|
|
|
|
@property
|
2024-06-25 11:00:24 -05:00
|
|
|
def model_format(self) -> ModelFormat:
|
2024-07-10 09:20:43 -05:00
|
|
|
return self._model_format
|
2024-01-28 10:31:59 -05:00
|
|
|
|
2024-06-25 11:00:24 -05:00
|
|
|
@model_format.setter
|
2024-07-10 09:20:43 -05:00
|
|
|
def model_format(self, model_format: ModelFormat) -> None:
|
|
|
|
log.debug(f"Setting model format to {model_format}")
|
|
|
|
self._model_format = model_format
|
2024-01-28 10:31:59 -05:00
|
|
|
|
|
|
|
@property
|
2024-06-25 11:00:24 -05:00
|
|
|
def _model_format_default(self) -> ModelFormat:
|
2024-07-10 09:20:43 -05:00
|
|
|
return ModelFormat.ARMNN if ann.ann.is_available and settings.ann else ModelFormat.ONNX
|