スクリプトが必要:ビデオに最新のJPEGを追加してから古いJPEGを削除します。

スクリプトが必要:ビデオに最新のJPEGを追加してから古いJPEGを削除します。

私は見ている部屋の動きを検出するたびに、Arch LinuxボックスのFTPフォルダにJPEG画像をアップロードするIPカメラを持っています。すべてのモーションアクティビティが停止するまで、1秒間に1つのJPEGをアップロードします。

次の方法でJPEGファイルの名前を指定します。 ここに画像の説明を入力してください。

これを分析すると、次のことを意味します。

name-of-camera(inside)_1_YEAR-MONTH-DATE-TIME-SECONDS_imagenumber.jpg

1秒間に1フレームのビデオを作成できるスクリプトが必要ですがffmpeg(簡単なことはわかります)、互いに2秒以上離れていない画像でのみビデオを作成できるほどスマートでなければなりません。 JPEGの使用を削除してください。ネットワーク遅延とフレーム損失を防ぐために、「2秒間隔」と言います。

撮影後2秒以内に撮影されたすべての画像は、独自の動画にする必要があります。したがって、デフォルトでは、カメラが見るすべてのモーション「イベント」からビデオを生成できるはずです。

zoneminder私はプログラムがmotionこれが好きでできることを知っていますが、スクリプトを設計したいと思います。どんなアイデアにも感謝します。

答え1

日付に基づいてタイムスタンプを作成し、各ファイル間の範囲を確認できます。コメントですでに述べたように、1つの問題は夏時間です。日付/時刻がロケールによって異なると仮定します。

statファイル名の代わりにデフォルトとして使用すると便利です。ただし、このゲインはファイルのアップロード方法(タイムスタンプを保存するかどうかなど)によって異なります。

出発点として(予想よりはるかに長くかかるでしょう)、次のように試すことができます。

#!/bin/bash

declare -a fa_tmp=()    # Array holding tmp files with jpg collections.
declare dd=""           # Date extracted form file name.
declare -i ts=0         # Time stamp from date.
declare -i pre=0        # Previous time stamp.
declare -i lim=2        # Limit in seconds triggering new collection.
fmt_base='+%F-%H_%M_%S' # Format for date to generate video file name.

# Perhaps better using date from file-name:
# export TZ=UTC
# stat --printf=%Y $f

# Loop all jpg files
for f in *.jpg; do
    # Extract date, optionally use mktime() with gawk.
    # This assumes XX_XX_DATETIME_XXX... by split on underscore.
    dd=$(printf "$f" | tr '_' ' ' | awk '{
    printf("%d-%02d-%02d %02d:%02d:%02d",
        substr($3,  1, 4),
        substr($3,  5, 2),
        substr($3,  7, 2),
        substr($3,  9, 2),
        substr($3, 11, 2),
        substr($3, 13, 2))
    }')

    # Create time stamp from date.
    ts=$(date +%s -d "$dd")

    # If duration is greater then lim, create new tmp file.
    if ((ts - pre > lim)); then
        f_tmp="$(mktemp)"
        fa_tmp+=("$f_tmp")
        # First line in tmp file is first time stamp.
        printf "%s\n" "$ts" >> "$f_tmp"
    fi

    # Add filename to current tmp file.
    printf "%s\n" "$f" >> "$f_tmp"

    # Previous is current.
    pre="$ts"
done

declare -i i=1
# Loop tmp files.
for f_tmp in "${fa_tmp[@]}"; do
    printf "PROCESSING: %s\n---------------------------\n" "$f_tmp"
    base=""
    i=1

    # Rename files.
    while read -r img; do
        # First line is time stamp and is used as base for name.
        if [[ "$base" == "" ]]; then
            base=$(date "$fmt_base" -d "@$img")
            continue
        fi
        # New image name.
        iname=$(printf "%s-%04d.jpg" "$base" "$i")
        echo "mv '$img' => '$iname'"
        mv "$img" "$iname"
        ((++i))
    done <"$f_tmp"

    # Generate video.
    if ffmpeg -f image2 \
        -framerate 3 \
        -pattern_type sequence \
        -start_number 1 \
        -i "$base-%04d.jpg" \
        -vcodec mpeg4 \
        -r 6 \
        "$base.mp4"; then

        # Iff success, move jpg's to backup folder.
        mkdir "$base"
        mv $base-*.jpg "$base"
    else
        printf "FAILED:\n" >&2
        ls $base-*.jpg >&2
    fi

    # Remove tmp file.
    rm "$f_tmp"
done

関連情報