Manging Sessions
Everytime a client connects to the server a new SessionData is created. The session is used to store the client's custom data, such as nickname, score, team, etc. The session is not bound to client object itself, but to the client id itself. You can customize the session management implementing a custom NetworkManager class. In this guide, we will show you how to create a custom session management system.
Establishing a Session
When a client connects, a NetworkSessionEstablishRequestPacket is sent to the server. The method NetworkManager.OnCreateSessionEstablishRequest can be used to create a custom session establish packet:
| protected override NetworkSessionEstablishRequestPacket OnCreateSessionEstablishRequest()
{
return new ExampleNetworkSessionEstablishRequestPacket
{
ShouldBeAccepted = shouldBeAccepted.isOn,
Nickname = nicknameInput.text
};
}
|
The server receives the packet and can accept or reject the client. The method NetworkManager.OnSessionEstablishingRequest is called when the session is established, accepting or rejecting the client:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16 | protected override SessionEstablishingResponse OnSessionEstablishingRequest(NetworkSessionEstablishRequestPacket packet)
{
var examplePacket = (ExampleNetworkSessionEstablishRequestPacket) packet;
if (examplePacket.ShouldBeAccepted)
return new SessionEstablishingResponse
{
Type = SessionEstablishingResponse.SessionEstablishingResponseType.Accept,
};
return new SessionEstablishingResponse
{
Type = SessionEstablishingResponse.SessionEstablishingResponseType.Reject,
Reason = "just_testing"
};
}
|
Creating the Session
After a session is established, the method NetworkManager.OnCreateNewSessionData is called. This method is used to create the session object using the NetworkSessionEstablishRequestPacket used to establish the session:
| protected override SessionData OnCreateNewSessionData(int clientId, NetworkSessionEstablishRequestPacket packet)
{
var packetData = (ExampleNetworkSessionEstablishRequestPacket) packet;
return new ExampleSessionData()
{
nickname = packetData.Nickname
};
}
|
After this, the session is created and the server syncs all object states to the client.
Session Restoration
If the option NetworkManager.SupportsSessionRestoration is enabled, the server will try to restore the client's session after a hot-reload. The method NetworkManager.OnRestoreSessionData is called to restore the session data:
| protected override SessionData OnRestoreSessionData(int clientId, NetworkSessionEstablishRequestPacket packet)
{
//Keeps the session using the nickname
var packetData = (ExampleNetworkSessionEstablishRequestPacket) packet;
return GetAllDisconnectedSessionData<ExampleSessionData>().FirstOrDefault(data => data.nickname == packetData.Nickname);
}
|
If no session is found (returns null), the method NetworkManager.OnCreateEmptySessionData is called to create a new session.
By default, the NetworkManager uses the client id to restore the session, so after a hot-reload, the client will be able to restore its session.
Warning
The client id increases after each connection, so if you are using the client id to restore the session, you should not rely on it to store any important data. If a client connects the disconnects and connects again, his id may not be the same after a hot-reload. We heavily recommend using another key to store the session data, such as the client's nickname.
Handling Session Data
The session data is stored in A SessionData object, stored by the server. When the SessionData is created, it's content is synced to the client. The local SessionData object can be accessed using the NetworkManager.GetLocalSessionData method.
Only the server may change the session data contents, and after a change, the server needs to call the SessionData.ApplyChanges method to sync the changes to the client:
1
2
3
4
5
6
7
8
9
10
11
12 | public void Update()
{
if(!IsServerRunning)
return;
foreach (var sessionData in GetAllSessionData<ExampleSessionData>())
{
sessionData.someInt++;
//Sync data to the client
sessionData.ApplyChanges();
}
}
|
The method NetworkManager.OnLocalSessionDataChanged is called automatically when the client receives the session data changes:
| protected override void OnLocalSessionDataChanged(SessionData data)
{
var exampleSessionData = GetLocalSessionData<ExampleSessionData>();
//var exampleSessionData = (ExampleSessionData) data;
Debug.LogError($"Test: {exampleSessionData.someInt}");
}
|
Full Session Management Example
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131 | namespace Samples.SessionDataTesting.Scripts
{
public class SessionDataTestingNetworkManager : NetworkManager
{
public InputField nicknameInput;
public Toggle shouldBeAccepted;
#region Session Establishing
protected override NetworkSessionEstablishRequestPacket OnCreateSessionEstablishRequest()
{
return new ExampleNetworkSessionEstablishRequestPacket
{
ShouldBeAccepted = shouldBeAccepted.isOn,
Nickname = nicknameInput.text
};
}
protected override SessionEstablishingResponse OnSessionEstablishingRequest(NetworkSessionEstablishRequestPacket packet)
{
var examplePacket = (ExampleNetworkSessionEstablishRequestPacket) packet;
if (examplePacket.ShouldBeAccepted)
return new SessionEstablishingResponse
{
Type = SessionEstablishingResponse.SessionEstablishingResponseType.Accept,
};
return new SessionEstablishingResponse
{
Type = SessionEstablishingResponse.SessionEstablishingResponseType.Reject,
Reason = "just_testing"
};
}
#endregion
#region Session Data Restoring
protected override SessionData OnRestoreSessionData(int clientId, NetworkSessionEstablishRequestPacket packet)
{
//Keeps the session using the nickname
var packetData = (ExampleNetworkSessionEstablishRequestPacket) packet;
return GetAllDisconnectedSessionData<ExampleSessionData>().FirstOrDefault(data => data.nickname == packetData.Nickname);
}
#endregion
#region Session Data Creation
//Server Side
protected override SessionData OnCreateNewSessionData(int clientId, NetworkSessionEstablishRequestPacket packet)
{
var packetData = (ExampleNetworkSessionEstablishRequestPacket) packet;
return new ExampleSessionData()
{
nickname = packetData.Nickname
};
}
protected override SessionData OnCreateEmptySessionData()
{
return new ExampleSessionData();
}
#endregion
#region Session Data Manipulation
public void Update()
{
if (Time.frameCount % 100 != 0)
return;
if(!IsServerRunning)
return;
foreach (var sessionData in GetAllSessionData<ExampleSessionData>())
{
sessionData.someInt++;
//Sync data to the client
sessionData.ApplyChanges();
}
}
//Client Side
public override void OnLocalSessionDataChanged(SessionData data)
{
var exampleSessionData = GetLocalSessionData<ExampleSessionData>();
//var exampleSessionData = (ExampleSessionData) data;
Debug.LogError($"Test: {exampleSessionData.someInt}");
}
#endregion
}
public class ExampleNetworkSessionEstablishRequestPacket : NetworkSessionEstablishRequestPacket
{
public bool ShouldBeAccepted { get; set; }
public string Nickname { get; set; }
public override void Serialize(BinaryWriter writer)
{
writer.Write(ShouldBeAccepted);
writer.Write(Nickname);
}
public override void Deserialize(BinaryReader reader)
{
ShouldBeAccepted = reader.ReadBoolean();
Nickname = reader.ReadString();
}
}
public class ExampleSessionData : SessionData
{
//Doesn't needs to be synced to the client
public string nickname;
//Needs to be synced to the client
public int someInt;
public override void Serialize(BinaryWriter writer, bool shouldSerializeEverything)
{
if (shouldSerializeEverything)
writer.Write(nickname);
writer.Write(someInt);
}
public override void Deserialize(BinaryReader reader, bool shouldDeserializeEverything)
{
if (shouldDeserializeEverything)
nickname = reader.ReadString();
someInt = reader.ReadInt32();
}
}
}
|
Note
The parameter shouldSerializeEverything will only be true when the session data is being kept through a hot-reload, so all the data must be serialized.