1、用户A和用户B实现通信双方间建立链接最重要的是双方获取彼此的sdp信息和ice信息。
sdp就是一段文本描述,里面包含了当前本地设备所支持的一些信息,比如设备是否支持h264编码,传输协议是什么。
ice也是一段文本,是配合p2p打洞服务器stun/turn让双方知道各自的公网ip和端口,从而实现端对端通信。
对于实现一个简单入门通信案例来说,不要太去深入的理解每个名词的具体意思,只要能总的明白是做什么就行了,等入门后再去深入理解每个知识点。
简单理解就是sdp是设备描述文本,ice就是ip端口描述文本
2、sdp,ice的创建和使用
用户A,B都需要设置setLocationDescription和setRemoteDescription。
呼叫者调用createOffer创建sdp,被呼叫者调用createAnswer创建sdp
ice信息是在创建peerConnection后会自动从stun/turn服务器请求回调。
下面来具体实现
xml布局
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="#000"
tools:context=".ui.SimpleMeetingActivity">
<!--播放远端视频-->
<org.webrtc.SurfaceViewRenderer
android:layout_marginBottom="200dp"
android:id="@+id/ivRemoteRender"
android:layout_width="match_parent"
android:layout_height="match_parent"/>
<!--播放本地视频-->
<org.webrtc.SurfaceViewRenderer
android:id="@+id/ivLocalRender"
app:layout_constraintTop_toTopOf="parent"
app:layout_constraintRight_toRightOf="parent"
android:layout_width="150dp"
android:layout_height="200dp"/>
<View
android:id="@+id/ivJoin"
android:layout_marginBottom="20dp"
app:layout_constraintRight_toRightOf="parent"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintBottom_toBottomOf="parent"
android:background="#4CAF50"
android:layout_width="100dp"
android:layout_height="100dp"/>
</androidx.constraintlayout.widget.ConstraintLayout>
1.引入webrtc Android 和websocket
implementation 'org.webrtc:google-webrtc:1.0.32006'
//引入socket依赖
implementation 'org.java-websocket:Java-WebSocket:1.4.0'
2.全局初始化
//在初次使用PeerConnectionFactory之前,必须调用静态方法initialize()对其进行全局的初始化与资源加载
PeerConnectionFactory.InitializationOptions initializationOptions =
PeerConnectionFactory.InitializationOptions
.builder(this)
.setEnableInternalTracer(true)// 启用内部追踪器,用来记录一些相关数据
.createInitializationOptions();
PeerConnectionFactory.initialize(initializationOptions);
3.创建PeerConnectionFactory
//-----------创建视频编码和解码器
VideoEncoderFactory encoderFactory = new DefaultVideoEncoderFactory(mRootEGL.getEglBaseContext(),
true, true);
VideoDecoderFactory decoderFactory = new DefaultVideoDecoderFactory(mRootEGL.getEglBaseContext());
//-----------创建PeerConnectionFactory
AudioDeviceModule adm = JavaAudioDeviceModule.builder(this).createAudioDeviceModule();//音频配置当前JAVA实现,还有native
PeerConnectionFactory.Options options = new PeerConnectionFactory.Options();
//options.disableEncryption : true表示不用数据加密
//options.disableNetworkMonitor : true表示禁用网络监视器
factory = PeerConnectionFactory.builder()
.setOptions(options)//设置网络配置,使用默认
.setAudioDeviceModule(adm)//设置音频采集和播放使用的配置,当前使用java中的audioTrack 和audioRecord
.setVideoEncoderFactory(encoderFactory)
.setVideoDecoderFactory(decoderFactory)
.createPeerConnectionFactory();
4.创建声音源
//配置音频约束
MediaConstraints audioConstraints = new MediaConstraints();
//回声消除
audioConstraints.mandatory.add(new MediaConstraints.KeyValuePair("googEchoCancellation", "true"));
//自动增益
audioConstraints.mandatory.add(new MediaConstraints.KeyValuePair("googAutoGainControl", "true"));
//高音过滤
audioConstraints.mandatory.add(new MediaConstraints.KeyValuePair("googHighpassFilter", "true"));
//噪音处理
audioConstraints.mandatory.add(new MediaConstraints.KeyValuePair("googNoiseSuppression", "true"));
audioSource = factory.createAudioSource(audioConstraints);
//创建音频轨道
localAudioTrack = factory.createAudioTrack("102", audioSource);
5.创建视频源
//ScreenCapturerAndroid 录屏 ;FileVideoCapturer文件
//从相机里获取视频流
CameraEnumerator enumerator = new Camera2Enumerator(this);
String[] deviceNames = enumerator.getDeviceNames();
//遍历所有摄像头找到前置摄像头
for (String deviceName : deviceNames) {
if (enumerator.isFrontFacing(deviceName)) {
videoCapturer = enumerator.createCapturer(deviceName, null);
}
}
assert videoCapturer != null;
videoSource = factory.createVideoSource(videoCapturer.isScreencast());
//创建视频轨道
videoTrack = factory.createVideoTrack("103", videoSource);
6.播放本地视频
//因为使用的是opengl 渲染
surfaceTextureHelper = SurfaceTextureHelper.create("capture-thread", mRootEGL.getEglBaseContext());
videoCapturer.initialize(surfaceTextureHelper, this, videoSource.getCapturerObserver());
//开始录制
videoCapturer.startCapture(
720,//宽
1080,//高
25//fps 帧率
);
//播放本地视频
localRender.init(mRootEGL.getEglBaseContext(), null);
//SCALE_ASPECT_FILL 自动适应屏幕比例, 画面存在被裁剪的可能
//SCALE_ASPECT_FIT 自动适应画面比例,屏幕上可能存在黑边
//SCALE_ASPECT_BALANCED 视频尺寸非等比缩放。保证视频内容全部显示,且填满视窗。
localRender.setScalingType(RendererCommon.ScalingType.SCALE_ASPECT_FILL);
localRender.setMirror(true);//启用镜像
videoTrack.addSink(localRender);//最后播放
7.初始化远端render
//初始化远端render ,因为使用opengl渲染所以 必须在在主线程初始化
remoteRender.init(mRootEGL.getEglBaseContext(), null);
remoteRender.setMirror(true);//启用镜像
remoteRender.setScalingType(RendererCommon.ScalingType.SCALE_ASPECT_FILL);
8.安装ice服务器
//在ubunto服务器上安装coturn软件即可实现ice服务器
sudo apt-get install libssl-dev
sudo apt-get install libevent-dev
git clone https://github.com/coturn/coturn
cd coturn
./configure
make
sudo make install
//启动服务器, wzp:123456 表示用户名:密码也可用通过配置文件来设置
sudo nohup turnserver -L 0.0.0.0 -a -u wzp:123456 -v -f -r nort.text &
//查看端口号,默认3478端口,
sudo lsof -i:3478
9.创建peerConntion
//ice服务器列表
List<PeerConnection.IceServer> iceServers = new ArrayList<>();
//添加一个turn服务器,turn服务器主要用户下面的stun服务器打洞失败的时候使用这个turn服务器转发数据流,可以添加多个
iceServers.add(
PeerConnection.IceServer.builder("turn:**.**.**.**3478")//这是你服务器的地址
.setUsername("wzp")//用户名
.setPassword("123456")//密码
.createIceServer());
//添加一个stun服务器,
iceServers.add(PeerConnection.IceServer.builder("stun:**.**.**.**:3478").createIceServer());
peerConnection = factory.createPeerConnection(iceServers, new PeerConnection.Observer() {
@Override
public void onSignalingChange(PeerConnection.SignalingState signalingState) {
}
@Override
public void onIceConnectionChange(PeerConnection.IceConnectionState iceConnectionState) {
//ICE 连接状态变化后回调
}
@Override
public void onIceConnectionReceivingChange(boolean b) {
}
@Override
public void onIceGatheringChange(PeerConnection.IceGatheringState iceGatheringState) {
}
@Override
public void onIceCandidate(IceCandidate iceCandidate) {
//自动请求stun/turn服务器后回调这个方法
//发送Ice信息给对端用户 ,下面的代码只是用于发送信息给远端用户,我使用的是websocket,自己可以用其他方式实现。最后结尾我会给出服务器端的代码。
JSONObject sendObj = new JSONObject();
try {
sendObj.put("cmd", cmd_ice);
sendObj.put("uid", uid);
sendObj.put("remoteUid", remoteUid);
sendObj.put("roomId",roomId);
JSONObject msgObj = new JSONObject();
msgObj.put("sdpMid", iceCandidate.sdpMid);
msgObj.put("sdpMLineIndex", iceCandidate.sdpMLineIndex);
msgObj.put("sdp", iceCandidate.sdp);
sendObj.put("msg", msgObj);
socket.send(sendObj.toString());
} catch (JSONException e) {
throw new RuntimeException(e);
}
}
@Override
public void onIceCandidatesRemoved(IceCandidate[] iceCandidates) {
}
@Override
public void onAddStream(MediaStream mediaStream) {
//收到远端数据流信息
mediaStream.videoTracks.get(0).addSink(remoteRender);
mediaStream.audioTracks.get(0).setEnabled(true);
}
@Override
public void onRemoveStream(MediaStream mediaStream) {
}
@Override
public void onDataChannel(DataChannel dataChannel) {
}
@Override
public void onRenegotiationNeeded() {
}
@Override
public void onAddTrack(RtpReceiver rtpReceiver, MediaStream[] mediaStreams) {
}
});
// List<String> mediaStreamLabels = Collections.singletonList("ARDAMS");
// peerConnection.addTrack(videoTrack,mediaStreamLabels);
// peerConnection.addTrack(localAudioTrack,mediaStreamLabels);
//将本地流添加到peerConnection,远端的onAddStream回调将接受该数据流
MediaStream stream = factory.createLocalMediaStream("110");
stream.addTrack(videoTrack);
stream.addTrack(localAudioTrack);
peerConnection.addStream(stream);
10.建立websocket链接,只是用于演示效果,所以这里只是简单的实现。重要的是如何简单的去理解流程
URI uri = null;//信令服务器地址
try {
uri = new URI("ws://192.168.2.134:8090");
} catch (Exception e) {
}
socket = new WebSocketClient(uri) {
@Override
public void onOpen(ServerHandshake handshakedata) {
if (isDestroyed()) {
return;
}
Log.e(tag,"链接socket成功");
}
@Override
public void onMessage(String message) {
if (isDestroyed()) {
return;
}
try {
JSONObject msgObj = new JSONObject(message);
String cmd = msgObj.getString("cmd");
Log.e(tag,"收到消息:" + message);
if (cmd.equals(cmd_new_peer)) {
//有新人加入房间
handleNewPeer(msgObj);
return;
}
if (cmd.equals(cmd_offer)) {
//收到offer请求
handleOffer(msgObj);
return;
}
if (cmd.equals(cmd_answer)) {
//收到answer请求
handleAnswer(msgObj);
return;
}
if (cmd.equals(cmd_ice)) {
//收到ice信息
handleIce(msgObj);
}
} catch (JSONException e) {
throw new RuntimeException(e);
}
}
@Override
public void onClose(int code, String reason, boolean remote) {
if (isDestroyed()) {
return;
}
}
@Override
public void onError(Exception ex) {
if (isDestroyed()) {
return;
}
Log.e(tag,"socket错误" + ex.toString());
}
};
socket.connect();
11.发起通话者创建offer并且设置本地setLocalDescription,最后发送给被呼叫者
private void handleNewPeer(JSONObject msgObj) {
/*新人加入房间,创建offer,发起通话*/
try {
//被呼叫者的uid
remoteUid = msgObj.getString("uid");
MediaConstraints constraints = new MediaConstraints();
constraints.mandatory.add(new MediaConstraints.KeyValuePair("OfferToReceiveAudio", "true"));
constraints.mandatory.add(new MediaConstraints.KeyValuePair("OfferToReceiveVideo", "true"));
peerConnection.createOffer(new SdpObserver() {
private SessionDescription localSdp;
@Override
public void onCreateSuccess(SessionDescription sessionDescription) {
//创建offer成功即成功
localSdp = sessionDescription;
peerConnection.setLocalDescription(this, sessionDescription);
}
@Override
public void onSetSuccess() {
//setLocalDescription 成功后回调
JSONObject sendObj = new JSONObject();
try {
sendObj.put("cmd", cmd_offer);
sendObj.put("uid", uid);
sendObj.put("remoteUid", remoteUid);
sendObj.put("roomId",roomId);
sendObj.put("msg", localSdp.description);
socket.send(sendObj.toString());
} catch (JSONException e) {
throw new RuntimeException(e);
}
}
@Override
public void onCreateFailure(String s) {
}
@Override
public void onSetFailure(String s) {
}
}, constraints);
} catch (JSONException e) {
throw new RuntimeException(e);
}
}
12.被呼叫者收到offer后设置setRemoteDescription ,创建answer并设置setLocalDescription,最后发送给呼叫者
private void handleOffer(JSONObject msgObj) {
//收到offer, 当前身份是被呼叫者
try {
//发起通话者的uid
remoteUid = msgObj.getString("uid");
String sdpDescription = msgObj.getString("msg");
SessionDescription sdp = new SessionDescription(SessionDescription.Type.OFFER, sdpDescription);
peerConnection.setRemoteDescription(new SdpObserver() {
private boolean isCreateAnswer;
private String sdpDescription;
@Override
public void onCreateSuccess(SessionDescription sessionDescription) {
//createAnswer 创建成功时候回调
Log.e(tag,"创建answer成功");
sdpDescription = sessionDescription.description;
peerConnection.setLocalDescription(this, sessionDescription);
}
@Override
public void onSetSuccess() {
//setRemoteDescription setLocalDescription 成功时候的回调
if (!isCreateAnswer) {
Log.e(tag,"onSetSuccess1");
//还未创建answer 说明是setRemoteDescription回调
isCreateAnswer = true;
MediaConstraints constraints = new MediaConstraints();
constraints.mandatory.add(new MediaConstraints.KeyValuePair("OfferToReceiveAudio", "true"));
constraints.mandatory.add(new MediaConstraints.KeyValuePair("OfferToReceiveVideo", "true"));
peerConnection.createAnswer(this, constraints);
} else {
Log.e(tag,"onSetSuccess2");
//是setLocalDescription回调
JSONObject sendObj = new JSONObject();
try {
sendObj.put("cmd", cmd_answer);
sendObj.put("uid", uid);
sendObj.put("remoteUid", remoteUid);
sendObj.put("roomId",roomId);
sendObj.put("msg", sdpDescription);
socket.send(sendObj.toString());
} catch (JSONException e) {
throw new RuntimeException(e);
}
}
}
@Override
public void onCreateFailure(String s) {
}
@Override
public void onSetFailure(String s) {
}
}, sdp);
} catch (JSONException e) {
e.printStackTrace();
throw new RuntimeException(e);
}
}
13.发起通话者收到answer后设置setRemoteDescription
private void handleAnswer(JSONObject msgObj) {
//收到answer,当前是发起通话者
try {
String sdpDescription = msgObj.getString("msg");
SessionDescription sdp = new SessionDescription(SessionDescription.Type.ANSWER, sdpDescription);
peerConnection.setRemoteDescription(new SdpObserver() {
@Override
public void onCreateSuccess(SessionDescription sessionDescription) {
}
@Override
public void onSetSuccess() {
// setRemoteDescription 设置成功
}
@Override
public void onCreateFailure(String s) {
}
@Override
public void onSetFailure(String s) {
}
}, sdp);
} catch (JSONException e) {
throw new RuntimeException(e);
}
}
14,各自收到对方发来的ice信息时候addIceCandidate
private void handleIce(JSONObject msgObj) {
//收到对方的ice信息
try {
Log.e(tag,"设置ICE信息");
JSONObject iceObj = msgObj.getJSONObject("msg");
IceCandidate iceCandidate = new IceCandidate(iceObj.getString("sdpMid"),
iceObj.getInt("sdpMLineIndex"),
iceObj.getString("sdp"));
peerConnection.addIceCandidate(iceCandidate);
} catch (JSONException e) {
Log.e(tag,"ice设置失败:" + e.getMessage());
throw new RuntimeException(e);
}
}
15.销毁
@Override
protected void onDestroy() {
if (socket != null && socket.isOpen()) {
socket.close();
}
socket = null;
if(peerConnection != null){
peerConnection.close();
peerConnection = null;
}
if(videoSource != null){
videoSource.dispose();
videoSource = null;
}
if (audioSource != null) {
audioSource.dispose();
audioSource = null;
}
if(videoCapturer != null){
try {
videoCapturer.stopCapture();
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
videoCapturer.dispose();
videoCapturer = null;
}
if(surfaceTextureHelper != null){
surfaceTextureHelper.dispose();
surfaceTextureHelper = null;
}
localRender.release();
remoteRender.release();
if(factory != null){
factory.dispose();
factory = null;
}
if(mRootEGL != null){
mRootEGL.release();
mRootEGL = null;
}
super.onDestroy();
}
最后给出java全部源码
package com.example.rtcmy.ui;
import androidx.appcompat.app.AppCompatActivity;
import androidx.databinding.BindingConversion;
import androidx.databinding.DataBindingUtil;
import androidx.databinding.ObservableField;
import android.database.Observable;
import android.graphics.Color;
import android.graphics.drawable.ColorDrawable;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.TextView;
import com.example.rtcmy.R;
import org.java_websocket.client.WebSocketClient;
import org.java_websocket.handshake.ServerHandshake;
import org.json.JSONException;
import org.json.JSONObject;
import org.webrtc.AudioSource;
import org.webrtc.AudioTrack;
import org.webrtc.Camera1Enumerator;
import org.webrtc.Camera2Enumerator;
import org.webrtc.CameraEnumerator;
import org.webrtc.DataChannel;
import org.webrtc.DefaultVideoDecoderFactory;
import org.webrtc.DefaultVideoEncoderFactory;
import org.webrtc.EglBase;
import org.webrtc.IceCandidate;
import org.webrtc.Logging;
import org.webrtc.MediaConstraints;
import org.webrtc.MediaStream;
import org.webrtc.PeerConnection;
import org.webrtc.PeerConnectionFactory;
import org.webrtc.RendererCommon;
import org.webrtc.RtpReceiver;
import org.webrtc.ScreenCapturerAndroid;
import org.webrtc.SdpObserver;
import org.webrtc.SessionDescription;
import org.webrtc.SurfaceTextureHelper;
import org.webrtc.SurfaceViewRenderer;
import org.webrtc.VideoCapturer;
import org.webrtc.VideoDecoder;
import org.webrtc.VideoDecoderFactory;
import org.webrtc.VideoEncoderFactory;
import org.webrtc.VideoSource;
import org.webrtc.VideoTrack;
import org.webrtc.audio.AudioDeviceModule;
import org.webrtc.audio.JavaAudioDeviceModule;
import java.net.URI;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.UUID;
public class SimpleMeetingActivity extends AppCompatActivity {
private WebSocketClient socket;
private String roomId = "1";//房间号
private String cmd_join_room = "cmd_join_room";//加入房间命令
private String cmd_new_peer = "cmd_new_peer";//有新人加入房间
private String cmd_offer = "cmd_offer";
private String cmd_answer = "cmd_answer";
private String cmd_ice = "cmd_ice";
private SurfaceViewRenderer remoteRender;
private SurfaceViewRenderer localRender;
private TextView ivStatus;
private View ivJoin;
private EglBase mRootEGL;
private PeerConnectionFactory factory;
private VideoCapturer videoCapturer;
private AudioSource audioSource;
private AudioTrack localAudioTrack;
private VideoSource videoSource;
private VideoTrack videoTrack;
private List<PeerConnection.IceServer> iceServers = new ArrayList<>();
private SurfaceTextureHelper surfaceTextureHelper;
private PeerConnection peerConnection;
private String remoteUid;
private String uid = UUID.randomUUID().toString();
private String tag = "simpleWebrtc";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_simple_metting);
ivStatus = findViewById(R.id.ivStatus);
ivJoin = findViewById(R.id.ivJoin);
remoteRender = findViewById(R.id.ivRemoteRender);
localRender = findViewById(R.id.ivLocalRender);
//开启日志
iceServers.add(
PeerConnection.IceServer.builder("turn:123.60.151.37:3478")
.setUsername("wzp")
.setPassword("123456")
.createIceServer());
iceServers.add(PeerConnection.IceServer.builder("stun:123.60.151.37:3478").createIceServer());
mRootEGL = EglBase.create();
//在初次使用PeerConnectionFactory之前,必须调用静态方法initialize()对其进行全局的初始化与资源加载
PeerConnectionFactory.InitializationOptions initializationOptions =
PeerConnectionFactory.InitializationOptions
.builder(this)
.setEnableInternalTracer(true)// 启用内部追踪器,用来记录一些相关数据
.createInitializationOptions();
PeerConnectionFactory.initialize(initializationOptions);
//-----------创建视频编码和解码器
VideoEncoderFactory encoderFactory = new DefaultVideoEncoderFactory(mRootEGL.getEglBaseContext(),
true, true);
VideoDecoderFactory decoderFactory = new DefaultVideoDecoderFactory(mRootEGL.getEglBaseContext());
//-----------创建PeerConnectionFactory
AudioDeviceModule adm = JavaAudioDeviceModule.builder(this).createAudioDeviceModule();//音频配置当前JAVA实现,还有native
PeerConnectionFactory.Options options = new PeerConnectionFactory.Options();
//options.disableEncryption : true表示不用数据加密
//options.disableNetworkMonitor : true表示禁用网络监视器
factory = PeerConnectionFactory.builder()
.setOptions(options)//设置网络配置,使用默认
.setAudioDeviceModule(adm)//设置音频采集和播放使用的配置,当前使用java中的audioTrack 和audioRecord
.setVideoEncoderFactory(encoderFactory)
.setVideoDecoderFactory(decoderFactory)
.createPeerConnectionFactory();
//-----------创建声音源
//配置音频约束
MediaConstraints audioConstraints = new MediaConstraints();
//回声消除
audioConstraints.mandatory.add(new MediaConstraints.KeyValuePair("googEchoCancellation", "true"));
//自动增益
audioConstraints.mandatory.add(new MediaConstraints.KeyValuePair("googAutoGainControl", "true"));
//高音过滤
audioConstraints.mandatory.add(new MediaConstraints.KeyValuePair("googHighpassFilter", "true"));
//噪音处理
audioConstraints.mandatory.add(new MediaConstraints.KeyValuePair("googNoiseSuppression", "true"));
audioSource = factory.createAudioSource(audioConstraints);
localAudioTrack = factory.createAudioTrack("102", audioSource);
//----------创建视频源
//ScreenCapturerAndroid 录屏 ;FileVideoCapturer文件
//从相机里获取视频流
CameraEnumerator enumerator = new Camera2Enumerator(this);
String[] deviceNames = enumerator.getDeviceNames();
//遍历所有摄像头找到前置摄像头
for (String deviceName : deviceNames) {
if (enumerator.isFrontFacing(deviceName)) {
videoCapturer = enumerator.createCapturer(deviceName, null);
}
}
assert videoCapturer != null;
videoSource = factory.createVideoSource(videoCapturer.isScreencast());
//创建视频轨道
videoTrack = factory.createVideoTrack("103", videoSource);
//因为使用的是opengl 渲染
surfaceTextureHelper = SurfaceTextureHelper.create("capture-thread", mRootEGL.getEglBaseContext());
videoCapturer.initialize(surfaceTextureHelper, this, videoSource.getCapturerObserver());
//开始录制
videoCapturer.startCapture(
720,//宽
1080,//高
25//fps 帧率
);
//播放本地视频
localRender.init(mRootEGL.getEglBaseContext(), null);
//SCALE_ASPECT_FILL 自动适应屏幕比例, 画面存在被裁剪的可能
//SCALE_ASPECT_FIT 自动适应画面比例,屏幕上可能存在黑边
//SCALE_ASPECT_BALANCED 视频尺寸非等比缩放。保证视频内容全部显示,且填满视窗。
localRender.setScalingType(RendererCommon.ScalingType.SCALE_ASPECT_FILL);
localRender.setMirror(true);//启用镜像
videoTrack.addSink(localRender);//最后播放
//初始化远端render ,因为使用opengl渲染所以 必须在在主线程初始化
remoteRender.init(mRootEGL.getEglBaseContext(), null);
remoteRender.setMirror(true);//启用镜像
remoteRender.setScalingType(RendererCommon.ScalingType.SCALE_ASPECT_FILL);
//创建peerConnection
peerConnection = factory.createPeerConnection(iceServers, new PeerConnection.Observer() {
@Override
public void onSignalingChange(PeerConnection.SignalingState signalingState) {
}
@Override
public void onIceConnectionChange(PeerConnection.IceConnectionState iceConnectionState) {
//ICE 连接状态变化后回调
}
@Override
public void onIceConnectionReceivingChange(boolean b) {
}
@Override
public void onIceGatheringChange(PeerConnection.IceGatheringState iceGatheringState) {
}
@Override
public void onIceCandidate(IceCandidate iceCandidate) {
//自动请求stun/turn服务器后回调这个方法
//发送Ice信息给对端用户
JSONObject sendObj = new JSONObject();
try {
sendObj.put("cmd", cmd_ice);
sendObj.put("uid", uid);
sendObj.put("remoteUid", remoteUid);
sendObj.put("roomId",roomId);
JSONObject msgObj = new JSONObject();
msgObj.put("sdpMid", iceCandidate.sdpMid);
msgObj.put("sdpMLineIndex", iceCandidate.sdpMLineIndex);
msgObj.put("sdp", iceCandidate.sdp);
sendObj.put("msg", msgObj);
socket.send(sendObj.toString());
} catch (JSONException e) {
throw new RuntimeException(e);
}
}
@Override
public void onIceCandidatesRemoved(IceCandidate[] iceCandidates) {
}
@Override
public void onAddStream(MediaStream mediaStream) {
//收到远端数据流信息
mediaStream.videoTracks.get(0).addSink(remoteRender);
mediaStream.audioTracks.get(0).setEnabled(true);
}
@Override
public void onRemoveStream(MediaStream mediaStream) {
}
@Override
public void onDataChannel(DataChannel dataChannel) {
}
@Override
public void onRenegotiationNeeded() {
}
@Override
public void onAddTrack(RtpReceiver rtpReceiver, MediaStream[] mediaStreams) {
}
});
// List<String> mediaStreamLabels = Collections.singletonList("ARDAMS");
// peerConnection.addTrack(videoTrack,mediaStreamLabels);
// peerConnection.addTrack(localAudioTrack,mediaStreamLabels);
//将本地流添加到peerConnection,远端的onAddStream回调将接受该数据流
MediaStream stream = factory.createLocalMediaStream("110");
stream.addTrack(videoTrack);
stream.addTrack(localAudioTrack);
peerConnection.addStream(stream);
createSocket();
ivJoin.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
if (!socket.isOpen()) {
return;
}
ivStatus.setText("正在加入房间...");
JSONObject sendObj = new JSONObject();
try {
sendObj.put("cmd", cmd_join_room);
sendObj.put("uid", uid);
sendObj.put("roomId", roomId);
socket.send(sendObj.toString());
} catch (JSONException e) {
throw new RuntimeException(e);
}
}
});
}
private void createSocket() {
if (socket != null) {
socket.close();
}
URI uri = null;//信令服务器地址
try {
uri = new URI("ws://192.168.2.134:8090");
} catch (Exception e) {
}
socket = new WebSocketClient(uri) {
@Override
public void onOpen(ServerHandshake handshakedata) {
if (isDestroyed()) {
return;
}
Log.e(tag,"链接socket成功");
}
@Override
public void onMessage(String message) {
if (isDestroyed()) {
return;
}
try {
JSONObject msgObj = new JSONObject(message);
String cmd = msgObj.getString("cmd");
Log.e(tag,"收到消息:" + message);
if (cmd.equals(cmd_new_peer)) {
//有新人加入房间
handleNewPeer(msgObj);
return;
}
if (cmd.equals(cmd_offer)) {
//收到offer请求
handleOffer(msgObj);
return;
}
if (cmd.equals(cmd_answer)) {
//收到answer请求
handleAnswer(msgObj);
return;
}
if (cmd.equals(cmd_ice)) {
//收到ice信息
handleIce(msgObj);
}
} catch (JSONException e) {
throw new RuntimeException(e);
}
}
@Override
public void onClose(int code, String reason, boolean remote) {
if (isDestroyed()) {
return;
}
}
@Override
public void onError(Exception ex) {
if (isDestroyed()) {
return;
}
Log.e(tag,"socket错误" + ex.toString());
}
};
socket.connect();
}
private void handleIce(JSONObject msgObj) {
//收到对方的ice信息
try {
Log.e(tag,"设置ICE信息");
JSONObject iceObj = msgObj.getJSONObject("msg");
IceCandidate iceCandidate = new IceCandidate(iceObj.getString("sdpMid"),
iceObj.getInt("sdpMLineIndex"),
iceObj.getString("sdp"));
peerConnection.addIceCandidate(iceCandidate);
} catch (JSONException e) {
Log.e(tag,"ice设置失败:" + e.getMessage());
throw new RuntimeException(e);
}
}
private void handleAnswer(JSONObject msgObj) {
//收到answer,当前是发起通话者
try {
String sdpDescription = msgObj.getString("msg");
SessionDescription sdp = new SessionDescription(SessionDescription.Type.ANSWER, sdpDescription);
peerConnection.setRemoteDescription(new SdpObserver() {
@Override
public void onCreateSuccess(SessionDescription sessionDescription) {
}
@Override
public void onSetSuccess() {
// setRemoteDescription 设置成功
}
@Override
public void onCreateFailure(String s) {
}
@Override
public void onSetFailure(String s) {
}
}, sdp);
} catch (JSONException e) {
throw new RuntimeException(e);
}
}
private void handleOffer(JSONObject msgObj) {
//收到offer, 当前身份是被呼叫者
try {
//发起通话者的uid
remoteUid = msgObj.getString("uid");
String sdpDescription = msgObj.getString("msg");
SessionDescription sdp = new SessionDescription(SessionDescription.Type.OFFER, sdpDescription);
peerConnection.setRemoteDescription(new SdpObserver() {
private boolean isCreateAnswer;
private String sdpDescription;
@Override
public void onCreateSuccess(SessionDescription sessionDescription) {
//createAnswer 创建成功时候回调
Log.e(tag,"创建answer成功");
sdpDescription = sessionDescription.description;
peerConnection.setLocalDescription(this, sessionDescription);
}
@Override
public void onSetSuccess() {
//setRemoteDescription setLocalDescription 成功时候的回调
if (!isCreateAnswer) {
Log.e(tag,"onSetSuccess1");
//还未创建answer 说明是setRemoteDescription回调
isCreateAnswer = true;
MediaConstraints constraints = new MediaConstraints();
constraints.mandatory.add(new MediaConstraints.KeyValuePair("OfferToReceiveAudio", "true"));
constraints.mandatory.add(new MediaConstraints.KeyValuePair("OfferToReceiveVideo", "true"));
peerConnection.createAnswer(this, constraints);
} else {
Log.e(tag,"onSetSuccess2");
//是setLocalDescription回调
JSONObject sendObj = new JSONObject();
try {
sendObj.put("cmd", cmd_answer);
sendObj.put("uid", uid);
sendObj.put("remoteUid", remoteUid);
sendObj.put("roomId",roomId);
sendObj.put("msg", sdpDescription);
socket.send(sendObj.toString());
} catch (JSONException e) {
throw new RuntimeException(e);
}
}
}
@Override
public void onCreateFailure(String s) {
}
@Override
public void onSetFailure(String s) {
}
}, sdp);
} catch (JSONException e) {
e.printStackTrace();
throw new RuntimeException(e);
}
}
private void handleNewPeer(JSONObject msgObj) {
/*新人加入房间,创建offer,发起通话*/
try {
//被呼叫者的uid
remoteUid = msgObj.getString("uid");
MediaConstraints constraints = new MediaConstraints();
constraints.mandatory.add(new MediaConstraints.KeyValuePair("OfferToReceiveAudio", "true"));
constraints.mandatory.add(new MediaConstraints.KeyValuePair("OfferToReceiveVideo", "true"));
peerConnection.createOffer(new SdpObserver() {
private SessionDescription localSdp;
@Override
public void onCreateSuccess(SessionDescription sessionDescription) {
//创建offer成功即成功
localSdp = sessionDescription;
peerConnection.setLocalDescription(this, sessionDescription);
}
@Override
public void onSetSuccess() {
//setLocalDescription 成功后回调
JSONObject sendObj = new JSONObject();
try {
sendObj.put("cmd", cmd_offer);
sendObj.put("uid", uid);
sendObj.put("remoteUid", remoteUid);
sendObj.put("roomId",roomId);
sendObj.put("msg", localSdp.description);
socket.send(sendObj.toString());
} catch (JSONException e) {
throw new RuntimeException(e);
}
}
@Override
public void onCreateFailure(String s) {
}
@Override
public void onSetFailure(String s) {
}
}, constraints);
} catch (JSONException e) {
throw new RuntimeException(e);
}
}
@Override
protected void onDestroy() {
if (socket != null && socket.isOpen()) {
socket.close();
}
socket = null;
if(peerConnection != null){
peerConnection.close();
peerConnection = null;
}
if(videoSource != null){
videoSource.dispose();
videoSource = null;
}
if (audioSource != null) {
audioSource.dispose();
audioSource = null;
}
if(videoCapturer != null){
try {
videoCapturer.stopCapture();
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
videoCapturer.dispose();
videoCapturer = null;
}
if(surfaceTextureHelper != null){
surfaceTextureHelper.dispose();
surfaceTextureHelper = null;
}
localRender.release();
remoteRender.release();
if(factory != null){
factory.dispose();
factory = null;
}
if(mRootEGL != null){
mRootEGL.release();
mRootEGL = null;
}
super.onDestroy();
}
}
服务器端node js全部源码
//简单信令服务器 npm install nodejs-websocket
let ws = require("nodejs-websocket")
class Client {
constructor(uid,conn,roomId) {
this.uid = uid;
this.conn = conn;
this.conn.uid = this.uid;
this.roomId = roomId;
this.conn.roomId = roomId;
}
}
let roomMaps = new Map
function handleJoinRoom(receiveObj, conn) {
//有人加入房间
let uid = receiveObj.uid;
let roomId = receiveObj.roomId;
let room = roomMaps.get(roomId);
let client = new Client(uid,conn,roomId)
if(!room){
//创建房间
room = new Map
}
if(room.get(uid)){
//已经在房间了
console.log("已经在房间里了")
return
}
room.set(uid,client)
roomMaps.set(roomId,room);
console.log("加入房间了");
if(room.size > 1){
let clients = Array.from(room.keys())
clients.forEach(remoteUid => {
if(remoteUid !== uid){
//通知房间其他人有新人加入
let sendObj = {
cmd: "cmd_new_peer",
uid: uid,
remoteUid
}
let remoteClient = room.get(remoteUid)
remoteClient.conn.sendText(JSON.stringify(sendObj))
console.log("新人发送成功");
}
})
}
}
function handleOffer(receiveObj) {
//转发offer
let remoteUid = receiveObj.remoteUid
let roomId = receiveObj.roomId
let room = roomMaps.get(roomId)
let client = room.get(remoteUid)
client.conn.sendText(JSON.stringify(receiveObj))
}
function handleAnswer(receiveObj) {
//转发answer;
let remoteUid = receiveObj.remoteUid
let roomId = receiveObj.roomId
let room = roomMaps.get(roomId)
let client = room.get(remoteUid)
client.conn.sendText(JSON.stringify(receiveObj))
}
function handleIce(receiveObj) {
//转发ice
let remoteUid = receiveObj.remoteUid
let roomId = receiveObj.roomId
let room = roomMaps.get(roomId)
let client = room.get(remoteUid)
client.conn.sendText(JSON.stringify(receiveObj))
}
function handleClose(conn) {
let uid = conn.uid
let roomId = conn.roomId
let room = roomMaps.get(roomId)
room.delete(uid)
}
ws.createServer(function (conn) {
//有客服端链接
conn.on("text",function (str) {
//收到消息
let receiveObj = JSON.parse(str);
console.log(str)
switch (receiveObj.cmd) {
case "cmd_join_room":
//有人加入
handleJoinRoom(receiveObj,conn)
break
case "cmd_offer":
//转发offer
handleOffer(receiveObj)
break
case "cmd_answer":
handleAnswer(receiveObj);
break
case "cmd_ice":
handleIce(receiveObj);
break
}
})
conn.on("close",function (code,reason) {
console.log("链接关闭")
handleClose(conn);
});
conn.on("error",function (err){
console.log(err);
})
}).listen(8090)