Can I use MultiNetworkDriver for both IPv4 and IPv6?

Hi. I managed multiple NetworkDriver for handling both IPv4 and IPv6 connection. I created abstraction connection struct which has internal driver index for its source.

Then I found that Unity Transport 2.0.0-pre.7 introduces “MultiNetworkDriver” that seems to be best fit for above case.

So can MultiNetworkDriver handles multiple UDP network driver which has only different binding address?

THanks.

Related post:

Yes! That’s exactly one of the use cases we had in mind when we designed this feature. It’s even one of the examples in our documentation. I’m happy to read that this will be immediately useful to you. Please don’t hesitate if you have any questions.

1 Like

The link to the example (the one you posted now) is not available anywhere in the manual (at lest not in the index on the right side) as far as I see. I mention this in case you forget to add it. I have seen the example mentioned in the changelog, but I was not able to find it.

Also your update is just in time for me too as I today started implementing the migration of my custom networking layer from UNET to Transport.

The sidebar does have a “cross-play” section that links to the page I link to. I’m guessing it might not appear for you because the browser is using a cached version of the sidebar. I know I had to force a cache update to get it to appear (Shift+F5 or Ctrl+F5 depending on your browser).

Yes that was the case, it is working now, thank you.

Looks like it is not only the left menu, but the individual reference pages also need a hard refresh to show the newest version. Which makes the whole documentation more difficult to use. I only find that I need to do the cache refresh because I have seen that there is a

section in the Transport source code, but it is not visible on the reference page. So I guessed that I have to refresh.

Even though I know it is not related directly to Transport, I think this should be somehow fixed by Unity as it seems like a big problem.

@simon-lemay-unity Hi. My project has P2P feature, so client and server share the most of code. The problem is that MultiNetworkDriver and NetworkDriver does not implement any shared interface so all of my code should have annoying conditional statement like this:

if(useMultiNetworkDriver)
{
    var multiNetworkDriver = ...
    multiNetworkDrvier.Send(...)
}
else
{
    var newtorkDriver = ...
    newtorkDriver.Send(...)
}

Main problem is that MultiNetworkDriver does not have Connect() method so I can’t use it for client side.

Is there any other method for handling both MultiNetworkDriver and NetworkDriver transparently?

I would like to see this feature too (even though I will not use P2P) as it would simplify the code. Both the client and the server have to do similar things like sending and receiving data not being able to use MultiNetworkDriver for both sides is a complication that could be avoided.

Currently there is not. We thought about adding an interface for common driver operations, but ultimately opted not to include one at this point because such a solution wouldn’t work within Burst-compiled code. We’re very much open to revisiting this decision however if there is a need for it.

We also thought about supporting client drivers in MultiNetworkDriver, but again opted not to at this point because the semantics for clients are less clear. For example, let’s say we were to add a Connect method, which driver should it connect?

Between the two, is there one approach that would better fit your use case? If we add an interface for common driver operations, what would you expect this interface to cover? Sending data and popping events? Scheduling updates (which would preclude the interface from covering the concurrent versions of the NetworkDriver and MultiNetworkDriver)? I’m curious to read about what an ideal solution would look like from a user’s point of view.

Wouldn’t the client MultiNetworkDriver only need one NetworkDriver? Connect() could probably safely use the first driver from the list.

That’s certainly one option, but I’m not keen on creating an implicit “role” for MultiNetworkDriver (e.g. that it could be either a server or a client without it being made very clear in the API). Plus, this will break down if we ever want to support mixed servers and clients in the same MultiNetworkDriver. I could see this being useful for mesh networking scenarios, for example. If we’re going to support client drivers in MultiNetworkDriver, personally I’d rather we provide a more complete support for it.

In my case, I created my own wrapper for this case:

struct DriverWrapper : IDisposable
{
    public bool UseMultiNetworkDriver;
    public MultiNetworkDriver MultiNetworkDriver;
    public NetworkDriver NetworkDriver;
    public bool IsCreated => this.UseMultiNetworkDriver ? this.MultiNetworkDriver.IsCreated : this.NetworkDriver.IsCreated;

    public NetworkConnect Connect(...)
    {
        if(this.UseMultiNetworkDriver)
        {
            throw new NotSupoortedException();
        }
        else
        {
            return this.NetworkDriver.Connect(...);
        }
    }

    public int Send(...)
    {
        if(this.UseMultiNetworkDriver)
        {
            return this.MultiNetworkDriver.Send(...);
        }
        else
        {
            return this.NetworkDriver.Send(...);
        }
    }

    // And other public methods ...
}


struct MySendJob : IJob
{
    [NativeDisableContainerSafetyRestriction] // <= This attribute need for Unity safety check!
    public DriverWrapper Driver;

    // My shared application code for common operation ...
}

It may be inefficient, but works :slight_smile:

Or it may be useful if I can convert NetworkConnection of single NetworkDriver to MultiNetworkDriver’s like this:

NetworkDriver driver = ...;

var connection = driver.Connect(...);

MultiNetworkDriver multiDriver = ...;

multiDriver.AddDriver(driver);

var connectionForMultiDriver = multiDriver.ConvertConnection(driver, connection);

If you were able to add non-server drivers to a MultiNetworkDriver and connect them by ID, would this solve your problem? For example:

var driver = NetworkDriver.Create();
var multiDriver = MultiNetworkDriver.Create();

var driverId = multiDriver.AddDriver(driver);
var connection = multiDriver.Connect(driverId, endpoint);

(To clarify, the above code does not work in 2.0.0-pre.7. It’s just an example of what the API could look like.)

1 Like

Also, thank you for posting your solution with the driver wrapper! This helps us define what is the best way to address your problem. I now see that an interface might not be ideal, since it would not include operations like Connect. Also, using an interface would require you to use generic jobs, which can be annoying to work with.

DriverId seems to be simple and easy to use. It can solve my case.

1 Like

Can I use the same port for UDPNetworkInterface and WebSocketNetworkInterface? In the documentation example you use different ports, but I used the same and I got no errors.

Great! Thanks for the feedback! We’ll try to get this done ASAP.

Yes, you can use the same port. The UDP interface uses a UDP port, and the WebSocket interface uses a TCP port. Of course, if you were (for whatever reason) to add two WebSocket drivers, they wouldn’t be able to use the same port (unless bound to distinct non-wildcard local IP addresses).

1 Like

Small update on this: version 2.0.0-pre.8 has been released and contains the change to allow client drivers in MultiNetworkDriver. The documentation has been updated with an example, but it’s basically the same usage as the sample code I posted above.

1 Like

New MultiNetworkDriver makes my code be simple and clear. I can’t find any issue at this time. Thanks.

2 Likes