目录
在云考试或视频面试中,除了对考生、考官的实时音视频监控以防止作弊行为的发生以外,对直播流的音视频录制也尤为重要,可做为后期证据材料进行追溯、举证。
在实际的应用场景中,会有多路直播流的产生,因此根据业务需要可以将多路直播流混合录制成一个视频文件,腾讯云称其为云端混录。混录后的视频可以更加直观的进行回放,可以同时查看多路直播流的视频情况。
混录场景举例:
场景1:在线考试回放,三路混流。主图像显示考生面部及背后方视频、副图1显示考生正前方视频、副图2显示屏幕共享视频。
场景2:一对一视频面试,两路混流。主图显示考生答题情况视频、副图1显示考官提问情况视频。
云端混流涉及腾讯云直播和云点播服务。
腾讯云直播(Cloud Streaming Services,CSS)提供极速、稳定、专业的云端直播处理服务,根据业务的不同直播场景需求,提供标准直播、快直播、云导播台三种服务,分别针对大规模实时观看、超低延时直播、便捷云端导播的场景,配合腾讯云视立方·直播 SDK,提供一站式的音视频直播解决方案。具体可访问该网址进行了解:https://cloud.tencent.com/product/css
腾讯云点播(VOD)面向音视频、图片等媒体,提供制作上传、存储、转码、媒体处理、媒体 AI、加速分发播放、版权保护等一体化的高品质媒体服务。具体可访问该网址进行了解:https://cloud.tencent.com/product/vod
下图是我们基于腾讯云产品架构图的部分采用和实现方案:
我们的混流设计输出如下图演示:
副图1显示在右上方,如果有副图2则依次向下排列。
关于更多布局设计和产品文档请参考腾讯云产品网址进行了解:https://www.tencentcloud.com/zh/document/product/267/37665
- //混录方法,参数 mixtype,默认为混录,填写cancel为申请取消,roomid为直播房间号,userid1 为主图直播流名称,userid2 为副图直播流名称,videosize为视频分辨率,如“720p"
- public string MixRecord(string mixtype,string roomid,string userid1,string userid2,string videosize)
- {
- roomid = int.Parse(roomid).ToString(); //对roomid进行一次转int的特殊处理
- //计算主图尺寸
- int vw = 640;
- int vh = 480;
- switch (videosize)
- {
- case "240p":
- vw=320;vh=240;break;
- case "360p":
- vw=640;vh=360;break;
- case "480p":
- vw=640;vh=480;break;
- case "720p":
- vw=1280;vh=720;break;
- case "1080p":
- vw=1920;vh=1080;break;
- case "1440p":
- vw=2560;vh=1440;break;
- case "4K":
- vw=3840;vh=2160;break;
-
- }
- //计算副图尺寸
- int svw = vw / 4;
- int svh = vh / 4;
- switch (videosize)
- {
- case "240p":
- svh=60; break;
- case "360p":
- svh=90; break;
- case "480p":
- svh=120; break;
- case "720p":
- svh=180; break;
- case "1080p":
- svh=270; break;
- case "1440p":
- svh=360; break;
- case "4K":
- svh = 540; break;
-
- }
- //提供在腾讯云申请的开发帐号及开发KEY等
- string _appid = "";
- string _sdkappid = "";
- string _key = "";
- //调用腾讯云混流API
- string _interface = "Mix_StreamV2";
- MD5 md5 = new MD5();
- string _t=getTimestamp(60); //加偏移量60秒
- string _sign = md5.GetMD5Hash(_key+_t).ToLower(); //计算MD5
-
- var url = string.Format("http://fcgi.video.qcloud.com/common_access?appid={0}&interface={1}&t={2}&sign={3}",_appid,_interface,_t,_sign);
-
- string _timestamp = getTimestamp(0);
- string _eventid = getTimestamp(0);
- string _app_id = _appid;
- string mix_stream_session_id = roomid;
- string output_stream_id = _sdkappid + "_" + roomid + "_" + userid1 + "_main";
- string input_stream_id1 = _sdkappid + "_" + roomid + "_" + userid1 + "_main";
- string input_stream_id2 = _sdkappid + "_" + roomid + "_" + userid2 + "_main";
-
- var postData = "{\"timestamp\":" + _timestamp + ",\"eventId\":" + _eventid + ",\"interface\":{\"interfaceName\":\"Mix_StreamV2\",\"para\":{\"app_id\":\"" + _app_id + "\",\"interface\": \"mix_streamv2.start_mix_stream_advanced\",\"mix_stream_session_id\" : \"" + mix_stream_session_id + "\",\"output_stream_id\": \"" + output_stream_id + "\",\"input_stream_list\":[{\"input_stream_id\":\"" + input_stream_id1 + "\",\"layout_params\":{\"image_layer\":1}},{\"input_stream_id\":\"" + input_stream_id2 + "\",\"layout_params\":{\"image_layer\": 2,\"image_width\": "+svw+",\"image_height\": "+svh+",\"location_x\": "+(vw-svw-20).ToString()+",\"location_y\": 20}}]}}}";
-
- if (mixtype == "cancel")
- {
- postData = "{\"timestamp\":"+_timestamp+",\"eventId\":"+_eventid+",\"interface\":{\"interfaceName\":\"Mix_StreamV2\",\"para\":{\"app_id\":\""+_app_id+"\",\"interface\": \"mix_streamv2.cancel_mix_stream\",\"mix_stream_session_id\" : \""+mix_stream_session_id+"\",\"output_stream_id\": \""+output_stream_id+"\"}}}";
- }
- System.Net.HttpWebRequest request;
- request = (System.Net.HttpWebRequest)WebRequest.Create(url);
- request.Method = "POST";
- request.ContentType = "application/json;charset=UTF-8";
- byte[] payload;
- payload = System.Text.Encoding.UTF8.GetBytes(postData);
- request.ContentLength = payload.Length;
- try
- {
- Stream writer = request.GetRequestStream();
- writer.Write(payload, 0, payload.Length);
- writer.Close();
- System.Net.HttpWebResponse response;
- response = (System.Net.HttpWebResponse)request.GetResponse();
- System.IO.Stream stream;
- stream = response.GetResponseStream();
- List<byte> bytes = new List<byte>();
- int temp = stream.ReadByte();
- while (temp != -1)
- {
- bytes.Add((byte)temp);
- temp = stream.ReadByte();
- }
- byte[] result = bytes.ToArray();
-
- return System.Text.Encoding.Default.GetString(result);
- }
- catch (Exception ee)
- {
- return "{\"errcode\":2,\"errmsg\":\"" + ee.Message + "\"}";
-
- }
- }//request mix
-
- public class MD5
- {
- public string GetMD5Hash(string str)
- {
- //就是比string往后一直加要好的优化容器
- StringBuilder sb = new StringBuilder();
- using (MD5CryptoServiceProvider md5 = new MD5CryptoServiceProvider())
- {
- //将输入字符串转换为字节数组并计算哈希。
- byte[] data = md5.ComputeHash(Encoding.UTF8.GetBytes(str));
-
- //X为 十六进制 X都是大写 x都为小写
- //2为 每次都是两位数
- //假设有两个数10和26,正常情况十六进制显示0xA、0x1A,这样看起来不整齐,为了好看,可以指定"X2",这样显示出来就是:0x0A、0x1A。
- //遍历哈希数据的每个字节
- //并将每个字符串格式化为十六进制字符串。
- int length = data.Length;
- for (int i = 0; i < length; i++)
- sb.Append(data[i].ToString("X2"));
-
- }
- return sb.ToString();
- }
- }
- public string getTimestamp(int seconds)
- {
- TimeSpan ts = DateTime.UtcNow - new DateTime(1970, 1, 1, 0, 0, 0, 0);
- return Convert.ToInt64(ts.TotalSeconds+seconds).ToString();
- }
以上提供的代码仅供参考,在实际的应用中,我们要编写符合自己业务的逻辑,比如多路混流,还要考虑实际的运营成本,比如录制费用、存储费用等。有关腾讯云点播产品的价格情况,可以访问:https://cloud.tencent.com/act/pro/vod
云端混录在直播时进行合成,腾讯的建议是延迟一段时间再进行API申请,我在这里设置为5秒以后再申请。
为防止混录失败,我们可以在腾讯云直播管理后台,设置自动生成各路直播流的录制,以做为素材备用(会产生存储费用和录制费用),后期可以下载视频进行再合成。
以上就是自己的一些分享,时间仓促,不妥之处还请大家批评指正!