SRMP-Public/SRMP/Networking/NetworkClient.cs
2023-05-29 22:23:11 +02:00

264 lines
No EOL
10 KiB
C#

using Lidgren.Network;
using SRMultiplayer.Packets;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Net;
using System.Reflection;
using System.Text;
using UnityEngine;
using UnityEngine.SceneManagement;
namespace SRMultiplayer.Networking
{
public class NetworkClient : SRSingleton<NetworkClient>
{
private NetClient m_Client;
public enum ConnectionStatus
{
Disconnected,
Connecting,
Connected
}
public struct LocalGame
{
public string Name;
public string IP;
public int Port;
}
public ConnectionStatus Status;
public NetPeerStatistics Statistics { get { return m_Client.Statistics; } }
public int Port { get { return m_Client.Port; } }
public List<LocalGame> LocalGames = new List<LocalGame>();
private void Start()
{
NetPeerConfiguration config = new NetPeerConfiguration("srmp");
config.EnableMessageType(NetIncomingMessageType.NatIntroductionSuccess);
config.EnableMessageType(NetIncomingMessageType.DiscoveryResponse);
m_Client = new NetClient(config);
m_Client.Start();
}
public void SendDiscoverMessage()
{
LocalGames.Clear();
m_Client.DiscoverLocalPeers(6996);
}
public void Connect(string ip, int port, string username)
{
Status = ConnectionStatus.Connecting;
NetOutgoingMessage hail = CreateMessage();
hail.Write(Globals.Version);
hail.Write(Globals.UserData.UUID.ToByteArray());
hail.Write(username);
hail.Write(Globals.Mods.Count);
foreach (var mod in Globals.Mods)
{
hail.Write(mod);
}
var dlcs = SRSingleton<GameContext>.Instance.DLCDirector.Installed;
hail.Write(dlcs.Count());
foreach (var dlc in dlcs)
{
hail.Write((byte)dlc);
}
m_Client.Connect(ip, port, hail);
}
public void Connect(IPEndPoint ip, string username)
{
SRMP.Log("Connect " + ip.ToString());
if (Status != ConnectionStatus.Disconnected)
{
SRMP.Log("Already connecting... abort");
return;
}
NetOutgoingMessage hail = CreateMessage();
hail.Write(Globals.Version);
hail.Write(Globals.UserData.UUID.ToByteArray());
hail.Write(username);
hail.Write(Globals.Mods.Count);
foreach (var mod in Globals.Mods)
{
hail.Write(mod);
}
var dlcs = SRSingleton<GameContext>.Instance.DLCDirector.Installed;
hail.Write(dlcs.Count());
foreach(var dlc in dlcs)
{
hail.Write((byte)dlc);
}
m_Client.Connect(ip, hail);
}
private void Update()
{
NetIncomingMessage im;
while ((im = m_Client?.ReadMessage()) != null)
{
switch (im.MessageType)
{
case NetIncomingMessageType.DebugMessage:
case NetIncomingMessageType.ErrorMessage:
case NetIncomingMessageType.WarningMessage:
case NetIncomingMessageType.VerboseDebugMessage:
{
string text = im.ReadString();
SRMP.Log("[NetworkClient] " + text);
}
break;
case NetIncomingMessageType.NatIntroductionSuccess:
{
string token = im.ReadString();
SRMP.Log("[NetworkClient] Nat introduction success to " + im.SenderEndPoint + " token is: " + token);
MultiplayerUI.Instance.ConnectResponse(MultiplayerUI.ConnectError.None);
Connect(im.SenderEndPoint, Globals.Username);
}
break;
case NetIncomingMessageType.DiscoveryResponse:
{
var game = new LocalGame()
{
Name = im.ReadString(),
Port = im.ReadInt32(),
IP = im.SenderEndPoint.Address.ToString()
};
LocalGames.Add(game);
Console.WriteLine("Found server at " + im.SenderEndPoint + " name: " + game.Name);
}
break;
case NetIncomingMessageType.StatusChanged:
{
NetConnectionStatus status = (NetConnectionStatus)im.ReadByte();
string reason = im.ReadString();
SRMP.Log("[NetworkClient] " + NetUtility.ToHexString(im.SenderConnection.RemoteUniqueIdentifier) + " " + status + ": " + reason);
if (status == NetConnectionStatus.Connected)
{
Status = ConnectionStatus.Connected;
NetIncomingMessage hail = im.SenderConnection.RemoteHailMessage;
Globals.LocalID = hail.ReadByte();
int playerCount = hail.ReadInt32();
for (int i = 0; i < playerCount; i++)
{
byte id = hail.ReadByte();
string username = hail.ReadString();
bool hasloaded = hail.ReadBoolean();
var playerObject = new GameObject($"{username} ({id})");
var player = playerObject.AddComponent<NetworkPlayer>();
DontDestroyOnLoad(playerObject);
player.ID = id;
player.Username = username;
player.HasLoaded = hasloaded;
Globals.Players.Add(id, player);
if (id == Globals.LocalID)
{
Globals.LocalPlayer = player;
}
}
Globals.PartyID = new Guid(hail.ReadBytes(16));
var gameMode = (PlayerState.GameMode)hail.ReadByte();
Globals.CurrentGameName = hail.ReadString();
SRSingleton<GameContext>.Instance.AutoSaveDirector.LoadNewGame("SRMultiplayerGame", Identifiable.Id.GOLD_SLIME, gameMode, () =>
{
Disconnect("Error loading save");
SceneManager.LoadScene(2);
});
}
else if (status == NetConnectionStatus.Disconnected)
{
if (reason != "goodbye")
{
if(reason.Equals("kicked"))
{
MultiplayerUI.Instance.ConnectResponse(MultiplayerUI.ConnectError.Kicked);
}
else
{
MultiplayerUI.Instance.ConnectResponse(MultiplayerUI.ConnectError.Message, reason);
}
}
Status = ConnectionStatus.Disconnected;
if (SceneManager.GetActiveScene().buildIndex == 3)
{
SceneManager.LoadScene(2);
}
}
}
break;
case NetIncomingMessageType.Data:
{
PacketType type = (PacketType)im.ReadUInt16();
Globals.HandlePacket = true;
try
{
NetworkHandlerClient.HandlePacket(type, im);
}
catch (Exception ex)
{
SRMP.Log($"[NetworkClient] Could not handle packet {type}\n{ex}");
}
Globals.HandlePacket = false;
}
break;
default:
SRMP.Log("[NetworkClient] Unhandled type: " + im.MessageType + " " + im.LengthBytes + " bytes " + im.DeliveryMethod + "|" + im.SequenceChannel);
break;
}
m_Client.Recycle(im);
}
}
public void Disconnect(string message = "goodbye")
{
Status = ConnectionStatus.Disconnected;
m_Client?.Disconnect(message);
}
public void Send(Packet packet, NetDeliveryMethod method, int sequence)
{
if (m_Client == null || m_Client.ConnectionStatus != NetConnectionStatus.Connected) return;
var om = CreateMessage();
packet.Serialize(om);
m_Client?.SendMessage(om, method, sequence);
}
public void Send(NetOutgoingMessage om, NetDeliveryMethod method, int sequence)
{
if (m_Client == null || m_Client.ConnectionStatus != NetConnectionStatus.Connected) return;
m_Client?.SendMessage(om, method, sequence);
}
public void SendUnconnected(NetOutgoingMessage om, IPEndPoint endPoint)
{
if (m_Client == null) return;
m_Client?.SendUnconnectedMessage(om, endPoint);
}
public NetOutgoingMessage CreateMessage()
{
return m_Client?.CreateMessage();
}
}
}