Acquiring Servers Ping

Hi there,

We have been testing some MasterServer implementation on my local network.
We have configured a MasterServer, and created some Hosts.
We have successfully received the server list using the MasterServer.PollHostList method and so on…

The problem is we want the client to connect automatically to the best suitable server, taking into account the server load (users connected and ping), and we are unable to get the ping value correctly.

If we use this piece of code:

var ServerPing : Ping;

function OnGUI()
{
  ServerPing = Ping("192.168.1.59");
  while(!ServerPing.isDone) {}
  var Latency : String = (ServerPing.time).ToString();
  guiText.text = ("Server latency: " + Latency);
}

Unity almost freezes on mac, and totally freezes or even do not load on windows.

On the other hand, trying to do something similar to this (deleted other code to go straight to the bone):

var ServerPing : Ping;
var PingTime = -2;

function Awake()
{
 ServerPing = Ping("192.168.1.59");
}

function OnGUI()
{
  if (PingTime < -2)
{
  var Latency : String = (ServerPing.time).ToString();
  guiText.text = ("Server latency: " + Latency);
}
}

function Update()
{
  if(ServerPing.isDone) 
    PingTime = ServerPing.time;
}

But it seems that ServerPing is not accessible from Update…

I don’t know, I think this should be easier than it seems. Am I doing something terribly wrong? I have been searching throughout all the forum with no success…

Thanks in advance[/code]

This “while(!ServerPing.isDone) {}” is really not a good idea because it will stop Unity while it’s trying to execute OnGUI. What you should do instead is start somewhare (e.g. in the in GUI) and then in each call to OnGUI check if ServerPing.isDone. Obviously you can’t start Ping(…) each time OnGUI is called so one option is using a flag and once this is set, you don’t call Ping(…) anymore - but instead check for ServerPing.isDone and when that’s the case, calculate your latency.

Sunny regards,
Jashan

Hi Jashan,

Thanks for answering.

Actually, our second way to do it was exactly as you say, but it seems there is something wrong with our code.

This code now works as intended, It shows a list of all the registered servers and their pings when available, but only stores a single Ping value:

var ServerPing: Ping;
var NumServers : int = 0;
var bAllServersPinged : boolean = false;
var hostData : HostData[];
var firstLaunch = true;

function Awake() {
	MasterServer.ipAddress = "192.168.1.32";
	MasterServer.port = 25000;
	MasterServer.ClearHostList();
	MasterServer.RequestHostList("EnneGame");
	hostData = MasterServer.PollHostList(); 

}

function InitializePings()
{	
	
	hostData = MasterServer.PollHostList(); 
	if (hostData.length != 0)
	{
		for(var i=0;i<hostData.length;i++)
		{
			ServerPing=new Ping("192.168.1.104");
		}
		NumServers = 1;
	}
	else
	{
		guiText.text=("no server found");
	}	
}

function OnGUI() {
		if (Network.peerType == NetworkPeerType.Disconnected)
		{
			guiText.text = ("");
			
			if (hostData.length != 0)
			{
	
				guiText.text = ("Available servers...");
				
				GUI.Box(Rect(10,40, (Screen.width-20), (Screen.height-80)),"");
				GUILayout.BeginArea(Rect(20,50, (Screen.width-40), (Screen.height-100)));
										
				for (var i = 0; i<hostData.length; i++)
				{						
					GUILayout.BeginHorizontal();
					var name = hostData[i].gameName + " " + hostData[i].connectedPlayers + " / " + hostData[i].playerLimit;
					GUILayout.Label(name);	
					GUILayout.Space(5);
					var hostInfo;
					hostInfo = "[";
					for (var host in hostData[i].ip)
						hostInfo = hostInfo + host + ":" + hostData[i].port + " ";
					hostInfo = hostInfo + "]";
					GUILayout.Label(hostInfo);	
					GUILayout.Space(5);
					GUILayout.Label(hostData[i].comment);
					
					if(ServerPing.isDone)
					{
						var Latency : String = (ServerPing.time).ToString();			
					    GUILayout.Space(5);
					    GUILayout.Label( "(" + Latency + " ms)");
					}
					else
					{
						GUILayout.Space(5);
					    GUILayout.Label( "(pinging...)");
					}
					
					GUILayout.Space(5);
					GUILayout.FlexibleSpace();
					GUILayout.EndHorizontal();
				}			
									
				GUILayout.EndArea();
			}
			else
			{
				guiText.text = ("Server list empty, hit refresh to fetch new list");
			}
			
			if (GUILayout.Button("Refresh server list"))
			{
				MasterServer.RequestHostList("EnneGame");
				InitializePings();
			}
	
		}
		else
		{
			guiText.text = ("Connected to: " + BestServerIp);
			
			if (GUILayout.Button ("Disconnect"))
			{
				Network.Disconnect(200);
			}
		}
}

function OnConnectedToServer() {
	// Notify our objects that the level and the network is ready
	for (var go in FindObjectsOfType(GameObject))
		go.SendMessage("OnNetworkLoadedLevel", SendMessageOptions.DontRequireReceiver);		
}

function OnDisconnectedFromServer () {
	if (this.enabled != false)
		Application.LoadLevel(Application.loadedLevel);
	else
		FindObjectOfType(NetworkLevelLoad).OnDisconnectedFromServer();
}

function Update(){
	
	if(firstLaunch)
	{
		InitializePings();
		firstLaunch = false;
	}	
	
}

This works fine.
The problem is, if we now change…

var ServerPing: Ping;

for…

var ServerPing: Ping[];

And all the calls that were something like…

if(ServerPing.isDone)
{
  var Latency : String = (ServerPing.time).ToString();			
					    GUILayout.Space(5);
					    GUILayout.Label( "(" + Latency + " ms)");
					}

for…

if(ServerPing[0].isDone)
{
  var Latency : String = (ServerPing[0].time).ToString();			
					    GUILayout.Space(5);
					    GUILayout.Label( "(" + Latency + " ms)");
					}

… it comes up with a NullReferenceException error, and i don’t understand why…
May it be possible that, when we perform a PollHostList we could be losing all references that were inside the Ping array?..

I am completely stuck here, and can’t understand why a single variable works and an array don’t…

thanks in advance

Ok!, I found the solution myself.
I needed to specify a length for the ServerPing array! how dumb!..

I simply added this just before filling the ping values and it worked!

ServerPing = new Ping[hostData.length];

Thanks for your kind help!

Ping not return when Sever Die.