Queries on Peer to Peer Connectivity using Network.Framework

I am planning to implement Peer to Peer data exchange between 2 iOS devices. I have the following queries.

  1. My devices are connected through Ethernet. I would prefer to use this route if possible before switching over to Wifi. I can see there is an option to use prohibitedInterfaceTypes but that doesn't guarantee the route to use Ethernet when there are multiple available.
  2. Does the connection automatically switch between ethernet and Wifi or does that have to be handled via isViable and betterPathAvailable?
  3. I'm unsure when I should be using a custom framing protocol. I just need to exchange codable objects between the devices. Is the custom protocol required only if I use TCP as the underlying protocol?
  4. Can NSURLSessionStreamTask be used for this use case?
Answered by DTS Engineer in 814175022
My devices are connected through Ethernet. I would prefer to use this route if possible before switching over to Wifi. I can see there is an option to use prohibitedInterfaceTypes but that doesn't guarantee the route to use Ethernet when there are multiple available.

Network framework will general use the best path by default, based on its Happy Eyeballs implementation. I wouldn’t expect Wi-Fi to be faster (well, lower latency) than Ethernet but… *shrug*.

There isn’t a way to specify “use Ethernet even if it’s worse than Wi-Fi, but fallback to Wi-Fi if Ethernet doesn’t work at all”. Partly that’s because “doesn’t work at all” isn’t a reasonable concept in networking.

I can see a couple of ways you could implement this yourself:

  • Start a connection with a required interface type (requiredInterfaceType) of Ethernet and, if that fails, or doesn’t connect promptly, trying again without that constraint.

  • Open the connection with no constraints and, if the path is over Wi-Fi, try opening a second connection with Ethernet as the required interface type.

Honestly though, I think you’d be better off letting Network frame do its thing and then see if that causes a problem in practice.

Does the connection automatically switch between ethernet and Wifi … ?

No. For TCP that’s not possible because a TCP connection is identified by the local IP / local port / remote IP / remote port tuple. If you change interface, you change local IP and you have a different connection.

You can get this sort of automatic switching behaviour with MPTCP. I’m not sure if Network framework supports that on the server side.

I'm unsure when I should be using a custom framing protocol. I just need to exchange codable objects between the devices. Is the custom protocol required only if I use TCP as the underlying protocol?

What else are you going to use?

You could, for example, use QUIC for this, and then use one stream per message. Whether that makes sense in your scenario is hard to say without more background info.

One option that I like a lot is WebSocket. It works on top TCP (and TCP+TLS) and Network framework supports it on both the client and server side.

Can NSURLSessionStreamTask be used for this use case?

That’s not a good option. Stream tasks only support the client side, so you’d have to implement the server in something else. And if you’re going to do that with, say, Network framework, you might as well use Network framework on both sides.

Also, URLSessionStreamTask is not something we generally recommend. See TN3151 Choosing the right networking API.

Share and Enjoy

Quinn “The Eskimo!” @ Developer Technical Support @ Apple
let myEmail = "eskimo" + "1" + "@" + "apple.com"

My devices are connected through Ethernet. I would prefer to use this route if possible before switching over to Wifi. I can see there is an option to use prohibitedInterfaceTypes but that doesn't guarantee the route to use Ethernet when there are multiple available.

Network framework will general use the best path by default, based on its Happy Eyeballs implementation. I wouldn’t expect Wi-Fi to be faster (well, lower latency) than Ethernet but… *shrug*.

There isn’t a way to specify “use Ethernet even if it’s worse than Wi-Fi, but fallback to Wi-Fi if Ethernet doesn’t work at all”. Partly that’s because “doesn’t work at all” isn’t a reasonable concept in networking.

I can see a couple of ways you could implement this yourself:

  • Start a connection with a required interface type (requiredInterfaceType) of Ethernet and, if that fails, or doesn’t connect promptly, trying again without that constraint.

  • Open the connection with no constraints and, if the path is over Wi-Fi, try opening a second connection with Ethernet as the required interface type.

Honestly though, I think you’d be better off letting Network frame do its thing and then see if that causes a problem in practice.

Does the connection automatically switch between ethernet and Wifi … ?

No. For TCP that’s not possible because a TCP connection is identified by the local IP / local port / remote IP / remote port tuple. If you change interface, you change local IP and you have a different connection.

You can get this sort of automatic switching behaviour with MPTCP. I’m not sure if Network framework supports that on the server side.

I'm unsure when I should be using a custom framing protocol. I just need to exchange codable objects between the devices. Is the custom protocol required only if I use TCP as the underlying protocol?

What else are you going to use?

You could, for example, use QUIC for this, and then use one stream per message. Whether that makes sense in your scenario is hard to say without more background info.

One option that I like a lot is WebSocket. It works on top TCP (and TCP+TLS) and Network framework supports it on both the client and server side.

Can NSURLSessionStreamTask be used for this use case?

That’s not a good option. Stream tasks only support the client side, so you’d have to implement the server in something else. And if you’re going to do that with, say, Network framework, you might as well use Network framework on both sides.

Also, URLSessionStreamTask is not something we generally recommend. See TN3151 Choosing the right networking API.

Share and Enjoy

Quinn “The Eskimo!” @ Developer Technical Support @ Apple
let myEmail = "eskimo" + "1" + "@" + "apple.com"

One option that I like a lot is WebSocket. It works on top TCP (and TCP+TLS) and Network framework supports it on both the client and server side.

Thanks for the pointer. I did consider this at first but lack of documentation and sample code drove me away at first :D.

I do have a couple of follow up questions related to this though.

  1. Currently I am using Multipeer framework. Multipeer Framework has a parameter called MCSessionSendDataMode which I have currently set to reliable. How does this help with reliability underneath? Does it make a connection using TCP instead of UDP? I am trying to understand - by moving to Network framework and using TCP as underlying framework can I have a more reliable connection compared to Multipeer?

  2. A query in the behaviour of acknowledgements - Does the send API on NWConnection throw an error if there is no acknowledgement? Does the error differ based on the underlying protocol - meaning if I use TCP I would assume the send API would throw a failure when the sending fails or when there is no acknowledgement compared to UDP where I assume we get an error only when sending fails.

Queries on Peer to Peer Connectivity using Network.Framework
 
 
Q