Compare commits
5 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 9da397dd8c | |||
| 6bb84daf03 | |||
| f1c5c99417 | |||
| e52e940fda | |||
| e78e8048ff |
@@ -65,6 +65,8 @@ namespace Ragon.Protocol
|
|||||||
{
|
{
|
||||||
public class RagonBuffer
|
public class RagonBuffer
|
||||||
{
|
{
|
||||||
|
private const int MaxBufferSize = 1024 * 1024; // 1MB max buffer size
|
||||||
|
|
||||||
private int _read;
|
private int _read;
|
||||||
private int _write;
|
private int _write;
|
||||||
private uint[] _buckets;
|
private uint[] _buckets;
|
||||||
@@ -404,6 +406,12 @@ namespace Ragon.Protocol
|
|||||||
public void FromArray(byte[] data)
|
public void FromArray(byte[] data)
|
||||||
{
|
{
|
||||||
var length = data.Length;
|
var length = data.Length;
|
||||||
|
|
||||||
|
if (length > MaxBufferSize)
|
||||||
|
{
|
||||||
|
throw new InvalidOperationException($"Input data exceeds maximum buffer size: {length} bytes > {MaxBufferSize} bytes");
|
||||||
|
}
|
||||||
|
|
||||||
var bucketsCount = length / 4 + 1;
|
var bucketsCount = length / 4 + 1;
|
||||||
|
|
||||||
if (_buckets.Length < bucketsCount)
|
if (_buckets.Length < bucketsCount)
|
||||||
@@ -493,7 +501,14 @@ namespace Ragon.Protocol
|
|||||||
|
|
||||||
private void Resize(int capacity)
|
private void Resize(int capacity)
|
||||||
{
|
{
|
||||||
var buckets = new uint[_buckets.Length * 2 + capacity];
|
var newSize = _buckets.Length * 2 + capacity;
|
||||||
|
|
||||||
|
if (newSize * 4 > MaxBufferSize)
|
||||||
|
{
|
||||||
|
throw new InvalidOperationException($"Buffer size limit exceeded: {newSize * 4} bytes > {MaxBufferSize} bytes");
|
||||||
|
}
|
||||||
|
|
||||||
|
var buckets = new uint[newSize];
|
||||||
Array.Copy(_buckets, buckets, _buckets.Length);
|
Array.Copy(_buckets, buckets, _buckets.Length);
|
||||||
_buckets = buckets;
|
_buckets = buckets;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
FROM mcr.microsoft.com/dotnet/sdk:7.0 AS build-env
|
FROM mcr.microsoft.com/dotnet/sdk:9.0 AS build-env
|
||||||
|
|
||||||
WORKDIR /App
|
WORKDIR /App
|
||||||
|
|
||||||
@@ -7,7 +7,7 @@ COPY . ./
|
|||||||
RUN dotnet restore
|
RUN dotnet restore
|
||||||
RUN dotnet publish -c Release -o out
|
RUN dotnet publish -c Release -o out
|
||||||
|
|
||||||
FROM mcr.microsoft.com/dotnet/runtime:7.0
|
FROM mcr.microsoft.com/dotnet/runtime:9.0
|
||||||
|
|
||||||
WORKDIR /App
|
WORKDIR /App
|
||||||
|
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<OutputType>Exe</OutputType>
|
<OutputType>Exe</OutputType>
|
||||||
<RootNamespace>Ragon.Relay</RootNamespace>
|
<RootNamespace>Ragon.Relay</RootNamespace>
|
||||||
<TargetFramework>net8.0</TargetFramework>
|
<TargetFramework>net9.0</TargetFramework>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<PropertyGroup Condition=" '$(Configuration)' == 'Release' ">
|
<PropertyGroup Condition=" '$(Configuration)' == 'Release' ">
|
||||||
|
|||||||
@@ -10,5 +10,6 @@
|
|||||||
"limitRooms": 200,
|
"limitRooms": 200,
|
||||||
"limitBufferedEvents": 50,
|
"limitBufferedEvents": 50,
|
||||||
"limitUserDataSize": 1024,
|
"limitUserDataSize": 1024,
|
||||||
"limitPropertySize": 512
|
"limitPropertySize": 512,
|
||||||
|
"limitConnectionsPerProject": 100
|
||||||
}
|
}
|
||||||
@@ -4,7 +4,6 @@
|
|||||||
<ImplicitUsings>enable</ImplicitUsings>
|
<ImplicitUsings>enable</ImplicitUsings>
|
||||||
<Nullable>enable</Nullable>
|
<Nullable>enable</Nullable>
|
||||||
<RootNamespace>Ragon.ENet</RootNamespace>
|
<RootNamespace>Ragon.ENet</RootNamespace>
|
||||||
<TargetFrameworks>net6.0;net7.0;net8.0</TargetFrameworks>
|
|
||||||
<GeneratePackageOnBuild>true</GeneratePackageOnBuild>
|
<GeneratePackageOnBuild>true</GeneratePackageOnBuild>
|
||||||
<Copyright>Eduard Kargin</Copyright>
|
<Copyright>Eduard Kargin</Copyright>
|
||||||
<Authors>Eduard Kargin</Authors>
|
<Authors>Eduard Kargin</Authors>
|
||||||
@@ -15,6 +14,7 @@
|
|||||||
<PackageLicenseExpression>Apache-2.0</PackageLicenseExpression>
|
<PackageLicenseExpression>Apache-2.0</PackageLicenseExpression>
|
||||||
<RepositoryUrl>https://github.com/edmand46/Ragon</RepositoryUrl>
|
<RepositoryUrl>https://github.com/edmand46/Ragon</RepositoryUrl>
|
||||||
<RepositoryType>Source</RepositoryType>
|
<RepositoryType>Source</RepositoryType>
|
||||||
|
<TargetFramework>net9.0</TargetFramework>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
|
|||||||
@@ -4,7 +4,6 @@
|
|||||||
<ImplicitUsings>enable</ImplicitUsings>
|
<ImplicitUsings>enable</ImplicitUsings>
|
||||||
<Nullable>enable</Nullable>
|
<Nullable>enable</Nullable>
|
||||||
<RootNamespace>Ragon.WebSockets</RootNamespace>
|
<RootNamespace>Ragon.WebSockets</RootNamespace>
|
||||||
<TargetFrameworks>net6.0;net7.0;net8.0</TargetFrameworks>
|
|
||||||
<Copyright>Eduard Kargin</Copyright>
|
<Copyright>Eduard Kargin</Copyright>
|
||||||
<Authors>Eduard Kargin</Authors>
|
<Authors>Eduard Kargin</Authors>
|
||||||
<GeneratePackageOnBuild>true</GeneratePackageOnBuild>
|
<GeneratePackageOnBuild>true</GeneratePackageOnBuild>
|
||||||
@@ -15,6 +14,7 @@
|
|||||||
<PackageLicenseExpression>Apache-2.0</PackageLicenseExpression>
|
<PackageLicenseExpression>Apache-2.0</PackageLicenseExpression>
|
||||||
<RepositoryUrl>https://github.com/edmand46/Ragon</RepositoryUrl>
|
<RepositoryUrl>https://github.com/edmand46/Ragon</RepositoryUrl>
|
||||||
<RepositoryType>Source</RepositoryType>
|
<RepositoryType>Source</RepositoryType>
|
||||||
|
<TargetFramework>net9.0</TargetFramework>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
|
|||||||
@@ -43,7 +43,7 @@ public class WebSocketServer : INetworkServer
|
|||||||
_executor = new Executor();
|
_executor = new Executor();
|
||||||
}
|
}
|
||||||
|
|
||||||
public async void StartAccept(CancellationToken cancellationToken)
|
public async ValueTask StartAccept(CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
while (!cancellationToken.IsCancellationRequested)
|
while (!cancellationToken.IsCancellationRequested)
|
||||||
{
|
{
|
||||||
@@ -58,26 +58,26 @@ public class WebSocketServer : INetworkServer
|
|||||||
context.Response.Close();
|
context.Response.Close();
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
var webSocketContext = await context.AcceptWebSocketAsync(null);
|
var webSocketContext = await context.AcceptWebSocketAsync(null);
|
||||||
var webSocket = webSocketContext.WebSocket;
|
var webSocket = webSocketContext.WebSocket;
|
||||||
var peerId = _sequencer.Pop();
|
var peerId = _sequencer.Pop();
|
||||||
|
|
||||||
connection = new WebSocketConnection(webSocket, peerId);
|
connection = new WebSocketConnection(webSocket, peerId);
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
_logger.Error(ex);
|
_logger.Error(ex);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
_connections[connection.Id] = connection;
|
_connections[connection.Id] = connection;
|
||||||
|
|
||||||
StartListen(connection, cancellationToken);
|
_ = StartListen(connection, cancellationToken);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async void StartListen(WebSocketConnection connection, CancellationToken cancellationToken)
|
async ValueTask StartListen(WebSocketConnection connection, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
_activeConnections.Add(connection);
|
_activeConnections.Add(connection);
|
||||||
_networkListener.OnConnected(connection);
|
_networkListener.OnConnected(connection);
|
||||||
@@ -86,7 +86,7 @@ public class WebSocketServer : INetworkServer
|
|||||||
var rawData = new byte[2048];
|
var rawData = new byte[2048];
|
||||||
var rawDataBuffer = new Memory<byte>(rawData);
|
var rawDataBuffer = new Memory<byte>(rawData);
|
||||||
var data = new ArrayBufferWriter<byte>();
|
var data = new ArrayBufferWriter<byte>();
|
||||||
|
|
||||||
while (webSocket.State == WebSocketState.Open || !cancellationToken.IsCancellationRequested)
|
while (webSocket.State == WebSocketState.Open || !cancellationToken.IsCancellationRequested)
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
@@ -95,17 +95,17 @@ public class WebSocketServer : INetworkServer
|
|||||||
{
|
{
|
||||||
var result = await webSocket.ReceiveAsync(rawDataBuffer, cancellationToken);
|
var result = await webSocket.ReceiveAsync(rawDataBuffer, cancellationToken);
|
||||||
var payload = rawDataBuffer.Slice(0, result.Count);
|
var payload = rawDataBuffer.Slice(0, result.Count);
|
||||||
|
|
||||||
data.Write(payload.Span);
|
data.Write(payload.Span);
|
||||||
|
|
||||||
if (result.EndOfMessage)
|
if (result.EndOfMessage)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (data.WrittenCount > 0)
|
if (data.WrittenCount > 0)
|
||||||
{
|
{
|
||||||
_networkListener.OnData(connection, NetworkChannel.RELIABLE, data.WrittenMemory.ToArray());
|
_networkListener.OnData(connection, NetworkChannel.RELIABLE, data.WrittenMemory.ToArray());
|
||||||
|
|
||||||
data.Clear();
|
data.Clear();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -117,15 +117,15 @@ public class WebSocketServer : INetworkServer
|
|||||||
|
|
||||||
_sequencer.Push(connection.Id);
|
_sequencer.Push(connection.Id);
|
||||||
_activeConnections.Remove(connection);
|
_activeConnections.Remove(connection);
|
||||||
|
|
||||||
_networkListener.OnDisconnected(connection);
|
_networkListener.OnDisconnected(connection);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Update()
|
public void Update()
|
||||||
{
|
{
|
||||||
_executor.Update();
|
_executor.Update();
|
||||||
|
|
||||||
Flush();
|
_ = Flush();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Broadcast(byte[] data, NetworkChannel channel)
|
public void Broadcast(byte[] data, NetworkChannel channel)
|
||||||
@@ -134,7 +134,7 @@ public class WebSocketServer : INetworkServer
|
|||||||
activeConnection.Reliable.Send(data);
|
activeConnection.Reliable.Send(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
public async void Flush()
|
public async ValueTask Flush()
|
||||||
{
|
{
|
||||||
foreach (var conn in _activeConnections)
|
foreach (var conn in _activeConnections)
|
||||||
await conn.Flush();
|
await conn.Flush();
|
||||||
|
|||||||
@@ -14,7 +14,7 @@
|
|||||||
<RepositoryUrl>https://github.com/edmand46/Ragon</RepositoryUrl>
|
<RepositoryUrl>https://github.com/edmand46/Ragon</RepositoryUrl>
|
||||||
<RepositoryType>Source</RepositoryType>
|
<RepositoryType>Source</RepositoryType>
|
||||||
<LangVersion>10</LangVersion>
|
<LangVersion>10</LangVersion>
|
||||||
<TargetFrameworks>net6.0;net7.0;net8.0</TargetFrameworks>
|
<TargetFramework>net9.0</TargetFramework>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
|
|||||||
@@ -12,13 +12,25 @@ public class RagonData
|
|||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Read(RagonBuffer buffer)
|
public void Read(RagonBuffer buffer, int maxSize = 0)
|
||||||
{
|
{
|
||||||
var len = buffer.ReadUShort();
|
var len = buffer.ReadUShort();
|
||||||
|
var totalSize = 0;
|
||||||
|
|
||||||
for (int i = 0; i < len; i++)
|
for (int i = 0; i < len; i++)
|
||||||
{
|
{
|
||||||
var key = buffer.ReadString();
|
var key = buffer.ReadString();
|
||||||
var valueSize = buffer.ReadUShort();
|
var valueSize = buffer.ReadUShort();
|
||||||
|
|
||||||
|
if (maxSize > 0)
|
||||||
|
{
|
||||||
|
totalSize += valueSize;
|
||||||
|
if (totalSize > maxSize)
|
||||||
|
{
|
||||||
|
throw new InvalidOperationException($"User data exceeds limit: {totalSize} > {maxSize}");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (valueSize > 0)
|
if (valueSize > 0)
|
||||||
{
|
{
|
||||||
var value = buffer.ReadBytes(valueSize);
|
var value = buffer.ReadBytes(valueSize);
|
||||||
@@ -59,8 +71,6 @@ public class RagonData
|
|||||||
buffer.WriteString(prop.Key);
|
buffer.WriteString(prop.Key);
|
||||||
buffer.WriteUShort((ushort)prop.Value.Length);
|
buffer.WriteUShort((ushort)prop.Value.Length);
|
||||||
buffer.WriteBytes(prop.Value);
|
buffer.WriteBytes(prop.Value);
|
||||||
|
|
||||||
Console.WriteLine($"Key: {prop.Key} Value: {prop.Value.Length}");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -19,6 +19,7 @@ using Ragon.Server.IO;
|
|||||||
using Ragon.Server.Lobby;
|
using Ragon.Server.Lobby;
|
||||||
using Ragon.Server.Logging;
|
using Ragon.Server.Logging;
|
||||||
using Ragon.Server.Plugin;
|
using Ragon.Server.Plugin;
|
||||||
|
using Ragon.Server.Project;
|
||||||
|
|
||||||
namespace Ragon.Server.Handler
|
namespace Ragon.Server.Handler
|
||||||
{
|
{
|
||||||
@@ -30,19 +31,22 @@ namespace Ragon.Server.Handler
|
|||||||
private readonly RagonContextObserver _observer;
|
private readonly RagonContextObserver _observer;
|
||||||
private readonly RagonServerConfiguration _configuration;
|
private readonly RagonServerConfiguration _configuration;
|
||||||
private readonly RagonBuffer _writer;
|
private readonly RagonBuffer _writer;
|
||||||
|
private readonly ProjectRegistry _projectRegistry;
|
||||||
|
|
||||||
public AuthorizationOperation(RagonBuffer reader,
|
public AuthorizationOperation(RagonBuffer reader,
|
||||||
RagonBuffer writer,
|
RagonBuffer writer,
|
||||||
IRagonServer server,
|
IRagonServer server,
|
||||||
IServerPlugin serverPlugin,
|
IServerPlugin serverPlugin,
|
||||||
RagonContextObserver observer,
|
RagonContextObserver observer,
|
||||||
RagonServerConfiguration configuration) : base(reader, writer)
|
RagonServerConfiguration configuration,
|
||||||
|
ProjectRegistry projectRegistry) : base(reader, writer)
|
||||||
{
|
{
|
||||||
_serverPlugin = serverPlugin;
|
_serverPlugin = serverPlugin;
|
||||||
_configuration = configuration;
|
_configuration = configuration;
|
||||||
_observer = observer;
|
_observer = observer;
|
||||||
_writer = writer;
|
_writer = writer;
|
||||||
_server = server;
|
_server = server;
|
||||||
|
_projectRegistry = projectRegistry;
|
||||||
}
|
}
|
||||||
|
|
||||||
public override void Handle(RagonContext context, NetworkChannel channel)
|
public override void Handle(RagonContext context, NetworkChannel channel)
|
||||||
@@ -59,31 +63,45 @@ namespace Ragon.Server.Handler
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
var configuration = _configuration;
|
var projectKey = Reader.ReadString();
|
||||||
var key = Reader.ReadString();
|
|
||||||
var name = Reader.ReadString();
|
var name = Reader.ReadString();
|
||||||
var payload = Reader.ReadString();
|
var payload = Reader.ReadString();
|
||||||
|
|
||||||
if (key == configuration.ServerKey)
|
if (!_projectRegistry.ValidateKey(projectKey))
|
||||||
{
|
{
|
||||||
var authorizeViaPlugin = _serverPlugin.OnAuthorize(new ConnectionRequest(_server, context.Connection.Id, payload));
|
_logger.Warning($"Invalid project key from connection {context.Connection.Id}");
|
||||||
if (authorizeViaPlugin)
|
|
||||||
return;
|
|
||||||
|
|
||||||
var id = Guid.NewGuid().ToString();
|
|
||||||
Approve(context, new ConnectionResponse(id, name, payload));
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
_logger.Warning($"Invalid key for connection {context.Connection.Id}");
|
|
||||||
|
|
||||||
Reject(context);
|
Reject(context);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!_projectRegistry.CanConnect(projectKey))
|
||||||
|
{
|
||||||
|
_logger.Warning($"Connection limit reached for project key: {projectKey}");
|
||||||
|
Reject(context);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var authorizeViaPlugin = _serverPlugin.OnAuthorize(new ConnectionRequest(_server, context.Connection.Id, payload));
|
||||||
|
if (authorizeViaPlugin)
|
||||||
|
return;
|
||||||
|
|
||||||
|
var project = _projectRegistry.GetOrCreateProject(projectKey);
|
||||||
|
if (project == null)
|
||||||
|
{
|
||||||
|
_logger.Warning($"Failed to create project for key: {projectKey}");
|
||||||
|
Reject(context);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var id = Guid.NewGuid().ToString();
|
||||||
|
Approve(context, new ConnectionResponse(id, name, payload), project.Id);
|
||||||
|
|
||||||
|
_projectRegistry.RegisterConnection(project.Id);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Approve(RagonContext context, ConnectionResponse result)
|
public void Approve(RagonContext context, ConnectionResponse result, int projectId)
|
||||||
{
|
{
|
||||||
var lobbyPlayer = new RagonLobbyPlayer(context.Connection, result.Id, result.Name, result.Payload);
|
var lobbyPlayer = new RagonLobbyPlayer(context.Connection, result.Id, result.Name, result.Payload, projectId);
|
||||||
context.SetPlayer(lobbyPlayer);
|
context.SetPlayer(lobbyPlayer);
|
||||||
context.ConnectionStatus = ConnectionStatus.Authorized;
|
context.ConnectionStatus = ConnectionStatus.Authorized;
|
||||||
|
|
||||||
@@ -102,7 +120,7 @@ namespace Ragon.Server.Handler
|
|||||||
var sendData = _writer.ToArray();
|
var sendData = _writer.ToArray();
|
||||||
context.Connection.Reliable.Send(sendData);
|
context.Connection.Reliable.Send(sendData);
|
||||||
|
|
||||||
_logger.Trace($"Approved {context.Connection.Id} as {playerId}|{context.LobbyPlayer.Name}");
|
_logger.Trace($"Approved {context.Connection.Id} as {playerId}|{context.LobbyPlayer.Name} for project {projectId}");
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Reject(RagonContext context)
|
public void Reject(RagonContext context)
|
||||||
|
|||||||
@@ -27,7 +27,7 @@ namespace Ragon.Server.Handler
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
context.UserData.Read(Reader);
|
context.UserData.Read(Reader, _userDataLimit);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -88,7 +88,7 @@ namespace Ragon.Server.Handler
|
|||||||
var roomPlayer = new RagonRoomPlayer(context, lobbyPlayer.Id, lobbyPlayer.Name);
|
var roomPlayer = new RagonRoomPlayer(context, lobbyPlayer.Id, lobbyPlayer.Name);
|
||||||
|
|
||||||
var roomPlugin = _serverPlugin.CreateRoomPlugin(information);
|
var roomPlugin = _serverPlugin.CreateRoomPlugin(information);
|
||||||
var room = new RagonRoom(roomId, information, roomPlugin);
|
var room = new RagonRoom(roomId, information, roomPlugin, lobbyPlayer.ProjectId);
|
||||||
|
|
||||||
room.Plugin.OnAttached(room);
|
room.Plugin.OnAttached(room);
|
||||||
roomPlayer.OnAttached(room);
|
roomPlayer.OnAttached(room);
|
||||||
|
|||||||
@@ -43,6 +43,13 @@ public sealed class RoomJoinOperation : BaseOperation
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (existsRoom.ProjectId != lobbyPlayer.ProjectId)
|
||||||
|
{
|
||||||
|
JoinFailed(context, Writer);
|
||||||
|
_logger.Warning($"Player {context.Connection.Id}|{lobbyPlayer.Name} tried to join room from different project");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
var player = new RagonRoomPlayer(context, lobbyPlayer.Id, lobbyPlayer.Name);
|
var player = new RagonRoomPlayer(context, lobbyPlayer.Id, lobbyPlayer.Name);
|
||||||
context.SetRoom(existsRoom, player);
|
context.SetRoom(existsRoom, player);
|
||||||
|
|
||||||
|
|||||||
@@ -57,6 +57,12 @@ public sealed class RoomJoinOrCreateOperation : BaseOperation
|
|||||||
|
|
||||||
if (context.Lobby.FindRoomByScene(_roomParameters.Scene, out var existsRoom))
|
if (context.Lobby.FindRoomByScene(_roomParameters.Scene, out var existsRoom))
|
||||||
{
|
{
|
||||||
|
if (existsRoom.ProjectId != lobbyPlayer.ProjectId)
|
||||||
|
{
|
||||||
|
_logger.Warning($"Player {context.Connection.Id}|{lobbyPlayer.Name} tried to join room from different project");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
var player = new RagonRoomPlayer(context, lobbyPlayer.Id, lobbyPlayer.Name);
|
var player = new RagonRoomPlayer(context, lobbyPlayer.Id, lobbyPlayer.Name);
|
||||||
|
|
||||||
context.SetRoom(existsRoom, player);
|
context.SetRoom(existsRoom, player);
|
||||||
@@ -81,7 +87,7 @@ public sealed class RoomJoinOrCreateOperation : BaseOperation
|
|||||||
|
|
||||||
var roomPlayer = new RagonRoomPlayer(context, lobbyPlayer.Id, lobbyPlayer.Name);
|
var roomPlayer = new RagonRoomPlayer(context, lobbyPlayer.Id, lobbyPlayer.Name);
|
||||||
var roomPlugin = _serverPlugin.CreateRoomPlugin(information);
|
var roomPlugin = _serverPlugin.CreateRoomPlugin(information);
|
||||||
var room = new RagonRoom(roomId, information, roomPlugin);
|
var room = new RagonRoom(roomId, information, roomPlugin, lobbyPlayer.ProjectId);
|
||||||
|
|
||||||
_serverPlugin.OnRoomCreate(lobbyPlayer, room);
|
_serverPlugin.OnRoomCreate(lobbyPlayer, room);
|
||||||
|
|
||||||
|
|||||||
@@ -45,6 +45,6 @@ public sealed class RoomUserDataOperation : BaseOperation
|
|||||||
|
|
||||||
var room = context.Room;
|
var room = context.Room;
|
||||||
if (room != null)
|
if (room != null)
|
||||||
room.UserData.Read(Reader);
|
room.UserData.Read(Reader, _userDataLimit);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -12,12 +12,17 @@ public class RagonLobbyDispatcher
|
|||||||
_lobby = lobby;
|
_lobby = lobby;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Write(RagonBuffer writer)
|
public void Write(RagonBuffer writer, int projectId = 0)
|
||||||
{
|
{
|
||||||
writer.Clear();
|
writer.Clear();
|
||||||
writer.WriteOperation(RagonOperation.ROOM_LIST_UPDATED);
|
writer.WriteOperation(RagonOperation.ROOM_LIST_UPDATED);
|
||||||
var rooms = _lobby.Rooms;
|
var rooms = _lobby.Rooms;
|
||||||
|
|
||||||
|
if (projectId > 0)
|
||||||
|
{
|
||||||
|
rooms = rooms.Where(r => r.ProjectId == projectId).ToList();
|
||||||
|
}
|
||||||
|
|
||||||
writer.WriteUShort((ushort)rooms.Count);
|
writer.WriteUShort((ushort)rooms.Count);
|
||||||
for (int i = 0; i < rooms.Count; i++)
|
for (int i = 0; i < rooms.Count; i++)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -32,12 +32,14 @@ public class RagonLobbyPlayer
|
|||||||
public string Id { get; private set; }
|
public string Id { get; private set; }
|
||||||
public string Name { get; private set; }
|
public string Name { get; private set; }
|
||||||
public string Payload { get; private set; }
|
public string Payload { get; private set; }
|
||||||
|
public int ProjectId { get; private set; }
|
||||||
public RagonLobbyPlayer(INetworkConnection connection, string id, string name, string payload)
|
|
||||||
|
public RagonLobbyPlayer(INetworkConnection connection, string id, string name, string payload, int projectId)
|
||||||
{
|
{
|
||||||
Id = id;
|
Id = id;
|
||||||
Name = name;
|
Name = name;
|
||||||
Connection = connection;
|
Connection = connection;
|
||||||
Payload = payload;
|
Payload = payload;
|
||||||
|
ProjectId = projectId;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -47,7 +47,7 @@ namespace Ragon.Server.Plugin
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
var operation = (AuthorizationOperation)_server.ResolveHandler(RagonOperation.AUTHORIZE);
|
var operation = (AuthorizationOperation)_server.ResolveHandler(RagonOperation.AUTHORIZE);
|
||||||
operation.Approve(ctx, new ConnectionResponse(id, name, payload));
|
operation.Approve(ctx, new ConnectionResponse(id, name, payload), 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Reject()
|
public void Reject()
|
||||||
|
|||||||
@@ -0,0 +1,89 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2023-2024 Eduard Kargin <kargin.eduard@gmail.com>
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace Ragon.Server.Project;
|
||||||
|
|
||||||
|
public class ProjectRegistry
|
||||||
|
{
|
||||||
|
private readonly Dictionary<string, RagonProject> _projectsByKey;
|
||||||
|
private readonly Dictionary<int, RagonProject> _projectsById;
|
||||||
|
private readonly int _connectionLimitPerProject;
|
||||||
|
|
||||||
|
public ProjectRegistry(int connectionLimitPerProject)
|
||||||
|
{
|
||||||
|
_projectsByKey = new Dictionary<string, RagonProject>();
|
||||||
|
_projectsById = new Dictionary<int, RagonProject>();
|
||||||
|
_connectionLimitPerProject = connectionLimitPerProject;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool ValidateKey(string key)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrWhiteSpace(key))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (key.Length < 4)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public RagonProject? GetOrCreateProject(string projectKey)
|
||||||
|
{
|
||||||
|
if (!_projectsByKey.TryGetValue(projectKey, out var project))
|
||||||
|
{
|
||||||
|
project = new RagonProject(projectKey);
|
||||||
|
_projectsByKey[projectKey] = project;
|
||||||
|
_projectsById[project.Id] = project;
|
||||||
|
}
|
||||||
|
|
||||||
|
return project;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool CanConnect(string projectKey)
|
||||||
|
{
|
||||||
|
if (!_projectsByKey.TryGetValue(projectKey, out var project))
|
||||||
|
return true;
|
||||||
|
|
||||||
|
return project.ActiveConnections < _connectionLimitPerProject;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RegisterConnection(int projectId)
|
||||||
|
{
|
||||||
|
if (_projectsById.TryGetValue(projectId, out var project))
|
||||||
|
{
|
||||||
|
project.IncrementConnections();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void UnregisterConnection(int projectId)
|
||||||
|
{
|
||||||
|
if (_projectsById.TryGetValue(projectId, out var project))
|
||||||
|
{
|
||||||
|
project.DecrementConnections();
|
||||||
|
|
||||||
|
if (project.ActiveConnections <= 0)
|
||||||
|
{
|
||||||
|
_projectsByKey.Remove(project.Key);
|
||||||
|
_projectsById.Remove(projectId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public RagonProject? GetProjectById(int projectId)
|
||||||
|
{
|
||||||
|
return _projectsById.TryGetValue(projectId, out var project) ? project : null;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,44 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2023-2024 Eduard Kargin <kargin.eduard@gmail.com>
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace Ragon.Server.Project;
|
||||||
|
|
||||||
|
public class RagonProject
|
||||||
|
{
|
||||||
|
private static int _idGenerator = 0;
|
||||||
|
|
||||||
|
public int Id { get; }
|
||||||
|
public string Key { get; }
|
||||||
|
public int ActiveConnections { get; private set; }
|
||||||
|
|
||||||
|
public RagonProject(string key)
|
||||||
|
{
|
||||||
|
Id = Interlocked.Increment(ref _idGenerator);
|
||||||
|
Key = key;
|
||||||
|
ActiveConnections = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void IncrementConnections()
|
||||||
|
{
|
||||||
|
ActiveConnections++;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void DecrementConnections()
|
||||||
|
{
|
||||||
|
if (ActiveConnections > 0)
|
||||||
|
ActiveConnections--;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -21,14 +21,15 @@ using Ragon.Server.IO;
|
|||||||
using Ragon.Server.Lobby;
|
using Ragon.Server.Lobby;
|
||||||
using Ragon.Server.Logging;
|
using Ragon.Server.Logging;
|
||||||
using Ragon.Server.Plugin;
|
using Ragon.Server.Plugin;
|
||||||
|
using Ragon.Server.Project;
|
||||||
using Ragon.Server.Time;
|
using Ragon.Server.Time;
|
||||||
|
|
||||||
namespace Ragon.Server;
|
namespace Ragon.Server;
|
||||||
|
|
||||||
public class RagonServer : IRagonServer, INetworkListener
|
public class RagonServer : IRagonServer, INetworkListener
|
||||||
{
|
{
|
||||||
private const string ServerVersion = "1.4.1";
|
private const string ServerVersion = "1.4.3";
|
||||||
|
|
||||||
private readonly IRagonLogger _logger = LoggerManager.GetLogger(nameof(RagonServer));
|
private readonly IRagonLogger _logger = LoggerManager.GetLogger(nameof(RagonServer));
|
||||||
private readonly INetworkServer _server;
|
private readonly INetworkServer _server;
|
||||||
private readonly BaseOperation[] _handlers;
|
private readonly BaseOperation[] _handlers;
|
||||||
@@ -40,12 +41,14 @@ public class RagonServer : IRagonServer, INetworkListener
|
|||||||
private readonly RagonScheduler _scheduler;
|
private readonly RagonScheduler _scheduler;
|
||||||
private readonly Dictionary<ushort, RagonContext> _contextsByConnection;
|
private readonly Dictionary<ushort, RagonContext> _contextsByConnection;
|
||||||
private readonly Dictionary<string, RagonContext> _contextsByPlayerId;
|
private readonly Dictionary<string, RagonContext> _contextsByPlayerId;
|
||||||
|
private readonly ProjectRegistry _projectRegistry;
|
||||||
private readonly Stopwatch _timer;
|
private readonly Stopwatch _timer;
|
||||||
private readonly RagonLobbyDispatcher _lobbySerializer;
|
private readonly RagonLobbyDispatcher _lobbySerializer;
|
||||||
private readonly long _tickRate = 0;
|
private readonly long _tickRate = 0;
|
||||||
private bool _isRunning = false;
|
private bool _isRunning = false;
|
||||||
|
|
||||||
public bool IsRunning => _isRunning;
|
public bool IsRunning => _isRunning;
|
||||||
|
public ProjectRegistry ProjectRegistry => _projectRegistry;
|
||||||
|
|
||||||
public RagonServer(
|
public RagonServer(
|
||||||
INetworkServer server,
|
INetworkServer server,
|
||||||
@@ -57,6 +60,7 @@ public class RagonServer : IRagonServer, INetworkListener
|
|||||||
_serverPlugin = plugin;
|
_serverPlugin = plugin;
|
||||||
_contextsByConnection = new Dictionary<ushort, RagonContext>();
|
_contextsByConnection = new Dictionary<ushort, RagonContext>();
|
||||||
_contextsByPlayerId = new Dictionary<string, RagonContext>();
|
_contextsByPlayerId = new Dictionary<string, RagonContext>();
|
||||||
|
_projectRegistry = new ProjectRegistry(configuration.LimitConnectionsPerProject);
|
||||||
_lobby = new LobbyInMemory();
|
_lobby = new LobbyInMemory();
|
||||||
_lobbySerializer = new RagonLobbyDispatcher(_lobby);
|
_lobbySerializer = new RagonLobbyDispatcher(_lobby);
|
||||||
_scheduler = new RagonScheduler();
|
_scheduler = new RagonScheduler();
|
||||||
@@ -73,7 +77,7 @@ public class RagonServer : IRagonServer, INetworkListener
|
|||||||
_serverPlugin.OnAttached(this);
|
_serverPlugin.OnAttached(this);
|
||||||
|
|
||||||
_handlers = new BaseOperation[byte.MaxValue];
|
_handlers = new BaseOperation[byte.MaxValue];
|
||||||
_handlers[(byte)RagonOperation.AUTHORIZE] = new AuthorizationOperation(_reader, _writer, this, _serverPlugin, contextObserver, configuration);
|
_handlers[(byte)RagonOperation.AUTHORIZE] = new AuthorizationOperation(_reader, _writer, this, _serverPlugin, contextObserver, configuration, _projectRegistry);
|
||||||
_handlers[(byte)RagonOperation.JOIN_OR_CREATE_ROOM] = new RoomJoinOrCreateOperation(_reader, _writer, plugin, _configuration);
|
_handlers[(byte)RagonOperation.JOIN_OR_CREATE_ROOM] = new RoomJoinOrCreateOperation(_reader, _writer, plugin, _configuration);
|
||||||
_handlers[(byte)RagonOperation.CREATE_ROOM] = new RoomCreateOperation(_reader, _writer, plugin, _configuration);
|
_handlers[(byte)RagonOperation.CREATE_ROOM] = new RoomCreateOperation(_reader, _writer, plugin, _configuration);
|
||||||
_handlers[(byte)RagonOperation.JOIN_ROOM] = new RoomJoinOperation(_reader, _writer);
|
_handlers[(byte)RagonOperation.JOIN_ROOM] = new RoomJoinOperation(_reader, _writer);
|
||||||
@@ -152,12 +156,15 @@ public class RagonServer : IRagonServer, INetworkListener
|
|||||||
if (room != null)
|
if (room != null)
|
||||||
{
|
{
|
||||||
room.DetachPlayer(context.RoomPlayer);
|
room.DetachPlayer(context.RoomPlayer);
|
||||||
|
|
||||||
_lobby.RemoveIfEmpty(room);
|
_lobby.RemoveIfEmpty(room);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (context.ConnectionStatus == ConnectionStatus.Authorized)
|
if (context.ConnectionStatus == ConnectionStatus.Authorized)
|
||||||
|
{
|
||||||
_contextsByPlayerId.Remove(context.LobbyPlayer.Id);
|
_contextsByPlayerId.Remove(context.LobbyPlayer.Id);
|
||||||
|
_projectRegistry.UnregisterConnection(context.LobbyPlayer.ProjectId);
|
||||||
|
}
|
||||||
|
|
||||||
_logger.Trace($"Disconnected: {connection.Id}");
|
_logger.Trace($"Disconnected: {connection.Id}");
|
||||||
}
|
}
|
||||||
@@ -177,10 +184,13 @@ public class RagonServer : IRagonServer, INetworkListener
|
|||||||
room.DetachPlayer(context.RoomPlayer);
|
room.DetachPlayer(context.RoomPlayer);
|
||||||
_lobby.RemoveIfEmpty(room);
|
_lobby.RemoveIfEmpty(room);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (context.ConnectionStatus == ConnectionStatus.Authorized)
|
if (context.ConnectionStatus == ConnectionStatus.Authorized)
|
||||||
|
{
|
||||||
_contextsByPlayerId.Remove(context.LobbyPlayer.Id);
|
_contextsByPlayerId.Remove(context.LobbyPlayer.Id);
|
||||||
|
_projectRegistry.UnregisterConnection(context.LobbyPlayer.ProjectId);
|
||||||
|
}
|
||||||
|
|
||||||
_logger.Trace($"Timeout: {connection.Id}|{context.LobbyPlayer.Name}|{context.LobbyPlayer.Id}");
|
_logger.Trace($"Timeout: {connection.Id}|{context.LobbyPlayer.Name}|{context.LobbyPlayer.Id}");
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@@ -198,9 +208,16 @@ public class RagonServer : IRagonServer, INetworkListener
|
|||||||
_writer.Clear();
|
_writer.Clear();
|
||||||
_reader.Clear();
|
_reader.Clear();
|
||||||
_reader.FromArray(data);
|
_reader.FromArray(data);
|
||||||
|
|
||||||
var operation = _reader.ReadByte();
|
var operation = _reader.ReadByte();
|
||||||
_handlers[operation]?.Handle(context, channel);
|
|
||||||
|
if (operation >= _handlers.Length || _handlers[operation] == null)
|
||||||
|
{
|
||||||
|
_logger.Warning($"Invalid operation code: {operation} from connection {connection.Id}");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
_handlers[operation].Handle(context, channel);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
@@ -228,13 +245,14 @@ public class RagonServer : IRagonServer, INetworkListener
|
|||||||
|
|
||||||
public void SendRoomList()
|
public void SendRoomList()
|
||||||
{
|
{
|
||||||
_lobbySerializer.Write(_writer);
|
foreach (var (_, context) in _contextsByPlayerId)
|
||||||
|
|
||||||
var sendData = _writer.ToArray();
|
|
||||||
foreach (var (_, value) in _contextsByPlayerId)
|
|
||||||
{
|
{
|
||||||
if (value.Room == null) // If only in lobby, then send room list data
|
if (context.Room == null) // If only in lobby, then send room list data
|
||||||
value.Connection.Reliable.Send(sendData);
|
{
|
||||||
|
_lobbySerializer.Write(_writer, context.LobbyPlayer.ProjectId);
|
||||||
|
var sendData = _writer.ToArray();
|
||||||
|
context.Connection.Reliable.Send(sendData);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -287,7 +305,7 @@ public class RagonServer : IRagonServer, INetworkListener
|
|||||||
{
|
{
|
||||||
return _contextsByPlayerId.TryGetValue(playerId, out var context) ? context : null;
|
return _contextsByPlayerId.TryGetValue(playerId, out var context) ? context : null;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void CopyrightInfo()
|
private void CopyrightInfo()
|
||||||
{
|
{
|
||||||
_logger.Info($"Server Version: {ServerVersion}");
|
_logger.Info($"Server Version: {ServerVersion}");
|
||||||
|
|||||||
@@ -36,6 +36,7 @@ public struct RagonServerConfiguration
|
|||||||
public int LimitBufferedEvents;
|
public int LimitBufferedEvents;
|
||||||
public int LimitUserDataSize;
|
public int LimitUserDataSize;
|
||||||
public int LimitPropertySize;
|
public int LimitPropertySize;
|
||||||
|
public int LimitConnectionsPerProject;
|
||||||
|
|
||||||
private static Dictionary<string, ServerType> _serverTypes = new Dictionary<string, ServerType>()
|
private static Dictionary<string, ServerType> _serverTypes = new Dictionary<string, ServerType>()
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -24,6 +24,7 @@ public interface IRagonRoom
|
|||||||
{
|
{
|
||||||
public string Id { get; }
|
public string Id { get; }
|
||||||
public string Scene { get; }
|
public string Scene { get; }
|
||||||
|
public int ProjectId { get; }
|
||||||
public int PlayerMin { get; }
|
public int PlayerMin { get; }
|
||||||
public int PlayerMax { get; }
|
public int PlayerMax { get; }
|
||||||
public int PlayerCount { get; }
|
public int PlayerCount { get; }
|
||||||
|
|||||||
@@ -31,7 +31,8 @@ public class RagonRoom : IRagonRoom, IRagonAction
|
|||||||
public int PlayerMax { get; private set; }
|
public int PlayerMax { get; private set; }
|
||||||
public int PlayerMin { get; private set; }
|
public int PlayerMin { get; private set; }
|
||||||
public int PlayerCount => WaitPlayersList.Count;
|
public int PlayerCount => WaitPlayersList.Count;
|
||||||
|
public int ProjectId { get; private set; }
|
||||||
|
|
||||||
public bool IsDone { get; private set; }
|
public bool IsDone { get; private set; }
|
||||||
|
|
||||||
public RagonData UserData { get; set; }
|
public RagonData UserData { get; set; }
|
||||||
@@ -53,13 +54,14 @@ public class RagonRoom : IRagonRoom, IRagonAction
|
|||||||
private readonly List<RagonEvent> _bufferedEvents;
|
private readonly List<RagonEvent> _bufferedEvents;
|
||||||
private readonly int _limitBufferedEvents;
|
private readonly int _limitBufferedEvents;
|
||||||
|
|
||||||
public RagonRoom(string roomId, RoomInformation info, IRoomPlugin roomPlugin)
|
public RagonRoom(string roomId, RoomInformation info, IRoomPlugin roomPlugin, int projectId)
|
||||||
{
|
{
|
||||||
Id = roomId;
|
Id = roomId;
|
||||||
Scene = info.Scene;
|
Scene = info.Scene;
|
||||||
PlayerMax = info.Max;
|
PlayerMax = info.Max;
|
||||||
PlayerMin = info.Min;
|
PlayerMin = info.Min;
|
||||||
Plugin = roomPlugin;
|
Plugin = roomPlugin;
|
||||||
|
ProjectId = projectId;
|
||||||
|
|
||||||
Players = new Dictionary<ushort, RagonRoomPlayer>(info.Max);
|
Players = new Dictionary<ushort, RagonRoomPlayer>(info.Max);
|
||||||
WaitPlayersList = new List<RagonRoomPlayer>(info.Max);
|
WaitPlayersList = new List<RagonRoomPlayer>(info.Max);
|
||||||
|
|||||||
Reference in New Issue
Block a user