mirakc(Docker)+Linux版EDCB+KonomiTVの環境がうまくいったので、どうせならとDockerを使用せずインストールするバージョンも、と作ってみました。ビルド等のためにインストールに時間はかかりますが、原理的にはこのほうが軽いはず。チャンネル切り替えも速い印象です。
その後、少し使った感じでは実用出来そうな様子です。一応、今からインストールするならより新しいスクリプトを使用したほうがよいです。
チューナードライバインストール・LXDコンテナ作成(ホストで実行)
OS:Ubuntu 26.04
TVチューナー:DTV02A-1T1S-U
まずはホスト側にチューナードライバを適用します。過去にダウンロードしたドライバがあれば、/opt/lxd-data/konomitv-backupに入れておけば、それを利用するかの確認も表示されます(サイトを確認して保持しているのが最新かチェックしてもよいかと。スクリプトでは自動で最新版をダウンロードします。保持しているファイルを利用する場合はパーミッションにも注意)。
Secure Boot環境の場合、ドライバインストール時に確認画面が表示されるので、こちらの記事を参考に設定。
コンテナにはDockerをインストールしません。
sudo mkdir -p /opt/lxd-data/konomitv-backup
cd /opt/lxd-data/konomitv-backup
sudo nano tuner-lxd.sh
sudo bash tuner-lxd.sh
#!/bin/bash
# =============================================================
# KonomiTV LXD セットアップスクリプト (ホスト側)
# TunerOK スナップショットまで
# =============================================================
set -euo pipefail
MOUNT_PATH="/opt/lxd-data"
DRIVER_DIR="/opt/lxd-data/konomitv-backup"
# ============================================================
# ユーティリティ
# ============================================================
ask_yn() {
local prompt="$1"
local default="${2:-Y}"
local hint
if [ "$default" = "Y" ]; then
hint="Y/n"
else
hint="y/N"
fi
while true; do
read -rp "${prompt} [${hint}]: " ans
ans="${ans:-$default}"
case "${ans,,}" in
y|yes) return 0 ;;
n|no) return 1 ;;
*) echo " y または n で答えてください。" ;;
esac
done
}
# ============================================================
# 1. チューナードライバのインストール
# ============================================================
echo "=========================================="
echo " KonomiTV LXD セットアップ"
echo "=========================================="
echo ""
if ask_yn "チューナードライバ (px4_drv) をインストールしますか?"; then
echo ""
echo "=== チューナードライバのインストール ==="
mkdir -p "$DRIVER_DIR"
EXISTING_DEB=$(ls "${DRIVER_DIR}"/px4-drv-dkms_*.deb 2>/dev/null | head -n1 || true)
if [ -n "$EXISTING_DEB" ]; then
echo " ダウンロード済みのドライバが見つかりました: $(basename "$EXISTING_DEB")"
if ask_yn "ダウンロード済みの$(basename "$EXISTING_DEB")を使用しますか?"; then
# 既存ファイルのフルパスをそのまま使用
DRIVER_DEB="$EXISTING_DEB"
else
DRIVER_VERSION=$(curl -s https://api.github.com/repos/tsukumijima/px4_drv/releases/latest | grep '"tag_name"' | sed 's/.*"v\([^"]*\)".*/\1/')
DRIVER_DEB="${DRIVER_DIR}/px4-drv-dkms_${DRIVER_VERSION}_all.deb"
DRIVER_URL="https://github.com/tsukumijima/px4_drv/releases/download/v${DRIVER_VERSION}/px4-drv-dkms_${DRIVER_VERSION}_all.deb"
curl -L -o "${DRIVER_DEB}" "${DRIVER_URL}"
chmod 644 "${DRIVER_DEB}"
fi
else
DRIVER_VERSION=$(curl -s https://api.github.com/repos/tsukumijima/px4_drv/releases/latest | grep '"tag_name"' | sed 's/.*"v\([^"]*\)".*/\1/')
DRIVER_DEB="${DRIVER_DIR}/px4-drv-dkms_${DRIVER_VERSION}_all.deb"
DRIVER_URL="https://github.com/tsukumijima/px4_drv/releases/download/v${DRIVER_VERSION}/px4-drv-dkms_${DRIVER_VERSION}_all.deb"
curl -L -o "${DRIVER_DEB}" "${DRIVER_URL}"
chmod 644 "${DRIVER_DEB}"
fi
# DRIVER_DEB は常にフルパスなので ./ を付けずにそのまま渡す
sudo apt install -y "${DRIVER_DEB}"
sudo modprobe -r px4_drv 2>/dev/null || true
sudo modprobe px4_drv
echo " ドライバ インストール完了"
else
echo " ドライバインストールをスキップします"
fi
echo ""
# ============================================================
# 2. コンテナ名の入力
# ============================================================
read -rp "作成するLXDコンテナ名を入力してください [konomitv]: " CONTAINER
CONTAINER="${CONTAINER:-konomitv}"
echo " コンテナ名: ${CONTAINER}"
echo ""
# ============================================================
# 3. Tailscale authkey の入力
# ============================================================
if ask_yn "Tailscale の authkey がありますか?"; then
read -rsp "authkey を入力してください(tskeyから入力。入力は非表示): " TS_AUTHKEY
echo ""
USE_TS_AUTHKEY=true
else
USE_TS_AUTHKEY=false
fi
echo ""
# ============================================================
# 4. コンテナ作成・マウント・ID マッピング
# ============================================================
echo "=== コンテナ '${CONTAINER}' を作成 ==="
if lxc info "$CONTAINER" &>/dev/null; then
echo " コンテナは既に存在します"
else
lxc launch ubuntu:26.04 "$CONTAINER"
fi
echo ""
echo "=== ${MOUNT_PATH} の確認・作成 ==="
if [ ! -d "$MOUNT_PATH" ]; then
sudo mkdir -p "$MOUNT_PATH"
echo " ${MOUNT_PATH} を作成しました"
else
echo " ${MOUNT_PATH} は既に存在します"
fi
TV_DIR="${MOUNT_PATH}/tv"
if [ ! -d "$TV_DIR" ]; then
sudo mkdir -p "$TV_DIR"
echo " ${TV_DIR} を作成しました"
else
echo " ${TV_DIR} は既に存在します"
fi
sudo chown 1000:1000 "$TV_DIR"
sudo chmod 755 "$TV_DIR"
echo " 録画フォルダ所有者: 1000:1000 パーミッション: 755"
echo ""
echo "=== ホストの ${MOUNT_PATH} をコンテナにマウント ==="
if lxc config device show "$CONTAINER" 2>/dev/null | grep -q "opt-lxd-data"; then
echo " opt-lxd-data は登録済み"
else
lxc config device add "$CONTAINER" opt-lxd-data disk source="$MOUNT_PATH" path="$MOUNT_PATH"
fi
echo ""
echo "=== ID マッピング設定 ==="
lxc config set "$CONTAINER" raw.idmap "both 1000 1000"
echo ""
echo "=== コンテナを再起動 ==="
lxc restart "$CONTAINER"
sleep 3
# ============================================================
# 5. コンテナ内セットアップ (apt / Tailscale)
# ============================================================
echo ""
echo "=== コンテナ内セットアップ ==="
lxc exec "${CONTAINER}" -- bash -euo pipefail << 'INNER'
apt update
apt upgrade -y
apt install -y curl
curl -fsSL https://tailscale.com/install.sh | sh
INNER
# ============================================================
# 6. Tailscale 起動
# ============================================================
echo ""
echo "=== Tailscale を起動 ==="
if [ "$USE_TS_AUTHKEY" = true ]; then
lxc exec "${CONTAINER}" -- tailscale up --authkey="${TS_AUTHKEY}"
else
echo " authkeyがないため、手動認証を行ってください。"
lxc exec "${CONTAINER}" -- tailscale up || true
fi
TS_IP=$(lxc exec "${CONTAINER}" -- tailscale ip -4 2>/dev/null || echo "取得中...")
echo " Tailscale IP: ${TS_IP}"
# ============================================================
# 7. TailscaleOK スナップショット
# ============================================================
echo ""
if ask_yn "スナップショット 'TailscaleOK' を作成しますか?"; then
echo "=== コンテナを停止中 ==="
lxc stop "${CONTAINER}"
echo "=== スナップショット 'TailscaleOK' を作成中 ==="
lxc snapshot "${CONTAINER}" TailscaleOK
echo " スナップショット 'TailscaleOK' を作成しました"
echo "=== コンテナを起動中 ==="
lxc start "${CONTAINER}"
sleep 3
fi
# ============================================================
# 8. USB チューナーパススルー
# ============================================================
DO_TUNER_PASS=false
if ask_yn "USB チューナーをコンテナにパススルーしますか?"; then
DO_TUNER_PASS=true
echo ""
echo "=== USB チューナーを検出中 ==="
LSUSB_LINE=$(lsusb | grep -i "ISDBT2056" || true)
if [ -z "$LSUSB_LINE" ]; then
echo " 警告: ISDBT2056 デバイスが見つかりません。パススルーをスキップします。"
DO_TUNER_PASS=false
else
echo " 検出: $LSUSB_LINE"
IDS=$(echo "$LSUSB_LINE" | grep -oP 'ID \K[0-9a-fA-F]{4}:[0-9a-fA-F]{4}')
VENDOR_ID=$(echo "$IDS" | cut -d: -f1)
PRODUCT_ID=$(echo "$IDS" | cut -d: -f2)
echo " vendorid : $VENDOR_ID"
echo " productid: $PRODUCT_ID"
if lxc config device show "$CONTAINER" 2>/dev/null | grep -q "usb-tuner"; then
echo " usb-tuner は登録済みのため上書きします"
lxc config device remove "$CONTAINER" usb-tuner
fi
lxc config device add "$CONTAINER" usb-tuner usb \
vendorid="$VENDOR_ID" \
productid="$PRODUCT_ID"
echo " usb-tuner を追加しました"
echo ""
echo "=== isdb2056video デバイスを検出中 ==="
ISDB_DEVS=$(ls /dev/isdb2056video* 2>/dev/null || true)
if [ -z "$ISDB_DEVS" ]; then
echo " 警告: /dev/isdb2056video* が見つかりません。スキップします。"
else
IDX=0
for DEV in $ISDB_DEVS; do
NAME="isdb2056-${IDX}"
echo " 追加: $DEV ($NAME)"
if lxc config device show "$CONTAINER" 2>/dev/null | grep -q "^${NAME}:"; then
lxc config device remove "$CONTAINER" "$NAME"
fi
lxc config device add "$CONTAINER" "$NAME" unix-char \
source="$DEV" path="$DEV"
IDX=$(( IDX + 1 ))
done
fi
fi
fi
# ============================================================
# 9. コンテナ内チューナー確認
# ============================================================
if [ "$DO_TUNER_PASS" = true ]; then
echo ""
echo "=== コンテナ内チューナー確認 ==="
lxc exec "${CONTAINER}" -- bash -c '
ISDB_DEVS=$(ls /dev/isdb2056video* 2>/dev/null || true)
if [ -z "$ISDB_DEVS" ]; then
echo " NG: /dev/isdb2056video* が見つかりません"
else
for DEV in $ISDB_DEVS; do
chmod 666 "$DEV" 2>/dev/null
echo " OK: $DEV"
done
fi
'
fi
# ============================================================
# 10. TunerOK スナップショット
# ============================================================
echo ""
if ask_yn "スナップショット 'TunerOK' を作成しますか?"; then
echo "=== コンテナを停止中 ==="
lxc stop "${CONTAINER}"
echo "=== スナップショット 'TunerOK' を作成中 ==="
lxc snapshot "${CONTAINER}" TunerOK
echo " スナップショット 'TunerOK' を作成しました"
echo "=== コンテナを起動中 ==="
lxc start "${CONTAINER}"
sleep 3
fi
# ============================================================
# 11. コンテナ内でシェルを開始
# ============================================================
echo ""
echo "コンテナ内でシェルを開始します。"
exec lxc exec "${CONTAINER}" -- bash
mirakc・EDCB・KonomiTVセットアップ(コンテナ内で実行)
そのままコンテナ内で下記を貼り付けてソフトをインストールします。最初のキー入力以外は放置していればよいのですが、完了するまで30分くらいはかかります。さらにインストール後のEPG取得が完全に終わるまで待つなら、トータル1時間くらい見ておけば、というところです。コンテナ内で実行するので、通常はどの環境でも成功するとは思いますが。
#!/bin/bash
set -euo pipefail
# ============================================================
# mirakc + EDCB + KonomiTV 環境構築スクリプト (Docker不使用版)
# 前提条件:
# - Debian系Linux (Ubuntu/Debian)
# - ネット接続
# - TVチューナー (px4_drv対応デバイス) は接続済みで、ドライバは
# ホスト側でインストール・ロード済みであること
# (本スクリプトを実行するコンテナ/ホストにはデバイスノード
# /dev/isdb2056video* または /dev/px4video* が見えている必要がある)
# ============================================================
REAL_USER="${SUDO_USER:-$(logname 2>/dev/null || true)}"
[ -z "$REAL_USER" ] && REAL_USER="$(id -un)"
REAL_HOME=$(getent passwd "$REAL_USER" | cut -d: -f6)
DTV_DIR="$REAL_HOME/dtv"
LOG_FILE="/var/log/mirakc-konomitv-install.log"
mkdir -p /var/log
if touch "$LOG_FILE" 2>/dev/null; then
exec > >(tee -a "$LOG_FILE") 2>&1
fi
echo "=== インストール開始: $(date) ==="
# ============================================================
# --rescan-only オプション: チャンネルスキャンのみ再実行したい場合
# (mirakc/EDCB/KonomiTV は既にインストール済みで、チャンネルが
# 表示されない問題だけを直したいときに使う)
# ============================================================
RESCAN_ONLY=false
if [ "${1:-}" = "--rescan-only" ]; then
RESCAN_ONLY=true
echo "=== --rescan-only モード: チャンネルスキャンと mirakc/EDCB 設定の再適用のみ実行します ==="
fi
# ============================================================
# 0. ユーザー入力 (--rescan-only の場合はスキップ)
# ============================================================
if [ "$RESCAN_ONLY" = false ]; then
echo "=== 1/9: キー設定 ==="
echo "キーの内容を貼り付けてください(入力後、Enterキーを2回押すと確定します):"
BCAS_CONTENT=$(sed '/^$/q')
fi
# ============================================================
# 1. 依存パッケージ
# ============================================================
if [ "$RESCAN_ONLY" = false ]; then
echo "=== 2/9: 依存パッケージをインストール中 ==="
sudo apt update
sudo apt install -y \
autoconf automake cmake libtool libpcsclite-dev git build-essential pkg-config \
curl wget libclang-dev libdvbv5-dev libudev-dev \
nodejs npm ffmpeg liblua5.2-dev liblua5.4-dev libz-dev \
g++ make gcc ninja-build \
libpcap-dev libpcre2-dev libcurl4-openssl-dev || {
echo "WARNING: 一部パッケージのインストールに失敗しました。続行します..."
}
fi
# GCC 13 は利用しない (GCC 13 と 15 の混在でリンクエラーが発生するため)
# デフォルトコンパイラ (GCC 15) + -Wno-error で統一する
# ============================================================
# 2. TVチューナードライバについて
#
# ドライバのインストール・ロードはホスト側で完了していることを
# 前提とし、コンテナ側ではデバイスノード (/dev/isdb2056video*,
# /dev/px4video* 等) が見えているかどうかだけを確認する。
# ============================================================
mkdir -p "$DTV_DIR"
# ============================================================
# チューナーデバイスの検出 (この後の処理全体で使う)
# ============================================================
detect_tuner_device() {
local isdb_dev px4_dev
isdb_dev=$(ls /dev/isdb2056video* 2>/dev/null | head -1 || true)
px4_dev=$(ls /dev/px4video* 2>/dev/null | head -1 || true)
if [ -n "$isdb_dev" ]; then
echo "$isdb_dev"
elif [ -n "$px4_dev" ]; then
echo "$px4_dev"
else
echo ""
fi
}
TUNER_DEVICE="$(detect_tuner_device)"
if [ -z "$TUNER_DEVICE" ]; then
echo "ERROR: チューナーデバイス (/dev/isdb2056video* または /dev/px4video*) が検出できません。"
echo " ドライバはホスト側でインストール・ロードされている前提です。以下をホストで確認してください:"
echo " - USB/PCIe チューナーが正しく接続されているか ('lsusb' または 'lspci')"
echo " - 'lsmod | grep px4_drv' でドライバがロードされているか"
echo " - LXD コンテナにデバイスが渡されているか ('lxc config device show <コンテナ名>')"
echo " コンテナ内では 'ls -la /dev/ | grep -E \"isdb|px4\"' でデバイスノードを確認してください。"
echo ""
echo "チューナーが認識されない状態でチャンネルスキャンを行うと、"
echo "チャンネルが1件も検出されず空の番組表になります。"
exit 1
fi
echo "チューナー検出: $TUNER_DEVICE"
# ============================================================
# 3. libyakisoba / libsobacas のビルド
# ============================================================
if [ "$RESCAN_ONLY" = false ]; then
echo "=== 3/9: 復号ライブラリのビルド ==="
cd "$DTV_DIR"
for repo in "libyakisoba" "libsobacas"; do
if [ ! -d "$repo" ]; then
git clone "https://github.com/tsunoda14/${repo}.git" || {
echo "WARNING: ${repo} のクローンに失敗しました"
continue
}
fi
cd "$repo"
autoreconf -i || true
mkdir -p build && cd build
if [ "$repo" == "libyakisoba" ]; then
../configure --sysconfdir=/usr/local/etc || true
else
../configure || true
fi
make -j"$(nproc)" || true
sudo make install || true
cd "$DTV_DIR"
done
sudo ldconfig
fi
# ============================================================
# 4. Rust と ツール群のビルド
# ============================================================
if [ "$RESCAN_ONLY" = false ]; then
echo "=== 4/9: Rust と ツールのビルド ==="
if ! command -v cargo &> /dev/null; then
curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh -s -- -y
fi
source "$HOME/.cargo/env" 2>/dev/null || export PATH="$HOME/.cargo/bin:$PATH"
# libsobacas.pc
sudo mkdir -p /usr/local/lib/pkgconfig
sudo tee /usr/local/lib/pkgconfig/libsobacas.pc > /dev/null <<EOF
prefix=/usr/local
libdir=/usr/local/lib
includedir=/usr/include
Name: libsobacas
Description: PCSC compatible ECM decoder library
Version: 0.0.0
Libs: -L\${libdir} -lsobacas
Cflags: -I\${includedir}/PCSC
EOF
# recisdb
cd "$DTV_DIR"
if [ ! -d "recisdb-rs" ]; then
git clone --recursive https://github.com/kazuki0824/recisdb-rs.git || {
echo "ERROR: recisdb-rs のクローンに失敗しました"
echo "手動で https://github.com/kazuki0824/recisdb-rs を確認してください"
}
fi
if [ -d "recisdb-rs" ]; then
cd recisdb-rs
if [ -f "b25-sys/build.rs" ]; then
sed -i 's/pcsclite/sobacas/g' b25-sys/build.rs || true
fi
cargo build -F dvb --release || {
echo "ERROR: recisdb のビルドに失敗しました"
exit 1
}
sudo systemctl stop mirakc 2>/dev/null || true
sudo cp target/release/recisdb /usr/local/bin/
fi
# mirakc
echo "=== mirakc ビルド中(数分かかります)==="
cd "$DTV_DIR"
if [ ! -d "mirakc-src" ]; then
git clone --recursive https://github.com/mirakc/mirakc.git mirakc-src || {
echo "ERROR: mirakc のクローンに失敗しました"
exit 1
}
fi
cd mirakc-src
git pull --recurse-submodules || true
cargo build --release || {
echo "ERROR: mirakc のビルドに失敗しました"
exit 1
}
sudo systemctl stop mirakc 2>/dev/null || true
sudo cp target/release/mirakc /usr/local/bin/
fi
# ============================================================
# 5. mirakc-arib: ソースビルド (Docker不使用)
# GCC 15 対策: CXXFLAGS_EXTRA=-Wno-error で対応
# ============================================================
if [ "$RESCAN_ONLY" = false ]; then
echo "=== mirakc-arib セットアップ ==="
MIRAKC_ARIB_VERSION="0.24.29"
if ! command -v mirakc-arib &> /dev/null; then
echo "mirakc-arib をソースからビルド中..."
cd "$DTV_DIR"
if [ ! -d "mirakc-arib" ]; then
git clone --recursive --depth 1 -b "$MIRAKC_ARIB_VERSION" https://github.com/mirakc/mirakc-arib.git 2>/dev/null || {
echo "WARNING: バージョン $MIRAKC_ARIB_VERSION のタグが見つかりません。最新版を取得します..."
rm -rf mirakc-arib
git clone --recursive --depth 1 https://github.com/mirakc/mirakc-arib.git || {
echo "ERROR: mirakc-arib のクローンに失敗しました"
exit 1
}
}
fi
cd mirakc-arib
echo "サブモジュールを同期中..."
git submodule sync --recursive
git submodule update --init --recursive --depth 1 || {
echo "ERROR: サブモジュールの取得に失敗しました"
exit 1
}
# 過去の失敗ビルドのキャッシュが残っていると依存関係が壊れる
if [ -d build ]; then
echo "既存の build ディレクトリをクリーンアップ中..."
rm -rf build
fi
# CMakeLists.txt を修正: GCC 13 参照を除去し、-Wno-error のみを追加
echo "CMakeLists.txt を修正中..."
# 既に GCC 13 の設定が注入済みの場合は除去
sed -i 's/ CC=gcc-13 CXX=g++-13 GCC=gcc-13//g' CMakeLists.txt || true
sed -i 's/ CC=gcc-13//g' CMakeLists.txt || true
sed -i 's/ CXX=g++-13//g' CMakeLists.txt || true
sed -i 's/ GCC=gcc-13//g' CMakeLists.txt || true
# CXXFLAGS_EXTRA=-Wno-error を追加(既にあればスキップ)
if ! grep -q "CXXFLAGS_EXTRA=-Wno-error" CMakeLists.txt; then
echo "CMakeLists.txt に -Wno-error を追加中..."
sed -i 's|make -j ${NPROC} SYSPREFIX=${MIRAKC_ARIB_VENDOR_DIR} ${MIRAKC_ARIB_TSDUCK_DEFS}|make -j ${NPROC} SYSPREFIX=${MIRAKC_ARIB_VENDOR_DIR} CXXFLAGS_EXTRA=-Wno-error ${MIRAKC_ARIB_TSDUCK_DEFS}|g' CMakeLists.txt || true
sed -i 's|make -j ${NPROC} install-devel SYSPREFIX=${MIRAKC_ARIB_VENDOR_DIR} ${MIRAKC_ARIB_TSDUCK_DEFS}|make -j ${NPROC} install-devel SYSPREFIX=${MIRAKC_ARIB_VENDOR_DIR} CXXFLAGS_EXTRA=-Wno-error ${MIRAKC_ARIB_TSDUCK_DEFS}|g' CMakeLists.txt || true
fi
# CMake の CXXFLAGS_EXTRA 変数も設定
if ! grep -q "MIRAKC_ARIB_TSDUCK_ARIB_CXXFLAGS_EXTRA" CMakeLists.txt | head -1; then
sed -i 's|set(MIRAKC_ARIB_TSDUCK_ARIB_CXXFLAGS_EXTRA ${MIRAKC_ARIB_TSDUCK_ARIB_CXXFLAGS})|set(MIRAKC_ARIB_TSDUCK_ARIB_CXXFLAGS_EXTRA "-Wno-error")|g' CMakeLists.txt || true
fi
echo "CMake 構成中..."
cmake -S . -B build -G Ninja -D CMAKE_BUILD_TYPE=Release -DMIRAKC_ARIB_TSDUCK_ARIB_CXXFLAGS="-Wno-error" || {
echo "ERROR: mirakc-arib の cmake 設定に失敗しました"
exit 1
}
echo "vendor ターゲットをビルド中(tsduck-arib 等の取得・ビルド、数分かかります)..."
VENDOR_OK=false
for attempt in 1 2 3; do
if ninja -C build vendor; then
VENDOR_OK=true
break
fi
echo "WARNING: vendor ビルド失敗 (試行 $attempt/3)。build をクリーンして再試行します..."
rm -rf build
cmake -S . -B build -G Ninja -D CMAKE_BUILD_TYPE=Release -DMIRAKC_ARIB_TSDUCK_ARIB_CXXFLAGS="-Wno-error" || true
done
if [ "$VENDOR_OK" != true ]; then
echo "ERROR: mirakc-arib の vendor ビルドが複数回失敗しました"
echo "手動で確認してください: cd $DTV_DIR/mirakc-arib && ninja -C build vendor"
exit 1
fi
# vendor 完了検証
TSDUCK_LIB=$(find build -iname "*tsduck*" -name "*.a" 2>/dev/null | head -1)
if [ -z "$TSDUCK_LIB" ]; then
echo "WARNING: vendor ビルド後に tsduck-arib のライブラリが見つかりません。続行します..."
else
echo "vendor ビルド完了を確認: $TSDUCK_LIB"
fi
# vendor 完了後に CMake キャッシュを再生成
echo "CMake 再構成中(vendor 反映)..."
cmake -S . -B build -G Ninja -D CMAKE_BUILD_TYPE=Release -DMIRAKC_ARIB_TSDUCK_ARIB_CXXFLAGS="-Wno-error" || {
echo "ERROR: mirakc-arib の cmake 再設定に失敗しました"
exit 1
}
echo "利用可能なターゲットを確認中..."
ninja -C build -t targets all 2>/dev/null | grep -i "mirakc\|arib" | head -5
echo "mirakc-arib 本体ビルド中..."
if ! ninja -C build mirakc-arib; then
echo "WARNING: 'mirakc-arib' ターゲットが見つかりません。build ターゲットを試行します..."
if ! ninja -C build; then
echo "ERROR: mirakc-arib のビルドに失敗しました"
exit 1
fi
fi
if [ ! -f build/bin/mirakc-arib ]; then
echo "ERROR: build/bin/mirakc-arib が生成されませんでした"
echo "build ディレクトリ内のバイナリを確認してください: find build -name 'mirakc-arib' -type f"
exit 1
fi
sudo cp build/bin/mirakc-arib /usr/local/bin/mirakc-arib
sudo chmod +x /usr/local/bin/mirakc-arib
echo "mirakc-arib ビルド完了"
fi
# mirakc-arib の動作確認
if ! mirakc-arib --version > /dev/null 2>&1; then
echo "ERROR: mirakc-arib が正常にインストールされていません。"
exit 1
fi
echo "mirakc-arib インストール確認: $(mirakc-arib --version | head -1)"
fi
# ============================================================
# 6. キーの保存
# ============================================================
if [ "$RESCAN_ONLY" = false ]; then
echo "=== 5/9: キー保存 ==="
sudo mkdir -p /usr/local/etc
echo "$BCAS_CONTENT" | sudo tee /usr/local/etc/bcas_keys > /dev/null
# キー読み込みテスト
echo "bcas_keys の内容を確認中..."
KEY_COUNT=$(grep -c '^Key\[' /usr/local/etc/bcas_keys 2>/dev/null || echo 0)
echo " 登録キー数: $KEY_COUNT"
if echo "$BCAS_CONTENT" | grep -q 'Key\[17\]'; then
echo " CS (Key[17]) キー: OK"
else
echo " WARNING: CS (Key[17]) キーがありません。CS視聴はできません。"
fi
fi
# ============================================================
# 7. ISDBScanner 実行
# ============================================================
echo "=== 6/9: チャンネルスキャン ==="
if [ ! -f /usr/local/bin/isdb-scanner ]; then
sudo wget -q https://github.com/tsukumijima/ISDBScanner/releases/download/v1.3.3/isdb-scanner -O /usr/local/bin/isdb-scanner || {
echo "ERROR: ISDBScanner のダウンロードに失敗しました"
exit 1
}
sudo chmod +x /usr/local/bin/isdb-scanner
fi
# 既存の mirakc/EDCB を一旦止めてチューナーを解放する
# (稼働中の mirakc/EDCB がチューナーを掴んでいると ISDBScanner が
# チューナーを開けず、サイレントに 0 件スキャンになることがある)
sudo systemctl stop mirakc 2>/dev/null || true
sudo systemctl stop edcb 2>/dev/null || true
# --rescan-only の場合は強制的に再スキャンする。通常実行時は
# 既存のスキャン結果があれば使うが、中身が空/不正なら再スキャンする。
NEED_SCAN=true
if [ "$RESCAN_ONLY" = false ] && [ -f "$DTV_DIR/scanned/mirakc/config.yml" ]; then
EXISTING_CH_COUNT=$(grep -c '^\s*- name:' "$DTV_DIR/scanned/mirakc/config.yml" 2>/dev/null || echo 0)
if [ "$EXISTING_CH_COUNT" -gt 0 ]; then
echo "既存のスキャン結果 (${EXISTING_CH_COUNT} チャンネル) を使用します。再スキャンする場合は $DTV_DIR/scanned を削除してから再実行してください。"
NEED_SCAN=false
fi
fi
if [ "$NEED_SCAN" = true ]; then
echo "ISDBScanner を実行します(チューナーを使用してチャンネルスキャンを行うため数分かかります)..."
rm -rf "$DTV_DIR/scanned"
mkdir -p "$DTV_DIR/scanned"
SCAN_LOG="/tmp/isdb-scanner.log"
set +e
timeout 600 isdb-scanner "$DTV_DIR/scanned/" 2>&1 | tee "$SCAN_LOG"
SCAN_RC=${PIPESTATUS[0]}
set -e
if [ "$SCAN_RC" -ne 0 ]; then
echo ""
echo "ERROR: isdb-scanner が異常終了しました (終了コード: $SCAN_RC)"
echo "ログ: $SCAN_LOG"
echo ""
echo "よくある原因:"
echo " - チューナーが他のプロセスに使用中 (mirakc/EDCB が起動していないか確認)"
echo " - チューナーが信号を受信できない (アンテナ線・B-CASカードの接触不良)"
echo " - bcas_keys が正しく設定されていない"
echo ""
echo "対処後、このスクリプトを '--rescan-only' オプション付きで再実行してください:"
echo " sudo bash $0 --rescan-only"
exit 1
fi
fi
# スキャン結果の検証: config.yml が存在し、かつチャンネルが1件以上あるか確認
if [ ! -f "$DTV_DIR/scanned/mirakc/config.yml" ]; then
echo ""
echo "ERROR: $DTV_DIR/scanned/mirakc/config.yml が生成されませんでした。"
echo "ISDBScanner はチャンネルを検出できなかったか、出力先パスが想定と異なります。"
echo "$DTV_DIR/scanned/ の内容を確認してください: find $DTV_DIR/scanned -type f"
find "$DTV_DIR/scanned" -type f 2>/dev/null || true
exit 1
fi
SCANNED_CH_COUNT=$(grep -c '^\s*- name:' "$DTV_DIR/scanned/mirakc/config.yml" 2>/dev/null || echo 0)
if [ "$SCANNED_CH_COUNT" -eq 0 ]; then
echo ""
echo "ERROR: スキャン結果に登録されたチャンネルが0件です。"
echo "$DTV_DIR/scanned/mirakc/config.yml の内容を確認してください:"
cat "$DTV_DIR/scanned/mirakc/config.yml" 2>/dev/null || true
echo ""
echo "チューナー/アンテナ線/B-CASキーを確認の上、再実行してください:"
echo " sudo bash $0 --rescan-only"
exit 1
fi
echo "チャンネルスキャン完了: ${SCANNED_CH_COUNT} チャンネルを検出しました"
# ============================================================
# 8. mirakc ネイティブセットアップ (mirakc 4.x 対応)
# ============================================================
echo "=== 7/9: mirakc セットアップ ==="
MIRAKC_ETC="/etc/mirakc"
MIRAKC_DATA="/var/lib/mirakc/epg"
sudo mkdir -p "$MIRAKC_ETC" "$MIRAKC_DATA"
# config.yml は必ず ISDBScanner の結果を使う。
# (ダミー設定へのフォールバックは廃止 — 上のスキャン検証で
# 0件の場合は既に exit している)
sudo cp "$DTV_DIR/scanned/mirakc/config.yml" "$MIRAKC_ETC/config.yml"
# scan-services / sync-clocks / update-schedules を無効化する。
# 旧版は既存の "disabled: false" 行を残したまま下に "disabled: true" を
# 追記してしまい、YAML 内に重複キーが残る不具合があったため、
# Python で安全にパースして上書きする。
python3 << PYEOF
import re
path = "$MIRAKC_ETC/config.yml"
with open(path, "r", encoding="utf-8") as f:
content = f.read()
# jobs: セクションが既に存在する場合は disabled の値を true に統一する
if re.search(r'^jobs:\s*$', content, re.MULTILINE):
for job in ("scan-services", "sync-clocks", "update-schedules"):
# " job-name:\n disabled: false" -> " job-name:\n disabled: true"
pattern = re.compile(
r'(^\s*' + re.escape(job) + r':\s*\n(?:\s+\S.*\n)*?\s*disabled:\s*)(true|false)',
re.MULTILINE,
)
if pattern.search(content):
content = pattern.sub(lambda m: m.group(1) + "true", content)
else:
# job 自体はあるが disabled 行がない場合は追加
job_pattern = re.compile(r'(^\s*' + re.escape(job) + r':\s*\n)', re.MULTILINE)
if job_pattern.search(content):
content = job_pattern.sub(lambda m: m.group(1) + " disabled: true\n", content)
else:
content += f"\n {job}:\n disabled: true\n"
else:
content += (
"\njobs:\n"
" scan-services:\n disabled: true\n"
" sync-clocks:\n disabled: true\n"
" update-schedules:\n disabled: true\n"
)
with open(path, "w", encoding="utf-8") as f:
f.write(content)
print("jobs.*.disabled を true に設定しました")
PYEOF
# strings.yml をコピー (mirakc 4.x で必須)
if [ -f "$DTV_DIR/mirakc-src/resources/strings.yml" ]; then
sudo cp "$DTV_DIR/mirakc-src/resources/strings.yml" "$MIRAKC_ETC/strings.yml"
elif [ ! -f "$MIRAKC_ETC/strings.yml" ]; then
echo "WARNING: strings.yml が見つかりません。"
fi
# tuners.*.command が ISDBScanner の出力したデバイスパスと一致しているか確認。
# 一致しない場合 (例: チューナーを差し替えた、デバイスノード名が変わった) は
# 検出済みの $TUNER_DEVICE に書き換える。
if ! grep -q "$TUNER_DEVICE" "$MIRAKC_ETC/config.yml"; then
echo "WARNING: config.yml 内のチューナーデバイスパスが現在検出されているデバイス ($TUNER_DEVICE) と一致しません。"
echo " config.yml 内の記述:"
grep -A2 "^tuners:" "$MIRAKC_ETC/config.yml" | head -10 || true
echo " 自動置換は行いません。視聴できない場合は $MIRAKC_ETC/config.yml の tuners セクションを手動確認してください。"
fi
# update-epg.sh: 番組情報の手動更新スクリプト (Docker不使用)
cat > "$REAL_HOME/update-epg.sh" << UPDATEEPG
#!/bin/bash
# update-epg.sh - Manual EPG update script
# Run this when NOT watching TV (it will occupy the tuner for several minutes)
#
# Usage: sudo bash ~/update-epg.sh
set -e
echo "=== EPG Manual Update ==="
echo "WARNING: This will occupy the tuner for several minutes."
echo "Do NOT run this while watching TV."
echo ""
echo "[1/2] Syncing clocks..."
recisdb tune --device ${TUNER_DEVICE} --channel T27 - 2>/dev/null | timeout 30 mirakc-arib sync-clocks 2>&1 | head -20
echo ""
echo "[2/2] Collecting EIT data (番組情報更新)..."
recisdb tune --device ${TUNER_DEVICE} --channel T27 - 2>/dev/null | timeout 120 mirakc-arib collect-eits \$(cat /var/lib/mirakc/epg/services.json 2>/dev/null | python3 -c "
import json,sys
d=json.load(sys.stdin)
if isinstance(d, list):
sids = [str(item[0] if isinstance(item, list) else item.get('id','')) for item in d[:200]]
print(' '.join(['--sids=' + s for s in sids[:50]]))
elif isinstance(d, dict):
sids = list(d.keys())[:50]
print(' '.join(['--sids=' + str(s) for s in sids]))
" 2>/dev/null) 2>&1 | tail -5
echo ""
echo "=== EPG Update Complete ==="
UPDATEEPG
chmod +x "$REAL_HOME/update-epg.sh"
# mirakc systemd サービス
sudo tee /etc/systemd/system/mirakc.service > /dev/null << EOT
[Unit]
Description=mirakc TV Recorder
After=network-online.target
Wants=network-online.target
[Service]
Type=simple
User=$REAL_USER
Environment=MIRAKC_CONFIG=$MIRAKC_ETC/config.yml
ExecStart=/usr/local/bin/mirakc --config $MIRAKC_ETC/config.yml
Restart=always
RestartSec=5
[Install]
WantedBy=multi-user.target
EOT
sudo systemctl daemon-reload
sudo systemctl enable mirakc
sudo systemctl restart mirakc
echo "mirakc 起動待機中..."
MIRAKC_READY=false
for i in $(seq 1 30); do
if curl -s http://127.0.0.1:40772/api/tuners >/dev/null 2>&1; then
echo "mirakc API 応答OK"
MIRAKC_READY=true
break
fi
sleep 2
echo " 待機中... ($((i*2))秒経過)"
done
if [ "$MIRAKC_READY" = false ]; then
echo "WARNING: mirakc が応答しません。ログを確認してください:"
echo " journalctl -u mirakc -n 50"
sudo systemctl status mirakc --no-pager || true
fi
# ============================================================
# 初回サービススキャン (scan-services を一時的に有効化して実行)
# mirakc にservicesを知らせるため。一度だけ実行して無効化。
# ============================================================
if [ "$MIRAKC_READY" = true ]; then
echo ""
echo "=== 初回サービススキャンを実行します(チューナーを使用、数分かかります)==="
sudo sed -i '/scan-services:/,/disabled:/ s/disabled: true/disabled: false/' "$MIRAKC_ETC/config.yml"
sudo systemctl restart mirakc
echo "mirakc 再起動待機中 (scan-services 有効)..."
for i in $(seq 1 30); do
if curl -s http://127.0.0.1:40772/api/tuners >/dev/null 2>&1; then
break
fi
sleep 2
done
SCAN_OK=false
for i in $(seq 1 90); do
API_SVC_COUNT=$(curl -s http://127.0.0.1:40772/api/services 2>/dev/null | grep -o '"id"' | wc -l || echo 0)
if [ "$API_SVC_COUNT" -gt 0 ]; then
echo "サービススキャン完了: ${API_SVC_COUNT} 件のサービスを検出"
SCAN_OK=true
break
fi
sleep 5
echo " スキャン中... ($((i*5))秒経過)"
done
if [ "$SCAN_OK" != true ]; then
echo "WARNING: サービススキャンがタイムアウトしました。services が0件のままです。"
echo " mirakc のログを確認してください: journalctl -u mirakc -n 100"
echo " 手動再試行: docker 不使用のため、以下を実行してください:"
echo " sudo systemctl restart mirakc"
echo " curl http://127.0.0.1:40772/api/services"
fi
# scan-services を無効化に戻して再起動 (起動時にチューナーを毎回占有しないようにする)
sudo sed -i '/scan-services:/,/disabled:/ s/disabled: false/disabled: true/' "$MIRAKC_ETC/config.yml"
sudo systemctl restart mirakc
echo "mirakc 再起動待機中 (scan-services 無効化に戻す)..."
MIRAKC_READY=false
for i in $(seq 1 30); do
if curl -s http://127.0.0.1:40772/api/tuners >/dev/null 2>&1; then
MIRAKC_READY=true
break
fi
sleep 2
done
# スキャンで取得した services が再起動後も保持されているか確認
if [ "$MIRAKC_READY" = true ]; then
API_SVC_COUNT=$(curl -s http://127.0.0.1:40772/api/services 2>/dev/null | grep -o '"id"' | wc -l || echo 0)
echo "mirakc API 上のサービス件数 (scan-services 無効化後): ${API_SVC_COUNT}"
if [ "$API_SVC_COUNT" -eq 0 ] && [ "$SCAN_OK" = true ]; then
echo "WARNING: スキャン直後は services を検出していましたが、再起動後に0件になりました。"
echo " /var/lib/mirakc/epg/ 以下のキャッシュファイルの権限・永続化を確認してください:"
ls -la /var/lib/mirakc/epg/ 2>/dev/null || true
fi
fi
fi
# ============================================================
# 9. EDCB (EpgTimerSrv) のセットアップ
# ============================================================
if [ "$RESCAN_ONLY" = false ]; then
echo "=== 8/9: EDCB セットアップ ==="
cd "$DTV_DIR"
if [ ! -d "EDCB" ]; then
git clone https://github.com/xtne6f/EDCB || {
echo "ERROR: EDCB のクローンに失敗しました"
exit 1
}
fi
cd EDCB/Document/Unix
make -j"$(nproc)" || {
echo "WARNING: EDCB ビルドでエラーが発生しました。続行します..."
}
sudo make install || true
make extra || true
sudo make install_extra || true
# EDCB lua ライブラリ修正
EDCB_LUA_DIR=$(find "$DTV_DIR/EDCB" -path "*/EpgTimerSrv/EpgTimerSrv" -type d 2>/dev/null | head -1)
if [ -n "$EDCB_LUA_DIR" ]; then
ln -sf /usr/lib/x86_64-linux-gnu/liblua5.2.so "$EDCB_LUA_DIR/" 2>/dev/null || true
ln -sf /usr/lib/x86_64-linux-gnu/liblua5.2.a "$EDCB_LUA_DIR/" 2>/dev/null || true
cd "$EDCB_LUA_DIR" && make -j"$(nproc)" 2>/dev/null || true
cd "$DTV_DIR/EDCB/Document/Unix"
sudo make install 2>/dev/null || true
fi
sudo mkdir -p /var/local/edcb
sudo chown -R "$REAL_USER:$REAL_USER" /var/local/edcb
make setup_ini || true
# EMWUI
cd "$DTV_DIR"
if [ ! -d "EDCB_Material_WebUI" ]; then
git clone https://github.com/EMWUI/EDCB_Material_WebUI || {
echo "WARNING: EDCB_Material_WebUI のクローンに失敗しました"
}
fi
if [ -d "EDCB_Material_WebUI" ]; then
sudo cp -r EDCB_Material_WebUI/HttpPublic /var/local/edcb/ 2>/dev/null || true
sudo cp -r EDCB_Material_WebUI/Setting /var/local/edcb/ 2>/dev/null || true
fi
# BonDriver_LinuxMirakc
cd "$DTV_DIR"
if [ ! -d "BonDriver_LinuxMirakc" ]; then
git clone https://github.com/matching/BonDriver_LinuxMirakc.git --recurse-submodules || {
echo "WARNING: BonDriver_LinuxMirakc のクローンに失敗しました"
}
fi
if [ -d "BonDriver_LinuxMirakc" ]; then
cd BonDriver_LinuxMirakc
if [ -f "src/BonDriver_LinuxMirakc.cpp" ]; then
python3 << 'PYEOF' || true
with open('src/BonDriver_LinuxMirakc.cpp', 'r') as f:
content = f.read()
content = content.replace('char szHeader[ 64 ];', 'char szHeader[ 256 ];')
import re
content = re.sub(r'char szHeader\[len\];', 'char szHeader[256];', content)
old = 'sprintf(szHeader, "Connection: close\\r\\nX-Mirakurun-Priority: %d", g_Priority);'
new = 'sprintf(szHeader, "Host: %s:%d\\r\\nConnection: close\\r\\nX-Mirakurun-Priority: %d", g_ServerHost, g_ServerPort, g_Priority);'
content = content.replace(old, new)
with open('src/BonDriver_LinuxMirakc.cpp', 'w') as f:
f.write(content)
PYEOF
fi
make -j"$(nproc)" || {
echo "WARNING: BonDriver_LinuxMirakc のビルドに失敗しました"
}
if [ -f "BonDriver_LinuxMirakc.so" ]; then
sudo cp BonDriver_LinuxMirakc.so /usr/local/lib/edcb/
fi
fi
sudo tee /usr/local/lib/edcb/BonDriver_LinuxMirakc.so.ini > /dev/null << 'EOT'
[GLOBAL]
SERVER_HOST="127.0.0.1"
SERVER_PORT=40772
SERVER_TYPE="http"
DECODE_B25=1
PRIORITY=10
SERVICE_SPLIT=0
EOT
# スキャン結果のコピー(存在する場合のみ)
if [ -f "$DTV_DIR/scanned/EDCB-Wine/ChSet5.txt" ]; then
cp "$DTV_DIR/scanned/EDCB-Wine/ChSet5.txt" /var/local/edcb/Setting/ 2>/dev/null || true
fi
if [ -f "$DTV_DIR/scanned/EDCB-Wine/BonDriver_mirakc(BonDriver_mirakc).ChSet4.txt" ]; then
cp "$DTV_DIR/scanned/EDCB-Wine/BonDriver_mirakc(BonDriver_mirakc).ChSet4.txt" \
"/var/local/edcb/Setting/BonDriver_LinuxMirakc(LinuxMirakc).ChSet4.txt" 2>/dev/null || true
fi
if [ -f /var/local/edcb/HttpPublic/legacy/util.lua ]; then
sed -i -e 's/^ALLOW_SETTING=.*/ALLOW_SETTING=true/' /var/local/edcb/HttpPublic/legacy/util.lua
fi
mkdir -p /var/local/edcb/HttpPublic/video
# チューナー数の自動検出
ISDB_DEV_COUNT=$(ls /dev/isdb2056video* 2>/dev/null | wc -l || echo 0)
PX4_DEV_COUNT=$(ls /dev/px4video* 2>/dev/null | wc -l || echo 0)
if [ "$ISDB_DEV_COUNT" -gt 0 ]; then
TUNER_COUNT="$ISDB_DEV_COUNT"
elif [ "$PX4_DEV_COUNT" -gt 0 ]; then
TUNER_COUNT=$(( PX4_DEV_COUNT / 4 ))
[ "$TUNER_COUNT" -eq 0 ] && TUNER_COUNT=1
else
TUNER_COUNT=1
fi
echo "検出チューナー数: ${TUNER_COUNT}"
sudo tee /var/local/edcb/EpgTimerSrv.ini > /dev/null << EOT
[SET]
EnableHttpSrv=1
EnableTCPSrv=1
RecEndMode=0
Data=1
HttpAccessControlList=+127.0.0.0/8,+10.0.0.0/8,+172.16.0.0/12,+192.168.0.0/16,+169.254.0.0/16,+100.64.0.0/10
CompatFlags=128
[BonDriver_LinuxMirakc.so]
Count=${TUNER_COUNT}
GetEpg=1
EPGCount=0
Priority=0
EOT
sudo tee /var/local/edcb/Common.ini > /dev/null << 'EOT'
[SET]
RecFolderPath0=/var/local/edcb/HttpPublic/video
RecFolderNum=1
EOT
sudo tee /var/local/edcb/EpgDataCap_Bon.ini > /dev/null << 'EOT'
[SET]
BonDriverDllPath=/usr/local/lib/edcb
EOT
# EDCB サービス (Docker依存なし)
sudo tee /etc/systemd/system/edcb.service > /dev/null << EOT
[Unit]
Description=EpgTimerSrv
After=network-online.target mirakc.service
Requires=mirakc.service
[Service]
Type=simple
User=$REAL_USER
WorkingDirectory=/usr/local/lib/edcb
ExecStart=/usr/local/bin/EpgTimerSrv
Restart=always
[Install]
WantedBy=default.target
EOT
sudo systemctl daemon-reload
sudo systemctl enable edcb
sudo systemctl restart edcb
# ============================================================
# EPG 受信開始
# ============================================================
echo ""
echo "=== EPG 受信を開始します ==="
(
echo "EDCBの起動を待機中..."
for i in $(seq 1 60); do
if curl -sf "http://127.0.0.1:5510/legacy/index.html" -o /dev/null 2>/dev/null; then
echo "EDCB: 起動確認 (${i}秒)"
break
fi
sleep 1
done
sleep 5
echo "EPG取得リクエストを送信中..."
CTOK=$(curl -s http://127.0.0.1:5510/legacy/index.html \
| grep -o 'name="ctok" value="[^"]*"' | head -1 \
| grep -o 'value="[^"]*"' | cut -d'"' -f2)
RESULT=$(curl -s -X POST "http://127.0.0.1:5510/legacy/index.html" \
-H "Content-Type: application/x-www-form-urlencoded" \
-d "ctok=${CTOK}&epgcap=y" \
| grep -o 'id="result">[^<]*')
if echo "$RESULT" | grep -q "開始しました"; then
echo "EPG受信: 開始しました"
else
echo "警告: EPG取得リクエストの応答: $RESULT"
fi
) >> /var/log/dtv_epg_start.log 2>&1 &
echo "EPG受信をバックグラウンドで開始しました(KonomiTVインストール中に受信進行中)"
echo "進捗ログ: tail -f /var/log/dtv_epg_start.log"
fi
# ============================================================
# 10. KonomiTV インストール
# ============================================================
if [ "$RESCAN_ONLY" = false ]; then
echo "=== 9/9: KonomiTV インストール ==="
# pm2 のインストール (sudo用PATH対応)
sudo npm install -g pm2 || {
echo "WARNING: pm2 のインストールに失敗しました"
}
PM2_BIN=$(find /usr/local/lib/node_modules -name "pm2" -path "*/bin/*" -type f 2>/dev/null | head -1)
if [ -n "$PM2_BIN" ]; then
sudo ln -sf "$PM2_BIN" /usr/local/bin/pm2
sudo chmod +x "$PM2_BIN"
fi
echo "pm2: $(/usr/local/bin/pm2 --version 2>/dev/null || echo 'not found')"
cd "$DTV_DIR"
if [ ! -f KonomiTV-Installer.elf ]; then
curl -LO https://github.com/tsukumijima/KonomiTV/releases/latest/download/KonomiTV-Installer.elf || {
echo "ERROR: KonomiTV インストーラーのダウンロードに失敗しました"
exit 1
}
chmod +x KonomiTV-Installer.elf
fi
if ! command -v expect &> /dev/null; then
sudo apt install -y expect
fi
INSTALLER_PATH="$DTV_DIR/KonomiTV-Installer.elf"
if ! expect << EXPECT_EOF
set timeout 600
log_file /tmp/konomi_install.log
spawn $INSTALLER_PATH
expect -re {\[1/2/3/4/5\].*:} { send "1\r" }
expect -re {フォルダのパス:} { send "/opt/KonomiTV\r" }
expect -re {\[EDCB/Mirakurun\].*:} { send "\r" }
expect -re {TCP API の URL:} { send "tcp://127.0.0.1:4510/\r" }
expect -re {\[FFmpeg.*\].*:} { send "FFmpeg\r" }
expect -re {録画フォルダのパス:} { send "/var/local/edcb/HttpPublic/video\r" }
expect -re {録画フォルダのパス:} { send "\r" }
expect -re {キャプチャ画像の保存先フォルダのパス:} { send "/var/local/edcb/HttpPublic/video\r" }
expect -re {キャプチャ画像の保存先フォルダのパス:} { send "\r" }
expect eof
EXPECT_EOF
then
echo "WARNING: KonomiTV の自動インストールで問題が発生しました"
echo "手動でインストールしてください: cd $DTV_DIR && sudo ./KonomiTV-Installer.elf"
fi
KONOMI_CONFIG="/opt/KonomiTV/config.yaml"
if [ ! -f "$KONOMI_CONFIG" ]; then
KONOMI_CONFIG=$(find /opt /usr/local /home -name "config.yaml" -path "*/KonomiTV/*" 2>/dev/null | head -1)
fi
if [ -z "$KONOMI_CONFIG" ] || [ ! -f "$KONOMI_CONFIG" ]; then
KONOMI_CONFIG=$(find /opt /usr/local /home -name "config.yml" -path "*/KonomiTV/*" 2>/dev/null | head -1)
fi
if [ -z "$KONOMI_CONFIG" ]; then
echo "WARNING: KonomiTV の config.yaml が見つかりませんでした。"
echo "手動で設定ファイルを確認してください。"
else
echo "KonomiTV config.yaml を更新中: $KONOMI_CONFIG"
sudo sed -i 's/always_receive_tv_from_mirakurun: false/always_receive_tv_from_mirakurun: true/' "$KONOMI_CONFIG" || true
# backend を Mirakurun に変更(EDCB のパイプ通信が Linux で動作しないため)
sudo sed -i "s/backend: 'EDCB'/backend: 'Mirakurun'/" "$KONOMI_CONFIG" || true
sudo /usr/local/bin/pm2 restart KonomiTV 2>/dev/null || sudo /usr/local/bin/pm2 start KonomiTV 2>/dev/null || {
echo "WARNING: KonomiTV の再起動に失敗しました。手動で再起動してください。"
}
fi
fi
# --rescan-only の場合は KonomiTV も再起動してチャンネル一覧を再取得させる
if [ "$RESCAN_ONLY" = true ]; then
echo "KonomiTV を再起動してチャンネル一覧を再取得させます..."
sudo /usr/local/bin/pm2 restart KonomiTV 2>/dev/null || true
fi
# ============================================================
# 完了
# ============================================================
CONTAINER_NAME=$(hostname 2>/dev/null || echo "localhost")
KONOMI_URLS=()
KONOMI_URLS+=("https://my.local.konomi.tv:7000/|ローカルホスト")
while read -r addr iface; do
IP=$(echo "$addr" | cut -d/ -f1)
[ -z "$IP" ] || [ "$IP" = "127.0.0.1" ] && continue
IP_DASH=$(echo "$IP" | tr '.' '-')
KONOMI_URLS+=("https://${IP_DASH}.local.konomi.tv:7000/|${iface}")
done < <(ip -4 addr show | awk '/inet / {print $2, $NF}')
echo ""
echo "============================================================"
echo " セットアップ完了 (v16)"
echo "============================================================"
echo ""
echo " サービス:"
echo " mirakc: http://127.0.0.1:40772/"
echo " EDCB Web:"
echo " http://${CONTAINER_NAME}:5510/"
echo ""
echo " KonomiTV:"
for entry in "${KONOMI_URLS[@]}"; do
URL="${entry%%|*}"
LABEL="${entry##*|}"
printf " %-52s (%s)\n" "$URL" "$LABEL"
done
echo ""
echo " サービス操作:"
echo " mirakc 再起動: sudo systemctl restart mirakc"
echo " EDCB 再起動: sudo systemctl restart edcb"
echo " KonomiTV 再起動: sudo pm2 restart KonomiTV"
echo ""
echo " チャンネルが表示されない場合の再スキャン:"
echo " sudo bash $0 --rescan-only"
echo ""
echo " BCASキーの再設定:"
echo " sudo nano /usr/local/etc/bcas_keys"
echo " sudo systemctl restart mirakc"
echo ""
echo " 番組情報の更新 (視聴していない時に実行):"
echo " sudo bash ~/update-epg.sh"
echo ""
echo " EPG:"
echo " 確認: ls /var/local/edcb/Setting/EpgData"
echo " ログ: tail -f /var/log/dtv_epg_start.log"
echo " インストールログ: tail -f $LOG_FILE"
echo "============================================================"
インストール後のEPG取得状況確認など(コンテナ内)
番組表データを取得中なので、下記でEPG取得状況を確認。tmpファイルが無くなればEPGの取得完了。
ls /var/local/edcb/Setting/EpgData
1チューナー環境で先にKonomiTVの動作確認を行いたいなら、チューナーがEPG受信中なので下記でEDCBを再起動してからKonomiTVで動作確認を。LXDコンテナを再起動してもよいかも。
# EDCB再起動
systemctl restart edcb
# KonomiTV再起動
pm2 restart KonomiTV
# mirakc 再起動
systemctl restart mirakc
GPUパススルーする場合
セットアップ完了後、GPUをパスするーするならLXD-UIでGPUをコンテナにパススルーします。インテル環境の場合、この設定を行い、KonomiTVで利用するエンコーダを変更するだけでGPUを利用した視聴が可能でした。


チャンネルスキャンだけを実行する方法
コンテナ内でインストールスクリプトを保存し(例:setup.sh)、オプションを付けて実行します。
bash setup.sh --rescan-only
この方法でインストールしてPM2へのパスが正しく張れていなかった場合
このスクリプトではpm2をnpmのグローバルインストールから/usr/local/bin/pm2にシンボリックリンクしているのですが、古い/usr/local/bin/pm2が壊れたシンボリックリンク(自分自身を指している)になっているのが原因で正しく張れていなかったようです。なので、pm2 restart KonomiTVで再起動に失敗していました。
直接直すコマンドは下記。次回のセットアップスクリプトで修正しておきます。
と思いましたが、これは別件が原因だった可能性が高そうです。今さら確認しないので一応残しておきます。
sudo rm -f /usr/local/bin/pm2
sudo ln -s /usr/local/lib/node_modules/pm2/bin/pm2 /usr/local/bin/pm2
sudo chmod +x /usr/local/lib/node_modules/pm2/bin/pm2
sudo /usr/local/bin/pm2 restart KonomiTV


