WebRTC - वॉयस डेमो

इस अध्याय में, हम एक ग्राहक एप्लिकेशन बनाने जा रहे हैं जो दो उपयोगकर्ताओं को अलग-अलग डिवाइसों पर WebRTC ऑडियो स्ट्रीम का उपयोग करके संवाद करने की अनुमति देता है। हमारे आवेदन में दो पेज होंगे। एक लॉगिन के लिए और दूसरा दूसरे यूज़र को ऑडियो कॉल करने के लिए।

दो पेज div टैग होंगे। अधिकांश इनपुट सरल ईवेंट हैंडलर के माध्यम से किया जाता है।

सिग्नलिंग सर्वर

WebRTC कनेक्शन बनाने के लिए क्लाइंट को WebRTC सहकर्मी कनेक्शन का उपयोग किए बिना संदेशों को स्थानांतरित करने में सक्षम होना चाहिए। यह वह जगह है जहां हम HTML5 WebSockets - दो एंडपॉइंट के बीच एक द्विदिश सॉकेट कनेक्शन - एक वेब सर्वर और एक वेब ब्राउज़र का उपयोग करेंगे। अब WebSocket लाइब्रेरी का उपयोग शुरू करते हैं। Server.js फ़ाइल बनाएँ और निम्नलिखित कोड डालें -

//require our websocket library 
var WebSocketServer = require('ws').Server; 

//creating a websocket server at port 9090 
var wss = new WebSocketServer({port: 9090});
  
//when a user connects to our sever 
wss.on('connection', function(connection) { 
   console.log("user connected"); 
	
   //when server gets a message from a connected user 
   connection.on('message', function(message) { 
      console.log("Got message from a user:", message); 
   }); 
	
   connection.send("Hello from server"); 
});

पहली पंक्ति के लिए वेबस्केट लाइब्रेरी की आवश्यकता होती है जिसे हमने पहले ही स्थापित कर दिया है। फिर हम पोर्ट 9090 पर एक सॉकेट सर्वर बनाते हैं। इसके बाद, हम कनेक्शन ईवेंट को सुनते हैं । यह कोड तब निष्पादित किया जाएगा जब कोई उपयोगकर्ता सर्वर से WebSocket कनेक्शन बनाता है। फिर हम उपयोगकर्ता द्वारा भेजे गए किसी भी संदेश को सुनते हैं। अंत में, हम कनेक्टेड यूज़र को "सर्वर से हैलो" कहते हुए एक प्रतिक्रिया भेजें।

हमारे सिग्नलिंग सर्वर में, हम प्रत्येक कनेक्शन के लिए एक स्ट्रिंग-आधारित उपयोगकर्ता नाम का उपयोग करेंगे ताकि हमें पता चले कि संदेश कहां भेजना है। आइए हमारे कनेक्शन हैंडलर को थोड़ा बदलें -

connection.on('message', function(message) { 
   var data; 
	
   //accepting only JSON messages 
   try { 
      data = JSON.parse(message); 
   } catch (e) { 
      console.log("Invalid JSON");
      data = {}; 
   } 
});

इस तरह हम केवल JSON संदेशों को स्वीकार करते हैं। अगला, हमें सभी कनेक्ट किए गए उपयोगकर्ताओं को कहीं स्टोर करने की आवश्यकता है। हम इसके लिए एक सरल जावास्क्रिप्ट ऑब्जेक्ट का उपयोग करेंगे। हमारी फ़ाइल के शीर्ष को बदलें -

//require our websocket library 
var WebSocketServer = require('ws').Server; 

//creating a websocket server at port 9090 
var wss = new WebSocketServer({port: 9090}); 

//all connected to the server users 
var users = {};

हम क्लाइंट से आने वाले प्रत्येक संदेश के लिए एक प्रकार का फ़ील्ड जोड़ने जा रहे हैं । उदाहरण के लिए यदि कोई उपयोगकर्ता लॉगिन करना चाहता है, तो वह लॉगिन प्रकार संदेश भेजता है । आइए इसे परिभाषित करते हैं -

connection.on('message', function(message) {
  
   var data; 
   //accepting only JSON messages 
   try { 
      data = JSON.parse(message); 
   } catch (e) { 
      console.log("Invalid JSON"); 
      data = {}; 
   } 
	
   //switching type of the user message 
   switch (data.type) { 
      //when a user tries to login 
      case "login": 
         console.log("User logged:", data.name); 
			
         //if anyone is logged in with this username then refuse 
         if(users[data.name]) { 
            sendTo(connection, { 
               type: "login",
               success: false 
            }); 
         } else { 
            //save user connection on the server 
            users[data.name] = connection; 
            connection.name = data.name; 
				
            sendTo(connection, { 
               type: "login", 
               success: true 
            }); 
         } 
			
         break;
			
      default: 
         sendTo(connection, { 
            type: "error", 
            message: "Command no found: " + data.type 
         }); 
			
         break; 
   }  
});

यदि उपयोगकर्ता लॉगिन प्रकार के साथ एक संदेश भेजता है , तो हम -

  • जांचें कि क्या कोई पहले से ही इस उपयोगकर्ता नाम के साथ लॉग इन कर चुका है।
  • यदि ऐसा है, तो उपयोगकर्ता को बताएं कि उसने सफलतापूर्वक लॉगिन नहीं किया है।
  • यदि कोई इस उपयोगकर्ता नाम का उपयोग नहीं कर रहा है, तो हम कनेक्शन ऑब्जेक्ट की कुंजी के रूप में उपयोगकर्ता नाम जोड़ते हैं।
  • यदि कोई कमांड नहीं पहचानी जाती है तो हम एक त्रुटि भेजते हैं।

निम्नलिखित कोड एक कनेक्शन के लिए संदेश भेजने के लिए एक सहायक कार्य है। इसे server.js फ़ाइल में जोड़ें -

function sendTo(connection, message) { 
   connection.send(JSON.stringify(message)); 
}

जब उपयोगकर्ता डिस्कनेक्ट करता है तो हमें उसके कनेक्शन को साफ करना चाहिए। जब करीबी घटना को निकाल दिया जाता है तो हम उपयोगकर्ता को हटा सकते हैं । कनेक्शन हैंडलर में निम्न कोड जोड़ें

connection.on("close", function() { 
   if(connection.name) { 
      delete users[connection.name]; 
   } 
});

सफल लॉगिन के बाद उपयोगकर्ता दूसरे को कॉल करना चाहता है। उसे इसे प्राप्त करने के लिए किसी अन्य उपयोगकर्ता को एक प्रस्ताव देना चाहिए । प्रस्ताव हैंडलर जोड़ें -

case "offer": 
   //for ex. UserA wants to call UserB 
   console.log("Sending offer to: ", data.name); 
	
   //if UserB exists then send him offer details 
   var conn = users[data.name]; 
	
   if(conn != null) { 
      //setting that UserA connected with UserB 
      connection.otherName = data.name; 
      sendTo(conn, { 
         type: "offer", 
         offer: data.offer, 
         name: connection.name 
      });
   }		
	
   break;

सबसे पहले, हम उस उपयोगकर्ता का कनेक्शन प्राप्त करते हैं जिसे हम कॉल करने का प्रयास कर रहे हैं। यदि यह मौजूद है तो हम उसे प्रस्ताव विवरण भेजते हैं । हम यह भी जोड़ना otherName को कनेक्शन वस्तु। इसे बाद में खोजने की सरलता के लिए बनाया गया है।

प्रतिक्रिया के जवाब में एक समान पैटर्न है जो हमने प्रस्ताव हैंडलर में उपयोग किया था । हमारा सर्वर अन्य उपयोगकर्ताओं के उत्तर के रूप में सभी संदेशों से गुजरता है । प्रस्ताव हैंडलर के बाद निम्नलिखित कोड जोड़ें -

case "answer": 
   console.log("Sending answer to: ", data.name); 
   //for ex. UserB answers UserA
   var conn = users[data.name]; 
	
   if(conn != null) { 
      connection.otherName = data.name;
		
      sendTo(conn, { 
         type: "answer", 
         answer: data.answer 
      }); 
   } 
	
   break;

अंतिम भाग उपयोगकर्ताओं के बीच आईसीई उम्मीदवार को संभाल रहा है। हम उसी तकनीक का उपयोग करते हैं जो उपयोगकर्ताओं के बीच संदेश भेजती है। मुख्य अंतर यह है कि उम्मीदवार संदेश किसी भी क्रम में प्रति उपयोगकर्ता कई बार हो सकता है। उम्मीदवार हैंडलर जोड़ें -

case "candidate": 
   console.log("Sending candidate to:",data.name); 
   var conn = users[data.name];
	
   if(conn != null) { 
      sendTo(conn, { 
         type: "candidate", 
         candidate: data.candidate 
      }); 
   } 
	
   break;

हमारे उपयोगकर्ताओं को किसी अन्य उपयोगकर्ता से डिस्कनेक्ट करने की अनुमति देने के लिए हमें हैंगिंग फ़ंक्शन को लागू करना चाहिए। यह सभी उपयोगकर्ता संदर्भों को हटाने के लिए सर्वर को भी बताएगा। छुट्टी हैंडलर जोड़ें -

case "leave": 
   console.log("Disconnecting from", data.name); 
   var conn = users[data.name]; 
   conn.otherName = null; 
	
   //notify the other user so he can disconnect his peer connection 
   if(conn != null) { 
      sendTo(conn, {
         type: "leave" 
      }); 
   }  
	
   break;

यह अन्य उपयोगकर्ता को छुट्टी की घटना भी भेजेगा ताकि वह अपने सहकर्मी कनेक्शन को तदनुसार काट सके। हमें उस मामले को भी संभालना चाहिए जब कोई उपयोगकर्ता सिग्नलिंग सर्वर से अपना कनेक्शन छोड़ देता है। आइए हमारे करीबी हैंडलर को संशोधित करें -

connection.on("close", function() { 

   if(connection.name) { 
      delete users[connection.name]; 
		
      if(connection.otherName) { 
         console.log("Disconnecting from ", connection.otherName); 
         var conn = users[connection.otherName]; 
         conn.otherName = null;
			
         if(conn != null) { 
            sendTo(conn, { 
               type: "leave" 
            }); 
         }
			
      } 
   } 
});

हमारे सिग्नलिंग सर्वर का पूरा कोड निम्नलिखित है -

//require our websocket library 
var WebSocketServer = require('ws').Server; 

//creating a websocket server at port 9090 
var wss = new WebSocketServer({port: 9090}); 

//all connected to the server users 
var users = {};

//when a user connects to our sever 
wss.on('connection', function(connection) {
  
   console.log("User connected");
	
   //when server gets a message from a connected user 
   connection.on('message', function(message) { 
	
      var data;
		
      //accepting only JSON messages 
      try { 
         data = JSON.parse(message); 
      } catch (e) { 
         console.log("Invalid JSON"); 
         data = {}; 
      }
		
      //switching type of the user message 
      switch (data.type) { 
         //when a user tries to login 
         case "login": 
            console.log("User logged", data.name); 
				
            //if anyone is logged in with this username then refuse 
            if(users[data.name]) { 
               sendTo(connection, { 
                  type: "login", 
                  success: false 
               }); 
            } else { 
               //save user connection on the server 
               users[data.name] = connection; 
               connection.name = data.name;
					
               sendTo(connection, { 
                  type: "login", 
                  success: true 
               }); 
            } 
				
            break;
				
         case "offer": 
            //for ex. UserA wants to call UserB 
            console.log("Sending offer to: ", data.name); 
				
            //if UserB exists then send him offer details 
            var conn = users[data.name]; 
				
            if(conn != null) { 
               //setting that UserA connected with UserB 
               connection.otherName = data.name; 
					
               sendTo(conn, { 
                  type: "offer", 
                  offer: data.offer, 
                  name: connection.name 
               }); 
            } 
				
            break;
				
         case "answer": 
            console.log("Sending answer to: ", data.name); 
            //for ex. UserB answers UserA 
            var conn = users[data.name]; 
				
            if(conn != null) { 
               connection.otherName = data.name; 
               sendTo(conn, { 
                  type: "answer", 
                  answer: data.answer 
               });
            } 
				
            break;
				
         case "candidate": 
            console.log("Sending candidate to:",data.name); 
            var conn = users[data.name];  
				
            if(conn != null) { 
               sendTo(conn, { 
                  type: "candidate", 
                  candidate: data.candidate 
               }); 
            } 
				
            break;
				
         case "leave": 
            console.log("Disconnecting from", data.name); 
            var conn = users[data.name]; 
            conn.otherName = null; 
				
            //notify the other user so he can disconnect his peer connection 
            if(conn != null) { 
               sendTo(conn, { 
                  type: "leave" 
               }); 
            }  
				
            break;
				
         default: 
            sendTo(connection, { 
               type: "error", 
               message: "Command not found: " + data.type 
            });
				
            break; 
      }  
   });
	
   //when user exits, for example closes a browser window 
   //this may help if we are still in "offer","answer" or "candidate" state 
   connection.on("close", function() { 
	
      if(connection.name) { 
         delete users[connection.name]; 
			
         if(connection.otherName) { 
            console.log("Disconnecting from ", connection.otherName); 
            var conn = users[connection.otherName]; 
            conn.otherName = null;  
				
            if(conn != null) { 
               sendTo(conn, { 
                  type: "leave" 
              }); 
            }  
         } 
      } 
   });  
	
   connection.send("Hello world"); 
}); 
 
function sendTo(connection, message) { 
   connection.send(JSON.stringify(message)); 
}

ग्राहक आवेदन

इस एप्लिकेशन का परीक्षण करने का एक तरीका दो ब्राउज़र टैब खोल रहा है और एक दूसरे को ऑडियो कॉल करने की कोशिश कर रहा है।

सबसे पहले, हमें बूटस्ट्रैप लाइब्रेरी स्थापित करने की आवश्यकता है । बूटस्ट्रैप वेब अनुप्रयोगों को विकसित करने के लिए एक दृश्यमान रूपरेखा है। आप और अधिक सीख सकते हैंhttp://getbootstrap.com/.उदाहरण के लिए, "ऑडियोचैट" नामक एक फ़ोल्डर बनाएँ। यह हमारा रूट एप्लीकेशन फोल्डर होगा। इस फ़ोल्डर के अंदर एक फ़ाइल पैकेज बनाएँ । Json (यह npm निर्भरता के प्रबंधन के लिए आवश्यक है) और निम्नलिखित जोड़ें -

{ 
   "name": "webrtc-audiochat", 
   "version": "0.1.0", 
   "description": "webrtc-audiochat", 
   "author": "Author", 
   "license": "BSD-2-Clause" 
}

फिर npm इंस्टॉल बूटस्ट्रैप चलाएं । यह ऑडीओचैट / नोड_मॉड्यूल फ़ोल्डर में बूटस्ट्रैप लाइब्रेरी स्थापित करेगा ।

अब हमें एक मूल HTML पेज बनाने की आवश्यकता है। निम्नलिखित कोड के साथ रूट फ़ोल्डर में एक index.html फ़ाइल बनाएँ -

<html>
 
   <head> 
      <title>WebRTC Voice Demo</title> 
      <link rel = "stylesheet" href = "node_modules/bootstrap/dist/css/bootstrap.min.css"/> 
   </head>
 
   <style> 
      body { 
         background: #eee; 
         padding: 5% 0; 
      } 
   </style>
	
   <body> 
      <div id = "loginPage" class = "container text-center"> 
		
         <div class = "row"> 
            <div class = "col-md-4 col-md-offset-4">
				
               <h2>WebRTC Voice Demo. Please sign in</h2>
				
               <label for = "usernameInput" class = "sr-only">Login</label> 
               <input type = "email" id = "usernameInput" 
                  class = "form-control formgroup"
                  placeholder = "Login" required = "" autofocus = ""> 
               <button id = "loginBtn" class = "btn btn-lg btn-primary btnblock">
                  Sign in</button> 
            </div> 
         </div> 
			
      </div>
		
      <div id = "callPage" class = "call-page">
		
         <div class = "row"> 
			
            <div class = "col-md-6 text-right"> 
               Local audio: <audio id = "localAudio" 
               controls autoplay></audio> 
            </div>
				
            <div class = "col-md-6 text-left"> 
               Remote audio: <audio id = "remoteAudio" 
                  controls autoplay></audio> 
            </div> 
				
         </div> 
			
         <div class = "row text-center"> 
            <div class = "col-md-12"> 
               <input id = "callToUsernameInput" 
                  type = "text" placeholder = "username to call" /> 
               <button id = "callBtn" class = "btn-success btn">Call</button> 
               <button id = "hangUpBtn" class = "btn-danger btn">Hang Up</button> 
            </div> 
         </div>
			
      </div> 
		
      <script src = "client.js"></script> 
		
   </body>
	
</html>

यह पृष्ठ आपको परिचित होना चाहिए। हमने बूटस्ट्रैप css फ़ाइल जोड़ दी है । हमने दो पृष्ठ भी परिभाषित किए हैं। अंत में, हमने उपयोगकर्ता से जानकारी प्राप्त करने के लिए कई पाठ फ़ील्ड और बटन बनाए हैं। आपको स्थानीय और दूरस्थ ऑडियो स्ट्रीम के लिए दो ऑडियो तत्वों को देखना चाहिए। ध्यान दें कि हमने एक client.js फ़ाइल के लिए एक लिंक जोड़ा है ।

अब हमें अपने सिग्नलिंग सर्वर के साथ एक कनेक्शन स्थापित करने की आवश्यकता है। निम्नलिखित कोड के साथ रूट फ़ोल्डर में client.js फ़ाइल बनाएँ -

//our username 
var name; 
var connectedUser;
  
//connecting to our signaling server 
var conn = new WebSocket('ws://localhost:9090');
  
conn.onopen = function () { 
   console.log("Connected to the signaling server"); 
}; 
 
//when we got a message from a signaling server 
conn.onmessage = function (msg) { 
   console.log("Got message", msg.data);  
   var data = JSON.parse(msg.data);  
	
   switch(data.type) { 
      case "login": 
         handleLogin(data.success); 
         break; 
      //when somebody wants to call us 
      case "offer": 
         handleOffer(data.offer, data.name); 
         break; 
      case "answer": 
         handleAnswer(data.answer); 
         break; 
      //when a remote peer sends an ice candidate to us 
      case "candidate": 
         handleCandidate(data.candidate); 
         break;
      case "leave": 
         handleLeave(); 
         break; 
      default: 
         break; 
   } 
};
  
conn.onerror = function (err) { 
   console.log("Got error", err); 
};
  
//alias for sending JSON encoded messages 
function send(message) { 
   //attach the other peer username to our messages
   if (connectedUser) { 
      message.name = connectedUser; 
   } 
	
   conn.send(JSON.stringify(message)); 
};

अब नोड सर्वर के माध्यम से हमारे सिग्नलिंग सर्वर को चलाएं । फिर, रूट फ़ोल्डर के अंदर स्टेटिक कमांड चलाएं और ब्राउज़र के अंदर पेज खोलें। आपको निम्न कंसोल आउटपुट देखना चाहिए -

अगला चरण एक अद्वितीय उपयोगकर्ता नाम के साथ एक उपयोगकर्ता लॉग को लागू कर रहा है। हम केवल सर्वर पर एक उपयोगकर्ता नाम भेजते हैं, जो तब हमें बताता है कि यह लिया गया है या नहीं। निम्न कोड को अपने client.js फ़ाइल में जोड़ें -

//****** 
//UI selectors block 
//******

var loginPage = document.querySelector('#loginPage'); 
var usernameInput = document.querySelector('#usernameInput'); 
var loginBtn = document.querySelector('#loginBtn');
 
var callPage = document.querySelector('#callPage'); 
var callToUsernameInput = document.querySelector('#callToUsernameInput');
var callBtn = document.querySelector('#callBtn');
 
var hangUpBtn = document.querySelector('#hangUpBtn');
  
callPage.style.display = "none";
  
// Login when the user clicks the button 
loginBtn.addEventListener("click", function (event) { 
   name = usernameInput.value;
	
   if (name.length > 0) { 
      send({
         type: "login", 
         name: name 
      }); 
   } 
	
}); 
 
function handleLogin(success) { 
   if (success === false) { 
      alert("Ooops...try a different username"); 
   } else { 
      loginPage.style.display = "none"; 
      callPage.style.display = "block"; 
		
      //********************** 
      //Starting a peer connection 
      //**********************
		         
   } 
	
};

सबसे पहले, हम पृष्ठ के तत्वों के कुछ संदर्भों का चयन करते हैं। हम कॉल पेज छिपाते हैं। फिर, हम लॉगिन बटन पर एक ईवेंट श्रोता जोड़ते हैं। जब उपयोगकर्ता इसे क्लिक करता है, तो हम सर्वर पर उसका उपयोगकर्ता नाम भेजते हैं। अंत में, हम हैंडललॉग कॉलबैक लागू करते हैं। यदि लॉगिन सफल रहा, तो हम कॉल पेज दिखाते हैं और एक सहकर्मी कनेक्शन सेट करना शुरू करते हैं।

एक सहकर्मी कनेक्शन शुरू करने के लिए -

  • एक माइक्रोफोन से एक ऑडियो स्ट्रीम प्राप्त करें
  • RTCPeerConnection ऑब्जेक्ट बनाएँ

निम्नलिखित कोड को "UI चयनकर्ताओं ब्लॉक" में जोड़ें -

var localAudio = document.querySelector('#localAudio'); 
var remoteAudio = document.querySelector('#remoteAudio'); 

var yourConn; 
var stream;

हैंडललॉगिन फ़ंक्शन को संशोधित करें -

function handleLogin(success) { 
   if (success === false) { 
      alert("Ooops...try a different username"); 
   } else { 
      loginPage.style.display = "none"; 
      callPage.style.display = "block";
		
      //********************** 
      //Starting a peer connection 
      //********************** 
		
      //getting local audio stream 
      navigator.webkitGetUserMedia({ video: false, audio: true }, function (myStream) { 
         stream = myStream; 
			
         //displaying local audio stream on the page
         localAudio.src = window.URL.createObjectURL(stream);
			
         //using Google public stun server 
         var configuration = { 
            "iceServers": [{ "url": "stun:stun2.1.google.com:19302" }] 
         }; 
			
         yourConn = new webkitRTCPeerConnection(configuration); 
			
         // setup stream listening 
         yourConn.addStream(stream); 
			
         //when a remote user adds stream to the peer connection, we display it 
         yourConn.onaddstream = function (e) { 
            remoteAudio.src = window.URL.createObjectURL(e.stream); 
         }; 
			
         // Setup ice handling 
         yourConn.onicecandidate = function (event) { 
            if (event.candidate) { 
               send({ 
                  type: "candidate", 
               }); 
            } 
         };  
			
      }, function (error) { 
         console.log(error); 
      }); 
		
   } 
};

अब यदि आप कोड चलाते हैं, तो पृष्ठ को आपको लॉग इन करने और पृष्ठ पर अपनी स्थानीय ऑडियो स्ट्रीम प्रदर्शित करने की अनुमति देनी चाहिए।

अब हम एक कॉल शुरू करने के लिए तैयार हैं। सबसे पहले, हम दूसरे उपयोगकर्ता को एक प्रस्ताव भेजते हैं । एक बार उपयोगकर्ता को प्रस्ताव मिलने के बाद, वह एक उत्तर बनाता है और ICE उम्मीदवारों का व्यापार शुरू करता है । निम्नलिखित कोड को client.js फ़ाइल में जोड़ें -

//initiating a call 
callBtn.addEventListener("click", function () { 
   var callToUsername = callToUsernameInput.value; 
	
   if (callToUsername.length > 0) { 
      connectedUser = callToUsername; 
		
      // create an offer 
      yourConn.createOffer(function (offer) { 
         send({ 
            type: "offer", 
            offer: offer 
         }); 
			
         yourConn.setLocalDescription(offer); 
			
      }, function (error) { 
         alert("Error when creating an offer"); 
      }); 
   } 
	
});
 
//when somebody sends us an offer 
function handleOffer(offer, name) { 
   connectedUser = name; 
   yourConn.setRemoteDescription(new RTCSessionDescription(offer)); 
	
   //create an answer to an offer 
   yourConn.createAnswer(function (answer) { 
      yourConn.setLocalDescription(answer); 
		
      send({ 
         type: "answer",
         answer: answer 
      }); 
		
   }, function (error) { 
      alert("Error when creating an answer"); 
   }); 
	
};
 
//when we got an answer from a remote user 
function handleAnswer(answer) { 
   yourConn.setRemoteDescription(new RTCSessionDescription(answer)); 
};
 
//when we got an ice candidate from a remote user 
function handleCandidate(candidate) { 
   yourConn.addIceCandidate(new RTCIceCandidate(candidate)); 
};

हम कॉल बटन पर एक क्लिक हैंडलर जोड़ते हैं , जो एक प्रस्ताव शुरू करता है। फिर हम onmessage हैंडलर द्वारा अपेक्षित कई हैंडलर लागू करते हैं । जब तक दोनों उपयोगकर्ताओं ने एक कनेक्शन नहीं किया है, तब तक उन्हें एसिंक्रोनस रूप से संसाधित किया जाएगा।

अंतिम चरण हैंग-अप सुविधा को लागू कर रहा है। यह डेटा संचारित करना बंद कर देगा और अन्य उपयोगकर्ता को कॉल बंद करने के लिए कहेगा। निम्नलिखित कोड जोड़ें -

//hang up 
hangUpBtn.addEventListener("click", function () { 
   send({ 
      type: "leave" 
   });  
	
   handleLeave(); 
});
  
function handleLeave() { 
   connectedUser = null; 
   remoteAudio.src = null;
	
   yourConn.close(); 
   yourConn.onicecandidate = null; 
   yourConn.onaddstream = null;
};

जब उपयोगकर्ता हैंग अप बटन पर क्लिक करता है -

  • यह दूसरे उपयोगकर्ता को "छुट्टी" संदेश भेजेगा
  • यह RTCPeerConnection को बंद कर देगा और स्थानीय रूप से कनेक्शन को नष्ट कर देगा

अब कोड को रन करें। आपको दो ब्राउज़र टैब का उपयोग करके सर्वर में लॉग इन करने में सक्षम होना चाहिए। फिर आप टैब पर एक ऑडियो कॉल कर सकते हैं और कॉल को लटका सकते हैं।

निम्नलिखित संपूर्ण ग्राहक है। जेएस फाइल -

//our username 
var name; 
var connectedUser;
 
//connecting to our signaling server 
var conn = new WebSocket('ws://localhost:9090');
 
conn.onopen = function () { 
   console.log("Connected to the signaling server"); 
};
 
//when we got a message from a signaling server 
conn.onmessage = function (msg) { 
   console.log("Got message", msg.data); 
   var data = JSON.parse(msg.data); 
	
   switch(data.type) { 
      case "login": 
         handleLogin(data.success); 
         break; 
      //when somebody wants to call us 
      case "offer": 
         handleOffer(data.offer, data.name); 
         break; 
      case "answer": 
         handleAnswer(data.answer); 
         break; 
      //when a remote peer sends an ice candidate to us 
      case "candidate": 
         handleCandidate(data.candidate); 
         break; 
      case "leave": 
         handleLeave(); 
         break; 
      default: 
         break; 
   } 
}; 

conn.onerror = function (err) { 
   console.log("Got error", err); 
};
 
//alias for sending JSON encoded messages 
function send(message) { 
   //attach the other peer username to our messages 
   if (connectedUser) { 
      message.name = connectedUser; 
   } 
	
   conn.send(JSON.stringify(message)); 
};
 
//****** 
//UI selectors block 
//****** 

var loginPage = document.querySelector('#loginPage'); 
var usernameInput = document.querySelector('#usernameInput'); 
var loginBtn = document.querySelector('#loginBtn');

var callPage = document.querySelector('#callPage'); 
var callToUsernameInput = document.querySelector('#callToUsernameInput');
var callBtn = document.querySelector('#callBtn'); 

var hangUpBtn = document.querySelector('#hangUpBtn'); 
var localAudio = document.querySelector('#localAudio'); 
var remoteAudio = document.querySelector('#remoteAudio'); 

var yourConn; 
var stream; 

callPage.style.display = "none";
 
// Login when the user clicks the button 
loginBtn.addEventListener("click", function (event) { 
   name = usernameInput.value; 
	
   if (name.length > 0) { 
      send({ 
         type: "login", 
         name: name 
      }); 
   } 
	
});
 
function handleLogin(success) { 
   if (success === false) { 
      alert("Ooops...try a different username"); 
   } else { 
      loginPage.style.display = "none"; 
      callPage.style.display = "block"; 
		
      //********************** 
      //Starting a peer connection 
      //********************** 
		
      //getting local audio stream 
      navigator.webkitGetUserMedia({ video: false, audio: true }, function (myStream) { 
         stream = myStream; 
			
         //displaying local audio stream on the page 
         localAudio.src = window.URL.createObjectURL(stream);
			
         //using Google public stun server 
         var configuration = { 
            "iceServers": [{ "url": "stun:stun2.1.google.com:19302" }] 
         }; 
			
         yourConn = new webkitRTCPeerConnection(configuration); 
			
         // setup stream listening 
         yourConn.addStream(stream); 
			
         //when a remote user adds stream to the peer connection, we display it 
         yourConn.onaddstream = function (e) { 
            remoteAudio.src = window.URL.createObjectURL(e.stream); 
         }; 
			
         // Setup ice handling 
         yourConn.onicecandidate = function (event) { 
            if (event.candidate) { 
               send({ 
                  type: "candidate", 
                  candidate: event.candidate 
               }); 
            } 
         }; 
			
      }, function (error) { 
         console.log(error); 
      }); 
		
   } 
};
 
//initiating a call 
callBtn.addEventListener("click", function () { 
   var callToUsername = callToUsernameInput.value; 
	
   if (callToUsername.length > 0) { 
      connectedUser = callToUsername; 
		
      // create an offer 
      yourConn.createOffer(function (offer) { 
         send({
            type: "offer", 
            offer: offer 
         }); 
			
         yourConn.setLocalDescription(offer); 
      }, function (error) { 
         alert("Error when creating an offer"); 
      }); 
   } 
});
 
//when somebody sends us an offer 
function handleOffer(offer, name) { 
   connectedUser = name; 
   yourConn.setRemoteDescription(new RTCSessionDescription(offer)); 
	
   //create an answer to an offer 
   yourConn.createAnswer(function (answer) { 
      yourConn.setLocalDescription(answer); 
		
      send({ 
         type: "answer", 
         answer: answer 
      });
		
   }, function (error) { 
      alert("Error when creating an answer"); 
   }); 
	
};
 
//when we got an answer from a remote user 
function handleAnswer(answer) { 
   yourConn.setRemoteDescription(new RTCSessionDescription(answer)); 
};
 
//when we got an ice candidate from a remote user 
function handleCandidate(candidate) { 
   yourConn.addIceCandidate(new RTCIceCandidate(candidate)); 
};
 
//hang up
hangUpBtn.addEventListener("click", function () { 
   send({ 
      type: "leave" 
   }); 
	
   handleLeave(); 
});
 
function handleLeave() { 
   connectedUser = null; 
   remoteAudio.src = null; 
	
   yourConn.close(); 
   yourConn.onicecandidate = null; 
   yourConn.onaddstream = null; 
};