Skip to content

Commit fafcb3f

Browse files
Keith BannisterKeith Bannister
Keith Bannister
authored and
Keith Bannister
committedJul 6, 2020
Add TURN server
1 parent e6c7e0f commit fafcb3f

File tree

6 files changed

+70
-16
lines changed

6 files changed

+70
-16
lines changed
 

‎.gitignore

+2
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,8 @@ venv/
44
package-lock.json
55
node_modules/
66
public/game/phaser3_assets
7+
.env
8+
79

810
# Logs
911
logs

‎package.json

+1
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@
1313
"cookie-parser": "~1.4.4",
1414
"datauri": "^3.0.0",
1515
"debug": "~2.6.9",
16+
"dotenv": "^8.2.0",
1617
"express": "~4.16.1",
1718
"express-session": "^1.17.1",
1819
"face-api.js": "^0.22.2",

‎public/game/js/game.js

+29-15
Original file line numberDiff line numberDiff line change
@@ -1,19 +1,21 @@
11
'use strict'
22

3-
function createMyPeerConnection() {
4-
var config = {
5-
sdpSemantics: 'unified-plan'
3+
function createMyPeerConnection(iceServer) {
4+
let config = {
5+
sdpSemantics: 'unified-plan',
6+
iceServers: [iceServer]
67
};
78

8-
if (document.getElementById('use-stun').checked) {
9-
config.iceServers = [{
10-
urls: ['stun:stun.l.google.com:19302'
11-
//'turn:freya.bannister.id.au:3479'
12-
]
13-
}];
14-
}
9+
//
10+
// if (document.getElementById('use-stun').checked) {
11+
// config.iceServers = [{
12+
// urls: ['stun:stun.l.google.com:19302'
13+
// //'turn:freya.bannister.id.au:3479'
14+
// ]
15+
// }];
16+
// }
1517

16-
pc = new RTCPeerConnection(config);
18+
let pc = new RTCPeerConnection(config);
1719

1820
// register some listeners to help debugging
1921
pc.addEventListener('icegatheringstatechange', function() {
@@ -47,16 +49,16 @@ class WebRTCConnection {
4749
await this.onmessage(webrtcdata);
4850
}
4951
});
50-
this.setupPeerConnection();
52+
this.pc = null;
5153
}
5254

53-
setupPeerConnection = () => {
55+
setupPeerConnection = (server) => {
5456
this.isPolite = false;
5557
this.makingOffer = false;
5658
this.ignoreOffer = false;
5759
this.remoteId = null;
5860
this.tracksToAdd = []; //[...this.allTracks];
59-
this.pc = createMyPeerConnection();
61+
this.pc = createMyPeerConnection(server);
6062
pc = this.pc;
6163
const signaler = this;
6264
pc.onnegotiationneeded = async () => {
@@ -96,7 +98,10 @@ class WebRTCConnection {
9698
}
9799

98100
addEventListener = (evtname, func) => {
99-
this.pc.addEventListener(evtname, func);
101+
102+
if (this.pc !== null) {
103+
this.pc.addEventListener(evtname, func);
104+
}
100105

101106
if (!(evtname in Object.keys(this.listeners))) {
102107
this.listeners[evtname] = [];
@@ -110,6 +115,11 @@ class WebRTCConnection {
110115
this.sendTracksIfPossible();
111116
}
112117

118+
set iceServer(server) {
119+
this._iceServer = server;
120+
this.setupPeerConnection(server);
121+
}
122+
113123
sendTracksIfPossible = () => {
114124
const msg = `
115125
Attempting to send tracks signallingState: ${this.pc.signalingState}
@@ -536,6 +546,10 @@ function create() {
536546
star_animation_no = (star_animation_no + 1)% anims.length;
537547
});
538548

549+
this.socket.on('iceServers', function(iceServer) {
550+
self.webrtcConnection.iceServer = iceServer;
551+
});
552+
539553
this.cursors = this.input.keyboard.createCursorKeys();
540554
this.leftKeyPressed = false;
541555
this.rightKeyPressed = false;

‎rockethedz_game_server/js/game.js

+1
Original file line numberDiff line numberDiff line change
@@ -106,6 +106,7 @@ function create() {
106106
addPlayer(self, players[socket.id]);
107107
// send the players object to the new player
108108
socket.emit('currentPlayers', players);
109+
socket.emit('iceServers', window.getIceServer(socket.id));
109110
// update all other players of the new player
110111
socket.broadcast.emit('newPlayer', players[socket.id]);
111112
// send the star object to the new player

‎rockethedz_game_server/js/server.js

Whitespace-only changes.

‎routes/index.js

+37-1
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,41 @@ const { JSDOM } = jsdom;
99

1010
games = {};
1111

12+
function calcTURNCredentials(name, secret, expire_sec) {
13+
// From Here: https://www.google.com/search?client=firefox-b-d&q=how+to+use+COTURN+REST+API
14+
const unixTimeStamp = parseInt(Date.now()/1000) + expire_sec;
15+
const username = [unixTimeStamp, name].join(':');
16+
let hmac = crypto.createHmac('sha1', secret);
17+
hmac.setEncoding('base64');
18+
hmac.write(username);
19+
hmac.end();
20+
const password = hmac.read();
21+
return {
22+
username: username,
23+
password: password
24+
};
25+
}
26+
27+
function getIceServer(name) {
28+
// Returns an ice server configuration for the given username
29+
// That can be plugged straight into a webtrc setup
30+
// See: https://tools.ietf.org/html/draft-uberti-behave-turn-rest-00
31+
// See: https://developer.mozilla.org/en-US/docs/Web/API/RTCIceServer
32+
// turn defualt port: 3478 turns: 5349
33+
let expire_sec = process.env.TURN_EXPIRE_SEC || 86400;
34+
let turn_server = process.env.TURN_SERVER || 'localhost';
35+
let turn_port = process.env.TURN_PORT || 5349;
36+
let turn_proto = process.env.TURN_PROTO || 'turns'; // 'turn' or 'turns'
37+
let cred = calcTURNCredentials(name, process.env.TURN_SECRET, expire_sec);
38+
const iceServer = {
39+
username:cred.username,
40+
credential:cred.password,
41+
urls: [turn_proto + ':' + turn_server + ':' + turn_port + '?transport=udp',
42+
turn_proto + ':' + turn_server + ':' + turn_port + '?transport=tcp']
43+
};
44+
45+
return iceServer;
46+
}
1247

1348
/* GET home page. */
1449
router.get('/', function(req, res, next) {
@@ -78,8 +113,9 @@ function setupAuthoritativePhaser(gameId, io) {
78113
dom.window.URL.revokeObjectURL = (objectURL) => {};
79114
dom.window.io = io.of(socketNamespace);
80115
dom.window.gameId = gameId;
116+
dom.window.getIceServer = getIceServer;
81117
dom.window.onFinished = () => {
82-
console.log('Game ${gameid} finished');
118+
console.log('Game', gameId, 'finished');
83119
dom.window.close();
84120
delete games[gameId];
85121
}

0 commit comments

Comments
 (0)
Please sign in to comment.