码迷,mamicode.com
首页 > 其他好文 > 详细

去掉m3u8的片头和片尾

时间:2018-07-22 18:42:07      阅读:1022      评论:0      收藏:0      [点我收藏+]

标签:datetime   shel   line   ons   sum   pat   内网   sts   htm   

# pip3 install -i https://mirrors.aliyun.com/pypi/simple/ m3u8
# pip3 install -i https://mirrors.aliyun.com/pypi/simple/ oss2
# pip3 install --upgrade pip

# 在Centos中安装ffmpeg和aria2c
# https://www.cnblogs.com/littlehb/p/9347978.html

import m3u8, sys
import os, subprocess
import oss2, requests, datetime

# ========================================================================
# 前面需要截取的时间长度,单位:秒
prefixCutSeconds = 10
# 后面需要截取的时间长度,单位:秒
suffixCutSeconds = 5
# 上传的路径前缀
prefix = down/M3u8_ZhengZhou103/

# ========================================================================
# 访问oss的用户名与密码,无需修改
access_key_id = xxxxxxxxxxxx
access_key_secret = xxxxxxxxxxxx
bucket_name = dsideal-yy


# 检查是内网还是外网
def CheckInOut():
    # 探测内网外网
    print(正在探测使用环境是阿里云内网还是外网,请稍等...)
    url = http:// + bucket_name + .oss-cn-qingdao-internal.aliyuncs.com/down/Material/42/42385DBE-E03F-FF6A-A37C-CC8A04612BE4.doc
    try:
        requests.get(url, timeout=1)
        endpoint = http://oss-cn-qingdao-internal.aliyuncs.com/
        print(内部网络访问,将使用endpoint: + endpoint)
    except Exception as err:
        endpoint = http://oss-cn-qingdao.aliyuncs.com/
        print(外网访问,将使用endpoint: + endpoint)
    return endpoint


# 从哪个结点下进行操作
endpoint = CheckInOut()

# 从哪里下载回来,如果是内网,应该写成 http://dsideal_yy.oss-cn-qingdao-internal.aliyuncs.com
if internal in endpoint:
    downloadPrefixUrl = http:// + bucket_name + .oss-cn-qingdao-internal.aliyuncs.com/down/M3u8/
else:
    downloadPrefixUrl = http:// + bucket_name + .oss-cn-qingdao.aliyuncs.com/down/M3u8/


# 获取视频的时间长度
def GetVideoLength(fileName):
    cmd = "ffmpeg -i " + fileName + " 2>&1 | grep ‘Duration‘ | cut -d ‘ ‘ -f 4 | sed s/,//"
    p = subprocess.Popen(cmd, shell=True, stdout=subprocess.PIPE)
    out, err = p.communicate()
    for line in out.splitlines():
        return (line.decode()[0:8])


# 将时间换算成秒数
def t2s(t):
    h, m, s = t.strip().split(":")
    return int(h) * 3600 + int(m) * 60 + int(s)


# 下载m3u8的所有ts文件
def DoActionM3u8(m3u8_url):
    # m3u8文件的真实名称
    m3u8FileName = WorkingPath + "/" + m3u8_url.split(/)[-1]
    print(M3u8真实文件名称: + m3u8FileName)
    # 构造两个文本文件
    m3u8List = []
    m3u8FileList = []
    m3u8_obj = m3u8.load(m3u8_url)
    for l in m3u8_obj.segments:
        m3u8List.append(downloadPrefixUrl + l.uri[0:2] + / + l.uri)
        m3u8FileList.append("file ‘" + l.uri + "")

    # 形成文本文件,这个用于调用aria2c进行批量下载使用
    tempFile = WorkingPath + /url.txt
    result = map(lambda x: x.strip() + \n, m3u8List)
    with open(tempFile, w) as f:
        f.writelines(result)

    # 下载回来转码后的视频ts
    cmd = aria2c -c -s 4 -d  + WorkingPath +  -j 8 -i  + tempFile
    os.system(cmd)

    # 形成文本文件,用于将ts文件拼接成大的ts文件时使用
    tempFile = WorkingPath + /mylist.txt
    result = map(lambda x: x.strip() + \n, m3u8FileList)
    with open(tempFile, w) as f:
        f.writelines(result)

    # 使用ffmpeg 拼成大的ts
    tempAllTs = WorkingPath + /output.ts
    if os.path.exists(tempAllTs):
        os.remove(tempAllTs)

    tempAllTs_new = WorkingPath + /output_new.ts
    if os.path.exists(tempAllTs_new):
        os.remove(tempAllTs_new)

    tempAllMp4 = WorkingPath + /output.mp4
    if os.path.exists(tempAllMp4):
        os.remove(tempAllMp4)

    tempAllMp4_new = WorkingPath + /output_new.mp4
    if os.path.exists(tempAllMp4_new):
        os.remove(tempAllMp4_new)

    # 拼接为原始的ts
    cmd = ffmpeg -f concat -i  + tempFile +  -c copy  + tempAllTs +  -y
    os.system(cmd)
    #
    # 将大的ts转为mp4
    cmd = ffmpeg -i  + tempAllTs +  -c:v copy -c:a copy -bsf:a aac_adtstoasc  + tempAllMp4
    os.system(cmd)
    os.remove(tempAllTs)
    # 以下为new的步骤=========================================================================
    #
    #  截取一部分
    # (1) 测出文件有多长
    videoLength = GetVideoLength(tempAllMp4)
    # 去掉后suffixCutSeconds秒
    seconds = t2s(videoLength) - suffixCutSeconds
    m, s = divmod(seconds, 60)
    h, m = divmod(m, 60)
    endTime = "%02d:%02d:%02d" % (h, m, s)

    # 开始的秒数
    m, s = divmod(prefixCutSeconds, 60)
    h, m = divmod(m, 60)
    startTime = "%02d:%02d:%02d" % (h, m, s)
    # #
    # # (2)去头部prefixCutSeconds秒,尾部suffixCutSeconds秒
    print(截取并生成mp4!)
    cmd = ffmpeg  -i  + tempAllMp4 +  -vcodec copy -acodec copy -ss  + startTime +  -to  + endTime +   + tempAllMp4_new +  -y
    os.system(cmd)
    os.remove(tempAllMp4)
    print(生成新mp4成功!到这里都是非常正常的!!!!!)

    # (3) mp4 转 ts
    cmd = ffmpeg -y -i  + tempAllMp4_new +  -vcodec copy -acodec copy -vbsf h264_mp4toannexb  + tempAllTs_new
    os.system(cmd)
    os.remove(tempAllMp4_new)

    # (4) 切割ts
    prefixName = m3u8_url.split(/)[-1][0:36]
    cmd = ffmpeg -i  + tempAllTs_new +  -c copy -map 0 -f segment -segment_list  + WorkingPath + / + prefixName + .m3u8 -segment_time 10  + WorkingPath + / + prefixName + %03d.ts
    os.system(cmd)
    os.remove(tempAllTs_new)

    # (5)上传到oss,放到另一个目录下
    path = os.listdir(os.getcwd()+/M3u8)
    for filename in path:
        if os.path.isfile(WorkingPath+/+filename):
            if .m3u8 in filename or .ts in filename:
                logInfo(开始进行文件上传: + filename)
                key = prefix +  filename[0:2] + / + filename
                bucket.put_object_from_file(key, WorkingPath + / + filename, progress_callback=percentage)
                print(文件上传云存储成功完成! + filename)


# 黄海定义的输出信息的办法,带当前时间
def logInfo(msg):
    i = datetime.datetime.now()
    print(" %s            %s" % (i, msg))


# 进度条功能
def percentage(consumed_bytes, total_bytes):
    if total_bytes:
        per = int(100 * (float(consumed_bytes)) / (float(total_bytes)))
        s1 = "\r[%s%s]%d%%" % ("=" * int(per), " " * (100 - int(per)), per)
        sys.stdout.write(s1)
        sys.stdout.flush()


if __name__ == __main__:
    # 删除临时目录并重新创建
    WorkingPath = /usr/local/software/TestM3u8/M3u8
    os.system(rm -rf  + WorkingPath)
    os.mkdir(WorkingPath)

    # 构建OSS存储对象
    auth = oss2.Auth(access_key_id, access_key_secret)
    # 阿里云上应该用这个内部网络选项
    bucket = oss2.Bucket(auth, endpoint, bucket_name)

    # 以一个名师云课为例
    m3u8_url = downloadPrefixUrl+C4/C4CD770E-C98A-AF8C-8B53-8541210B355B.m3u8
    print(m3u8_url)
    DoActionM3u8(m3u8_url)
# http://video.edusoa.com/down/M3u8_ZhengZhou103/C4/C4CD770E-C98A-AF8C-8B53-8541210B355B.m3u8
# http://dsideal-yy.oss-cn-qingdao.aliyuncs.com/down/M3u8_ZhengZhou103/C4/C4CD770E-C98A-AF8C-8B53-8541210B355B.m3u8
#

 

去掉m3u8的片头和片尾

标签:datetime   shel   line   ons   sum   pat   内网   sts   htm   

原文地址:https://www.cnblogs.com/littlehb/p/9350899.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!