mirror of
				https://github.com/immich-app/immich.git
				synced 2025-10-25 16:04:21 -04:00 
			
		
		
		
	* modularize model classes * various fixes * expose port * change response * round coordinates * simplify preload * update server * simplify interface simplify * update tests * composable endpoint * cleanup fixes remove unnecessary interface support text input, cleanup * ew camelcase * update server server fixes fix typing * ml fixes update locustfile fixes * cleaner response * better repo response * update tests formatting and typing rename * undo compose change * linting fix type actually fix typing * stricter typing fix detection-only response no need for defaultdict * update spec file update api linting * update e2e * unnecessary dimension * remove commented code * remove duplicate code * remove unused imports * add batch dim
		
			
				
	
	
		
			254 lines
		
	
	
		
			9.4 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			254 lines
		
	
	
		
			9.4 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| from __future__ import annotations
 | |
| 
 | |
| from abc import ABC, abstractmethod
 | |
| from pathlib import Path
 | |
| from shutil import rmtree
 | |
| from typing import Any, ClassVar
 | |
| 
 | |
| import onnxruntime as ort
 | |
| from huggingface_hub import snapshot_download
 | |
| 
 | |
| import ann.ann
 | |
| from app.models.constants import SUPPORTED_PROVIDERS
 | |
| 
 | |
| from ..config import clean_name, log, settings
 | |
| from ..schemas import ModelFormat, ModelIdentity, ModelSession, ModelTask, ModelType
 | |
| from .ann import AnnSession
 | |
| 
 | |
| 
 | |
| class InferenceModel(ABC):
 | |
|     depends: ClassVar[list[ModelIdentity]]
 | |
|     identity: ClassVar[ModelIdentity]
 | |
| 
 | |
|     def __init__(
 | |
|         self,
 | |
|         model_name: str,
 | |
|         cache_dir: Path | str | None = None,
 | |
|         providers: list[str] | None = None,
 | |
|         provider_options: list[dict[str, Any]] | None = None,
 | |
|         sess_options: ort.SessionOptions | None = None,
 | |
|         preferred_format: ModelFormat | None = None,
 | |
|         **model_kwargs: Any,
 | |
|     ) -> None:
 | |
|         self.loaded = False
 | |
|         self.model_name = clean_name(model_name)
 | |
|         self.cache_dir = Path(cache_dir) if cache_dir is not None else self.cache_dir_default
 | |
|         self.providers = providers if providers is not None else self.providers_default
 | |
|         self.provider_options = provider_options if provider_options is not None else self.provider_options_default
 | |
|         self.sess_options = sess_options if sess_options is not None else self.sess_options_default
 | |
|         self.preferred_format = preferred_format if preferred_format is not None else self.preferred_format_default
 | |
| 
 | |
|     def download(self) -> None:
 | |
|         if not self.cached:
 | |
|             log.info(
 | |
|                 f"Downloading {self.model_type.replace('-', ' ')} model '{self.model_name}'. This may take a while."
 | |
|             )
 | |
|             self._download()
 | |
| 
 | |
|     def load(self) -> None:
 | |
|         if self.loaded:
 | |
|             return
 | |
| 
 | |
|         self.download()
 | |
|         log.info(f"Loading {self.model_type.replace('-', ' ')} model '{self.model_name}' to memory")
 | |
|         self.session = self._load()
 | |
|         self.loaded = True
 | |
| 
 | |
|     def predict(self, *inputs: Any, **model_kwargs: Any) -> Any:
 | |
|         self.load()
 | |
|         if model_kwargs:
 | |
|             self.configure(**model_kwargs)
 | |
|         return self._predict(*inputs, **model_kwargs)
 | |
| 
 | |
|     @abstractmethod
 | |
|     def _predict(self, *inputs: Any, **model_kwargs: Any) -> Any: ...
 | |
| 
 | |
|     def configure(self, **kwargs: Any) -> None:
 | |
|         pass
 | |
| 
 | |
|     def _download(self) -> None:
 | |
|         ignore_patterns = [] if self.preferred_format == ModelFormat.ARMNN else ["*.armnn"]
 | |
|         snapshot_download(
 | |
|             f"immich-app/{clean_name(self.model_name)}",
 | |
|             cache_dir=self.cache_dir,
 | |
|             local_dir=self.cache_dir,
 | |
|             local_dir_use_symlinks=False,
 | |
|             ignore_patterns=ignore_patterns,
 | |
|         )
 | |
| 
 | |
|     def _load(self) -> ModelSession:
 | |
|         return self._make_session(self.model_path)
 | |
| 
 | |
|     def clear_cache(self) -> None:
 | |
|         if not self.cache_dir.exists():
 | |
|             log.warning(
 | |
|                 f"Attempted to clear cache for model '{self.model_name}', but cache directory does not exist",
 | |
|             )
 | |
|             return
 | |
|         if not rmtree.avoids_symlink_attacks:
 | |
|             raise RuntimeError("Attempted to clear cache, but rmtree is not safe on this platform")
 | |
| 
 | |
|         if self.cache_dir.is_dir():
 | |
|             log.info(f"Cleared cache directory for model '{self.model_name}'.")
 | |
|             rmtree(self.cache_dir)
 | |
|         else:
 | |
|             log.warning(
 | |
|                 (
 | |
|                     f"Encountered file instead of directory at cache path "
 | |
|                     f"for '{self.model_name}'. Removing file and replacing with a directory."
 | |
|                 ),
 | |
|             )
 | |
|             self.cache_dir.unlink()
 | |
|         self.cache_dir.mkdir(parents=True, exist_ok=True)
 | |
| 
 | |
|     def _make_session(self, model_path: Path) -> ModelSession:
 | |
|         if not model_path.is_file():
 | |
|             onnx_path = model_path.with_suffix(".onnx")
 | |
|             if not onnx_path.is_file():
 | |
|                 raise ValueError(f"Model path '{model_path}' does not exist")
 | |
| 
 | |
|             log.warning(
 | |
|                 f"Could not find model path '{model_path}'. " f"Falling back to ONNX model path '{onnx_path}' instead.",
 | |
|             )
 | |
|             model_path = onnx_path
 | |
| 
 | |
|         match model_path.suffix:
 | |
|             case ".armnn":
 | |
|                 session = AnnSession(model_path)
 | |
|             case ".onnx":
 | |
|                 session = ort.InferenceSession(
 | |
|                     model_path.as_posix(),
 | |
|                     sess_options=self.sess_options,
 | |
|                     providers=self.providers,
 | |
|                     provider_options=self.provider_options,
 | |
|                 )
 | |
|             case _:
 | |
|                 raise ValueError(f"Unsupported model file type: {model_path.suffix}")
 | |
|         return session
 | |
| 
 | |
|     @property
 | |
|     def model_dir(self) -> Path:
 | |
|         return self.cache_dir / self.model_type.value
 | |
| 
 | |
|     @property
 | |
|     def model_path(self) -> Path:
 | |
|         return self.model_dir / f"model.{self.preferred_format}"
 | |
| 
 | |
|     @property
 | |
|     def model_task(self) -> ModelTask:
 | |
|         return self.identity[1]
 | |
| 
 | |
|     @property
 | |
|     def model_type(self) -> ModelType:
 | |
|         return self.identity[0]
 | |
| 
 | |
|     @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
 | |
|     def cache_dir_default(self) -> Path:
 | |
|         return settings.cache_folder / self.model_task.value / self.model_name
 | |
| 
 | |
|     @property
 | |
|     def cached(self) -> bool:
 | |
|         return self.model_path.is_file()
 | |
| 
 | |
|     @property
 | |
|     def providers(self) -> list[str]:
 | |
|         return self._providers
 | |
| 
 | |
|     @providers.setter
 | |
|     def providers(self, providers: list[str]) -> None:
 | |
|         log.info(
 | |
|             (f"Setting '{self.model_name}' execution providers to {providers}, " "in descending order of preference"),
 | |
|         )
 | |
|         self._providers = providers
 | |
| 
 | |
|     @property
 | |
|     def providers_default(self) -> list[str]:
 | |
|         available_providers = set(ort.get_available_providers())
 | |
|         log.debug(f"Available ORT providers: {available_providers}")
 | |
|         if (openvino := "OpenVINOExecutionProvider") in available_providers:
 | |
|             device_ids: list[str] = ort.capi._pybind_state.get_available_openvino_device_ids()
 | |
|             log.debug(f"Available OpenVINO devices: {device_ids}")
 | |
| 
 | |
|             gpu_devices = [device_id for device_id in device_ids if device_id.startswith("GPU")]
 | |
|             if not gpu_devices:
 | |
|                 log.warning("No GPU device found in OpenVINO. Falling back to CPU.")
 | |
|                 available_providers.remove(openvino)
 | |
|         return [provider for provider in SUPPORTED_PROVIDERS if provider in available_providers]
 | |
| 
 | |
|     @property
 | |
|     def provider_options(self) -> list[dict[str, Any]]:
 | |
|         return self._provider_options
 | |
| 
 | |
|     @provider_options.setter
 | |
|     def provider_options(self, provider_options: list[dict[str, Any]]) -> None:
 | |
|         log.debug(f"Setting execution provider options to {provider_options}")
 | |
|         self._provider_options = provider_options
 | |
| 
 | |
|     @property
 | |
|     def provider_options_default(self) -> list[dict[str, Any]]:
 | |
|         options = []
 | |
|         for provider in self.providers:
 | |
|             match provider:
 | |
|                 case "CPUExecutionProvider" | "CUDAExecutionProvider":
 | |
|                     option = {"arena_extend_strategy": "kSameAsRequested"}
 | |
|                 case "OpenVINOExecutionProvider":
 | |
|                     option = {"device_type": "GPU_FP32", "cache_dir": (self.cache_dir / "openvino").as_posix()}
 | |
|                 case _:
 | |
|                     option = {}
 | |
|             options.append(option)
 | |
|         return options
 | |
| 
 | |
|     @property
 | |
|     def sess_options(self) -> ort.SessionOptions:
 | |
|         return self._sess_options
 | |
| 
 | |
|     @sess_options.setter
 | |
|     def sess_options(self, sess_options: ort.SessionOptions) -> None:
 | |
|         log.debug(f"Setting execution_mode to {sess_options.execution_mode.name}")
 | |
|         log.debug(f"Setting inter_op_num_threads to {sess_options.inter_op_num_threads}")
 | |
|         log.debug(f"Setting intra_op_num_threads to {sess_options.intra_op_num_threads}")
 | |
|         self._sess_options = sess_options
 | |
| 
 | |
|     @property
 | |
|     def sess_options_default(self) -> ort.SessionOptions:
 | |
|         sess_options = ort.SessionOptions()
 | |
|         sess_options.enable_cpu_mem_arena = False
 | |
| 
 | |
|         # avoid thread contention between models
 | |
|         if settings.model_inter_op_threads > 0:
 | |
|             sess_options.inter_op_num_threads = settings.model_inter_op_threads
 | |
|         # these defaults work well for CPU, but bottleneck GPU
 | |
|         elif settings.model_inter_op_threads == 0 and self.providers == ["CPUExecutionProvider"]:
 | |
|             sess_options.inter_op_num_threads = 1
 | |
| 
 | |
|         if settings.model_intra_op_threads > 0:
 | |
|             sess_options.intra_op_num_threads = settings.model_intra_op_threads
 | |
|         elif settings.model_intra_op_threads == 0 and self.providers == ["CPUExecutionProvider"]:
 | |
|             sess_options.intra_op_num_threads = 2
 | |
| 
 | |
|         if sess_options.inter_op_num_threads > 1:
 | |
|             sess_options.execution_mode = ort.ExecutionMode.ORT_PARALLEL
 | |
| 
 | |
|         return sess_options
 | |
| 
 | |
|     @property
 | |
|     def preferred_format(self) -> ModelFormat:
 | |
|         return self._preferred_format
 | |
| 
 | |
|     @preferred_format.setter
 | |
|     def preferred_format(self, preferred_format: ModelFormat) -> None:
 | |
|         log.debug(f"Setting preferred format to {preferred_format}")
 | |
|         self._preferred_format = preferred_format
 | |
| 
 | |
|     @property
 | |
|     def preferred_format_default(self) -> ModelFormat:
 | |
|         return ModelFormat.ARMNN if ann.ann.is_available and settings.ann else ModelFormat.ONNX
 |