2023-06-25 05:18:09 +02:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2023-06-27 19:21:50 +02:00
|
|
|
from abc import ABC, abstractmethod
|
2023-06-25 05:18:09 +02:00
|
|
|
from pathlib import Path
|
2023-06-27 23:01:24 +02:00
|
|
|
from shutil import rmtree
|
2023-06-25 05:18:09 +02:00
|
|
|
from typing import Any
|
|
|
|
|
2023-06-28 01:21:33 +02:00
|
|
|
from onnxruntime.capi.onnxruntime_pybind11_state import InvalidProtobuf # type: ignore
|
2023-06-27 23:01:24 +02:00
|
|
|
|
2023-06-25 05:18:09 +02:00
|
|
|
from ..config import get_cache_dir
|
|
|
|
from ..schemas import ModelType
|
|
|
|
|
|
|
|
|
|
|
|
class InferenceModel(ABC):
|
|
|
|
_model_type: ModelType
|
|
|
|
|
2023-06-28 01:21:33 +02:00
|
|
|
def __init__(self, model_name: str, cache_dir: Path | str | None = None, **model_kwargs: Any) -> None:
|
2023-06-25 05:18:09 +02:00
|
|
|
self.model_name = model_name
|
2023-06-28 01:21:33 +02:00
|
|
|
self._cache_dir = Path(cache_dir) if cache_dir is not None else get_cache_dir(model_name, self.model_type)
|
2023-06-25 05:18:09 +02:00
|
|
|
|
2023-06-27 23:01:24 +02:00
|
|
|
try:
|
|
|
|
self.load(**model_kwargs)
|
|
|
|
except (OSError, InvalidProtobuf):
|
|
|
|
self.clear_cache()
|
|
|
|
self.load(**model_kwargs)
|
|
|
|
|
|
|
|
@abstractmethod
|
|
|
|
def load(self, **model_kwargs: Any) -> None:
|
|
|
|
...
|
|
|
|
|
2023-06-25 05:18:09 +02:00
|
|
|
@abstractmethod
|
|
|
|
def predict(self, inputs: Any) -> Any:
|
|
|
|
...
|
|
|
|
|
|
|
|
@property
|
|
|
|
def model_type(self) -> ModelType:
|
|
|
|
return self._model_type
|
|
|
|
|
|
|
|
@property
|
|
|
|
def cache_dir(self) -> Path:
|
|
|
|
return self._cache_dir
|
|
|
|
|
|
|
|
@cache_dir.setter
|
2023-06-27 23:01:24 +02:00
|
|
|
def cache_dir(self, cache_dir: Path) -> None:
|
2023-06-25 05:18:09 +02:00
|
|
|
self._cache_dir = cache_dir
|
|
|
|
|
|
|
|
@classmethod
|
2023-06-28 01:21:33 +02:00
|
|
|
def from_model_type(cls, model_type: ModelType, model_name: str, **model_kwargs: Any) -> InferenceModel:
|
|
|
|
subclasses = {subclass._model_type: subclass for subclass in cls.__subclasses__()}
|
2023-06-25 05:18:09 +02:00
|
|
|
if model_type not in subclasses:
|
|
|
|
raise ValueError(f"Unsupported model type: {model_type}")
|
|
|
|
|
|
|
|
return subclasses[model_type](model_name, **model_kwargs)
|
2023-06-27 23:01:24 +02:00
|
|
|
|
|
|
|
def clear_cache(self) -> None:
|
|
|
|
if not self.cache_dir.exists():
|
|
|
|
return
|
|
|
|
elif not rmtree.avoids_symlink_attacks:
|
2023-06-28 01:21:33 +02:00
|
|
|
raise RuntimeError("Attempted to clear cache, but rmtree is not safe on this platform.")
|
2023-06-27 23:01:24 +02:00
|
|
|
|
|
|
|
rmtree(self.cache_dir)
|