2024-01-26 01:26:27 +01:00
|
|
|
import concurrent.futures
|
2023-08-30 10:22:01 +02:00
|
|
|
import logging
|
2023-08-25 06:28:51 +02:00
|
|
|
import os
|
2023-12-14 20:51:24 +01:00
|
|
|
import sys
|
2023-06-25 05:18:09 +02:00
|
|
|
from pathlib import Path
|
2023-12-14 20:51:24 +01:00
|
|
|
from socket import socket
|
2023-06-25 05:18:09 +02:00
|
|
|
|
2023-12-14 20:51:24 +01:00
|
|
|
from gunicorn.arbiter import Arbiter
|
2024-10-14 00:00:21 +02:00
|
|
|
from pydantic import BaseModel
|
|
|
|
from pydantic_settings import BaseSettings, SettingsConfigDict
|
2023-08-30 10:22:01 +02:00
|
|
|
from rich.console import Console
|
|
|
|
from rich.logging import RichHandler
|
2023-12-14 20:51:24 +01:00
|
|
|
from uvicorn import Server
|
|
|
|
from uvicorn.workers import UvicornWorker
|
2023-06-18 05:49:19 +02:00
|
|
|
|
2023-06-25 05:18:09 +02:00
|
|
|
|
2024-03-04 01:48:56 +01:00
|
|
|
class PreloadModelData(BaseModel):
|
2024-10-14 00:00:21 +02:00
|
|
|
clip: str | None = None
|
|
|
|
facial_recognition: str | None = None
|
2024-03-04 01:48:56 +01:00
|
|
|
|
|
|
|
|
2024-10-23 14:50:28 +02:00
|
|
|
class MaxBatchSize(BaseModel):
|
|
|
|
facial_recognition: int | None = None
|
|
|
|
|
|
|
|
|
2023-06-18 05:49:19 +02:00
|
|
|
class Settings(BaseSettings):
|
2024-10-14 00:00:21 +02:00
|
|
|
model_config = SettingsConfigDict(
|
|
|
|
env_prefix="MACHINE_LEARNING_",
|
|
|
|
case_sensitive=False,
|
|
|
|
env_nested_delimiter="__",
|
|
|
|
protected_namespaces=("settings_",),
|
|
|
|
)
|
|
|
|
|
2024-06-07 05:09:47 +02:00
|
|
|
cache_folder: Path = Path("/cache")
|
2023-11-17 03:42:44 +01:00
|
|
|
model_ttl: int = 300
|
|
|
|
model_ttl_poll_s: int = 10
|
2023-06-18 05:49:19 +02:00
|
|
|
host: str = "0.0.0.0"
|
|
|
|
port: int = 3003
|
|
|
|
workers: int = 1
|
2023-06-28 01:21:33 +02:00
|
|
|
test_full: bool = False
|
2023-08-25 06:28:51 +02:00
|
|
|
request_threads: int = os.cpu_count() or 4
|
2024-01-22 00:22:39 +01:00
|
|
|
model_inter_op_threads: int = 0
|
|
|
|
model_intra_op_threads: int = 0
|
2024-01-11 18:26:46 +01:00
|
|
|
ann: bool = True
|
2024-07-20 21:59:27 +02:00
|
|
|
ann_fp16_turbo: bool = False
|
|
|
|
ann_tuning_level: int = 2
|
2024-03-04 01:48:56 +01:00
|
|
|
preload: PreloadModelData | None = None
|
2024-10-23 14:50:28 +02:00
|
|
|
max_batch_size: MaxBatchSize | None = None
|
2023-06-18 05:49:19 +02:00
|
|
|
|
2024-10-07 23:37:45 +02:00
|
|
|
@property
|
|
|
|
def device_id(self) -> str:
|
|
|
|
return os.environ.get("MACHINE_LEARNING_DEVICE_ID", "0")
|
|
|
|
|
2023-06-18 05:49:19 +02:00
|
|
|
|
2023-08-30 10:22:01 +02:00
|
|
|
class LogSettings(BaseSettings):
|
2024-10-14 00:00:21 +02:00
|
|
|
model_config = SettingsConfigDict(case_sensitive=False)
|
|
|
|
|
2024-05-17 17:44:22 +02:00
|
|
|
immich_log_level: str = "info"
|
2023-08-30 10:22:01 +02:00
|
|
|
no_color: bool = False
|
|
|
|
|
|
|
|
|
2023-08-25 06:28:51 +02:00
|
|
|
_clean_name = str.maketrans(":\\/", "___", ".")
|
|
|
|
|
|
|
|
|
2023-11-12 02:04:49 +01:00
|
|
|
def clean_name(model_name: str) -> str:
|
|
|
|
return model_name.split("/")[-1].translate(_clean_name)
|
|
|
|
|
|
|
|
|
2023-08-30 10:22:01 +02:00
|
|
|
LOG_LEVELS: dict[str, int] = {
|
|
|
|
"critical": logging.ERROR,
|
|
|
|
"error": logging.ERROR,
|
|
|
|
"warning": logging.WARNING,
|
|
|
|
"warn": logging.WARNING,
|
|
|
|
"info": logging.INFO,
|
|
|
|
"log": logging.INFO,
|
|
|
|
"debug": logging.DEBUG,
|
|
|
|
"verbose": logging.DEBUG,
|
|
|
|
}
|
|
|
|
|
2023-06-18 05:49:19 +02:00
|
|
|
settings = Settings()
|
2023-08-30 10:22:01 +02:00
|
|
|
log_settings = LogSettings()
|
|
|
|
|
2024-05-17 17:44:22 +02:00
|
|
|
LOG_LEVEL = LOG_LEVELS.get(log_settings.immich_log_level.lower(), logging.INFO)
|
2024-01-28 01:50:50 +01:00
|
|
|
|
2023-09-01 01:30:53 +02:00
|
|
|
|
|
|
|
class CustomRichHandler(RichHandler):
|
|
|
|
def __init__(self) -> None:
|
|
|
|
console = Console(color_system="standard", no_color=log_settings.no_color)
|
2024-01-26 01:26:27 +01:00
|
|
|
self.excluded = ["uvicorn", "starlette", "fastapi"]
|
|
|
|
super().__init__(
|
|
|
|
show_path=False,
|
|
|
|
omit_repeated_times=False,
|
|
|
|
console=console,
|
|
|
|
rich_tracebacks=True,
|
|
|
|
tracebacks_suppress=[*self.excluded, concurrent.futures],
|
2024-01-28 01:50:50 +01:00
|
|
|
tracebacks_show_locals=LOG_LEVEL == logging.DEBUG,
|
2024-01-26 01:26:27 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
# hack to exclude certain modules from rich tracebacks
|
|
|
|
def emit(self, record: logging.LogRecord) -> None:
|
|
|
|
if record.exc_info is not None:
|
|
|
|
tb = record.exc_info[2]
|
|
|
|
while tb is not None:
|
|
|
|
if any(excluded in tb.tb_frame.f_code.co_filename for excluded in self.excluded):
|
|
|
|
tb.tb_frame.f_locals["_rich_traceback_omit"] = True
|
|
|
|
tb = tb.tb_next
|
|
|
|
|
|
|
|
return super().emit(record)
|
|
|
|
|
|
|
|
|
|
|
|
log = logging.getLogger("ml.log")
|
2024-01-28 01:50:50 +01:00
|
|
|
log.setLevel(LOG_LEVEL)
|
2023-12-14 20:51:24 +01:00
|
|
|
|
|
|
|
|
|
|
|
# patches this issue https://github.com/encode/uvicorn/discussions/1803
|
|
|
|
class CustomUvicornServer(Server):
|
|
|
|
async def shutdown(self, sockets: list[socket] | None = None) -> None:
|
|
|
|
for sock in sockets or []:
|
|
|
|
sock.close()
|
|
|
|
await super().shutdown()
|
|
|
|
|
|
|
|
|
|
|
|
class CustomUvicornWorker(UvicornWorker):
|
|
|
|
async def _serve(self) -> None:
|
|
|
|
self.config.app = self.wsgi
|
|
|
|
server = CustomUvicornServer(config=self.config)
|
|
|
|
self._install_sigquit_handler()
|
|
|
|
await server.serve(sockets=self.sockets)
|
|
|
|
if not server.started:
|
|
|
|
sys.exit(Arbiter.WORKER_BOOT_ERROR)
|