之前发过Armbian下合并小米摄像头的脚本和群晖NAS给小米摄像机视频合并教程,今天我们在飞牛上安装,方法都类似。
飞牛默认集成了Python和FFmpeg环境,
首先打开飞牛的SSH,登录 输入sudo-i切换root
sudo apt install python3 pip
继续
pip install loguru --break-system-packages
保存以下代码另存为main.py,然后上传到root根目录
import argparse import os import subprocess from pathlib import Path from loguru import logger import platform parser = argparse.ArgumentParser(description='合并米家摄像头视频,以天为单位。') parser.add_argument('indir', help='原米家摄像头视频目录。') parser.add_argument('--outdir', default='./', help='合并后视频存放目录,目录不存在会被创建。默认当前目录。') args = parser.parse_args() skip_filenames = [] def merge_vids(vidlist_file: str, tofile: str): """执行 ffmpeg 命令合并视频。""" # 需要对音频重新编码,否则会报错: # Could not find tag for codec pcm_alaw in stream #1, codec not currently supported in container when concatenating 2 files using ffmpeg # ffmpeg -y overwrite if platform.system().lower() == "windows": cmd = f"ffmpeg -f concat -safe 0 -i {vidlist_file} -c:v copy -c:a flac -strict -2 {tofile}" subprocess.run(cmd) else: cmd = f"ffmpeg -y -f concat -safe 0 -i {vidlist_file} -c:v copy -c:a aac -strict -2 {tofile}" subprocess.run(cmd,shell=True) def has_subdirectories(directory): subdirectories = [item for item in os.listdir(directory) if os.path.isdir(os.path.join(directory, item))] return bool(subdirectories) def merge_dirs(indir: str, outdir: str, date_name: str, parent_path: str): """合并目录下的监控文件,在当前目录生成以天为单位的视频。 indir 结构: indir 2021051001 2021051002 2021051003 ... 即,子目录结构为:年月日时。 """ if not Path(outdir).exists(): logger.info(f'{outdir} 不存在,即将被创建') Path(outdir).mkdir(parents=True) date_dict = {} if Path(indir).is_file(): print("indir is file:", indir) return # 小米第一代文件目录有多层 for d in Path(indir).iterdir(): if d.is_file(): # 兼容一级目录是视频文件 date_dict[date_name] = [Path(indir)] break if not d.is_dir(): continue date = d.stem[:8] if date not in date_dict: date_dict[date] = [] date_dict[date].append(d) for ds_date, ds in date_dict.items(): videos = [] for d in ds: mp4_list = list(Path(d).glob("*.mp4")) videos.extend(mp4_list) print("data_dict:",d,has_subdirectories(Path(d))) if len(videos) == 0 and Path(d).is_dir() and has_subdirectories(Path(d)): # 往下层递归 merge_dirs(Path(d), outdir, date_name, ds_date) logger.info(f"{ds_date}, {len(videos)} videos") if not videos: continue videos = sorted(videos, key=lambda f: int(f.stem.split("_")[-1])) videos = [ "file " + str(f.resolve(strict=True)).replace("\\", "/") for f in videos ] # 直接使用 outdir 作为输出目录 vidslist_path = f"{outdir}/vidslist.txt" Path(vidslist_path).write_text("\n".join(videos), encoding="utf8") merge_vids(vidslist_path, Path(outdir).joinpath(f"{ds_date}.mp4")) def startup(indir: str, outdir: str): for date in Path(indir).iterdir(): date_name = Path(date).name if date_name in skip_filenames: continue print(f"start merge:{date_name} video") merge_dirs(Path(date), outdir, date_name, "") if __name__ == "__main__": startup(args.indir, args.outdir)
复制到root目录方法很多,可以上传到飞牛文件管理里面,然后输入cp /vol1/1000/Work/main.py /root复制过去
或者上传到/tmp目录 再cp /tmp /root
执行命令
python3 main.py /vol1/1000/Work/1 --outdir /vol1/1000/Work/2
/vol1/1000/Work/1为摄像机存储的目录,
/vol1/1000/Work/2为合并后保存的目录。
可以加
nohup python3 main.py /vol1/1000/Work/1 --outdir /vol1/1000/Work/2 &
开头和&为后台自动运行合并命令,关掉ssh也不影响。
安装openlist,添加两个存储,一个是本地目录/vol1/1000/Work/1,然后再添加一个网盘,根据需求选择,已夸克网盘为例,合并完成后选择本地存储找到合并后的视频,右键复制到夸克网盘。
本文作者为dwf135,转载请注明。