webrtc 入门第五章 一对一视频通话实现

这篇具有很好参考价值的文章主要介绍了webrtc 入门第五章 一对一视频通话实现。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

webrtc 入门第五章 一对一视频通话实现

一、介绍

​ 在前面的章节我们学习了如何操作本地的设备摄像头,麦克风等,学会了如何进行本地的流媒体操作如录制,下载,同步等。在第三第四章节学习了webrtc的一对一连接的原理和实操并且实现了简单的数据传输。

​ 但是之前的实践在两个不同的设备之间还不能实现真正意义上的通话,当两端不在一台设备上是还不能通信。要实现远程的两个设备间的数据传输还需要借助信令服务器和STUN服务器。

二、实践
1、通话流程

​ 一对一的视频通话连接流程和第三章的连接流程一样,学者可以详细阅读第三章的内容和实际操示例。整个通话的流程相对来说还是比较复杂,需要借助信令服务器和STUN服务器。整个系统设计如下

1.正常的用户体系,包括用户登录,注册,查看用户列表,请求通话,通话详情页面,用户挂断,用户退出等基础功能

2.用户A:作为会话的发起方,创建提议offer

3.用户B:会话应答方接到A发来的提议后创建应答answer

4.信令服务器:websocket服务连接用户A和用户B,转发双方的SDP及Candidate信息,以及用户上下线,请求通话,挂断,拒绝通话等消息。

5.STUN服务器:用于接收用户A 、B的ICE请求,从而获取各自的Candidate信息,再通过信令服务器转发至双方,另外STUN服务器也有转发媒体数据的功能

6.服务连接,双方获取本地媒体流及交换媒体流。

2、技术框架

webrtc 入门第五章 一对一视频通话实现

在本次实现该功能的技术路线中可以看到客户端为web PC-A和PC-B 服务器需要两个具体如下

1.PC web端:web端使用的是vue+h5实现了用户注册登录,进入聊天列表,发起视频请求等功能

2.信令服务:本次使用github.com/gorilla/websocket 包实现了websocket功能,主要用来用户登录,离线,消息通知,和转发Offer、Answer,Candidate等数据。

3.web服务:使用golang的gin包实现了https服务,主要提供了页面渲染,群聊,mysql存贮用户数据等

4.STUN服务器:本次使用turnserver 服务实现STUN及TURN媒体数据中转,可以用现有服务,也可以自行安装

安装教程 https://blog.csdn.net/qq_32435729/article/details/78729093

4、信令设计

信令就是两个客户端之间信息交换的数据,如会话,开始通话,结束通话,用户上线,offer,answer等数据的交换,需要提供统一的数据格式,根据不同的消息类型客户端和服务端处理不同的数据业务。

// 接收定义消息结构
type ReceiveMessage struct {
   // 请求的方法
   Method string `json:"method"`
   // 消息类型(1,text,0:系统消息)
   Type int8 `json:"type"`
   // 消息体
   Message string `json:"message"`
   // 消息来源用户Id
   FromId string `json:"fromId"`
   // 当前连接
   Client *websocket.Conn `json:"client"`
   // 数据参数
   Data map[string]interface{} `json:"data"`
}

// 发送消息结构
type SendMessage struct {
	// 请求的方法
	Method string `json:"method"`
	// 消息类型(1,text,0:系统消息))
	Type uint8 `json:"type"`
	// 消息体
	Message string `json:"message"`
	// Code 0 正确,1 错误
	Code int16 `json:"code"`
	// 数据参数
	Data interface{} `json:"data"`
}

本次设计的主要信令方法 method 有如下几种

类型 说明
Message/Offer 议题SDP
Message/Answer 应答SDP
Message/Candidate 交换ICECandidate网络信息
User/Connect 用户连接
Message/SendToAll 发消息
let data={
    "Method": "Message/Candidate",
    "Type": 0,
    "Message": that.user.name + "发给" + userId + "candidate信息",
    "fromId": that.user.id,
    "toId": userId,
    "data": {
        "candidate": {
            "sdpMlineIndex": event.candidate.sdpMLineIndex,
            "sdpMid": event.candidate.sdpMid,
            "candidate": event.candidate.candidate
        }
    }
}
send(data)

不同的业务发送不同的数据类型,到服务器,服务器根据toId将消息转发给对应的用户id,在用户收到消息后,需要根据不同的返回业务码来处理业务,业务处理主要在websocket的onmessage回调方法中

onmessage: function () {
    let that = this

    console.log(that.users)

    that.ws.onmessage = function (event) {
        let msg = JSON.parse(event.data)
        console.log("收到消息-----")
        console.log(msg)
        if (msg.code == 10000) {
            that.users = msg.data
        }
        if (msg.code == 10005) {
            that.addMessage('from ' + msg.data.username + ": ", msg.data.message);
            return;
        }
        if (msg.code == 10006) {
            // 有用户登录
            console.log(that.users)
            if (that.users == null) {
                that.users = []
                that.users.push(msg.data)
                return;
            }
            let isIn = 0

            for (let i = 0; i < that.users.length; i++) {
                if (that.users[i].id == msg.data.id) {
                    isIn = 1
                }
            }
            if (isIn == 0) {
                that.users.push(msg.data)
            }
        }
        if (msg.code == 10008) {
            let userId = msg.data.fromId
            let newUser = []
            for (let i = 0; i < that.users.length; i++) {
                if (that.users[i].id != userId) {
                    newUser.push(that.users[i])
                }
            }
            that.users = newUser
        }
        if (msg.code == 10009) {
            that.onCandidate(msg)
            return;
        }
        if (msg.code == 10010) {
            that.onOffer(msg)
            return;
        }
        if (msg.code == 10011) {
            that.onAnswer(msg)
            return;
        }


        that.addMessage('系统消息', msg.message);
    }
5、服务后台

main.go 是整个函数的入口函数使用goang 的gin框架实现了http服务,包括路由websocket路由等

package main

import ( 
   "ginweb/controllers/wss"
   "ginweb/dao"
   "ginweb/route"
   "ginweb/runtime"
   "github.com/gin-gonic/gin"
)

func init() {
   go wss.HandleMessages()
}

func main() {

   config.InitConfig()
   runtime.InitLog()
   dao.Install()
   defer dao.Uninstall()

   r := route.RegisterRouters()  // 注册路由

   r.GET("/wss", wss.OnWssMessage) // websockt路由
   r.LoadHTMLGlob("www/**/**/*") // 加载静态文件
   r.StaticFS("/static", http.Dir("./static"))
   r.Run(":" + config.Data.Port)
}

route.go路由文件加载路由

package route

import (
   "ginweb/controllers/blog"
   "ginweb/controllers/elasticSearch"
   "ginweb/controllers/game/gobang"
   "ginweb/controllers/webrtc"
   "github.com/gin-gonic/gin"
)

// @Title 注册路由
// @return  route  *gin.Engine

func RegisterRouters() *gin.Engine {
   r := gin.Default()
   webrtcGroup := r.Group("/webrtc")
   {
      webrtcGroup.GET("/login", webrtc.LoginPage)
      webrtcGroup.POST("/login", webrtc.Login)
      webrtcGroup.POST("/register", webrtc.Register)
      webrtcGroup.GET("/admin", webrtc.ShowHead)
      webrtcGroup.Use(webrtc.LoginAuth(r)) // 验证登录
      // webrtc 基本操作
      webrtcGroup.GET("/in", webrtc.Home)
      webrtcGroup.GET("/wss", webrtc.OnWsMessage)
   }
   return r
}

wss.go文件 实现了websockt用户连接及断开服务的数据绑定等

func OnWsMessage(req *gin.Context) {
   var loginMsg webrtc.LoginMessage
   r := req.Request
   w := req.Writer
   c, err := upgrader.Upgrade(w, r, nil)
   if err != nil {
      zaplogger.Error(err)
      c.Close()
      return
   }

   err = c.ReadJSON(&loginMsg)
   if err != nil {
      zaplogger.Error(err)
      c.Close()
      return
   }

   // 关闭连接需要修改
   defer logout(loginMsg, c)

   if loginMsg.FromId <= 0 {
      return
   }

   // 2.接收到用户连接,执行登录
   code, res := login(loginMsg, c)
   if code > 0 {
      zaplogger.Error("connect error:"+res, code, loginMsg)
      return
   }

   // 系统监听用户消息
   for {
      // 1.处理当前用户获取系统消息
      var userMsg webrtc.ReceiveMessage
      err = c.ReadJSON(&userMsg)
      if err != nil {
         zaplogger.Error("收到消息 json解析err:", err)
         break
      }
      if userMsg.Method == "User/Connect" {
         continue
      }

      // TODO message 去掉 client 和指针
      zaplogger.Info("收到消息:->", userMsg)
      code, res := GetRouter(userMsg)
      zaplogger.Info("处理结果:->", code, res)
   }

}


message.go 文件主要包括转发会话消息等

// @Title SendToAll
// @Description 批量消息发送给所有在线用户
// @Param   message    用户消息
// @return   code  int16  返回码
// @return   message  string   消息

func (m *Message) Candidate(message webrtc.ReceiveMessage) (code int16, res string) {
   var returnData map[string]interface{}
   returnData = make(map[string]interface{})
   returnData["fromId"] = message.FromId
   returnData["data"] = message.Data
   m.SendToIds([]int32{message.ToId}, webrtc.MessageCandidate, message.Message, message.Method, returnData, 1)
   return
}

// @Title SendToAll
// @Description 批量消息发送给所有在线用户
// @Param   message    用户消息
// @return   code  int16  返回码
// @return   message  string   消息

func (m *Message) Offer(message webrtc.ReceiveMessage) (code int16, res string) {
   var returnData map[string]interface{}
   returnData = make(map[string]interface{})
   returnData["fromId"] = message.FromId
   returnData["data"] = message.Data
   m.SendToIds([]int32{message.ToId}, webrtc.MessageCreateOffer, message.Message, message.Method, returnData, 1)
   return
}

func (m *Message) Answer(message webrtc.ReceiveMessage) (code int16, res string) {
   var returnData map[string]interface{}
   returnData = make(map[string]interface{})
   returnData["fromId"] = message.FromId
   returnData["data"] = message.Data
   m.SendToIds([]int32{message.ToId}, webrtc.MessageAnswer, message.Message, message.Method, returnData, 1)
   return
}

在上述的代码中只是体现了部分重要的代码模块,有笔者需要的话可以查看我的dome

https://e.coding.net/caoxiukang123456/ginweb/ginweb.git 本文中部分实现只作为学习可参考。

webrtc 入门第五章 一对一视频通话实现

​ 一对一pc

webrtc 入门第五章 一对一视频通话实现

​ 手机与pc一对一

6、服务部署

1、本次前端采用layui+h5+vue.js 实现页面的组件展示,因此需要在项目中导入layui.js和vue.js

2、后台web服务采用golang的gin框架实现http服务及页面功能,在服务器需要安装golang 1.16.3版本

wget https://studygolang.com/dl/golang/go1.16.1.linux-amd64.tar.gz 
tar -C /usr/local -xzf go1.16.1.linux-amd64.tar.gz 
vim /etc/profile 
export GOROOT=/usr/local/go #设置为go安装的路径 
export GOPATH=/home/gocode #默认安装包的路径 
export PATH=$PATH:$GOROOT/bin:$GOPATH/bin
source /etc/profile  // 生效
go env -w GOPROXY=https://goproxy.cn,direct

3、golang 1.14版本后使用go.mod进行包管理创建项目很简单,进入到项目的根目录下执行

go mod init ginweb
go mod tidy
go run main.go  

4、安装stun服务器具体见其他教程:https://blog.csdn.net/qq_32435729/article/details/78729093 本教程讲的安装思路明确,测试方法也清晰

5、webrtc需要通过域名+https访问因此服务器需要打开80,443,以及stun服务器的 3478 端口。另外https服务需要提供证书。本次部署我是采用宝塔面板安装简单,http和websocket服务只需要做端口转发即可

三、总结

通过前面几章节的学习后我们学会了,操作本地媒体流,媒体操作渲染,数据通道,数据发送等基本功能,并且实现了信令服务,STUN服务器等单独功能,前期准备工作做好后就可以将所有功能整合起来,实现webrtc一对一视频通话功能。在实际处理过程中需要注意以下问题。

1、在用户发送会话提议offer和Answer的时候逻辑代码在一个页面书比较复杂而且挺绕,需要提前熟悉webrtc的连接流程,指导原理后书写就很简单。

2、系统中的用户体系在聊天界面中可以实现消息会话以及实现发起聊天,挂断等操作。

3、stun服务器目前网上有很多免费的但是好多都不可用,如果自己不想搭建的话 可以使用免费的,但是使用前需要测试可用性 https://webrtc.github.io/samples/src/content/peerconnection/trickle-ice/ 该网址可以测试

4、自行搭建的STUN服务器一般都自带turn服务。搭建完成后依然需要自行测测试

5、关于STUN服务器的原理以及webrtc端到端的连接原理会在下一章整理。文章来源地址https://www.toymoban.com/news/detail-410259.html

到了这里,关于webrtc 入门第五章 一对一视频通话实现的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • webRTC一对一音视频对话

    环境 阿里云操作系统: ubuntu 18.4 amd         注意:安全组一定添加对应的入口端口 nodejs -v 18.19.0 npm -v 10.2.3 需要安装的库 package.json 服务器端         webRTC一定要使用https服务器,如果没有ssl证书,可以使用自制证书         1.创建HTTPS服务器                 使用soc

    2024年01月19日
    浏览(44)
  • Windows7中使用SRS集成音视频一对一通话

    SRS早就具备了SFU的能力,比如一对一通话、多人通话、直播连麦等等。在沟通中,一对一是常用而且典型的场景, 让我们一起来看看如何用SRS做直播和RTC一体化的一对一通话。 一、启动windows7-docker  二、拉取SRS镜像 执行命令:docker pull ossrs/srs:4 使用docker images查看当前镜像列

    2024年02月16日
    浏览(33)
  • uniapp集成腾讯即时通信IM,实现一对一聊天,支持文字、表情、语音、图片、视频

    原则 介绍 效果图 uniapp集成腾讯即时通信IM,实现一对一聊天,支持文字、 使用方式 将文件放到相应的位置 app配置 main.js 配置 pages.json 配置 GenerateTestUserSig.js配置SDKAPPID和SECRETKEY 安装 npm i 运行 下载源码 联系方式 查看文章

    2024年02月09日
    浏览(77)
  • [MySql]实现多表查询-一对一,一对多

    目录 多表关联关系的分类 mybatis中的多表查询: 数据库准备  项目目录  一对一查询(多对一) 方式一:(xml文件映射) 查询结果:  方式二:(注解映射)  一对多查询(一对多) 方式一:(xml文件映射)  方式二:(注解映射) 既然数据库是存储项目中的数据的,项目中的

    2023年04月10日
    浏览(39)
  • TCP实现一对一聊天

    一,创建类 二,类 1.ChatSocketServer类 2.ChatSocketClient类 三,结果(先服务  再客户)

    2024年02月03日
    浏览(35)
  • Springboot + websocket 实现 一对一 单人聊天

    Springboot + websocket   实现 一对一 单人聊天 打开项目中的pom.xml,添加以下内容 配置websocke的endpoints 配置websocket的server @ServerEndpoint (value = \\\"/websocket/{username}\\\" ) 这句话 一定要注意, 这里 路径指定的是 页面需要访问的server的地址, {username} 表示需要传递参数 为刚刚的类 添加 属性

    2024年01月17日
    浏览(91)
  • java使用WebSocket实现一对一实时对话

    最近的项目中有实时对话的需求。这里也是查阅了很多的资料。使用webSocket实现一对一对话是最多的。 链接: https://pan.baidu.com/s/1Vn1e1qw7nRnU1_4R-4fcGg 提取码: qwer  逻辑讲解: 现在我们要给张三丰发一个你好,我们要解决下面两个问题 ·  这个你好怎么展示在我的窗口 ·   这个

    2023年04月09日
    浏览(36)
  • NRF24L01学习操作教程(二)——NRF实现一对一,一对多通讯

    上篇博客链接:https://blog.csdn.net/DIVIDADA/article/details/130599974?spm=1001.2014.3001.5501 以下单片机例程都是基于STM32 HAL库,在文档末尾,我会提供参考博客和源码程序的链接。 在CubeMx中配置单片机时钟、SPI通讯接口、NRF24L01接口等,并生成Keil工程 将NRF24L01的驱动程序的.c文件和.h文件

    2024年01月15日
    浏览(34)
  • [Django-04]一对一,一对多

    OneToOneField(),ForeignKey() 的参数如下 to 要连接的模型 to_field 要被关联的目标Model的字段 on_delete 删除主表(被关联的表,也就是非OneToOneField,ForeignKey的表)时,当前表怎么处理。 CASCADE-联动删除 PROTECT -抛出异常 SET_NULL-设为null SET_DEFAULT-设为预定义的默认值 SET-设置为指定的值

    2024年02月04日
    浏览(38)
  • hibernate 一对一 一对多 多对多

    User 实体类 Address 实体类 测试 User实体类 Vlog实体类 测试 测试 mappedby : 属性指向实体关联表的拥有者,声明在被拥有者。 简单说就是另一边定义了关联规则,这边不用再定义一遍了,直接引用就行。 @JoinColumn : 外键列 在一对一中 @JoinColumn 声明在那个实体类中,生成数据库表

    2024年02月13日
    浏览(32)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包