Added automatically port forwarding using UPnP

This commit is contained in:
Albin Corén 2018-01-07 08:29:01 +01:00
parent 18da9a5978
commit 3fd4955faf
6 changed files with 127 additions and 9 deletions

View File

@ -2,6 +2,7 @@
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Net;
using System.Security.Cryptography;
using UnityEngine.Networking;
@ -27,6 +28,8 @@ namespace MLAPI
//Should only be used for dedicated servers and will require the servers RSA keypair being hard coded into clients in order to exchange a AES key
//TODO
public bool EncryptMessages = false;
public bool UseUPnP = true;
public Action<bool, IPAddress> UPnPCompleteCallback;
//Cached config hash
private byte[] ConfigHash = null;

View File

@ -0,0 +1,86 @@
using Open.Nat;
using System;
using System.Linq;
using System.Net;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
using UnityEngine;
namespace MLAPI.Helper
{
public class UPnPHelper
{
internal static void AttemptPortMap(int port, Action<bool, IPAddress> callback)
{
bool invoked = false;
NatDiscoverer nat = new NatDiscoverer();
CancellationTokenSource cts = new CancellationTokenSource();
cts.CancelAfter(10000);
NatDevice device = null;
StringBuilder sb = new StringBuilder();
Task<NatDevice> natTask = nat.DiscoverDeviceAsync(PortMapper.Upnp, cts);
Mapping tcpMapping = new Mapping(Protocol.Tcp, port, port, 0, Application.productName + " (TCP)");
Mapping udpMapping = new Mapping(Protocol.Udp, port, port, 0, Application.productName + " (UDP)");
IPAddress publicIPAddress = null;
natTask.ContinueWith(tt =>
{
device = tt.Result;
device.GetExternalIPAsync()
.ContinueWith(task =>
{
publicIPAddress = task.Result;
return device.CreatePortMapAsync(udpMapping);
})
.Unwrap()
.ContinueWith(task =>
{
return device.CreatePortMapAsync(udpMapping);
})
.Unwrap()
.ContinueWith(task =>
{
return device.GetAllMappingsAsync();
})
.Unwrap()
.ContinueWith(task =>
{
Mapping[] mappings = task.Result.ToArray();
if(mappings.Length == 0)
{
if (!invoked)
callback(false, publicIPAddress);
invoked = true;
}
else
{
for (int i = 0; i < mappings.Length; i++)
{
if(mappings[i].PrivatePort == port)
{
if (!invoked)
callback(true, publicIPAddress);
invoked = true;
}
}
}
});
}, TaskContinuationOptions.OnlyOnRanToCompletion);
try
{
natTask.Wait();
}
catch (AggregateException e)
{
if (e.InnerException is NatDeviceNotFoundException)
{
if (!invoked)
callback(false, publicIPAddress);
invoked = true;
}
}
}
}
}

View File

@ -32,8 +32,14 @@
<WarningLevel>4</WarningLevel>
</PropertyGroup>
<ItemGroup>
<Reference Include="Open.Nat, Version=1.0.0.0, Culture=neutral, PublicKeyToken=f22a6a4582336c76, processorArchitecture=MSIL">
<HintPath>..\packages\Open.NAT.2.1.0.0\lib\net35\Open.Nat.dll</HintPath>
</Reference>
<Reference Include="System" />
<Reference Include="System.Core" />
<Reference Include="System.Threading, Version=1.0.2856.102, Culture=neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=MSIL">
<HintPath>..\packages\Open.NAT.2.1.0.0\lib\net35\System.Threading.dll</HintPath>
</Reference>
<Reference Include="System.Xml.Linq" />
<Reference Include="System.Data.DataSetExtensions" />
<Reference Include="System.Data" />
@ -44,6 +50,7 @@
</Reference>
</ItemGroup>
<ItemGroup>
<Compile Include="Helper\UPnPHelper.cs" />
<Compile Include="MonoBehaviours\Core\NetworkedBehaviour.cs" />
<Compile Include="Data\NetworkedClient.cs" />
<Compile Include="MonoBehaviours\Core\NetworkedObject.cs" />
@ -51,5 +58,8 @@
<Compile Include="MonoBehaviours\Core\NetworkingManager.cs" />
<Compile Include="Properties\AssemblyInfo.cs" />
</ItemGroup>
<ItemGroup>
<None Include="packages.config" />
</ItemGroup>
<Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
</Project>

View File

@ -1,4 +1,5 @@
using System;
using MLAPI.Helper;
using System;
using System.Collections;
using System.Collections.Generic;
using System.IO;
@ -344,6 +345,10 @@ namespace MLAPI
Debug.LogWarning("MLAPI: No ConnectionApproval callback defined. Connection approval will timeout");
}
}
if (NetworkConfig.UseUPnP)
{
UPnPHelper.AttemptPortMap(NetworkConfig.Port, NetworkConfig.UPnPCompleteCallback);
}
HostTopology hostTopology = new HostTopology(cConfig, NetworkConfig.MaxConnections);
hostId = NetworkTransport.AddHost(hostTopology, NetworkConfig.Port);
isServer = true;
@ -373,6 +378,10 @@ namespace MLAPI
Debug.LogWarning("MLAPI: No ConnectionApproval callback defined. Connection approval will timeout");
}
}
if(NetworkConfig.UseUPnP)
{
UPnPHelper.AttemptPortMap(NetworkConfig.Port, NetworkConfig.UPnPCompleteCallback);
}
HostTopology hostTopology = new HostTopology(cConfig, NetworkConfig.MaxConnections);
hostId = NetworkTransport.AddHost(hostTopology, NetworkConfig.Port, null);
isServer = true;

4
MLAPI/packages.config Normal file
View File

@ -0,0 +1,4 @@
<?xml version="1.0" encoding="utf-8"?>
<packages>
<package id="Open.NAT" version="2.1.0.0" targetFramework="net35" />
</packages>

View File

@ -5,14 +5,8 @@ The project is WIP.
It's licenced under the MIT licence :D
## Features that are planned / done are:
* Object and player spawning (done)
* Connection approval (done)
* Message names (done)
* Replace the integer QOS with names. When you setup the networking you specify names that are associated with a channel. This makes it easier to manage. You can thus specify that a message should be sent on the "damage" channel which handles all damage related logic and is running on the AllCostDelivery channel. (done)
* ProtocolVersion to allow making different versions not talk to each other. (done)
* NetworkedBehaviours does not have to be on the root, it's simply just a class that implements the send methods etc. (done)
* Multiple messages processed every frame with the ability to specify a maximum to prevent freezes in the normal game logic (done)
## Planned features
* Built in lag compensation (going to be worked on when all base functionality is there)
* Area of interest (not working on ATM but it's on the TODO)
* Core gameplay components similar to what the HLAPI offers (but hopefully of better quality)
@ -20,7 +14,19 @@ It's licenced under the MIT licence :D
* Serializer (both for the library to speed up and to allow structs to be sent easily)
* SyncVars (allow variables to automatically be synced to new clients and current clients when it's changed)
* Message compression
## Done features
* Host support (Client hosts the server) (done)
* Port forwarding using Open.NAT using the UPnP protcol (done)
* Object and player spawning (done)
* Connection approval (done)
* Message names (done)
* Replace the integer QOS with names. When you setup the networking you specify names that are associated with a channel. This makes it easier to manage. You can thus specify that a message should be sent on the "damage" channel which handles all damage related logic and is running on the AllCostDelivery channel. (done)
* ProtocolVersion to allow making different versions not talk to each other. (done)
* NetworkedBehaviours does not have to be on the root, it's simply just a class that implements the send methods etc. (done)
* Multiple messages processed every frame with the ability to specify a maximum to prevent freezes in the normal game logic (done)
That's all I can think of right now. But there is more to come, especially if people show intrest in the project.