iOS & Safari 11 WebRTC does not gather STUN/TURN Trickle ICE Candidates
Asked Answered
J

0

7

My web application is failing to gather WebRTC relay ICE candidates via a CoTURN server when using Safari 11 on iOS 11 (iPhone 5s & iPhone 7) or desktop. The web application (which establishes a one-way audio only WebRTC peer connection) works fine between the real browsers (Chrome and Firefox) either direct or via CoTURN relay, and I normally get 6-15 ICE candidates on these browsers.

I have a (frankly, unnecessary) call to getUserMedia on the receiving side, which allows host ICE candidates to be produced by Safari. (Note... the user must approve audio and/or video access before Safari will provide host Ice Candidates, even if on a receiving-only end. I'm past that hurdle, but just so you won't hit it too... This is out of "privacy" concerns.). Before I added the allow getUserMedia, I received no ICE. Now I receive two candidates. One with a private IPv4 and another with an IPv6. This is enough to get the app working properly when on the same machine or local network. So I'm pretty confident with other parts of the application code. I am not sure if my problem is with the application code or the CoTURN server.

Example of the ICE candidates received:

{"candidate":{"candidate":"candidate:622522263 1 udp 2113937151 172.27.0.65 56182 typ host generation 0 ufrag r23H network-cost 50","sdpMid":"audio","sdpMLineIndex":0,"usernameFragment":"r23H"}}

I pretty sure the RTCIceServer Dictionary for my RTCPeerConnection is inline with the following standards:

And I've tried multiple variations of parameters:

// For Example:
var RPCconfig = {
    iceServers: [{ 
        urls: "turn:Example.live",
        username: "un",
        credential: "pw"
        }] 
     };

// Or:    
var RPCconfig = {
    iceServers: [{ 
        urls: "turns:Example.live",
        username: "un",
        credential: "pw",
        credentialType: "password"
        }, {
        urls: "stun:Example.live"
        }] 
    };

// And even more desperate attempts...
var RPCconfig = {
    iceServers: [{ 
        urls: "turn:Example.live?transport=tcp",
        username: "un",
        credential: "pw",
        credentialType: "password"
        }] 
};

Here's an example of the signaling process log for an idea of what is going on. This is from the receiving side, which is Safari 11. The other browser was Chrome (compare 6 vs 2 ICE candidates). The state change refers to oniceconnectionstatechange.

SDP Offer received. 
Sending signal SDP 
Sending signal IceCandidate
Sending signal IceCandidate 
ICE Candidate Received 
4:08:25 AM State Change -> checking 
ICE Candidate Received 
ICE Candidate Received 
ICE Candidate Received 
ICE Candidate Received 
ICE Candidate Received
4:08:40 AM State Change -> failed

CoTURN is configured quite liberally in terms of accepting every possible transport method as far as I am aware. It works well for providing ICE Candidates and as a relay for the other browsers.

Any direction would be greatly appreciated. Even if it is just a sample RTCIceServer Dictionary code that works or a proven TURN server to try.

Joan answered 6/10, 2017 at 11:55 Comment(4)
I've been having lots of issues with this as well. If you find a solution, please report back!. I will do the same. As a side note, using legacy options, e.g ( offerToReceiveAudio: 1 ), is known to cause this behaviour but is shimmed out in the new adapter.js. Unfortunately the problem is still affecting me despite removing legacy options.Bourque
I do not think the problem is with the TURN server. I think it has something to do with how the networking is working on certain networks, because I have achieved success over cellular data, of all things. I think I need to force it to use TCP, but I am not sure if Safari is paying attention to that, and I am not sure how to view Safari's version of WebRTC internals. The documentation on Apple's half-way support of WebRTC is non-existent. I also had to change the application code again the other day to add in a play button for the user to click to get around a new lockout promise rejection.Joan
@GeigeV Did you ever figure this out? I have noticed that when my WebRTC Application is running on Safari in IOS 12, it hangs when trying to establish ICE Candidates.Particularly
I am also having the similar issue, but the diff is sometimes other types of ICE candidates are also returned from Trickle ICE so I can have a connection (but this happens not always). Really confused about this behavior of SafariWyant

© 2022 - 2025 — McMap. All rights reserved.