split latency from Client into ClientPing

closes #4282
This commit is contained in:
Matthias Mailänder
2014-04-16 19:35:43 +02:00
parent f365f9da2b
commit f68a6bbd76
7 changed files with 99 additions and 20 deletions

View File

@@ -18,6 +18,7 @@ namespace OpenRA.Network
public class Session public class Session
{ {
public List<Client> Clients = new List<Client>(); public List<Client> Clients = new List<Client>();
public List<ClientPing> ClientPings = new List<ClientPing>();
// Keyed by the PlayerReference id that the slot corresponds to // Keyed by the PlayerReference id that the slot corresponds to
public Dictionary<string, Slot> Slots = new Dictionary<string, Slot>(); public Dictionary<string, Slot> Slots = new Dictionary<string, Slot>();
@@ -45,6 +46,10 @@ namespace OpenRA.Network
session.Clients.Add(FieldLoader.Load<Client>(y.Value)); session.Clients.Add(FieldLoader.Load<Client>(y.Value));
break; break;
case "ClientPing":
session.ClientPings.Add(FieldLoader.Load<ClientPing>(y.Value));
break;
case "Slot": case "Slot":
var s = FieldLoader.Load<Slot>(y.Value); var s = FieldLoader.Load<Slot>(y.Value);
session.Slots.Add(s.PlayerReference, s); session.Slots.Add(s.PlayerReference, s);
@@ -107,6 +112,23 @@ namespace OpenRA.Network
public bool IsReady { get { return State == ClientState.Ready; } } public bool IsReady { get { return State == ClientState.Ready; } }
public bool IsInvalid { get { return State == ClientState.Invalid; } } public bool IsInvalid { get { return State == ClientState.Invalid; } }
public bool IsObserver { get { return Slot == null; } } public bool IsObserver { get { return Slot == null; } }
public string Serialize()
{
var clientData = new List<MiniYamlNode>();
clientData.Add(new MiniYamlNode("Client@{0}".F(this.Index), FieldSaver.Save(this)));
return clientData.WriteToString();
}
}
public ClientPing PingFromClient(Client client)
{
return ClientPings.SingleOrDefault(p => p.Index == client.Index);
}
public class ClientPing
{
public int Index;
public int Latency = -1; public int Latency = -1;
public int LatencyJitter = -1; public int LatencyJitter = -1;
public int[] LatencyHistory = { }; public int[] LatencyHistory = { };
@@ -114,7 +136,7 @@ namespace OpenRA.Network
public string Serialize() public string Serialize()
{ {
var clientData = new List<MiniYamlNode>(); var clientData = new List<MiniYamlNode>();
clientData.Add(new MiniYamlNode("Client@{0}".F(this.Index), FieldSaver.Save(this))); clientData.Add(new MiniYamlNode("ClientPing@{0}".F(this.Index), FieldSaver.Save(this)));
return clientData.WriteToString(); return clientData.WriteToString();
} }
} }
@@ -174,6 +196,9 @@ namespace OpenRA.Network
foreach (var client in Clients) foreach (var client in Clients)
sessionData.Append(client.Serialize()); sessionData.Append(client.Serialize());
foreach (var clientPing in ClientPings)
sessionData.Append(clientPing.Serialize());
foreach (var slot in Slots) foreach (var slot in Slots)
sessionData.Append(slot.Value.Serialize()); sessionData.Append(slot.Value.Serialize());

View File

@@ -218,6 +218,21 @@ namespace OpenRA.Network
break; break;
} }
case "SyncClientPings":
{
var pings = new List<Session.ClientPing>();
var nodes = MiniYaml.FromString(order.TargetString);
foreach (var node in nodes)
{
var strings = node.Key.Split('@');
if (strings[0] == "ClientPing")
pings.Add(FieldLoader.Load<Session.ClientPing>(node.Value));
}
orderManager.LobbyInfo.ClientPings = pings;
break;
}
case "SetStance": case "SetStance":
{ {
if (!Game.orderManager.LobbyInfo.GlobalSettings.FragileAlliances) if (!Game.orderManager.LobbyInfo.GlobalSettings.FragileAlliances)

View File

@@ -322,6 +322,9 @@ namespace OpenRA.Server
PreConns.Remove(newConn); PreConns.Remove(newConn);
Conns.Add(newConn); Conns.Add(newConn);
LobbyInfo.Clients.Add(client); LobbyInfo.Clients.Add(client);
var clientPing = new Session.ClientPing();
clientPing.Index = client.Index;
LobbyInfo.ClientPings.Add(clientPing);
Log.Write("server", "Client {0}: Accepted connection from {1}.", Log.Write("server", "Client {0}: Accepted connection from {1}.",
newConn.PlayerIndex, newConn.socket.RemoteEndPoint); newConn.PlayerIndex, newConn.socket.RemoteEndPoint);
@@ -442,6 +445,7 @@ namespace OpenRA.Server
switch (so.Name) switch (so.Name)
{ {
case "Command": case "Command":
{
bool handled = false; bool handled = false;
foreach (var t in serverTraits.WithInterface<IInterpretCommand>()) foreach (var t in serverTraits.WithInterface<IInterpretCommand>())
if (handled = t.InterpretCommand(this, conn, GetClient(conn), so.Data)) if (handled = t.InterpretCommand(this, conn, GetClient(conn), so.Data))
@@ -454,7 +458,7 @@ namespace OpenRA.Server
} }
break; break;
}
case "HandshakeResponse": case "HandshakeResponse":
ValidateClient(conn, so.Data); ValidateClient(conn, so.Data);
break; break;
@@ -472,20 +476,22 @@ namespace OpenRA.Server
break; break;
} }
var fromClient = GetClient(conn); var pingFromClient = LobbyInfo.PingFromClient(GetClient(conn));
var history = fromClient.LatencyHistory.ToList(); if (pingFromClient == null)
return;
var history = pingFromClient.LatencyHistory.ToList();
history.Add(Environment.TickCount - pingSent); history.Add(Environment.TickCount - pingSent);
// Cap ping history at 5 values (25 seconds) // Cap ping history at 5 values (25 seconds)
if (history.Count > 5) if (history.Count > 5)
history.RemoveRange(0, history.Count - 5); history.RemoveRange(0, history.Count - 5);
fromClient.Latency = history.Sum() / history.Count; pingFromClient.Latency = history.Sum() / history.Count;
fromClient.LatencyJitter = (history.Max() - history.Min()) / 2; pingFromClient.LatencyJitter = (history.Max() - history.Min()) / 2;
fromClient.LatencyHistory = history.ToArray(); pingFromClient.LatencyHistory = history.ToArray();
if (State == ServerState.WaitingPlayers) SyncClientPing();
SyncLobbyClients(); // TODO: SyncClientLatency
break; break;
} }
@@ -614,6 +620,19 @@ namespace OpenRA.Server
t.LobbyInfoSynced(this); t.LobbyInfoSynced(this);
} }
public void SyncClientPing()
{
var clientPings = new System.Text.StringBuilder();
foreach (var ping in LobbyInfo.ClientPings)
clientPings.Append(ping.Serialize());
DispatchOrders(null, 0,
new ServerOrder("SyncClientPings", clientPings.ToString()).Serialize());
foreach (var t in serverTraits.WithInterface<INotifySyncLobbyInfo>())
t.LobbyInfoSynced(this);
}
public void StartGame() public void StartGame()
{ {
listener.Stop(); listener.Stop();

View File

@@ -95,6 +95,9 @@ namespace OpenRA.Mods.RA
case "SyncClientInfo": case "SyncClientInfo":
case "SyncLobbySlots": case "SyncLobbySlots":
case "SyncLobbyGlobalSettings": case "SyncLobbyGlobalSettings":
case "SyncClientPing":
case "Ping":
case "Pong":
return; return;
} }
if (order.OrderString.StartsWith("Dev")) if (order.OrderString.StartsWith("Dev"))

View File

@@ -174,7 +174,11 @@ namespace OpenRA.Mods.RA.Server
if (occupant != null) if (occupant != null)
{ {
if (occupant.Bot != null) if (occupant.Bot != null)
{
server.LobbyInfo.Clients.Remove(occupant); server.LobbyInfo.Clients.Remove(occupant);
var ping = server.LobbyInfo.PingFromClient(occupant);
server.LobbyInfo.ClientPings.Remove(ping);
}
else else
{ {
var occupantConn = server.Conns.FirstOrDefault(c => c.PlayerIndex == occupant.Index); var occupantConn = server.Conns.FirstOrDefault(c => c.PlayerIndex == occupant.Index);
@@ -203,7 +207,11 @@ namespace OpenRA.Mods.RA.Server
// Slot may have a bot in it // Slot may have a bot in it
var occupant = server.LobbyInfo.ClientInSlot(s); var occupant = server.LobbyInfo.ClientInSlot(s);
if (occupant != null && occupant.Bot != null) if (occupant != null && occupant.Bot != null)
{
server.LobbyInfo.Clients.Remove(occupant); server.LobbyInfo.Clients.Remove(occupant);
var ping = server.LobbyInfo.PingFromClient(occupant);
server.LobbyInfo.ClientPings.Remove(ping);
}
server.SyncLobbyClients(); server.SyncLobbyClients();
return true; return true;

View File

@@ -71,8 +71,10 @@ namespace OpenRA.Mods.RA.Widgets.Logic
}; };
admin.IsVisible = () => orderManager.LobbyInfo.ClientWithIndex(clientIndex).IsAdmin; admin.IsVisible = () => orderManager.LobbyInfo.ClientWithIndex(clientIndex).IsAdmin;
latency.GetText = () => LobbyUtils.LatencyDescription(orderManager.LobbyInfo.ClientWithIndex(clientIndex).Latency); var client = orderManager.LobbyInfo.ClientWithIndex(clientIndex);
latency.GetColor = () => LobbyUtils.LatencyColor(orderManager.LobbyInfo.ClientWithIndex(clientIndex).Latency); var ping = orderManager.LobbyInfo.PingFromClient(client);
latency.GetText = () => LobbyUtils.LatencyDescription(ping);
latency.GetColor = () => LobbyUtils.LatencyColor(ping);
var address = orderManager.LobbyInfo.ClientWithIndex(clientIndex).IpAddress; var address = orderManager.LobbyInfo.ClientWithIndex(clientIndex).IpAddress;
if (address == "127.0.0.1" && UPnP.NatDevice != null) if (address == "127.0.0.1" && UPnP.NatDevice != null)
address = UPnP.NatDevice.GetExternalIP().ToString(); address = UPnP.NatDevice.GetExternalIP().ToString();

View File

@@ -163,26 +163,32 @@ namespace OpenRA.Mods.RA.Widgets.Logic
} }
} }
public static Color LatencyColor(int latency) public static Color LatencyColor(Session.ClientPing ping)
{ {
if (ping == null)
return Color.Gray;
// Levels set relative to the default order lag of 3 net ticks (360ms) // Levels set relative to the default order lag of 3 net ticks (360ms)
// TODO: Adjust this once dynamic lag is implemented // TODO: Adjust this once dynamic lag is implemented
if (latency < 0) if (ping.Latency < 0)
return Color.Gray; return Color.Gray;
if (latency < 300) if (ping.Latency < 300)
return Color.LimeGreen; return Color.LimeGreen;
if (latency < 600) if (ping.Latency < 600)
return Color.Orange; return Color.Orange;
return Color.Red; return Color.Red;
} }
public static string LatencyDescription(int latency) public static string LatencyDescription(Session.ClientPing ping)
{ {
if (latency < 0) if (ping == null)
return "Unknown"; return "Unknown";
if (latency < 300)
if (ping.Latency < 0)
return "Unknown";
if (ping.Latency < 300)
return "Good"; return "Good";
if (latency < 600) if (ping.Latency < 600)
return "Moderate"; return "Moderate";
return "Poor"; return "Poor";
} }
@@ -216,7 +222,8 @@ namespace OpenRA.Mods.RA.Widgets.Logic
block.IsVisible = () => visible; block.IsVisible = () => visible;
if (visible) if (visible)
block.Get<ColorBlockWidget>("LATENCY_COLOR").GetColor = () => LatencyColor(c.Latency); block.Get<ColorBlockWidget>("LATENCY_COLOR").GetColor = () => LatencyColor(
orderManager.LobbyInfo.PingFromClient(c));
var tooltip = parent.Get<ClientTooltipRegionWidget>("CLIENT_REGION"); var tooltip = parent.Get<ClientTooltipRegionWidget>("CLIENT_REGION");
tooltip.IsVisible = () => visible; tooltip.IsVisible = () => visible;