基於H5的攝像頭視頻數據流採集

最近,爲了支持部門團隊的項目,經過H5實現攝像頭的視頻流數據的捕獲,抓取到視頻流後,傳輸到視頻識別服務器進行後續的邏輯處理。css

 

視頻數據的採集過程,實際上是比較沒有譜的過程,由於以前沒有研究過HTML5操控攝像頭並取視頻流。html

研究了下網絡上的所謂的經驗帖子,大都說基於WebRTC的方案,沒有錯,可是也不對,咱們這裏涉及到的技術,確切的說是基於H5的navigator以及MediaRecorder API實現,輔助的工具是FileReader以及Blob。
參考的資料:(相關的內容,在這裏就不詳細描述)
navigator的getUserMediahttps://developer.mozilla.org/en-US/docs/Web/API/Navigator/getUserMedia
MediaRecorderhttps://developer.mozilla.org/en-US/docs/Web/API/MediaStream_Recording_API
FileReaderhttps://developer.mozilla.org/en-US/docs/Web/API/FileReader
Blobhttps://developer.mozilla.org/en-US/docs/Web/API/Blob前端

 

這個任務的實現邏輯,前端搭建一個Java的小Web應用,H5視頻採集以後,經過WebSocket的方式,將視頻流數據傳遞到Java的web小應用後臺,而後從後臺向視頻識別服務器經過UDP傳遞視頻數據。基本的架構以下圖:java

 

說明一下,本博文,視頻採集的部分,參考了一個老外的帖子,從他的帖子,改造後,獲得咱們的項目須要的效果。參考的帖子地址:https://addpipe.com/blog/mediarecorder-api/jquery

 

接下來,上前端頁面以及代碼. 大致說下,個人軟件架構,jersey2 + freemarker + springgit

前端頁面:github

<!DOCTYPE html>
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<meta name="viewport" content="width=device-width, minimum-scale=1.0, initial-scale=1.0, user-scalable=yes">
<meta name="mobile-web-app-capable" content="yes">
<meta id="theme-color" name="theme-color" content="#fff">
<base target="_blank">
<title>Media Recorder API Demo</title>
<link rel="stylesheet" href="${basePath}/css/video/main.css" />         #basePath是Web項目的根地址,例如 http://10.90.9.20:9080/RDConsumer
<style>
a#downloadLink {
    display: block;
    margin: 0 0 1em 0;
    min-height: 1.2em;
}
p#data {
    min-height: 6em;
}
</style>
</head>
<body>
<div id="container">
<div style = "text-align:center;">
    <h1>Media Recorder API Demo </h1>
    <h2>Record a 640x480 video using the media recorder API implemented in Firefox and Chrome</h2>
    <video controls autoplay></video><br>
    <button id="rec" onclick="onBtnRecordClicked()">Record</button>
    <button id="pauseRes"   onclick="onPauseResumeClicked()" disabled>Pause</button>
    <button id="stop"  onclick="onBtnStopClicked()" disabled>Stop</button>
 </div>
<a id="downloadLink" download="mediarecorder.webm" name="mediarecorder.webm" href></a>
<p id="data"></p>
<script src="${basePath}/js/jquery-1.11.1.min.js"></script>
<script src="${basePath}/js/video/main.js"></script>
<h2>Works on:</h2>
<p><ul><li>Firefox 30 and up</li><li>Chrome 47,48 (video only, enable <em>experimental Web Platform features</em> at  <a href="chrome://flags/#enable-experimental-web-platform-features">chrome://flags</a>)</li><li>Chrome 49+</li></ul></p>
<h2>
<span style="color:red">Issues:</span>
<p><ul><li>Pause does not stop audio recording on Chrome 49,50</li></ul></p>
<h2>Containers &amp; codecs:</h2>
<p><table style="width:100%">
    <thead>
    <tr>
        <th>&nbsp;</th><th>Chrome 47</th><th>Chrome 48</th><th>Chrome 49+</th><th>Chrome 52+</th><th>Firefox 30+</th>
    </tr>
    </thead>
    <tbody>
    <tr>
        <td><strong>Container</strong></td><td>webm</td><td>webm</td><td>webm</td><td>webm</td><td>webm</td>
    </tr>
    <tr>
        <td><strong>Video</strong></td><td>VP8</td><td>VP8</td><td>VP8/VP9</td><td>VP8/VP9/H264</td><td>VP8</td>
    </tr>
    <tr>
        <td><strong>Audio</strong></td><td>none</td><td>none</td><td>Opus @ 48kHz</td><td>Opus @ 48kHz</td><td>Vorbis @ 44.1 kHz</td>
    </tr>
    </tbody>
    </table>
</p>
<h2>Links:</h2>
<p>
    <ul>
    <li>Article: <a target="_blank" href="https://addpipe.com/blog/mediarecorder-api/">https://addpipe.com/blog/mediarecorder-api/</a></li>
    <li>GitHub: <a target="_blank" href="https://github.com/addpipe/Media-Recorder-API-Demo">https://github.com/addpipe/Media-Recorder-API-Demo</a></li>
    <li>W3C Draft: <a target="_blank"  href="http://w3c.github.io/mediacapture-record/MediaRecorder.html">http://w3c.github.io/mediacapture-record/MediaRecorder.html</a></li>
    <li>Media Recorder API at 65% penetration thanks to Chrome: <a target="_blank" href="https://addpipe.com/blog/media-recorder-api-is-now-supported-by-65-of-all-desktop-internet-users/">https://addpipe.com/blog/media-recorder-api-is-now-supported-by-65-of-all-desktop-internet-users/</a></li>
    </ul>
</p>
</div>
</body>
</html>

前端界面的效果圖:web

 

JS的代碼(重點之一在這個JS裏面的紅色部分,下面代碼是main.js的正文內容):spring

'use strict';

/* globals MediaRecorder */

// Spec is at http://dvcs.w3.org/hg/dap/raw-file/tip/media-stream-capture/RecordingProposal.html

navigator.getUserMedia = navigator.getUserMedia || navigator.webkitGetUserMedia || navigator.mozGetUserMedia || navigator.msGetUserMedia;


if(getBrowser() == "Chrome"){
    var constraints = {"audio": true, "video": {  "mandatory": {  "minWidth": 640,  "maxWidth": 640, "minHeight": 480,"maxHeight": 480 }, "optional": [] } };//Chrome
}else if(getBrowser() == "Firefox"){
    var constraints = {audio: false, video: { width: { min: 640, ideal: 640, max: 640 }, height: { min: 480, ideal: 480, max: 480 }}}; //Firefox
}

var recBtn = document.querySelector('button#rec');
var pauseResBtn = document.querySelector('button#pauseRes');
var stopBtn = document.querySelector('button#stop');

var videoElement = document.querySelector('video');
var dataElement = document.querySelector('#data');
var downloadLink = document.querySelector('a#downloadLink');

videoElement.controls = false;

function errorCallback(error){
    console.log('navigator.getUserMedia error: ', error);    
}

/*
var mediaSource = new MediaSource();
mediaSource.addEventListener('sourceopen', handleSourceOpen, false);
var sourceBuffer;
*/

var mediaRecorder;
var chunks = [];
var count = 0;

var wsurl = "ws://10.90.9.20:9080/RDConsumer/websocket"
var ws = null;
function createWs(){
var url = wsurl; if ('WebSocket' in window) { ws = new WebSocket(url); } else if ('MozWebSocket' in window) { ws = new MozWebSocket(url); } else { console.log("您的瀏覽器不支持WebSocket。"); return ; } } function init() { if (ws != null) { console.log("現已鏈接"); return ; } createWs(); ws.onopen = function() { //設置發信息送類型爲:ArrayBuffer ws.binaryType = "arraybuffer"; } ws.onmessage = function(e) { console.log(e.data.toString()); } ws.onclose = function(e) { console.log("onclose: closed"); ws = null; createWs(); //這個函數在這裏之因此再次調用,是爲了解決視頻傳輸的過程當中突發的鏈接斷開問題。 } ws.onerror = function(e) { console.log("onerror: error"); ws = null; createWs(); //同上面的解釋 } } $(document).ready(function(){ init(); }) function startRecording(stream) { log('Start recording...'); if (typeof MediaRecorder.isTypeSupported == 'function'){ /* MediaRecorder.isTypeSupported is a function announced in https://developers.google.com/web/updates/2016/01/mediarecorder and later introduced in the MediaRecorder API spec http://www.w3.org/TR/mediastream-recording/ */
//這裏涉及到視頻的容器以及編解碼參數,這個與瀏覽器有密切的關係
if (MediaRecorder.isTypeSupported('video/webm;codecs=vp9')) { var options = {mimeType: 'video/webm;codecs=h264'}; } else if (MediaRecorder.isTypeSupported('video/webm;codecs=h264')) { var options = {mimeType: 'video/webm;codecs=h264'}; } else if (MediaRecorder.isTypeSupported('video/webm;codecs=vp8')) { var options = {mimeType: 'video/webm;codecs=vp8'}; } log('Using '+options.mimeType); mediaRecorder = new MediaRecorder(stream, options); }else{ log('isTypeSupported is not supported, using default codecs for browser'); mediaRecorder = new MediaRecorder(stream); } pauseResBtn.textContent = "Pause"; mediaRecorder.start(10); var url = window.URL || window.webkitURL; videoElement.src = url ? url.createObjectURL(stream) : stream; videoElement.play();
//這個地方,是視頻數據捕獲好了後,會觸發MediaRecorder一個dataavailable的Event,在這裏作視頻數據的採集工做,主要是基於Blob進行轉寫,利用FileReader進行讀取。FileReader必定
//要註冊loadend的監聽器,或者寫onload的函數。在loadend的監聽函數裏面,進行格式轉換,方便websocket進行數據傳輸,由於websocket的數據類型支持blob以及arrayBuffer,咱們這裏用
//的是arrayBuffer,因此,將視頻數據的Blob轉寫爲Unit8Buffer,便於websocket的後臺服務用ByteBuffer接收。 mediaRecorder.ondataavailable
= function(e) { //log('Data available...'); //console.log(e.data); //console.log(e.data.type); //console.log(e); chunks.push(e.data); var reader = new FileReader(); reader.addEventListener("loadend", function() { //reader.result是一個含有視頻數據流的Blob對象 var buf = new Uint8Array(reader.result); console.log(reader.result); if(reader.result.byteLength > 0){ //加這個判斷,是由於有不少數據是空的,這個沒有必要發到後臺服務器,減輕網絡開銷,提高性能吧。 ws.send(buf); } }); reader.readAsArrayBuffer(e.data); }; mediaRecorder.onerror = function(e){ log('Error: ' + e); }; mediaRecorder.onstart = function(){ log('Started & state = ' + mediaRecorder.state); }; mediaRecorder.onstop = function(){ log('Stopped & state = ' + mediaRecorder.state); var blob = new Blob(chunks, {type: "video/webm"}); chunks = []; var videoURL = window.URL.createObjectURL(blob); downloadLink.href = videoURL; videoElement.src = videoURL; downloadLink.innerHTML = 'Download video file'; var rand = Math.floor((Math.random() * 10000000)); var name = "video_"+rand+".webm" ; downloadLink.setAttribute( "download", name); downloadLink.setAttribute( "name", name); }; mediaRecorder.onpause = function(){ log('Paused & state = ' + mediaRecorder.state); } mediaRecorder.onresume = function(){ log('Resumed & state = ' + mediaRecorder.state); } mediaRecorder.onwarning = function(e){ log('Warning: ' + e); }; } //function handleSourceOpen(event) { // console.log('MediaSource opened'); // sourceBuffer = mediaSource.addSourceBuffer('video/webm; codecs="vp9"'); // console.log('Source buffer: ', sourceBuffer); //}

//點擊按鈕,啓動視頻流的採集。重點是getUserMedia函數使用。本案例中,視頻採集的入口,是點擊頁面上的record按鈕,也就是下面這個函數的邏輯。 function onBtnRecordClicked (){ if (typeof MediaRecorder === 'undefined' || !navigator.getUserMedia) { alert('MediaRecorder not supported on your browser, use Firefox 30 or Chrome 49 instead.'); }else { navigator.getUserMedia(constraints, startRecording, errorCallback); recBtn.disabled = true; pauseResBtn.disabled = false; stopBtn.disabled = false; } } function onBtnStopClicked(){ mediaRecorder.stop(); videoElement.controls = true; recBtn.disabled = false; pauseResBtn.disabled = true; stopBtn.disabled = true; } function onPauseResumeClicked(){ if(pauseResBtn.textContent === "Pause"){ console.log("pause"); pauseResBtn.textContent = "Resume"; mediaRecorder.pause(); stopBtn.disabled = true; }else{ console.log("resume"); pauseResBtn.textContent = "Pause"; mediaRecorder.resume(); stopBtn.disabled = false; } recBtn.disabled = true; pauseResBtn.disabled = false; } function log(message){ dataElement.innerHTML = dataElement.innerHTML+'<br>'+message ; } //browser ID function getBrowser(){ var nVer = navigator.appVersion; var nAgt = navigator.userAgent; var browserName = navigator.appName; var fullVersion = ''+parseFloat(navigator.appVersion); var majorVersion = parseInt(navigator.appVersion,10); var nameOffset,verOffset,ix; // In Opera, the true version is after "Opera" or after "Version" if ((verOffset=nAgt.indexOf("Opera"))!=-1) { browserName = "Opera"; fullVersion = nAgt.substring(verOffset+6); if ((verOffset=nAgt.indexOf("Version"))!=-1) fullVersion = nAgt.substring(verOffset+8); } // In MSIE, the true version is after "MSIE" in userAgent else if ((verOffset=nAgt.indexOf("MSIE"))!=-1) { browserName = "Microsoft Internet Explorer"; fullVersion = nAgt.substring(verOffset+5); } // In Chrome, the true version is after "Chrome" else if ((verOffset=nAgt.indexOf("Chrome"))!=-1) { browserName = "Chrome"; fullVersion = nAgt.substring(verOffset+7); } // In Safari, the true version is after "Safari" or after "Version" else if ((verOffset=nAgt.indexOf("Safari"))!=-1) { browserName = "Safari"; fullVersion = nAgt.substring(verOffset+7); if ((verOffset=nAgt.indexOf("Version"))!=-1) fullVersion = nAgt.substring(verOffset+8); } // In Firefox, the true version is after "Firefox" else if ((verOffset=nAgt.indexOf("Firefox"))!=-1) { browserName = "Firefox"; fullVersion = nAgt.substring(verOffset+8); } // In most other browsers, "name/version" is at the end of userAgent else if ( (nameOffset=nAgt.lastIndexOf(' ')+1) < (verOffset=nAgt.lastIndexOf('/')) ) { browserName = nAgt.substring(nameOffset,verOffset); fullVersion = nAgt.substring(verOffset+1); if (browserName.toLowerCase()==browserName.toUpperCase()) { browserName = navigator.appName; } } // trim the fullVersion string at semicolon/space if present if ((ix=fullVersion.indexOf(";"))!=-1) fullVersion=fullVersion.substring(0,ix); if ((ix=fullVersion.indexOf(" "))!=-1) fullVersion=fullVersion.substring(0,ix); majorVersion = parseInt(''+fullVersion,10); if (isNaN(majorVersion)) { fullVersion = ''+parseFloat(navigator.appVersion); majorVersion = parseInt(navigator.appVersion,10); } return browserName; }
其中的byteLength的判斷,是有緣由的,前端打印的日誌能夠看出


個人這個案例,用的是Firefox的瀏覽器,由於我本地的Chrome的版本比較新,在應用啓動的時候爆出錯誤

時間緊,沒有深刻研究這個錯誤,因此一直都是Firefox基礎上進行驗證的。chrome

 

下面剩下的就是Java後臺的Websocket的服務了。直接上代碼:

/*
 * Copyright © reserved by roomdis.com, service for tgn company whose important business is rural e-commerce.
 */
package com.roomdis.mqr.infra.core;

import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
import java.net.SocketException;
import java.net.UnknownHostException;
import java.nio.ByteBuffer;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

import javax.websocket.OnClose;
import javax.websocket.OnError;
import javax.websocket.OnMessage;
import javax.websocket.OnOpen;
import javax.websocket.Session;
import javax.websocket.server.ServerEndpoint;

import org.apache.log4j.Logger;
import org.springframework.web.context.ContextLoader;

import com.google.gson.Gson;
import com.roomdis.mqr.infra.msg.KefuMessage;

/**
 * @author shihuc
 * @date 2017年8月22日 下午2:20:18
 */
@ServerEndpoint("/websocket")
public class WebsocketService {
    
    private static Logger logger = Logger.getLogger(WebsocketService.class);
    
    private HttpSendService httpSendService;
    
    private String videoRecServerHost = "10.90.7.10";
    
    private int videoRecServerPort = 7667;
    
    /*
     * 當存在多個客戶端訪問時,爲了保證會話繼續保持,將鏈接緩存。
     */
    private static Map<String, WebsocketService> webSocketMap = new ConcurrentHashMap<String, WebsocketService>();
    private Session session;
    
    private static final WebsocketService instance = new WebsocketService();

    public static final WebsocketService getInstance() {
        return instance;
    }

    @OnMessage
    public void onTextMessage(String message, Session session) throws IOException, InterruptedException {

        // Print the client message for testing purposes
        logger.info("Received: " + message);
        //TODO: 調用接口將消息發送給客戶端後臺服務系統
        Gson gson = new Gson();
        KefuMessage kfMsg = gson.fromJson(message, KefuMessage.class);
        httpSendService = ContextLoader.getCurrentWebApplicationContext().getBean(HttpSendService.class);
    }
    
    /**
     * 主要用來接受二進制數據。
     * 
     * @author shihuc
     * @param message
     * @param session
     * @throws IOException
     * @throws InterruptedException
     */ @OnMessage public void onBinaryMessage(ByteBuffer message, Session session, boolean last) throws IOException, InterruptedException { byte [] sentBuf = message.array(); logger.info("Binary Received: " + sentBuf.length + ", last: " + last); //下面的代碼邏輯,是用UDP協議發送視頻流數據到視頻處理服務器作後續邏輯處理 //sendToVideoRecognizer(sentBuf); } /**
     * @author shihuc
     * @param sentBuf
     * @throws SocketException
     * @throws UnknownHostException
     * @throws IOException
     */
    private void sendToVideoRecognizer(byte[] sentBuf) throws SocketException, UnknownHostException, IOException {
        DatagramSocket client = new DatagramSocket();
        InetAddress addr = InetAddress.getByName(videoRecServerHost);
        DatagramPacket sendPacket = new DatagramPacket(sentBuf, sentBuf.length, addr, videoRecServerPort);
        client.send(sendPacket);
        client.close();
    }

//    @OnOpen
//    public void onOpen(Session session){
//        this.session = session;
//        String staffId = session.getQueryString();        
//        webSocketMap.put(staffId, this);
//        logger.info(staffId + " client opened");
//    }
    
    @OnOpen
    public void onOpen(Session session){
        logger.info("client opened: " + session.toString());
    }

    @OnClose
    public void onClose() {
        logger.info("client onclose");        
    }
    
    @OnError
    public void onError(Session session, Throwable error){
        logger.info("connection onError");
        logger.info(error.getCause());
    }
    
    public boolean sendMessage(String message, String staffId) throws IOException{
        WebsocketService client = webSocketMap.get(staffId);
        if (client == null) {
            return false;
        }
        boolean result=false;
        try {            
            client.session.getBasicRemote().sendText(message);
            result=true;
        } catch (IOException e) {
            try {
                client.session.close();
            } catch (IOException e1) {
                e1.printStackTrace();
            }
        }
        return result;
    }
}

 

這裏,重點要注意的是,@OnMessage註解對應的函數,入參很是有講究的。對於arrayBuffer的二進制數據類型,參數個數必須是三個,最後的boolean的必須有,不然前端發送數據的時候,瀏覽器上會拋出錯誤:

 

最後,看看後臺運行的日誌:

  [2017-09-27 20:03:47] [ INFO] [http-nio-9080-exec-7] [com.roomdis.mqr.infra.core.WebsocketService.onBinaryMessage(WebsocketService.java:80)] - Binary Received: 8192, last: false
  [2017-09-27 20:03:47] [ INFO] [http-nio-9080-exec-7] [com.roomdis.mqr.infra.core.WebsocketService.onBinaryMessage(WebsocketService.java:80)] - Binary Received: 8192, last: false
  [2017-09-27 20:03:47] [ INFO] [http-nio-9080-exec-7] [com.roomdis.mqr.infra.core.WebsocketService.onBinaryMessage(WebsocketService.java:80)] - Binary Received: 8192, last: false
  [2017-09-27 20:03:47] [ INFO] [http-nio-9080-exec-7] [com.roomdis.mqr.infra.core.WebsocketService.onBinaryMessage(WebsocketService.java:80)] - Binary Received: 8192, last: false
  [2017-09-27 20:03:47] [ INFO] [http-nio-9080-exec-7] [com.roomdis.mqr.infra.core.WebsocketService.onBinaryMessage(WebsocketService.java:80)] - Binary Received: 8192, last: false
  [2017-09-27 20:03:47] [ INFO] [http-nio-9080-exec-7] [com.roomdis.mqr.infra.core.WebsocketService.onBinaryMessage(WebsocketService.java:80)] - Binary Received: 8192, last: false
  [2017-09-27 20:03:47] [ INFO] [http-nio-9080-exec-7] [com.roomdis.mqr.infra.core.WebsocketService.onBinaryMessage(WebsocketService.java:80)] - Binary Received: 8192, last: false
  [2017-09-27 20:03:47] [ INFO] [http-nio-9080-exec-7] [com.roomdis.mqr.infra.core.WebsocketService.onBinaryMessage(WebsocketService.java:80)] - Binary Received: 8192, last: false
  [2017-09-27 20:03:47] [ INFO] [http-nio-9080-exec-7] [com.roomdis.mqr.infra.core.WebsocketService.onBinaryMessage(WebsocketService.java:80)] - Binary Received: 8192, last: false
  [2017-09-27 20:03:47] [ INFO] [http-nio-9080-exec-7] [com.roomdis.mqr.infra.core.WebsocketService.onBinaryMessage(WebsocketService.java:80)] - Binary Received: 8192, last: false
  [2017-09-27 20:03:47] [ INFO] [http-nio-9080-exec-7] [com.roomdis.mqr.infra.core.WebsocketService.onBinaryMessage(WebsocketService.java:80)] - Binary Received: 8192, last: false
  [2017-09-27 20:03:47] [ INFO] [http-nio-9080-exec-7] [com.roomdis.mqr.infra.core.WebsocketService.onBinaryMessage(WebsocketService.java:80)] - Binary Received: 8192, last: false
  [2017-09-27 20:03:47] [ INFO] [http-nio-9080-exec-7] [com.roomdis.mqr.infra.core.WebsocketService.onBinaryMessage(WebsocketService.java:80)] - Binary Received: 8192, last: false
  [2017-09-27 20:03:47] [ INFO] [http-nio-9080-exec-7] [com.roomdis.mqr.infra.core.WebsocketService.onBinaryMessage(WebsocketService.java:80)] - Binary Received: 8192, last: false
  [2017-09-27 20:03:47] [ INFO] [http-nio-9080-exec-7] [com.roomdis.mqr.infra.core.WebsocketService.onBinaryMessage(WebsocketService.java:80)] - Binary Received: 8192, last: false
  [2017-09-27 20:03:47] [ INFO] [http-nio-9080-exec-7] [com.roomdis.mqr.infra.core.WebsocketService.onBinaryMessage(WebsocketService.java:80)] - Binary Received: 8192, last: false
  [2017-09-27 20:03:47] [ INFO] [http-nio-9080-exec-7] [com.roomdis.mqr.infra.core.WebsocketService.onBinaryMessage(WebsocketService.java:80)] - Binary Received: 7494, last: true

並附上一副前端運行的效果截圖:

 

總結:

1. 重點研究getUserMedia。

2.重點研究MediaRecorder。

3.重點研究Blob以及FileReader。

4.重點研究Websocket的@OnMessage的註解函數的參數,以及數據傳輸中鏈接可能會斷掉的處理方案。

 

2018-05-03

PS:既然有人對我這個研究有興趣,我就將源碼共享出來,幫助有須要的技術愛好者。源碼地址在github上面:https://github.com/shihuc/VideoConverter

但願有興趣的朋友,經過關注個人博客,共同互動,研究一些特別的應用!

相關文章
相關標籤/搜索