WebRTC学习实现视频
工作中遇到要视频直播的需求,前提是不能依赖Flash前端,于是就找到了WebRtc的相关资料.
什么GetUserMedia,RTCPeerConnection,DataChannel我不多说.
简单讲就是谷歌把实时通信层打包进浏览器.而这一套实时通信层又是来源于电信通信领域.
所以浏览器两端交互需要依赖一个叫做信令服务器的东西,来协助两端完成连接.
简单说下流程
以A呼叫B为例
A呼叫B
1.告知Server,我要找B
2.Server查一下有没有B,有就传达,没有就不传达,至于结果需不需要告知A,那全看心情了.反正这东西是自己实现的.WebRTC里就提了一嘴这东西,没具体规范.
3.B得到A的呼叫(准确说叫Offer)
4.B解析Offer,回应(Answer)到Server,Server回给A
5.A收到Answer,解析.
6.这时,A和B就算勾搭上了.剩下的事情就交给他们自己办了.
Offer 和 Answer 都属于SDP.具体规范http://datatracker.ietf.org/doc/draft-nandakumar-rtcweb-sdp/01/
解析这些是交给浏览器做的.
先看Server端的Java实现
package org.rtc.sip; import java.io.IOException; 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 com.google.gson.Gson; @ServerEndpoint(value = "/sipserver") public class SignalWorker { private Gson gson = new Gson(); private Session session; public Session getSession() { return session; } public void setSession(Session session) { this.session = session; } public SignalWorker() { } @OnClose public void onClose(Session session) { } @OnError public void onError(Throwable throwable) { } @OnMessage public void onMessage(String message, Session session) { System.out.println("收到消息:" + message); SIPObject sip = gson.fromJson(message, SIPObject.class); try { if(sip.getAction().equals("login")){ SIPSessionManager.addNewSession(sip.getFrom(), this); }else if(sip.getAction().equals("offer")){ SignalWorker sker=SIPSessionManager.getSignalWorker(sip.getTo()); if(sker!=null){ sker.getSession().getBasicRemote().sendText(message); } else{ miss(); } }else if(sip.getAction().equals("answer")){ SignalWorker sker=SIPSessionManager.getSignalWorker(sip.getTo()); if(sker!=null){ sker.getSession().getBasicRemote().sendText(message); } else{ miss(); } }else if(sip.getAction().equals("candidate")){ SignalWorker sker=SIPSessionManager.getSignalWorker(sip.getTo()); if(sker!=null){ sker.getSession().getBasicRemote().sendText(message); } } } catch (IOException e) { e.printStackTrace(); } } @OnOpen public void onOpen(Session session) { this.session = session; } public void miss(){ SIPObject resp=new SIPObject(); resp.setAction("miss"); try { session.getBasicRemote().sendText(gson.toJson(resp)); } catch (IOException e) { e.printStackTrace(); } } public void send(String msg) { System.out.println(msg); session.getAsyncRemote().sendText(msg); } }
代码逻辑很简单,就是A->B,B->A,
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title>WebRtc</title> <script src="rtc_suit/jquery-2.1.3.min.js"></script> <script src="rtc_suit/yuhan_rtc.js"></script> </head> <body> <video id="localVideo" style="width:400px;height:400px;" ></video> <button onclick="Call('Baidu')"> call</button> <br/> <video id="remoteVideo" style="width:400px;height:400px;" ></video> <script> var rtcg=new RTCSuit({id:"Google",to:"Baidu",sipws:"ws://192.168.0.119:8080/WebTest/sipserver"}); rtcg.connectSip(); var rtcb=new RTCSuit({id:"Baidu",to:"Google",sipws:"ws://192.168.0.119:8080/WebTest/sipserver"}); rtcb.connectSip(); rtcb.bindRemoteMedia("remoteVideo"); function Call(to){ rtcg.bindLocalMedia("localVideo",true,true,function(rst,stream){ if(rst){ rtcg.sendOffer(); } }); } //rtc.sendOffer(); </script> </body> </html>
上面HTML代码逻辑也很简单.
两端Baidu和Google先要登陆在SIPServer上.
Google呼叫baidu是在本地完成了摄像头视频获取后发出的Offer.
整个WebRtc交互用一个JS库屏蔽掉了复杂的代码.
代码只简单实现了功能,未完善.贴出来分享.
/** * */ // define override var RTCPeerConnection = window.RTCPeerConnection || window.mozRTCPeerConnection || window.webkitRTCPeerConnection; navigator.getUserMedia = navigator.webkitGetUserMedia || navigator.mozGetUserMedia; //ICE SERVER var iceServer = { "iceServers": [{ "url": "stun:stun.l.google.com:19302" }] }; //LOG function log(msg,level){ console.log(new Date().toLocaleString()+" "+msg); } function RTCSuit(opt) { this.sipws=opt.sipws;//信令服务器地址 this.id=opt.id||new Date().getTime();//身份信息 this.to=opt.to; this.conn=null;//websocket对象,connect时初始化 this.sessionId=null; this.peercn=new RTCPeerConnection(opt.ice||iceServer); this.peercn.rtc=this; //init peer this.peercn.onicecandidate = RTCSuit.prototype.onIceCandidate; this.peercn.onconnecting = RTCSuit.prototype.onPeerConnecting; this.peercn.onopen = RTCSuit.prototype.onPeerOpen; this.peercn.onaddstream = RTCSuit.prototype.onRemotePeerStreamAdd; this.peercn.onremovestream = RTCSuit.prototype.onPeerStreamRemove; this.localNode=null; this.remoteNode=null; return this; } //视频媒体交互 RTCSuit.prototype.bindLocalMedia=function(id, enableAudio, enableVideo,callback) { var rtc=this; navigator.getUserMedia({ "audio" : enableAudio, "video" : enableVideo }, function(stream) {//success var videoNode=document.getElementById(id); rtc.localNode=videoNode; videoNode.autoplay=true; videoNode.src= URL.createObjectURL(stream); rtc.peercn.addStream(stream); //触发事件,通知视频流绑定成功 if(typeof(callback)=="function"){ callback(true,stream); } log("bind local media success"); },function(error){//error if(typeof(callback)=="function"){ callback(false,null); } }); }; RTCSuit.prototype.bindRemoteMedia=function(id){ var videoNode=document.getElementById(id); videoNode.autoplay=true; this.remoteNode=videoNode; }; //PEER交互 RTCSuit.prototype.sendOffer=function(){ var rtc=this; if(!rtc.conn.readyState){ setTimeout(function(){rtc.sendOffer()},300);return; } this.peercn.createOffer(function(desc){//success rtc.peercn.setLocalDescription(desc); rtc.send({action:"offer",sdpWrap:desc}); },function(error){ rtc.onSendOfferError(error); }); }; RTCSuit.prototype.sendAnswer=function(){ var rtc=this; if(!rtc.conn.readyState){ setTimeout(function(){rtc.sendOffer()},300);return; } this.peercn.createAnswer(function(desc){//success rtc.peercn.setLocalDescription(desc); rtc.send({action:"answer",sdpWrap:desc}); },function(error){ }); } RTCSuit.prototype.onAddStreamToPeer=function(){ log("get here"); }; RTCSuit.prototype.onIceCandidate=function(event){ var rtc=this.rtc; if (event.candidate) { rtc.send({action:"candidate",sdpWrap:{ type : "candidate", label : event.candidate.sdpMLineIndex, id : event.candidate.sdpMid, candidate : event.candidate.candidate }}); } else { console.log("End of candidates."); } }; RTCSuit.prototype.onPeerConnecting=function(){ log(" peer connecting"); }; RTCSuit.prototype.onPeerOpen=function(){ log(" peer open"); }; RTCSuit.prototype.onRemotePeerStreamAdd=function(event){ var url = webkitURL.createObjectURL(event.stream); this.rtc.remoteNode.src=url; }; RTCSuit.prototype.onPeerStreamRemove=function(){ }; //信令交互 RTCSuit.prototype.send=function(json){ var rtc=this; json.from=rtc.id; json.to=rtc.to; var smsg=JSON.stringify(json); if(rtc.conn.readyState) rtc.conn.send(smsg); log(rtc.id+" 发送"+smsg); }; RTCSuit.prototype.connectSip=function(sipws,callback){ sipws=sipws||this.sipws; var rtc=this; this.conn=new WebSocket(sipws); this.conn.onopen=function(){ rtc.send({id:rtc.id,action:"login"}); log("sip server connected"); }; //core msg process this.conn.onmessage=function(e){ log(rtc.id+" 收到消息:"+e.data); var msg=e.data; var json=eval("("+msg+")"); if(typeof(rtc["on"+json.action]) == "function"){ rtc["on"+json.action](json); }else{ log("error response:"+msg); } }; }; RTCSuit.prototype.onoffer=function(json){ this.peercn.setRemoteDescription(new RTCSessionDescription(json.sdpWrap)); this.sendAnswer(); log("recv offer") }; RTCSuit.prototype.onanswer=function(json){ this.peercn.setRemoteDescription(new RTCSessionDescription(json.sdpWrap)); log("recv answer"); }; RTCSuit.prototype.oncandidate=function(json){ var msg=json.sdpWrap; var candidate = new RTCIceCandidate({ sdpMLineIndex : msg.label, candidate : msg.candidate }); this.peercn.addIceCandidate(candidate); };
最后代码奉上....环境 tomcat8.0 java 1.8