现在NAS已经很普及了,NAS上经常会存储海量的高清电影,但有时候我们希望把电影格式转换成便携一些的格式,直接在服务器上转换会更加的高效,因此我便写了这一个工具,可以做到无人值守的批量转换整个目录下任意视频。
只需要把这个脚本放在需要转换视频的目录下,然后运行即可。如果需要压制字幕,字幕文件名需要跟视频文件名保持完全一致(大小写敏感)。字幕支持srt和ass,utf8编码。如果在压制带字幕的视频时出现红字错误,先检查字幕编码是否正确,通过命令行cat查看字幕文件,如果输出乱码,则需要先对字幕文件进行转码处理。
支持以下几种创建转换队列的方式:

  1. 通过输入文件编号,用逗号或者空格或者点隔开,如1,2 3.4,会选择1、2、3、4号文件加入转换队列
  2. 通过输入文件区间,用英文减号隔开,如1-4,会选择1、2、3、4文件加入转换队列

每次转换,整个队列的转换参数是一致的。如果需要不同的转换参数,请分多次运行本脚本。
如果需要转换的队列太大,建议直接实机运行或者通过VNC运行,如果通过ssh运行本脚本,ssh超时断开会导致任务中断。转换后的视频在原视频目录下的out目录中,格式为mp4。转换时在out目录下遇到同名文件会直接覆盖,不提示。
注意:本脚本可以在Linux和Mac上运行,python3.7以上版本,需要提前安装好ffmpeg,最好是编译安装,with-libx264, with-lamemp3,压制ass格式的外挂字幕的话,还需要with-libass
话不多说,直接上代码:(文末附原文件下载)

#!/usr/bin/env python3
# -*- coding: UTF-8 -*-
import subprocess
import os,sys
#指定当前.py脚本的绝对路径
current_path = sys.path[0]
#指定ffmpeg的路径:
ffmpeg_path = "/usr/local/ffmpeg/bin/ffmpeg"
print("\033[36m","="*40,"\033[0m")
print("\033[35m欢迎使用ffmpeg视频转码脚本 v1.2\033[0m")
print("\033[35mLast Update: 2021/12/21\033[0m")
print("\033[35mAuthor: 1Fan Tech\033[0m")
print("\033[35m将本脚本与需要转换的视频放在同一目录下\033[0m")
print("\033[35m转换后的mp4视频输出在out目录\033[0m")
print("\033[35m如果目标视频已存在会自动覆盖\033[0m")
print("\033[36m","="*40,"\033[0m")
print("\033[32m当前目录文件列表:\033[0m")

#输出当前目录文件列表
print("\033[32m",current_path,"\033[0m")
#判断文件是否为视频文件
def is_video(t_file):
    v_list = ["mp4","mkv","mpg","mpeg","webm","flv","avi","mov"]
    rst = False
    for v in v_list:
        if t_file.lower().endswith(v):
            rst = True
    return rst
# x - y 类型字符串转换为列表
def dash_into_list(t_str):
    out_list = []
    if "-" in t_str:
        t_list = t_str.replace(" ","").split("-")
        try:
            t_list.remove("")
        except:
            pass
        if len(t_list)==2:
            try:
                start = int(t_list[0])
                end = int(t_list[1])
                if start<end:
                    for x in range(start,end+1):
                        out_list.append(x)
            except:
                pass
    return out_list
file_index = 0
file_list = []
print("\033[34m",file_index,".\t\033[32m  退出\033[0m")
for tmp_file in os.listdir(current_path):
    if not os.path.isdir(tmp_file) and is_video(tmp_file):
        real_file = os.path.join(current_path, tmp_file)
        file_list.append(real_file)
        file_index+=1
        file_size = int(os.path.getsize(real_file)/10485.76)/100 #文件大小以两位小数显示Mb
        print("\033[34m",file_index,".\t\033[32m ",tmp_file,"\033[33m\t",file_size,"MB\033[0m")
if file_index>0:
    loop_indicator = 0
    new_error = 0
    while loop_indicator == 0:
        new_error = 0
        print("\033[34m请输入您要转换的文件序号(1 - ",file_index,")可选择多个文件以逗号或空格或点分隔,或者通过x-y批量选择:\033[0m", end="")
        nums = input("")
        selected_files_list = []
        if nums =="0":
            sys.exit(0)
        elif "-" in nums:
            selected_files_list = dash_into_list(nums)
            if selected_files_list == []:
                new_error = 1
        else:
            input_files_list = nums.replace(".",",").replace(" ",",").replace(",",",").replace("。",",").split(",")
            for num in input_files_list:
                if num !="":
                    try:
                        num = int(num)
                    except:
                        new_error = 1
                        num = file_index + 1
                    if num > file_index:
                        new_error = 1
                    else:
                        if num not in selected_files_list:selected_files_list.append(num)
        if new_error == 0:
            loop_indicator = 1
        else:
            print("\033[31m请输入正确的序号!\033[0m")
    print("已选择:")
    for num in selected_files_list:
        print("\033[34m",num,".\033[32m ",file_list[num-1],"\033[0m")
    #开始构建vf参数序列
    select_sub_ext = ""
    while select_sub_ext=="":
        print("\033[0;32;40m请选择输入的字幕文件后缀:\033[0m")
        print("0. 不附加字幕")
        print("1. ass")
        print("2. srt")
        a = input("输入数字:")
        if a == "1":
            select_sub_ext = ".ass"
        elif a=="2":
            select_sub_ext = ".srt"
        elif a=="0" or a =="":
            select_sub_ext = "no"
        else:
            print("\033[0;31;40m输入有误,请重新输入!\033[0m")
    scale = ""
    while scale=="":
        print("\033[32m请选择视频分辨率:\033[0m")
        print("0. 保留原始分辨率[直接回车]")
        print("1. 720p")
        print("2. 1080p")
        print("3. 4K")
        a = input("输入数字:")
        if a == "1":
            scale = "-1:720"
        elif a=="2":
            scale = "-1:1080"
        elif a=="3":
            scale = "-1:2160"
        elif a=="0" or a == "":
            scale = "no"
        else:
            print("\033[34m输入有误,请重新输入!\033[0m")
    #开始构建参数序列
    args_list = []
    #追加解码器
    args_list.append("-vcodec")
    args_list.append("libx264")
    args_list.append("-acodec")
    args_list.append("libfdk_aac")
    #追加视频音频码率
    bitrate = ""
    while bitrate=="":
        print("\033[32m请选择视频/音频质量:\033[0m")
        print("0. 保留原始质量[直接回车]")
        print("1. 高质量(10M/320k)")
        print("2. 中质量(5M/320k)")
        print("3. 低质量(2M/256k)")
        a = input("输入数字:")
        if a == "1":
            bitrate = "1"
            args_list.append("-b:v")
            args_list.append("10240k")
            args_list.append("-b:a")
            args_list.append("320k")
        elif a=="2":
            bitrate = "2"
            args_list.append("-b:v")
            args_list.append("5120k")
            args_list.append("-b:a")
            args_list.append("320k")
        elif a=="3":
            bitrate = "3"
            args_list.append("-b:v")
            args_list.append("2048k")
            args_list.append("-b:a")
            args_list.append("256k")
        elif a=="0" or a == "":
            bitrate = "no"
        else:
            print("\033[34m输入有误,请重新输入!\033[0m")
    print(args_list)
    start = ""
    for file_index in selected_files_list:
        file = file_list[file_index-1]
        #获取视频文件不带后缀的文件名
        videofilename = file[:file.rfind('.')]
        #获取不带路径不带后缀的文件名(输出时使用)
        pure_videofilename = videofilename[videofilename.rfind('/')+1:]
        #初始化执行列表
        exec_list = list([ffmpeg_path, "-y", "-i"])
        print("\033[32m发现视频文件: \033[35m",file,"\033[0m")
        #file_vid_list.append(file)
        if select_sub_ext != "no":
            sub_file = videofilename + select_sub_ext
            if os.path.isfile(sub_file):
                print("\033[32m发现字幕文件: \033[35m",sub_file,"\033[0m")
                #file_ass_list.append(sub_file)
                add_sub = select_sub_ext
            else:
                add_sub = "no"
                print("\033[34m未找到视频对应的字幕文件:\033[35m",sub_file,"\033[34m本视频不添加字幕!\033[0m")
        else:
            add_sub = "no"
        if start =="":
            #开始构建执行任务:
            #追加视频文件
            exec_list.append(file)
            #追加字幕及分辨率
            if add_sub!="no" or scale!="no":
                exec_list.append("-vf")
                if add_sub==".ass":
                    vf_args = "ass=" + sub_file
                    args_connector = "," #添加参数间隔符
                elif add_sub==".srt":
                    vf_args = "subtitles=" + sub_file
                    args_connector = "," #添加参数间隔符
                else:
                    vf_args = ""
                    args_connector = ""
                if scale != "no":
                    vf_args = vf_args + args_connector + "scale=" + scale 
                exec_list.append(vf_args)
            #追加之前用户收入的自定义参数
            exec_list.extend(args_list)
            # 输出文件名
            exec_list.append(current_path + "/out/" + pure_videofilename + ".mp4")
            #开始检查输出目录是否存在
            if not os.path.exists(os.path.join(current_path, "out")):
                os.makedirs(os.path.join(current_path, "out"))
            print("\033[32m开始执行转换命令:",exec_list,"\033[0m")
            #开始最终执行
            subprocess.call(exec_list)
else:
    sys.exit(0)

下载:videoconv.py