Immich with Intel iGPU HW accelerated machine-learning (OpenVINO)

- ***requires adding Immich to the HW-accel section in build.func***
- relocate install dir ownership change command; fix ml script naming
- Fix quoting issue with some var groups
- Switch postgresql setup to standard
- use lowercase i in build.func for HW-accel container config
- Some rearranging and cleanup
- remove python-venv, use uv installer script
- reinstate python3-dev install
- borrowing from immich-native
- Clear execstack for onnxruntime shared object solves the pybind issue
- Replace pg-vector with pgvecto.rs solves the DB migration problem
This commit is contained in:
vhsdream 2025-04-08 10:54:29 -04:00
parent 5324c2e039
commit bd281eaf33
2 changed files with 113 additions and 56 deletions

View File

@ -13,23 +13,23 @@ setting_up_container
network_check network_check
update_os update_os
msg_info "Configuring apt and installing base dependencies" msg_info "Configuring apt and installing dependencies"
echo "deb http://deb.debian.org/debian testing main contrib" >/etc/apt/sources.list.d/immich.list echo "deb http://deb.debian.org/debian testing main contrib" >/etc/apt/sources.list.d/immich.list
{ cat <<EOF >/etc/apt/preferences.d/immich
echo "Package: *" Package: *
echo "Pin: release a=testing" Pin: release a=testing
echo "Pin-Priority: -10" Pin-Priority: -10
EOF
} >/etc/apt/preferences.d/immich
$STD apt-get update $STD apt-get update
$STD apt-get install --no-install-recommends -y \ $STD apt-get install --no-install-recommends -y \
git \ git \
redis \ redis \
python3-venv \
python3-dev \
gnupg \ gnupg \
autoconf \ autoconf \
build-essential \ build-essential \
python3-venv \
python3-dev \
cmake \ cmake \
jq \ jq \
libbrotli-dev \ libbrotli-dev \
@ -65,9 +65,7 @@ $STD apt-get install --no-install-recommends -y \
mesa-va-drivers \ mesa-va-drivers \
mesa-vulkan-drivers \ mesa-vulkan-drivers \
tini \ tini \
zlib1g \ zlib1g
ocl-icd-libopencl1 \
intel-media-va-driver
$STD apt-get install -y \ $STD apt-get install -y \
libgdk-pixbuf-2.0-dev librsvg2-dev libtool libgdk-pixbuf-2.0-dev librsvg2-dev libtool
curl -fsSL https://repo.jellyfin.org/jellyfin_team.gpg.key | gpg --dearmor -o /etc/apt/keyrings/jellyfin.gpg curl -fsSL https://repo.jellyfin.org/jellyfin_team.gpg.key | gpg --dearmor -o /etc/apt/keyrings/jellyfin.gpg
@ -85,19 +83,40 @@ $STD apt-get update
$STD apt-get install -y jellyfin-ffmpeg7 $STD apt-get install -y jellyfin-ffmpeg7
ln -s /usr/lib/jellyfin-ffmpeg/ffmpeg /usr/bin/ffmpeg ln -s /usr/lib/jellyfin-ffmpeg/ffmpeg /usr/bin/ffmpeg
ln -s /usr/lib/jellyfin-ffmpeg/ffprobe /usr/bin/ffprobe ln -s /usr/lib/jellyfin-ffmpeg/ffprobe /usr/bin/ffprobe
tmp_dir=$(mktemp -d) msg_ok "Dependencies Installed"
cd "$tmp_dir" || exit
curl -fsSL https://github.com/intel/intel-graphics-compiler/releases/download/igc-1.0.17193.4/intel-igc-core_1.0.17193.4_amd64.deb -O read -r -p "Install OpenVINO dependencies for Intel HW-accelerated machine-learning? " prompt
curl -fsSL https://github.com/intel/intel-graphics-compiler/releases/download/igc-1.0.17193.4/intel-igc-opencl_1.0.17193.4_amd64.deb -O if [[ ${prompt,,} =~ ^(y|yes)$ ]]; then
curl -fsSL https://github.com/intel/compute-runtime/releases/download/24.26.30049.6/intel-opencl-icd_24.26.30049.6_amd64.deb -O msg_info "Installing OpenVINO dependencies"
curl -fsSL https://github.com/intel/compute-runtime/releases/download/24.26.30049.6/libigdgmm12_22.3.20_amd64.deb -O export intel_hw=1
$STD dpkg -i ./*.deb $STD apt-get -y install --no-install-recommends ocl-icd-libopencl1
msg_ok "Base Dependencies Installed" tmp_dir=$(mktemp -d)
$STD pushd "$tmp_dir"
curl -fsSL https://github.com/intel/intel-graphics-compiler/releases/download/igc-1.0.17384.11/intel-igc-core_1.0.17384.11_amd64.deb -O
curl -fsSL https://github.com/intel/intel-graphics-compiler/releases/download/igc-1.0.17384.11/intel-igc-opencl_1.0.17384.11_amd64.deb -O
curl -fsSL https://github.com/intel/compute-runtime/releases/download/24.31.30508.7/intel-opencl-icd_24.31.30508.7_amd64.deb -O
curl -fsSL https://github.com/intel/compute-runtime/releases/download/24.31.30508.7/libigdgmm12_22.4.1_amd64.deb -O
$STD dpkg -i ./*.deb
$STD popd
rm -rf "$tmp_dir"
if [[ "$CTTYPE" == "0" ]]; then
chgrp video /dev/dri
chmod 755 /dev/dri
chmod 660 /dev/dri/*
$STD adduser "$(id -u -n)" video
$STD adduser "$(id -u -n)" render
fi
msg_ok "Installed OpenVINO dependencies"
fi
msg_info "Setting up Postgresql Database" msg_info "Setting up Postgresql Database"
$STD apt-get install -y postgresql-common curl -fsSL https://www.postgresql.org/media/keys/ACCC4CF8.asc | gpg --dearmor -o /etc/apt/trusted.gpg.d/postgresql.gpg
echo "YES" | /usr/share/postgresql-common/pgdg/apt.postgresql.org.sh &>/dev/null echo "deb https://apt.postgresql.org/pub/repos/apt bookworm-pgdg main" >/etc/apt/sources.list.d/pgdg.list
$STD apt-get install -y postgresql-17 postgresql-17-pgvector $STD apt-get update
$STD apt-get install -y postgresql-17
curl -fsSLO https://github.com/tensorchord/pgvecto.rs/releases/download/v0.4.0/vectors-pg17_0.4.0_amd64.deb
$STD dpkg -i vectors-pg17_0.4.0_amd64.deb
rm vectors-pg17_0.4.0.amd64.deb
DB_NAME="immich" DB_NAME="immich"
DB_USER="immich" DB_USER="immich"
DB_PASS=$(openssl rand -base64 18 | tr -dc 'a-zA-Z0-9' | head -c18) DB_PASS=$(openssl rand -base64 18 | tr -dc 'a-zA-Z0-9' | head -c18)
@ -105,6 +124,11 @@ $STD sudo -u postgres psql -c "CREATE USER $DB_USER WITH ENCRYPTED PASSWORD '$DB
$STD sudo -u postgres psql -c "CREATE DATABASE $DB_NAME WITH OWNER $DB_USER ENCODING 'UTF8' TEMPLATE template0;" $STD sudo -u postgres psql -c "CREATE DATABASE $DB_NAME WITH OWNER $DB_USER ENCODING 'UTF8' TEMPLATE template0;"
$STD sudo -u postgres psql -c "GRANT ALL PRIVILEGES ON DATABASE $DB_NAME to $DB_USER;" $STD sudo -u postgres psql -c "GRANT ALL PRIVILEGES ON DATABASE $DB_NAME to $DB_USER;"
$STD sudo -u postgres psql -c "ALTER USER $DB_USER WITH SUPERUSER;" $STD sudo -u postgres psql -c "ALTER USER $DB_USER WITH SUPERUSER;"
$STD sudo -u postgres psql -c "ALTER SYSTEM SET shared_preload_libraries = 'vectors.so';"
$STD sudo -u postgres psql -c "ALTER SYSTEM SET search_path TO "$user", public, vectors;"
systemctl restart postgresql.service
$STD sudo -u postgres psql -c "DROP EXTENSION IF EXISTS vectors;"
$STD sudo -u postgres psql -c "CREATE EXTENSION vectors;"
{ {
echo "${APPLICATION} DB Credentials" echo "${APPLICATION} DB Credentials"
echo "Database User: $DB_USER" echo "Database User: $DB_USER"
@ -132,13 +156,18 @@ $STD apt-get install -t testing --no-install-recommends -y \
libdav1d-dev \ libdav1d-dev \
libhwy-dev \ libhwy-dev \
libwebp-dev libwebp-dev
if [[ "$intel_hw" = 1 ]]; then
apt-get install -t testing -y patchelf
fi
msg_ok "Packages from Testing Repo Installed" msg_ok "Packages from Testing Repo Installed"
# Fix default DB collation issue # Fix default DB collation issue after libc update
$STD sudo -u postgres psql -c "ALTER DATABASE postgres REFRESH COLLATION VERSION;" $STD sudo -u postgres psql -c "ALTER DATABASE postgres REFRESH COLLATION VERSION;"
$STD sudo -u postgres psql -c "ALTER DATABASE $DB_NAME REFRESH COLLATION VERSION;" $STD sudo -u postgres psql -c "ALTER DATABASE $DB_NAME REFRESH COLLATION VERSION;"
msg_info "Compiling Custom Photo-processing Library (extreme patience)" msg_info "Compiling Custom Photo-processing Library (extreme patience)"
LD_LIBRARY_PATH=/usr/local/lib
export LD_RUN_PATH=/usr/local/lib
STAGING_DIR=/opt/staging STAGING_DIR=/opt/staging
BASE_REPO="https://github.com/immich-app/base-images" BASE_REPO="https://github.com/immich-app/base-images"
BASE_DIR=${STAGING_DIR}/base-images BASE_DIR=${STAGING_DIR}/base-images
@ -179,7 +208,7 @@ $STD cmake \
.. ..
$STD cmake --build . -- -j"$(nproc)" $STD cmake --build . -- -j"$(nproc)"
$STD cmake --install . $STD cmake --install .
$STD ldconfig /usr/local/lib ldconfig /usr/local/lib
$STD make clean $STD make clean
cd "$STAGING_DIR" || exit cd "$STAGING_DIR" || exit
rm -rf "$SOURCE"/{build,third_party} rm -rf "$SOURCE"/{build,third_party}
@ -201,7 +230,7 @@ $STD cmake --preset=release-noplugins \
-DWITH_X265=OFF \ -DWITH_X265=OFF \
-DWITH_EXAMPLES=OFF \ -DWITH_EXAMPLES=OFF \
.. ..
$STD make install $STD make install -j "$(nproc)"
ldconfig /usr/local/lib ldconfig /usr/local/lib
$STD make clean $STD make clean
cd "$STAGING_DIR" || exit cd "$STAGING_DIR" || exit
@ -240,7 +269,7 @@ $STD git reset --hard "$LIBVIPS_REVISION"
$STD meson setup build --buildtype=release --libdir=lib -Dintrospection=disabled -Dtiff=disabled $STD meson setup build --buildtype=release --libdir=lib -Dintrospection=disabled -Dtiff=disabled
cd build || exit cd build || exit
$STD ninja install $STD ninja install
$STD ldconfig /usr/local/lib ldconfig /usr/local/lib
cd "$STAGING_DIR" || exit cd "$STAGING_DIR" || exit
rm -rf "$SOURCE"/build rm -rf "$SOURCE"/build
msg_ok "Custom Photo-processing Library Compiled" msg_ok "Custom Photo-processing Library Compiled"
@ -258,9 +287,10 @@ ML_DIR="${APP_DIR}/machine-learning"
GEO_DIR="${INSTALL_DIR}/geodata" GEO_DIR="${INSTALL_DIR}/geodata"
mkdir -p "$INSTALL_DIR" mkdir -p "$INSTALL_DIR"
mv "$APPLICATION-$RELEASE"/ "$SRC_DIR" mv "$APPLICATION-$RELEASE"/ "$SRC_DIR"
mkdir -p $APP_DIR $UPLOAD_DIR $GEO_DIR $ML_DIR $INSTALL_DIR/.cache mkdir -p {"${APP_DIR}","${UPLOAD_DIR}","${GEO_DIR}","${ML_DIR}","${INSTALL_DIR}"/cache}
cd "$SRC_DIR"/server || exit cd "$SRC_DIR"/server || exit
$STD npm install -g node-gyp node-pre-gyp
$STD npm ci $STD npm ci
$STD npm run build $STD npm run build
$STD npm prune --omit=dev --omit=optional $STD npm prune --omit=dev --omit=optional
@ -275,23 +305,37 @@ cp -a server/{node_modules,dist,bin,resources,package.json,package-lock.json,sta
cp -a web/build "$APP_DIR"/www cp -a web/build "$APP_DIR"/www
cp LICENSE "$APP_DIR" cp LICENSE "$APP_DIR"
cp "$BASE_DIR"/server/bin/build-lock.json "$APP_DIR" cp "$BASE_DIR"/server/bin/build-lock.json "$APP_DIR"
msg_ok "Installed Immich Web Components"
cd "$SRC_DIR"/machine-learning || exit cd "$SRC_DIR"/machine-learning || exit
$STD python3 -m venv "$ML_DIR"/ml-venv $STD python3 -m venv "$ML_DIR/ml-venv"
( if [[ "$intel_hw" = 1 ]]; then
. "$ML_DIR"/ml-venv/bin/activate msg_info "Installing Immich HW-accelerated machine-learning"
(
source "$ML_DIR"/ml-venv/bin/activate
# $STD uv sync --extra openvino --active
$STD pip3 install uv $STD pip3 install uv
$STD uv sync --extra cpu --active uv -q sync --extra openvino --no-cache --active
) )
patchelf --clear-execstack "$ML_DIR"/ml-venv/lib/python3.11/site-packages/onnxruntime/capi/onnxruntime_pybind11_state.cpython-311-x86_64-linux-gnu.so
msg_ok "Installed HW-accelerated machine-learning"
else
msg_info "Installing Immich machine-learning"
(
source "$ML_DIR"/ml-venv/bin/activate
$STD pip3 install uv
uv -q sync --extra cpu --no-cache --active
)
msg_ok "Installed Immich machine-learning"
fi
cd "$SRC_DIR" || exit cd "$SRC_DIR" || exit
cp -a machine-learning/{ann,immich_ml} "$ML_DIR" cp -a machine-learning/{ann,immich_ml} "$ML_DIR"
ln -sf "$APP_DIR"/resources "$INSTALL_DIR" ln -sf "$APP_DIR"/resources "$INSTALL_DIR"
cd "$APP_DIR" || exit cd "$APP_DIR" || exit
grep -RlI /usr/src . --exclude="*.py*" --exclude="*.json" | grep -Rl /usr/src | xargs -n1 sed -i "s|\/usr/src|$INSTALL_DIR|g"
xargs -n1 sed -i "s|\/usr/src|$INSTALL_DIR|g"
# sed -i "s|\"/cache\"|\"$INSTALL_DIR/cache\"|g" $ML_DIR/immich_ml/config.py
grep -RlE "'/build'" | xargs -n1 sed -i "s|'/build'|'$APP_DIR'|g" grep -RlE "'/build'" | xargs -n1 sed -i "s|'/build'|'$APP_DIR'|g"
sed -i "s@\"/cache\"@\"$INSTALL_DIR/cache\"@g" "$ML_DIR"/immich_ml/config.py
ln -s "$UPLOAD_DIR" "$APP_DIR"/upload ln -s "$UPLOAD_DIR" "$APP_DIR"/upload
ln -s "$UPLOAD_DIR" "$ML_DIR"/upload ln -s "$UPLOAD_DIR" "$ML_DIR"/upload
@ -312,6 +356,7 @@ URL_LIST=(
echo "${URL_LIST[@]}" | xargs -n1 -P 8 wget -q echo "${URL_LIST[@]}" | xargs -n1 -P 8 wget -q
unzip -q cities500.zip unzip -q cities500.zip
date --iso-8601=seconds | tr -d "\n" >geodata-date.txt date --iso-8601=seconds | tr -d "\n" >geodata-date.txt
rm cities500.zip
cd "$INSTALL_DIR" || exit cd "$INSTALL_DIR" || exit
ln -s "$GEO_DIR" "$APP_DIR" ln -s "$GEO_DIR" "$APP_DIR"
msg_ok "Installed GeoNames data" msg_ok "Installed GeoNames data"
@ -321,21 +366,25 @@ touch /var/log/immich/{web.log,ml.log}
echo "$RELEASE" >/opt/"${APPLICATION}"_version.txt echo "$RELEASE" >/opt/"${APPLICATION}"_version.txt
msg_ok "Installed ${APPLICATION}" msg_ok "Installed ${APPLICATION}"
msg_info "Creating env file, scripts & services" msg_info "Creating user, env file, scripts & services"
$STD useradd -U -s /usr/sbin/nologin -r -M -d "$INSTALL_DIR" immich
if [[ "$intel_hw" = 1 ]]; then
usermod -aG video,render immich
fi
cat <<EOF >"${INSTALL_DIR}"/.env cat <<EOF >"${INSTALL_DIR}"/.env
TZ=$(cat /etc/timezone) TZ=$(cat /etc/timezone)
IMMICH_VERSION=release IMMICH_VERSION=release
IMMICH_ENV=production NODE_ENV=production
DB_HOSTNAME=localhost DB_HOSTNAME=127.0.0.1
DB_USERNAME=${DB_USER} DB_USERNAME=${DB_USER}
DB_PASSWORD=${DB_PASS} DB_PASSWORD=${DB_PASS}
DB_DATABASE_NAME=${DB_NAME} DB_DATABASE_NAME=${DB_NAME}
DB_VECTOR_EXTENSION=pgvector DB_VECTOR_EXTENSION=pgvector
REDIS_HOSTNAME=localhost REDIS_HOSTNAME=127.0.0.1
IMMICH_MACHINE_LEARNING_URL=http://127.0.0.1:3003
MACHINE_LEARNING_CACHE_FOLDER=${INSTALL_DIR}/.cache MACHINE_LEARNING_CACHE_FOLDER=${INSTALL_DIR}/cache
EOF EOF
cat <<EOF >"${ML_DIR}"/ml_start.sh cat <<EOF >"${ML_DIR}"/ml_start.sh
#!/usr/bin/env bash #!/usr/bin/env bash
@ -343,6 +392,10 @@ cat <<EOF >"${ML_DIR}"/ml_start.sh
cd ${ML_DIR} cd ${ML_DIR}
. ml-venv/bin/activate . ml-venv/bin/activate
set -a
. ${INSTALL_DIR}/.env
set +a
python -m immich_ml python -m immich_ml
EOF EOF
chmod +x "$ML_DIR"/ml_start.sh chmod +x "$ML_DIR"/ml_start.sh
@ -356,10 +409,12 @@ Requires=immich-ml.service
[Service] [Service]
Type=simple Type=simple
User=root User=immich
Group=immich
UMask=0077
WorkingDirectory=${APP_DIR} WorkingDirectory=${APP_DIR}
EnvironmentFile=${INSTALL_DIR}/.env EnvironmentFile=${INSTALL_DIR}/.env
ExecStart=/usr/bin/node dist/main "\$@" ExecStart=/usr/bin/node ${APP_DIR}/dist/main
Restart=on-failure Restart=on-failure
SyslogIdentifier=immich-web SyslogIdentifier=immich-web
StandardOutput=append:/var/log/immich/web.log StandardOutput=append:/var/log/immich/web.log
@ -375,7 +430,9 @@ After=network.target
[Service] [Service]
Type=simple Type=simple
User=root UMask=0077
User=immich
Group=immich
WorkingDirectory=${APP_DIR} WorkingDirectory=${APP_DIR}
EnvironmentFile=${INSTALL_DIR}/.env EnvironmentFile=${INSTALL_DIR}/.env
ExecStart=${ML_DIR}/ml_start.sh ExecStart=${ML_DIR}/ml_start.sh
@ -387,8 +444,9 @@ StandardError=append:/var/log/immich/ml.log
[Install] [Install]
WantedBy=multi-user.target WantedBy=multi-user.target
EOF EOF
chown -R immich:immich "$INSTALL_DIR" /var/log/immich
systemctl enable -q --now "$APPLICATION"-ml.service "$APPLICATION"-web.service systemctl enable -q --now "$APPLICATION"-ml.service "$APPLICATION"-web.service
msg_ok "Created env file, scripts and services" msg_ok "Created user, env file, scripts and services"
sed -i "$ a VERSION_ID=12" /etc/os-release # otherwise the motd_ssh function will fail sed -i "$ a VERSION_ID=12" /etc/os-release # otherwise the motd_ssh function will fail
motd_ssh motd_ssh
@ -396,7 +454,6 @@ customize
msg_info "Cleaning up" msg_info "Cleaning up"
rm -f "$tmp_file" rm -f "$tmp_file"
rm -rf "$tmp_dir"
$STD apt-get -y autoremove $STD apt-get -y autoremove
$STD apt-get -y autoclean $STD apt-get -y autoclean
msg_ok "Cleaned" msg_ok "Cleaned"

View File

@ -1345,8 +1345,8 @@ EOF
fi fi
if [ "$CT_TYPE" == "0" ]; then if [ "$CT_TYPE" == "0" ]; then
if [[ "$APP" == "Channels" || "$APP" == "Emby" || "$APP" == "ErsatzTV" || "$APP" == "Frigate" || "$APP" == "Jellyfin" || "$APP" == "Plex" || "$APP" == "Scrypted" || "$APP" == "Tdarr" || "$APP" == "Unmanic" || "$APP" == "Ollama" || "$APP" == "FileFlows" ]]; then if [[ "$APP" == "immich" || "$APP" == "Channels" || "$APP" == "Emby" || "$APP" == "ErsatzTV" || "$APP" == "Frigate" || "$APP" == "Jellyfin" || "$APP" == "Plex" || "$APP" == "Scrypted" || "$APP" == "Tdarr" || "$APP" == "Unmanic" || "$APP" == "Ollama" || "$APP" == "FileFlows" ]]; then
cat <<EOF >>"$LXC_CONFIG" cat <<EOF >>$LXC_CONFIG
# VAAPI hardware transcoding # VAAPI hardware transcoding
lxc.cgroup2.devices.allow: c 226:0 rwm lxc.cgroup2.devices.allow: c 226:0 rwm
lxc.cgroup2.devices.allow: c 226:128 rwm lxc.cgroup2.devices.allow: c 226:128 rwm
@ -1357,10 +1357,10 @@ lxc.mount.entry: /dev/dri/renderD128 dev/dri/renderD128 none bind,optional,creat
EOF EOF
fi fi
else else
if [[ "$APP" == "Channels" || "$APP" == "Emby" || "$APP" == "ErsatzTV" || "$APP" == "Frigate" || "$APP" == "Jellyfin" || "$APP" == "Plex" || "$APP" == "Scrypted" || "$APP" == "Tdarr" || "$APP" == "Unmanic" || "$APP" == "Ollama" || "$APP" == "FileFlows" ]]; then if [[ "$APP" == "immich" || "$APP" == "Channels" || "$APP" == "Emby" || "$APP" == "ErsatzTV" || "$APP" == "Frigate" || "$APP" == "Jellyfin" || "$APP" == "Plex" || "$APP" == "Scrypted" || "$APP" == "Tdarr" || "$APP" == "Unmanic" || "$APP" == "Ollama" || "$APP" == "FileFlows" ]]; then
if [[ -e "/dev/dri/renderD128" ]]; then if [[ -e "/dev/dri/renderD128" ]]; then
if [[ -e "/dev/dri/card0" ]]; then if [[ -e "/dev/dri/card0" ]]; then
cat <<EOF >>"$LXC_CONFIG" cat <<EOF >>$LXC_CONFIG
# VAAPI hardware transcoding # VAAPI hardware transcoding
dev0: /dev/dri/card0,gid=44 dev0: /dev/dri/card0,gid=44
dev1: /dev/dri/renderD128,gid=104 dev1: /dev/dri/renderD128,gid=104