- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我的 webrtc 聊天代码有问题。我无法在同行之间建立连接。 PC1通过信号服务器成功发送offer请求,pc2添加候选人,但pc2无法发送候选人并回答pc1,为什么?
我找了好几个小时都找不到错误,请帮我建立 webrtc 连接。
app.js
var app = require('express')();
var server = require('http').createServer(app);
var virtualDirPath = process.env.virtualDirPath || '';
var io = require('socket.io')(server)//(server, { path: virtualDirPath + '/socket.io' });
var users = {};
server.listen(process.env.PORT || 8080, function () {
console.log('Подняли сервер на *:8080');
});
app.get('/', function (req, res) {
res.sendFile(__dirname + '/index.html');
});
app.get('/style.css', function (req, res) {
res.sendFile(__dirname + '/style.css');
});
app.get('/index.js', function (req, res) {
res.sendFile(__dirname + '/index.js');
});
io.on('connection', function (socket) {
socket.on('login', function(data){
console.log('Зашел пользователь:', data.name);
//Не даем подключиться если пользователь уже в чате
if(users[data.name]) {
socket.emit('login', {state: 'taken'});
}else{
socket.name = data.name;
users[data.name] = socket;
socket.broadcast.emit('newuser', {name: data.name});
socket.emit('login', {name: data.name});
}
});
//format offer
socket.on('offer', function(data){//data.name, data.localDescription
console.log('Поулчили offer для:', data.to);
users[data.to].emit("offer", data);
});
socket.on('answer', function(data){//data.name, data.localDescription
console.log('Поулчили answer для:', data.to);
console.log(data);
users[data.to].emit("offer", data);
});
socket.on('candidate', function(data){//data.name, data.candidate
console.log('Поулчили candidate для:', data.to);
users[data.to].emit("candidate", data);
});
socket.on('disconnect', function (data) {
console.log(data);
io.emit('user disconnected',data.to);
io.sockets.emit('quit', {name: data.from})
delete users[data.from];
});
});
index.html:
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title></title>
<link rel="stylesheet" type="text/css" href="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.6/css/bootstrap.min.css">
<link rel="stylesheet" type="text/css" href="style.css">
<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/2.1.4/jquery.js"></script>
<script src="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.6/js/bootstrap.min.js"></script>
</head>
<body>
<div class="modal fade" id="login-modal" tabindex="-1" role="dialog" data-keyboard="false" data-backdrop="static" aria-hidden="true" style="display: none;">
<div class="modal-dialog">
<div class="loginmodal-container">
<h1>Войдите в чат</h1><br>
<form id="loginForm">
<input type="text" name="username" placeholder="Username">
<input type="submit" id="login" class="login loginmodal-submit" value="Вход">
</form>
</div>
</div>
</div>
<script src="https://cdn.socket.io/socket.io-1.3.5.js"></script>
<script src="index.js"></script>
</body>
</html>
index.js:
var PeerConnection = window.mozRTCPeerConnection || window.webkitRTCPeerConnection;
var IceCandidate = window.mozRTCIceCandidate || window.RTCIceCandidate;
var SessionDescription = window.mozRTCSessionDescription || window.RTCSessionDescription;
var socket;
var peers = {};
var server = {
iceServers: [
{url: "stun:23.21.150.121"},
{url: "stun:stun.l.google.com:19302"}
]
};
var options = {
optional: [
{DtlsSrtpKeyAgreement: true}, // требуется для соединения между Chrome и Firefox
{RtpDataChannels: true} // требуется в Firefox для использования DataChannels API
]
}
$(document).ready(function () {
$('#login-modal').modal('show');
//Логинимся.
$('#loginForm').on('submit', function(e) {
e.preventDefault();
var name = $('input[name=username]').val();
if (!name) {
alert('Вы не ввели логин');
return;
}else{
//подключаемся к сигнальному серверу
socket = io.connect(':8080', {
forceNew: true
});
socket.emit('login',{name: name})
}
socket.on('login', function(data){
var current_name = data.name;
if(data.state === 'taken'){
alert('Пльзователь с таким ником уже находится в чате');
}else if(data.name != undefined){
console.log('Успешно зашли в чат');
socket.on('newuser', function(data){
console.log('receive newuser ', data.name);
peers[data.name] = {
cache: []
};
// Создаем новое подключение
var pc = new PeerConnection(server, options);
// Инициализируем его
initConn(pc, data.name, current_name, "offer");
// Сохраняем пир в списке
peers[data.name].connection = pc;
// Создаем DataChannel по которому и будет происходить обмен сообщениями
var channel = pc.createDataChannel("chatchannel", {});
channel.owner = data.name;
peers[data.name].channel = channel;
// Устанавливаем обработчики событий канала
bindEvents(channel);
// Создаем SDP offer
pc.createOffer(function(offer) {
console.log('Установили локальный дескрипшен');
pc.setLocalDescription(offer);
},function(err){
console.log(err);
});
});
socket.on('candidate', function(data){
console.log('receive candidate from ', data.from);
createConnection(data.from, current_name);
var pc = peers[data.from].connection;
console.log(data.candidate);
pc.addIceCandidate(new IceCandidate(data.candidate));
});
socket.on('offer', function(data){ //name,localDescription
console.log('receive offer from ', data.from);
createConnection(data.from, current_name);
var pc = peers[data.from].connection;
console.log(pc);
pc.setRemoteDescription(new SessionDescription(data.localDescription));
pc.createAnswer(function(answer) {
pc.setLocalDescription(answer);
console.log('answer created');
},function(err){
console.log(err);
});
});
socket.on('answer', function(data){
console.log('receive answer from ', data.from);
var pc = peers[data.name].connection;
pc.setRemoteDescription(new SessionDescription(data.localDescription));
});
}else{
alert('Ошибка при логине');
location.reload();
}
});
});
});
function createConnection(name, current_name){
//Инициализируем подключение если его нет
console.log(peers[name])
if (peers[name] === undefined){
peers[name] = {
cache: []
};
var pc = new PeerConnection(server, options);
initConn(pc, name, current_name, 'answer');
peers[name].connection = pc;
pc.ondatachannel = function(e) {
peers[name].channel = e.channel;
peers[name].channel.owner = name;
bindEvents(peers[name].channel);
}
console.log('CreateConnection');
console.log(pc)
}
}
function initConn(pc, name, current_name, sdpType) {
console.log(pc);
console.log('-----------------------------------------')
pc.onicecandidate = function (event) {
if (event.candidate) {
// При обнаружении нового ICE кандидата добавляем его в список для дальнейшей отправки
peers[name].cache.push(event.candidate);
console.log('1');
} else {
// Когда обнаружение кандидатов завершено, обработчик будет вызван еще раз, но без кандидата
// В этом случае мы отправялем пиру сначала SDP offer или SDP answer (в зависимости от SDP запроса)
socket.emit(sdpType, {to: name, from: current_name, localDescription: pc.localDescription});
console.log('init conn');
console.log(name);
console.log(current_name);
console.log(sdpType);
// ...а затем все найденные ранее ICE кандидаты
for (var i = 0; i < peers[name].cache.length; i++) {
socket.emit("candidate", {to: name, from: current_name, candidate: peers[name].cache[i]});
}
}
}
pc.oniceconnectionstatechange = function (event) {
if (pc.iceConnectionState == "disconnected") {
//Пир отключился
//TODO добавить в список пользоавтелей data.name
delete peers[id];
}
}
}
function bindEvents (channel) {
channel.onopen = function () {
//TODO добавить в список пользоавтелей channel.owner
};
channel.onmessage = function (e) {
//TODO add text of message to chat div e.data
};
}
PC2 有本地描述和远程描述。PC2对等连接日志:
RTCPeerConnection {localDescription: RTCSessionDescription, remoteDescription: RTCSessionDescription, signalingState: "stable", iceGatheringState: "new", iceConnectionState: "new"…}
iceConnectionState
:
"new"
iceGatheringState
:
"new"
localDescription
:
RTCSessionDescription
sdp
:
"v=0
↵o=- 4651396889672739307 2 IN IP4 127.0.0.1
↵s=-
↵t=0 0
↵a=msid-semantic: WMS
↵m=application 0 UDP/TLS/RTP/SAVPF 127
↵c=IN IP4 0.0.0.0
↵a=rtcp:9 IN IP4 0.0.0.0
↵a=mid:data
↵a=recvonly
↵a=rtcp-mux
↵a=rtpmap:127 google-data/90000
↵"
type
:
"answer"
__proto__
:
RTCSessionDescription
onaddstream
:
null
ondatachannel
:
(e)
onicecandidate
:
(event)
oniceconnectionstatechange
:
(event)
onnegotiationneeded
:
null
onremovestream
:
null
onsignalingstatechange
:
null
remoteDescription
:
RTCSessionDescription
sdp
:
"v=0
↵o=- 1739995165662969380 2 IN IP4 127.0.0.1
↵s=-
↵t=0 0
↵a=group:BUNDLE data
↵a=msid-semantic: WMS
↵m=application 2740 UDP/TLS/RTP/SAVPF 127
↵c=IN IP4 5.141.232.232
↵b=AS:30
↵a=rtcp:2120 IN IP4 5.141.232.232
↵a=candidate:985767174 1 udp 2113937151 192.168.100.2 58907 typ host generation 0
↵a=candidate:985767174 2 udp 2113937150 192.168.100.2 58909 typ host generation 0
↵a=candidate:842163049 1 udp 1677729535 5.141.232.232 2740 typ srflx raddr 192.168.100.2 rport 58907 generation 0
↵a=candidate:842163049 2 udp 1677729534 5.141.232.232 2120 typ srflx raddr 192.168.100.2 rport 58909 generation 0
↵a=candidate:842163049 2 udp 1677729534 5.141.232.232 2910 typ srflx raddr 192.168.100.2 rport 58909 generation 0
↵a=candidate:842163049 1 udp 1677729535 5.141.232.232 2830 typ srflx raddr 192.168.100.2 rport 58907 generation 0
↵a=ice-ufrag:TYKF98cRmZWIRAm5
↵a=ice-pwd:viHciQB6l36zfTuHdZCnJDPY
↵a=fingerprint:sha-256 10:01:2C:0C:5D:B8:CC:32:15:D3:0D:0A:D3:50:BD:13:B8:9F:87:DF:97:1C:15:13:84:80:68:83:AB:FA:44:E8
↵a=setup:actpass
↵a=mid:data
↵a=sendrecv
↵a=rtcp-mux
↵a=rtpmap:127 google-data/90000
↵a=ssrc:2123048988 cname:pRuKm8BiQ79bLvJg
↵a=ssrc:2123048988 msid:chatchannel chatchannel
↵a=ssrc:2123048988 mslabel:chatchannel
↵a=ssrc:2123048988 label:chatchannel
↵"
type
:
"offer"
__proto__
:
RTCSessionDescription
signalingState
:
"stable"
__proto__
:
RTCPeerConnection
最佳答案
问题是 Chrome 和 Firefox 中 FTPData Channels 的标志已过时,这一问题花费了我生命中最后 14 个小时。不要使用它。好好照顾自己。现在一切都在轮子上进行。
关于javascript - Onicecandidate 不会在第二台电脑上触发,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36814735/
我的 webrtc 聊天代码有问题。我无法在同行之间建立连接。 PC1通过信号服务器成功发送offer请求,pc2添加候选人,但pc2无法发送候选人并回答pc1,为什么? 我找了好几个小时都找不到错误
下面我摘录了this link关于 RTCPeerConnection.onicecandidate 并希望根据我的理解提出两个问题,如果我的概念正确与否,则需要帮助。对我来说有点复杂 The RTC
我正在创建一个应用程序,它将使用 WebRTC 将相机视频共享到多个对等连接。服务器只是为用户提供了一个房间,房间里的所有用户都会看到摄像头视频。唯一不起作用的是 onicecandidate 没有触
我在 Chrome 浏览器中遇到了一个非常奇怪的 WebRTC 行为。我使用这样的简单代码: peerConn.onicecandidate = function (event) { if (
所以我关注了this tutorial了解如何实现 WebRTC 服务器-客户端设置。一旦我开始工作,我想将客户端分成两部分,一个发送者和一个接收者。现在他们可以相互建立连接,但接收方永远不会从发送方
我无法理解 webRTC 及其 PeerConnection 和“onicecandidate”事件。 据我了解,您必须使用 STUN(或 TURN)服务器启动对等连接,因为它会将您发回 ICE 候选
我在 Chrome 浏览器 v30 中运行我自己的 webrtc 演示代码时遇到问题。但该代码在 Firefox 上完美运行。 onicecandidate 事件在其他对等方接受报价之前触发。另一方面
大家好,我对这个 webrtc 和 javascript 很陌生,我尝试完成视频流,但在我的代码中,onicecandidate 从未被调用,有人可以帮助我解决我犯错误的地方 $('body #ca
我试图欺骗从 WebRTC 泄漏的 IP 地址,所以我想重写“onicecandidate”回调函数,但下面的代码不起作用,我不明白为什么。 Object.defineProperty(RTCPe
我的 webrtc 网络应用程序从来没有出现过 onicecandidate。当我查看本地(Webrtc 对象)对象时,设置了本地描述和远程描述。在 Firefox 中,它会触发 onicecandi
我很难理解 onicecandidate 函数的用途。ice candidates 如何帮助建立对等连接。谁先发送 ice candidates? 最佳答案 在幕后,webrtc 根据可用的网络连接收
使用的浏览器是 Chrome...我有调用者和接收者代码来生成 SDP 和 ICE 候选者。我获取调用者代码以使用 sdpMid=video 生成正确的 SDP 和 ICE 候选者,但对于接收者,我仅
我是一名优秀的程序员,十分优秀!