网络延迟是一种比较常见的情况。在本地网页上,我们可以建立多个RTCPeerConnection,增加转发次数,来模拟出网络延迟的效果。

建立通话后,再往后面增加本地转发节点。

准备

页面准备,方便我们后面调试

<div id="container">
<h1><a href="https://an.rustfisher.com/webrtc/peerconnection/upgrade-to-video" title="">WebRTC插入多个转发节点</a></h1> <div id="videos">
<video id="video1" playsinline autoplay muted></video>
<video id="video2" playsinline autoplay></video>
</div> <section><input type="checkbox" id="audio"><label for="audio">包含音频(>= Chrome 49)</label></section> <div id="buttons">
<button id="start">开始</button>
<button id="call" disabled>呼叫</button>
<button id="insertRelay" disabled>插入转发</button>
<button id="hangup" disabled>挂断</button>
</div>
<div id="status"></div>
</div> <script src="../../src/js/adapter-2021.js"></script>
<script src="js/connection2.js"></script>
<script src="js/main.js"></script>

放上2个video和几个button。引入WebRTC adapter和控制脚本。

如果要使用官方的适配器adapter,按下边的地址来引入

<!-- <script src="../../src/js/adapter-2021.js"></script> -->
<script src="https://webrtc.github.io/adapter/adapter-latest.js"></script>

控制

分为connection2.jsmain.js

connection2.js包含新建节点的逻辑。main.js控制主流程。

connection2.js

这里能新建2个RTCPeerConnection,建立新的连接。

function doNothing() { }

function errFunc(context) {
return function (error) {
trace('报错 ' + context + ': ' + error.toString);
};
} // 新建2个节点并建立连接
function Connection2(stream, handler) {
let servers = null;
let pc1 = new RTCPeerConnection(servers);
let pc2 = new RTCPeerConnection(servers); pc1.addStream(stream);
pc1.onicecandidate = function (event) {
if (event.candidate) {
pc2.addIceCandidate(new RTCIceCandidate(event.candidate),
doNothing, errFunc('AddIceCandidate'));
}
};
pc2.onicecandidate = function (event) {
if (event.candidate) {
pc1.addIceCandidate(new RTCIceCandidate(event.candidate),
doNothing, errFunc('AddIceCandidate'));
}
};
pc2.onaddstream = function (e) {
handler(e.stream);
};
pc1.createOffer(function (desc) {
pc1.setLocalDescription(desc);
pc2.setRemoteDescription(desc);
pc2.createAnswer(function (desc2) {
pc2.setLocalDescription(desc2);
pc1.setRemoteDescription(desc2);
}, errFunc('pc2.createAnswer'));
}, errFunc('pc1.createOffer'));
this.pc1 = pc1;
this.pc2 = pc2;
} Connection2.prototype.close = function () {
this.pc1.close();
this.pc2.close();
};

Connection2(stream, handler)新建pc1和pc2,将传入的stream作为数据源交给pc1。

随后在pc1和pc2之间建立连接。pc2接到数据流后再交回给handler

main.js

先拿到ui

'use strict';

// 获取ui
const video1 = document.querySelector('video#video1');
const video2 = document.querySelector('video#video2');
const statusDiv = document.querySelector('div#status');
const audioCheckbox = document.querySelector('input#audio');
const startBtn = document.querySelector('button#start');
const callBtn = document.querySelector('button#call');
const insertRelayBtn = document.querySelector('button#insertRelay');
const hangupBtn = document.querySelector('button#hangup');

记录一些变量

const connectionList = []; // 连接点
let localStream;
let remoteStream;

启动,获取数据流。可以选择是否带音频流。拿到数据流后,交给video1显示,并且记录为localStream

function gotStream(stream) {
video1.srcObject = stream;
localStream = stream;
callBtn.disabled = false;
} function start() {
startBtn.disabled = true;
const options = audioCheckbox.checked ? { audio: true, video: true } : { audio: false, video: true };
navigator.mediaDevices.getUserMedia(options)
.then(gotStream)
.catch(function (e) {
alert(`获取媒体失败 ${e}`);
});
}

发起呼叫

function gotremoteStream(stream) {
remoteStream = stream;
video2.srcObject = stream;
console.log('接收到了传输后的数据流');
statusDiv.textContent = `当前节点数: ${connectionList.length * 2}`;
insertRelayBtn.disabled = false;
} function call() {
callBtn.disabled = true;
insertRelayBtn.disabled = false;
hangupBtn.disabled = false;
console.log('呼叫!');
connectionList.push(new Connection2(localStream, gotremoteStream));
}

呼叫的方法是call(),使用Connection2来建立第一级连接。

连接的记录存放在connectionList

插入转发和call有点类似,都使用了Connection2。但是输入的是remoteStream

function insertRelay() {
connectionList.push(new Connection2(remoteStream, gotremoteStream));
insertRelayBtn.disabled = true;
}

多次调用insertRelay(),可以模拟出多层转发的情况。转发次数多了,视频延迟得也就越明显。

挂断/结束方法hangup()

function hangup() {
console.log('挂断');
while (connectionList.length > 0) {
const pipe = connectionList.pop();
pipe.close();
}
insertRelayBtn.disabled = true;
hangupBtn.disabled = true;
callBtn.disabled = false;
}

connectionList里面的连接全部拿出来结束掉。

如果数量比较多,结束耗时也会比较长。

效果预览

效果预览请参考WebRTC插入多个转发节点

可以尝试多点击几次插入转发按钮。观察视频的延迟情况。

小结

本地网页可以通过增加节点的办法,模拟出视频传输延迟的效果。

Connection2(stream, handler)里的代码写的非常简洁,也利于了解WebRTC传输建立的过程

从这个例子我们也可以看出,实际工程中要尽量减少中间节点。并且要优先选择质量高的节点。

本文链接

WebRTC本地插入多个转发节点的更多相关文章

  1. 【译】SSH隧道:本地和远程端口转发

    本文是:SSH Tunnel - Local and Remote Port Forwarding Explained With Examples 的译文 有两种方法可以创建SSH隧道,本地和远程端口 ...

  2. WebRTC本地分享屏幕,录制屏幕

    WebRTC有分享屏幕的功能.使用的是getDisplayMedia方法.用户同意分享屏幕后,可以拿到视频流. 再结合MediaRecorder和Blob,把视频流数据存下来,就能得到录制屏幕的视频. ...

  3. WebRTC本地选择codec(web本地模拟)

    视频编码后,再进行发送.WebRTC建立视频连接前,可以选择codec.一般来说支持多种codec,以VP8和H264为代表. Codec: 编码译码器,编解码器 示例代码 写一个示例,用户可以在发送 ...

  4. Android studio git 本地仓库和远程仓库节点对比

    1.初始状态 2.本地修改文件,然后commit 3.本地再次修改文件,然后commit 4.本地push 从上图可以看出,push完成后,本地仓库的节点和远程仓库的节点是一样的.

  5. Socket 接收本地短连接并转发为长连接 多线程

    import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.io. ...

  6. JMeter扩展Java请求实现WebRTC本地音视频推流压测脚本

    WebRTC是Web Real-Time Communication缩写,指网页即时通讯,是一个支持Web浏览器进行实时语音或视频对话的API,实现了基于网页的视频会议,比如声网的Agora Web ...

  7. DOM操作插入新的子节点

    appendChid.insertBefore首先这两个方法都是添加子节点. append(追加),appendChid:给父节点的子节点末尾添加子节点. insertBefore(newNode, ...

  8. 如何用nginx在本地把9000端口转发到80端口上

    起因看到一个用java写的轻博客,于是就兴致冲冲的试用一下.由于是lnmp的环境,Nginx占用了80端口,新博客只能用其他的端口,这里选择了9000端口,本地测试没问题.总不能访问了域名然后在加上端 ...

  9. (四)WebRTC手记之本地音频采集

    转自:http://www.cnblogs.com/fangkm/p/4374668.html 上一篇博文介绍了本地视频采集,这一篇就介绍下音频采集流程,也是先介绍WebRTC原生的音频采集,再介绍C ...

随机推荐

  1. mabatis的sql标签

    定义:mapper.xml映射文件中定义了操作数据库的sql,并且提供了各种标签方法实现动态拼接sql.每个sql是一个statement,映射文件是mybatis的核心. 一,内容标签 1.Name ...

  2. spring security 认证源码跟踪

    spring security 认证源码跟踪 ​ 在跟踪认证源码之前,我们先根据官网说明一下security的内部原理,主要是依据一系列的filter来实现,大家可以根据https://docs.sp ...

  3. Go语言核心36讲(Go语言实战与应用十九)--学习笔记

    41 | io包中的接口和工具 (下) 上一篇文章中,我主要讲到了io.Reader的扩展接口和实现类型.当然,io代码包中的核心接口不止io.Reader一个. 我们基于它引出的一条主线,只是io包 ...

  4. NFLSOJ #917 -「lych_cys模拟题2018」橘子树(树剖+ODT+莫反统计贡献的思想+动态开点线段树)

    题面传送门 sb 出题人不在题面里写 \(b_i=0\) 导致我挂成零蛋/fn/fn 首先考虑树链剖分将路径问题转化为序列上的问题,因此下文中简称"位置 \(i\)"表示 DFS ...

  5. 洛谷 P3711 - 仓鼠的数学题(多项式)

    洛谷题面传送门 提供一种不太一样的做法. 假设要求的多项式为 \(f(x)\).我们考察 \(f(x)-f(x-1)\),不难发现其等于 \(\sum\limits_{i=0}^na_ix^i\) 考 ...

  6. 斗地主的综合案例实现(Map有序)

    斗地主的综合案例实现(Map有序) 整体思路 代码实现 import java.util.ArrayList; import java.util.Collections; import java.ut ...

  7. SUNTANS 及 FVCOM 对流扩散方程求解简介[TBC]

    最近接到一个任务,就是解决FVCOM中对流扩散计算不守衡问题.导师认为是其求解时候水平和垂向计算分开求解所导致的,目前我也没搞清到底有什么问题,反正就是让把SUNTANS的对流扩散计算挪到FVCOM中 ...

  8. perl 数组快速去除重复元素

    这里记录两种perl数组去重的办法,一种利用哈希(hash),一种直接利用perl自带的模块List::MoreUtils内部的函数uniq. 一.利用hash去重 示例代码如下: 1 #!/usr/ ...

  9. 如何优雅地将printf的打印保存在文件中?

    我们都知道,一般使用printf的打印都会直接打印在终端,如果想要保存在文件里呢?我想你可能想到的是重定向.例如: $ program > result.txt 这样printf的输出就存储在r ...

  10. C/C++ Qt StatusBar 底部状态栏应用

    Qt窗体中默认会附加一个QstatusBar组件,状态栏组件位于主窗体的最下方,其作用是提供一个工具提示功能,当程序中有提示信息是可以动态的显示在这个区域内,状态栏组件内可以增加任何Qt中的通用组件, ...