001-视频:m3u8多码率自适应

96
霄峰
2018.12.05 11:53 字数 554

多码率自适应转码:随着播放器(终端)网络带宽的变化动态选择适应的码率播放
视频平台:七牛云
说明:对已经上传到七牛云的视频转码成包含多种码率的HLS视频流。以便能随着终端网络带宽的变化动态选择适应的码率播放。

多码率自适应转码:https://developer.qiniu.com/dora/manual/1245/multiple-rate-adaptive-transcoding-adapt
普通音视频转码:https://developer.qiniu.com/dora/manual/1248/audio-and-video-transcoding-avthumb

情况说明:

  1. 七牛云上有一批视频(mp4格式)所在空间名为Abucket
  2. 现在要将这批视频转为流畅、标清、高清三种m3u8文件格式并且将转好的新文件存储到空间名为Bbucket
  3. 还要实现多码率自适应(随着播放器(终端)网络带宽的变化动态选择适应的码率播放
  4. 还有要满足用户可以手动选择是播放流畅、标清、高清

1. 前期准备:

        use Qiniu\Auth;
        use Qiniu\Processing\PersistentFop;

        //对已经上传到七牛的视频发起异步转码操作
        $accessKey = getenv('QINIU_ACCESS_KEY');
        $secretKey = getenv('QINIU_SECRET_KEY');
        $auth = new Auth($accessKey, $secretKey);

        //源文件bucket
        $bucket = getenv('QINIU_TEST_BUCKET');
        //目标文件bucket
        $video51XlBucket = getenv('Qiniu51xlVideoBucket'); 

2. 提取源文件名(支持url格式文件路径)、设置目标文件名

        // 源文件名
        $sourceKey = 'http://out-001.yunfeng365.com/test-001.mp4';

        // 目标文件名
        $targetKey = pathinfo($sourceKey, PATHINFO_FILENAME);

        //要转码的文件所在的空间和文件名。从url中提取
        $info = pathinfo(ltrim(parse_url($sourceKey)['path'],'/'));
        $dirname = ltrim($info['dirname'], '.');
        $dirname = $dirname ? $dirname . '/' : null;
        // 成功提取目标文件名
        $key = $dirname . $info['basename']; // test-001.mp4

3. 转码操作

        $config = new \Qiniu\Config();
        //$config->useHTTPS=true;
        $pfop = new PersistentFop($auth, $config);
        //要进行转码的转码操作。 http://developer.qiniu.com/docs/v6/api/reference/fop/av/avthumb.html
        $saves480p = \Qiniu\base64_urlSafeEncode($video51XlBucket . ":480p/{$targetKey}.m3u8");
        $saves360p = \Qiniu\base64_urlSafeEncode($video51XlBucket . ":360p/{$targetKey}.m3u8");
        $saves180p = \Qiniu\base64_urlSafeEncode($video51XlBucket . ":180p/{$targetKey}.m3u8");
        $fops = [
            "avthumb/m3u8/noDomain/1/segtime/15/r/30/vb/780k/s/865x480|saveas/{$saves480p}",
            "avthumb/m3u8/noDomain/1/segtime/15/r/30/vb/380k/s/640x360|saveas/{$saves360p}",
            "avthumb/m3u8/noDomain/1/segtime/15/r/25/vb/200k/s/320x180|saveas/{$saves180p}",
        ];

        // 转码另存
        list($id, $err) = $pfop->execute($bucket, $key, $fops, $pipeline);
        if (!empty($err)) return ['status' => 500, 'err' => $err];

为什么使用m3u8文件

m3u8分为两种文件,一种是顶级m3u8文件,一种是二级m3u8文件
顶级m3u8文件来存放自适应的二级m3u8文件
二级m3u8文件来存放ts切片文件
客户端在播放顶级m3u8文件时,会选择码率高的流,当码流达不到时会请求码率低的流

这里要注意下:

  1. 七牛云同一个地区可以转m3u8并且将m3u8文件 、 TS切片文件一起另存到其它空间
  2. 七牛云多个操作的pfop使用数组,如本文一次将三种格式都转码
  3. 这一步是生成二级m3u8文件、ts切片文件
  4. 转码操作详细手册http://developer.qiniu.com/docs/v6/api/reference/fop/av/avthumb.html

4.多码率自适应

        // 1. 多码率自适应 https://developer.qiniu.com/dora/manual/1245/multiple-rate-adaptive-transcoding-adapt
//        $saves = \Qiniu\base64_urlSafeEncode($video51XlBucket . ":{$targetKey}.m3u8");
//        $fops = [
//            "adapt/m3u8/envBandWidth/200000,400000,800000/multiVb/200k,380k,780k/multiResolution/320:180,640:360,865:480/multiPrefix/{$targetKey}/hlstime/15"
//        ];

        // 2. 手动多码率自适应
        // 生成上传Token
        $token = self::$auth->uploadToken($video51XlBucket);
        $uploader = new UploadManager();

        $data = <<<EOF
#EXTM3U
#EXT-X-STREAM-INF:BANDWIDTH=200000
/180p/{$targetKey}.m3u8
#EXT-X-STREAM-INF:BANDWIDTH=400000
/360p/{$targetKey}.m3u8
#EXT-X-STREAM-INF:BANDWIDTH=800000
/480p/{$targetKey}.m3u8
EOF;
;
        list($ret, $err) = $uploader->put($token, "auto/{$targetKey}.m3u8", $data);
        if (!empty($err)) return ['status' => 500, 'err' => $err];
  1. 这一步是生成并上传顶级m3u8文件,其内容是上一步的二级m3u8文件

5. 结果处理

        $url = env('QiniuBucketFor51xlDomain', 'http://51xl-2.video.51nst.net/');
        return [
            'status' => 200,
            'data' => [
                'url' => $url,
                'auto' => $ret['key'],
                '180p' => "180p/{$targetKey}.m3u8",
                '360p' => "360p/{$targetKey}.m3u8",
                '480p' => "480p/{$targetKey}.m3u8",
            ]
        ];

视频处理