I have some bare bones networking fucntionality running, where I use one client to send positions and the the server just shows it. It usually works fine, but sometimes the servers starts lagging behind. By showing the queue with NetworkTransport.GetIncomingMessageQueueSize() I can see that the lag starts when the queue goes from the 0-2 that it usually resides in, to 9+. I am using the unreliable channel as I don’t really care about order or if some packets are lost, I just want the newest position as fast as possible.
Is there a way to ignore or empty the queue, I don’t care for old positions, only the newest one. I can’t really find anything. I am using NetworkTransport.
For MessageUpdates. You probably want to use StateUpdate as your QOS. In addition. You might want to call Transport.Recieve more than once per frame. I call it until the queue is empty but I also have a max amount of messages to process per frame so we don’t freeze on one frame for too long.
Makes sense that with unreliable I don’t get ordering correctly, thanks.
For your solution how do I iterate through the queue? In NetworkTransport I cannot find any methods for manipulating the queue. I think what you mean is to iterate through the queue and only act on the one with the most recent timestamp.
Will take a look at stateUpdate, from the description it seems like that is exactly what I want, but when I test it my queueSize just rapidly increases? Shouldn’t I deal with that?
For calling more than once per frame, are you using a seperate thread? I thought about that, but since my visuals will only update on frame, in my mind there was no reason to read more that one message per frame. Or do you just loop until the the queue has a size of 1 and use that. From Driiades’s post it seems like that won’t necessarily get me the latest one, and how this this work with stateUpdate to empty the queue?
Well, if you try to send a ton of messages with the State Update QOS. It will still fill the queue. But once it’s time to send them away. It only takes the last one. So I guess only send once every so often.
And personally I iterate though all messages in each update. In the Main thread.Usually you will have game logic to be ran in the Update method. So if you can process multiple messages between each logic thats great. Usually you want to empty the queue.
What I mean is that the server who is receiving the messages, gets growing incoming queue, which confuses me as the only thing sending to it is a single client using state update. Is it because state update only receive the most recent, but still leaves everything else in the incoming queue?
Can you give a code example of how to process multiple messages in a single update.
Is it like this?
do {
NetworkEventType recData = NetworkTransport.Receive(out recHostId, out connectionId, out channelId, recBuffer, bufferSize, out dataSize, out error);
} while { NetworkTransport.GetIncomingMessageQueueSize(recHostId, out error) > 1 }
int messagesProcessed = 0;
do
{
networkEvent = NetworkTransport.Receive(out recHostId, out connectionId, out channelId, recBuffer, ServerSettings.singleton.MaxMessageSize, out recSize, out error);
switch (networkEvent)
{
case NetworkEventType.Nothing:
break;
case NetworkEventType.ConnectEvent:
throw new NotImplementedException();
break;
case NetworkEventType.DataEvent:
throw new NotImplementedException();
break;
case NetworkEventType.DisconnectEvent:
throw new NotImplementedException();
break;
}
messagesProcessed++;
} while (networkEvent != NetworkEventType.Nothing &&
ServerSettings.singleton.ProcessMultipleRequestsPerTick &&
(messagesProcessed == -1 || messagesProcessed < ServerSettings.singleton.MaxMessagesToProcessPerTick));
In your Update loop. As for how the QOS is handled on the Recieving end, I am actually unsure of. All the documentation states is that we only recieve the latest one. Not sure how that would work tho. @aabramychev is probably the one to answer that one.
The maxMessageProcessed is just there if the queue is totally full the game doesnt just freeze.
My above code simply Polls the queue until we get “Nothing” OR until our limit is reached (If enabled)
Sorry I am a bit confused. Are you saying that is how state update work, or something I should add to my code when using state update? In your code snippet, how do I get those variables, current message and remove. I cannot find anywhere where I can manipulate the sending queue from networkTransport.
If you mean that i how it just works, that means I should just ignore the growing incommingqueue? with 600+ message in queue after a minute.
So the conclusion is that I just ignore the massively building incoming queue? Doesn’t it block my other channels, if I want a reliable message also for example?
On my client which is sending the stateupdate i get a constant queueSize of 2 with this code. (VDebug just writes to Debug.log and prints on my screen)
int queueSize = NetworkTransport.GetOutgoingMessageQueueSize(hostConnId, out error);
VDebug("Queue size: "+queueSize+", Sending: " + message, false);
On my server my queue from only one client and only stateUpdates being sent I have an increasing queue growing with approx 15 every second. In my dataEvent i have this:
int queueSize = NetworkTransport.GetIncomingMessageQueueSize(recHostId, out error);
VDebug("Queuesize: "+queueSize+", Receiving: " + msg + " from: " + connectionId, false);
In my client I am sending a stateupdate once per. frame/update. and On my server I am checking for message once per frame/update.
@munkbusiness , Difficult to say, need to see the code :(, report the bug please. I think that it is not very urgent task? Let’s our engineers check it first.
Would it be possible in the future that we get the ability to clear the Outgoing and Incoming NetworkTransport queue, as well as the Ack buffer?
My use case is, if I detect that a connection doesnt seem to be receiving any of my reliable messages and the ack buffer is almost full, I basically want to put that connection into an “unready” state. I would clear all incoming and outgoing messages for that connection as if I disconnected from them, I would also clear the ack buffer since I no longer care about those reliable messages reaching them, and will then send a “ResetConnection” reliable message and wait for their reply.
This way I dont have to go through the whole disconnect connection and reconnect process, destroying all their objects and what not.
I might have a long DisconnectTimeout, such as 20 seconds (unity sets default to 2 seconds), so it is possible that I might have sent to many reliable messages during those 20 seconds, which would cause problems as reliable packets start to not send since the ack buffer is full.
If you think 20 second DisconnectTimeout is to long, I think Team Fortress 2 has a 20 or 30 second DisconnectTimeout, and there are plenty of times where I lose connection for 5, 10, 15 seconds or so before finally reconnecting (I can see on my windows wireless network connection icon that my internet is having issues, so it seems to be my connection and not the game). So having it not disconnect me from the game during those 20 seconds is really nice.
From memory, I think I also saw my OutgoingQueueSize being a size of 2 when using NetworkTransport.QueueMessageForSending. When using regular NetworkTransport.Send, it seems, at least on localhost, that StateUpdate gets sent instantly regardless of your connectionconfig update rate, and ReliableStateUpdate just keeps increasing the OutgoingQueueSize. However, with NetworkTransport.QueueMessageForSending, both seem to work fine, but they do queue up 2 and not 1 for some reason.
@munkbusiness “On my client which is sending the stateupdate i get a constant queueSize of 2 with this code. (VDebug just writes to Debug.log and prints on my screen)” it is correct
But for server, not sure, it should be ~2 too if you send only stateupdate messages. So, yes it is probably bug, report them. As a problem is not urgent i guess, the bug reporting will be better here
@HiddenMonk , not sure. As You have already said, you can always call connect/disconnect then, +I will open (which was hidden before) send callback, where library will notify you when you can call send again. I’m not say not, but now I can not foresee any side effects of this feature:( If you already have a proper design in your head you are welcome PM me Make sense?