mirror of
https://github.com/MrOkiDoki/BattleBit-Community-Server-API.git
synced 2025-01-25 02:33:08 -03:00
commit
d1854786a1
9 changed files with 526 additions and 158 deletions
|
@ -1,4 +1,4 @@
|
||||||
namespace CommunityServerAPI.BattleBitAPI
|
namespace BattleBitAPI
|
||||||
{
|
{
|
||||||
public static class Const
|
public static class Const
|
||||||
{
|
{
|
||||||
|
|
53
BattleBitAPI/Common/Enums/LogLevel.cs
Normal file
53
BattleBitAPI/Common/Enums/LogLevel.cs
Normal file
|
@ -0,0 +1,53 @@
|
||||||
|
namespace BattleBitAPI.Common
|
||||||
|
{
|
||||||
|
[System.Flags]
|
||||||
|
public enum LogLevel : ulong
|
||||||
|
{
|
||||||
|
None = 0,
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Output logs from low level sockets.
|
||||||
|
/// </summary>
|
||||||
|
Sockets = 1 << 0,
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Output logs from remote game server (Highly recommended)
|
||||||
|
/// </summary>
|
||||||
|
GameServerErrors = 1 << 1,
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Output logs of game server connects, reconnects.
|
||||||
|
/// </summary>
|
||||||
|
GameServers = 1 << 2,
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Output logs of player connects, disconnects
|
||||||
|
/// </summary>
|
||||||
|
Players = 1 << 3,
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Output logs of squad changes (someone joining, leaving etc)
|
||||||
|
/// </summary>
|
||||||
|
Squads = 1 << 4,
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Output logs of kills/giveups/revives.
|
||||||
|
/// </summary>
|
||||||
|
KillsAndSpawns = 1 << 5,
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Output logs of role changes (player changing role to medic, support etc).
|
||||||
|
/// </summary>
|
||||||
|
Roles = 1 << 6,
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Output logs player's healt changes. (When received damage or healed)
|
||||||
|
/// </summary>
|
||||||
|
HealtChanges = 1 << 7,
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Output everything.
|
||||||
|
/// </summary>
|
||||||
|
All = ulong.MaxValue,
|
||||||
|
}
|
||||||
|
}
|
|
@ -25,6 +25,12 @@ namespace BattleBitAPI.Common.Extentions
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static void Replace<TKey, TValue>(this Dictionary<TKey, TValue> dic, TKey key, TValue value)
|
||||||
|
{
|
||||||
|
dic.Remove(key);
|
||||||
|
dic.Add(key, value);
|
||||||
|
}
|
||||||
|
|
||||||
public static void SafeClose(this TcpClient client)
|
public static void SafeClose(this TcpClient client)
|
||||||
{
|
{
|
||||||
try { client.Close(); } catch { }
|
try { client.Close(); } catch { }
|
||||||
|
|
|
@ -40,5 +40,7 @@
|
||||||
OnPlayerGivenUp = 70,
|
OnPlayerGivenUp = 70,
|
||||||
OnPlayerRevivedAnother = 71,
|
OnPlayerRevivedAnother = 71,
|
||||||
OnSquadPointsChanged = 72,
|
OnSquadPointsChanged = 72,
|
||||||
|
NotifyNewRoundID = 73,
|
||||||
|
Log = 74,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,4 @@
|
||||||
using System.ComponentModel.DataAnnotations;
|
using System.Net;
|
||||||
using System.Diagnostics;
|
|
||||||
using System.Linq;
|
|
||||||
using System.Net;
|
|
||||||
using System.Net.Sockets;
|
using System.Net.Sockets;
|
||||||
using System.Numerics;
|
using System.Numerics;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
|
@ -9,7 +6,6 @@ using BattleBitAPI.Common;
|
||||||
using BattleBitAPI.Common.Extentions;
|
using BattleBitAPI.Common.Extentions;
|
||||||
using BattleBitAPI.Networking;
|
using BattleBitAPI.Networking;
|
||||||
using BattleBitAPI.Pooling;
|
using BattleBitAPI.Pooling;
|
||||||
using CommunityServerAPI.BattleBitAPI;
|
|
||||||
|
|
||||||
namespace BattleBitAPI.Server
|
namespace BattleBitAPI.Server
|
||||||
{
|
{
|
||||||
|
@ -21,7 +17,6 @@ namespace BattleBitAPI.Server
|
||||||
public IPAddress GameIP => mInternal.GameIP;
|
public IPAddress GameIP => mInternal.GameIP;
|
||||||
public int GamePort => mInternal.GamePort;
|
public int GamePort => mInternal.GamePort;
|
||||||
|
|
||||||
public TcpClient Socket => mInternal.Socket;
|
|
||||||
public bool IsPasswordProtected => mInternal.IsPasswordProtected;
|
public bool IsPasswordProtected => mInternal.IsPasswordProtected;
|
||||||
public string ServerName => mInternal.ServerName;
|
public string ServerName => mInternal.ServerName;
|
||||||
public string Gamemode => mInternal.Gamemode;
|
public string Gamemode => mInternal.Gamemode;
|
||||||
|
@ -33,6 +28,8 @@ namespace BattleBitAPI.Server
|
||||||
public int MaxPlayerCount => mInternal.MaxPlayerCount;
|
public int MaxPlayerCount => mInternal.MaxPlayerCount;
|
||||||
public string LoadingScreenText => mInternal.LoadingScreenText;
|
public string LoadingScreenText => mInternal.LoadingScreenText;
|
||||||
public string ServerRulesText => mInternal.ServerRulesText;
|
public string ServerRulesText => mInternal.ServerRulesText;
|
||||||
|
public uint RoundIndex => mInternal.RoundIndex;
|
||||||
|
public long SessionID => mInternal.SessionID;
|
||||||
public ServerSettings<TPlayer> ServerSettings => mInternal.ServerSettings;
|
public ServerSettings<TPlayer> ServerSettings => mInternal.ServerSettings;
|
||||||
public MapRotation<TPlayer> MapRotation => mInternal.MapRotation;
|
public MapRotation<TPlayer> MapRotation => mInternal.MapRotation;
|
||||||
public GamemodeRotation<TPlayer> GamemodeRotation => mInternal.GamemodeRotation;
|
public GamemodeRotation<TPlayer> GamemodeRotation => mInternal.GamemodeRotation;
|
||||||
|
@ -162,7 +159,7 @@ namespace BattleBitAPI.Server
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
//Are we still connected on socket level?
|
//Are we still connected on socket level?
|
||||||
if (!Socket.Connected)
|
if (mInternal.Socket == null || !mInternal.Socket.Connected)
|
||||||
{
|
{
|
||||||
mClose("Connection was terminated.");
|
mClose("Connection was terminated.");
|
||||||
return;
|
return;
|
||||||
|
@ -175,10 +172,10 @@ namespace BattleBitAPI.Server
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
var networkStream = Socket.GetStream();
|
var networkStream = mInternal.Socket.GetStream();
|
||||||
|
|
||||||
//Read network packages.
|
//Read network packages.
|
||||||
while (Socket.Available > 0)
|
while (mInternal.Socket.Available > 0)
|
||||||
{
|
{
|
||||||
this.mInternal.mLastPackageReceived = Extentions.TickCount;
|
this.mInternal.mLastPackageReceived = Extentions.TickCount;
|
||||||
|
|
||||||
|
@ -409,10 +406,6 @@ namespace BattleBitAPI.Server
|
||||||
public virtual async Task OnTick()
|
public virtual async Task OnTick()
|
||||||
{
|
{
|
||||||
|
|
||||||
}
|
|
||||||
public virtual async Task OnReconnected()
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
}
|
||||||
public virtual async Task OnDisconnected()
|
public virtual async Task OnDisconnected()
|
||||||
{
|
{
|
||||||
|
@ -504,6 +497,10 @@ namespace BattleBitAPI.Server
|
||||||
public virtual async Task OnRoundEnded()
|
public virtual async Task OnRoundEnded()
|
||||||
{
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
public virtual async Task OnSessionChanged(long oldSessionID, long newSessionID)
|
||||||
|
{
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// ---- Functions ----
|
// ---- Functions ----
|
||||||
|
@ -557,10 +554,18 @@ namespace BattleBitAPI.Server
|
||||||
{
|
{
|
||||||
ExecuteCommand("endgame");
|
ExecuteCommand("endgame");
|
||||||
}
|
}
|
||||||
public void SayToChat(string msg)
|
public void SayToAllChat(string msg)
|
||||||
{
|
{
|
||||||
ExecuteCommand("say " + msg);
|
ExecuteCommand("say " + msg);
|
||||||
}
|
}
|
||||||
|
public void SayToChat(string msg, ulong steamID)
|
||||||
|
{
|
||||||
|
ExecuteCommand("sayto " + steamID + " " + msg);
|
||||||
|
}
|
||||||
|
public void SayToChat(string msg, Player<TPlayer> player)
|
||||||
|
{
|
||||||
|
SayToChat(msg, player.SteamID);
|
||||||
|
}
|
||||||
|
|
||||||
public void StopServer()
|
public void StopServer()
|
||||||
{
|
{
|
||||||
|
@ -901,7 +906,7 @@ namespace BattleBitAPI.Server
|
||||||
}
|
}
|
||||||
|
|
||||||
// ---- Static ----
|
// ---- Static ----
|
||||||
public static void SetInstance(GameServer<TPlayer> server, Internal @internal)
|
internal static void SetInstance(GameServer<TPlayer> server, Internal @internal)
|
||||||
{
|
{
|
||||||
server.mInternal = @internal;
|
server.mInternal = @internal;
|
||||||
}
|
}
|
||||||
|
@ -912,6 +917,7 @@ namespace BattleBitAPI.Server
|
||||||
// ---- Variables ----
|
// ---- Variables ----
|
||||||
public ulong ServerHash;
|
public ulong ServerHash;
|
||||||
public bool IsConnected;
|
public bool IsConnected;
|
||||||
|
public bool HasActiveConnectionSession;
|
||||||
public IPAddress GameIP;
|
public IPAddress GameIP;
|
||||||
public int GamePort;
|
public int GamePort;
|
||||||
public TcpClient Socket;
|
public TcpClient Socket;
|
||||||
|
@ -928,6 +934,8 @@ namespace BattleBitAPI.Server
|
||||||
public int MaxPlayerCount;
|
public int MaxPlayerCount;
|
||||||
public string LoadingScreenText;
|
public string LoadingScreenText;
|
||||||
public string ServerRulesText;
|
public string ServerRulesText;
|
||||||
|
public uint RoundIndex;
|
||||||
|
public long SessionID;
|
||||||
public ServerSettings<TPlayer> ServerSettings;
|
public ServerSettings<TPlayer> ServerSettings;
|
||||||
public MapRotation<TPlayer> MapRotation;
|
public MapRotation<TPlayer> MapRotation;
|
||||||
public GamemodeRotation<TPlayer> GamemodeRotation;
|
public GamemodeRotation<TPlayer> GamemodeRotation;
|
||||||
|
@ -948,6 +956,7 @@ namespace BattleBitAPI.Server
|
||||||
public long mLastPackageReceived;
|
public long mLastPackageReceived;
|
||||||
public long mLastPackageSent;
|
public long mLastPackageSent;
|
||||||
public bool mWantsToCloseConnection;
|
public bool mWantsToCloseConnection;
|
||||||
|
public long mPreviousSessionID;
|
||||||
public StringBuilder mBuilder;
|
public StringBuilder mBuilder;
|
||||||
public Queue<(ulong steamID, PlayerModifications<TPlayer>.mPlayerModifications)> mChangedModifications;
|
public Queue<(ulong steamID, PlayerModifications<TPlayer>.mPlayerModifications)> mChangedModifications;
|
||||||
|
|
||||||
|
@ -1294,7 +1303,9 @@ namespace BattleBitAPI.Server
|
||||||
int inQueuePlayers,
|
int inQueuePlayers,
|
||||||
int maxPlayers,
|
int maxPlayers,
|
||||||
string loadingScreenText,
|
string loadingScreenText,
|
||||||
string serverRulesText
|
string serverRulesText,
|
||||||
|
uint roundIndex,
|
||||||
|
long sessionID
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
this.ServerHash = ((ulong)port << 32) | (ulong)iP.ToUInt();
|
this.ServerHash = ((ulong)port << 32) | (ulong)iP.ToUInt();
|
||||||
|
@ -1315,6 +1326,8 @@ namespace BattleBitAPI.Server
|
||||||
this.MaxPlayerCount = maxPlayers;
|
this.MaxPlayerCount = maxPlayers;
|
||||||
this.LoadingScreenText = loadingScreenText;
|
this.LoadingScreenText = loadingScreenText;
|
||||||
this.ServerRulesText = serverRulesText;
|
this.ServerRulesText = serverRulesText;
|
||||||
|
this.RoundIndex = roundIndex;
|
||||||
|
this.SessionID = sessionID;
|
||||||
|
|
||||||
this.ServerSettings.Reset();
|
this.ServerSettings.Reset();
|
||||||
this._RoomSettings.Reset();
|
this._RoomSettings.Reset();
|
||||||
|
|
|
@ -29,7 +29,7 @@ namespace BattleBitAPI.Server
|
||||||
|
|
||||||
public override string ToString()
|
public override string ToString()
|
||||||
{
|
{
|
||||||
return Team + " : " + Name;
|
return "Squad " + Name;
|
||||||
}
|
}
|
||||||
|
|
||||||
// ---- Internal ----
|
// ---- Internal ----
|
||||||
|
|
|
@ -58,7 +58,7 @@ namespace BattleBitAPI
|
||||||
KickFromSquad();
|
KickFromSquad();
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if(value.Team != this.Team)
|
if (value.Team != this.Team)
|
||||||
ChangeTeam(value.Team);
|
ChangeTeam(value.Team);
|
||||||
JoinSquad(value.Name);
|
JoinSquad(value.Name);
|
||||||
}
|
}
|
||||||
|
@ -66,6 +66,8 @@ namespace BattleBitAPI
|
||||||
}
|
}
|
||||||
public bool InSquad => mInternal.SquadName != Squads.NoSquad;
|
public bool InSquad => mInternal.SquadName != Squads.NoSquad;
|
||||||
public int PingMs => mInternal.PingMs;
|
public int PingMs => mInternal.PingMs;
|
||||||
|
public long CurrentSessionID => mInternal.SessionID;
|
||||||
|
public bool IsConnected => mInternal.SessionID != 0;
|
||||||
|
|
||||||
public float HP
|
public float HP
|
||||||
{
|
{
|
||||||
|
@ -156,55 +158,77 @@ namespace BattleBitAPI
|
||||||
public virtual async Task OnDisconnected()
|
public virtual async Task OnDisconnected()
|
||||||
{
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
public virtual async Task OnSessionChanged(long oldSessionID, long newSessionID)
|
||||||
|
{
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// ---- Functions ----
|
// ---- Functions ----
|
||||||
public void Kick(string reason = "")
|
public void Kick(string reason = "")
|
||||||
{
|
{
|
||||||
|
if (IsConnected)
|
||||||
GameServer.Kick(this, reason);
|
GameServer.Kick(this, reason);
|
||||||
}
|
}
|
||||||
public void Kill()
|
public void Kill()
|
||||||
{
|
{
|
||||||
|
if (IsConnected)
|
||||||
GameServer.Kill(this);
|
GameServer.Kill(this);
|
||||||
}
|
}
|
||||||
public void ChangeTeam()
|
public void ChangeTeam()
|
||||||
{
|
{
|
||||||
|
if (IsConnected)
|
||||||
GameServer.ChangeTeam(this);
|
GameServer.ChangeTeam(this);
|
||||||
}
|
}
|
||||||
public void ChangeTeam(Team team)
|
public void ChangeTeam(Team team)
|
||||||
{
|
{
|
||||||
|
if (IsConnected)
|
||||||
GameServer.ChangeTeam(this, team);
|
GameServer.ChangeTeam(this, team);
|
||||||
}
|
}
|
||||||
public void KickFromSquad()
|
public void KickFromSquad()
|
||||||
{
|
{
|
||||||
|
if (IsConnected)
|
||||||
GameServer.KickFromSquad(this);
|
GameServer.KickFromSquad(this);
|
||||||
}
|
}
|
||||||
public void JoinSquad(Squads targetSquad)
|
public void JoinSquad(Squads targetSquad)
|
||||||
{
|
{
|
||||||
|
if (IsConnected)
|
||||||
GameServer.JoinSquad(this, targetSquad);
|
GameServer.JoinSquad(this, targetSquad);
|
||||||
}
|
}
|
||||||
public void DisbandTheSquad()
|
public void DisbandTheSquad()
|
||||||
{
|
{
|
||||||
|
if (IsConnected)
|
||||||
GameServer.DisbandPlayerCurrentSquad(this);
|
GameServer.DisbandPlayerCurrentSquad(this);
|
||||||
}
|
}
|
||||||
public void PromoteToSquadLeader()
|
public void PromoteToSquadLeader()
|
||||||
{
|
{
|
||||||
|
if (IsConnected)
|
||||||
GameServer.PromoteSquadLeader(this);
|
GameServer.PromoteSquadLeader(this);
|
||||||
}
|
}
|
||||||
public void WarnPlayer(string msg)
|
public void WarnPlayer(string msg)
|
||||||
{
|
{
|
||||||
|
if (IsConnected)
|
||||||
GameServer.WarnPlayer(this, msg);
|
GameServer.WarnPlayer(this, msg);
|
||||||
}
|
}
|
||||||
public void Message(string msg)
|
public void Message(string msg)
|
||||||
{
|
{
|
||||||
|
if (IsConnected)
|
||||||
GameServer.MessageToPlayer(this, msg);
|
GameServer.MessageToPlayer(this, msg);
|
||||||
}
|
}
|
||||||
|
public void SayToChat(string msg)
|
||||||
|
{
|
||||||
|
if (IsConnected)
|
||||||
|
GameServer.SayToChat(msg, this);
|
||||||
|
}
|
||||||
|
|
||||||
public void Message(string msg, float fadeoutTime)
|
public void Message(string msg, float fadeoutTime)
|
||||||
{
|
{
|
||||||
|
if (IsConnected)
|
||||||
GameServer.MessageToPlayer(this, msg, fadeoutTime);
|
GameServer.MessageToPlayer(this, msg, fadeoutTime);
|
||||||
}
|
}
|
||||||
public void SetNewRole(GameRole role)
|
public void SetNewRole(GameRole role)
|
||||||
{
|
{
|
||||||
|
if (IsConnected)
|
||||||
GameServer.SetRoleTo(this, role);
|
GameServer.SetRoleTo(this, role);
|
||||||
}
|
}
|
||||||
public void Teleport(Vector3 target)
|
public void Teleport(Vector3 target)
|
||||||
|
@ -213,47 +237,57 @@ namespace BattleBitAPI
|
||||||
}
|
}
|
||||||
public void SpawnPlayer(PlayerLoadout loadout, PlayerWearings wearings, Vector3 position, Vector3 lookDirection, PlayerStand stand, float spawnProtection)
|
public void SpawnPlayer(PlayerLoadout loadout, PlayerWearings wearings, Vector3 position, Vector3 lookDirection, PlayerStand stand, float spawnProtection)
|
||||||
{
|
{
|
||||||
|
if (IsConnected)
|
||||||
GameServer.SpawnPlayer(this, loadout, wearings, position, lookDirection, stand, spawnProtection);
|
GameServer.SpawnPlayer(this, loadout, wearings, position, lookDirection, stand, spawnProtection);
|
||||||
}
|
}
|
||||||
public void SetHP(float newHP)
|
public void SetHP(float newHP)
|
||||||
{
|
{
|
||||||
|
if (IsConnected)
|
||||||
GameServer.SetHP(this, newHP);
|
GameServer.SetHP(this, newHP);
|
||||||
}
|
}
|
||||||
public void GiveDamage(float damage)
|
public void GiveDamage(float damage)
|
||||||
{
|
{
|
||||||
|
if (IsConnected)
|
||||||
GameServer.GiveDamage(this, damage);
|
GameServer.GiveDamage(this, damage);
|
||||||
}
|
}
|
||||||
public void Heal(float hp)
|
public void Heal(float hp)
|
||||||
{
|
{
|
||||||
|
if (IsConnected)
|
||||||
GameServer.Heal(this, hp);
|
GameServer.Heal(this, hp);
|
||||||
}
|
}
|
||||||
public void SetPrimaryWeapon(WeaponItem item, int extraMagazines, bool clear = false)
|
public void SetPrimaryWeapon(WeaponItem item, int extraMagazines, bool clear = false)
|
||||||
{
|
{
|
||||||
|
if (IsConnected)
|
||||||
GameServer.SetPrimaryWeapon(this, item, extraMagazines, clear);
|
GameServer.SetPrimaryWeapon(this, item, extraMagazines, clear);
|
||||||
}
|
}
|
||||||
public void SetSecondaryWeapon(WeaponItem item, int extraMagazines, bool clear = false)
|
public void SetSecondaryWeapon(WeaponItem item, int extraMagazines, bool clear = false)
|
||||||
{
|
{
|
||||||
|
if (IsConnected)
|
||||||
GameServer.SetSecondaryWeapon(this, item, extraMagazines, clear);
|
GameServer.SetSecondaryWeapon(this, item, extraMagazines, clear);
|
||||||
}
|
}
|
||||||
public void SetFirstAidGadget(string item, int extra, bool clear = false)
|
public void SetFirstAidGadget(string item, int extra, bool clear = false)
|
||||||
{
|
{
|
||||||
|
if (IsConnected)
|
||||||
GameServer.SetFirstAid(this, item, extra, clear);
|
GameServer.SetFirstAid(this, item, extra, clear);
|
||||||
}
|
}
|
||||||
public void SetLightGadget(string item, int extra, bool clear = false)
|
public void SetLightGadget(string item, int extra, bool clear = false)
|
||||||
{
|
{
|
||||||
|
if (IsConnected)
|
||||||
GameServer.SetLightGadget(this, item, extra, clear);
|
GameServer.SetLightGadget(this, item, extra, clear);
|
||||||
}
|
}
|
||||||
public void SetHeavyGadget(string item, int extra, bool clear = false)
|
public void SetHeavyGadget(string item, int extra, bool clear = false)
|
||||||
{
|
{
|
||||||
|
if (IsConnected)
|
||||||
GameServer.SetHeavyGadget(this, item, extra, clear);
|
GameServer.SetHeavyGadget(this, item, extra, clear);
|
||||||
}
|
}
|
||||||
public void SetThrowable(string item, int extra, bool clear = false)
|
public void SetThrowable(string item, int extra, bool clear = false)
|
||||||
{
|
{
|
||||||
|
if (IsConnected)
|
||||||
GameServer.SetThrowable(this, item, extra, clear);
|
GameServer.SetThrowable(this, item, extra, clear);
|
||||||
}
|
}
|
||||||
|
|
||||||
// ---- Static ----
|
// ---- Static ----
|
||||||
public static void SetInstance(TPlayer player, Player<TPlayer>.Internal @internal)
|
internal static void SetInstance(TPlayer player, Player<TPlayer>.Internal @internal)
|
||||||
{
|
{
|
||||||
player.mInternal = @internal;
|
player.mInternal = @internal;
|
||||||
}
|
}
|
||||||
|
@ -275,6 +309,8 @@ namespace BattleBitAPI
|
||||||
public Team Team;
|
public Team Team;
|
||||||
public Squads SquadName;
|
public Squads SquadName;
|
||||||
public int PingMs = 999;
|
public int PingMs = 999;
|
||||||
|
public long PreviousSessionID = 0;
|
||||||
|
public long SessionID = 0;
|
||||||
|
|
||||||
public bool IsAlive;
|
public bool IsAlive;
|
||||||
public float HP;
|
public float HP;
|
||||||
|
|
|
@ -1,11 +1,12 @@
|
||||||
using System.Net;
|
using System.Data;
|
||||||
|
using System.Net;
|
||||||
using System.Net.Sockets;
|
using System.Net.Sockets;
|
||||||
using System.Numerics;
|
using System.Numerics;
|
||||||
using System.Runtime.CompilerServices;
|
using System.Resources;
|
||||||
using BattleBitAPI.Common;
|
using BattleBitAPI.Common;
|
||||||
using BattleBitAPI.Common.Extentions;
|
using BattleBitAPI.Common.Extentions;
|
||||||
using BattleBitAPI.Networking;
|
using BattleBitAPI.Networking;
|
||||||
using CommunityServerAPI.BattleBitAPI;
|
using BattleBitAPI.Pooling;
|
||||||
|
|
||||||
namespace BattleBitAPI.Server
|
namespace BattleBitAPI.Server
|
||||||
{
|
{
|
||||||
|
@ -15,6 +16,7 @@ namespace BattleBitAPI.Server
|
||||||
public bool IsListening { get; private set; }
|
public bool IsListening { get; private set; }
|
||||||
public bool IsDisposed { get; private set; }
|
public bool IsDisposed { get; private set; }
|
||||||
public int ListeningPort { get; private set; }
|
public int ListeningPort { get; private set; }
|
||||||
|
public LogLevel LogLevel { get; set; } = LogLevel.None;
|
||||||
|
|
||||||
// --- Events ---
|
// --- Events ---
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
@ -56,15 +58,6 @@ namespace BattleBitAPI.Server
|
||||||
/// </remarks>
|
/// </remarks>
|
||||||
public Func<GameServer<TPlayer>, Task> OnGameServerConnected { get; set; }
|
public Func<GameServer<TPlayer>, Task> OnGameServerConnected { get; set; }
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Fired when a game server reconnects. (When game server connects while a socket is already open)
|
|
||||||
/// </summary>
|
|
||||||
///
|
|
||||||
/// <remarks>
|
|
||||||
/// GameServer: Game server that is reconnecting.<br/>
|
|
||||||
/// </remarks>
|
|
||||||
public Func<GameServer<TPlayer>, Task> OnGameServerReconnected { get; set; }
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Fired when a game server disconnects. Check (GameServer.TerminationReason) to see the reason.
|
/// Fired when a game server disconnects. Check (GameServer.TerminationReason) to see the reason.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
|
@ -94,16 +87,29 @@ namespace BattleBitAPI.Server
|
||||||
/// </remarks>
|
/// </remarks>
|
||||||
public Func<ulong, TPlayer> OnCreatingPlayerInstance { get; set; }
|
public Func<ulong, TPlayer> OnCreatingPlayerInstance { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Fired on log
|
||||||
|
/// </summary>
|
||||||
|
///
|
||||||
|
/// <remarks>
|
||||||
|
/// LogLevel: The level of log<br/>
|
||||||
|
/// string: The message<br/>
|
||||||
|
/// object: The object that will be carried on log<br/>
|
||||||
|
/// </remarks>
|
||||||
|
public Action<LogLevel, string, object?> OnLog { get; set; }
|
||||||
|
|
||||||
// --- Private ---
|
// --- Private ---
|
||||||
private TcpListener mSocket;
|
private TcpListener mSocket;
|
||||||
private Dictionary<ulong, (TGameServer server, GameServer<TPlayer>.Internal resources)> mActiveConnections;
|
private Dictionary<ulong, (TGameServer server, GameServer<TPlayer>.Internal resources)> mActiveConnections;
|
||||||
private mInstances<TPlayer, TGameServer> mInstanceDatabase;
|
private mInstances<TPlayer, TGameServer> mInstanceDatabase;
|
||||||
|
private ItemPooling<GameServer<TPlayer>> mGameServerPool;
|
||||||
|
|
||||||
// --- Construction ---
|
// --- Construction ---
|
||||||
public ServerListener()
|
public ServerListener()
|
||||||
{
|
{
|
||||||
this.mActiveConnections = new Dictionary<ulong, (TGameServer, GameServer<TPlayer>.Internal)>(16);
|
this.mActiveConnections = new Dictionary<ulong, (TGameServer, GameServer<TPlayer>.Internal)>(16);
|
||||||
this.mInstanceDatabase = new mInstances<TPlayer, TGameServer>();
|
this.mInstanceDatabase = new mInstances<TPlayer, TGameServer>();
|
||||||
|
this.mGameServerPool = new ItemPooling<GameServer<TPlayer>>(64);
|
||||||
}
|
}
|
||||||
|
|
||||||
// --- Starting ---
|
// --- Starting ---
|
||||||
|
@ -122,6 +128,9 @@ namespace BattleBitAPI.Server
|
||||||
this.ListeningPort = port;
|
this.ListeningPort = port;
|
||||||
this.IsListening = true;
|
this.IsListening = true;
|
||||||
|
|
||||||
|
if (this.LogLevel.HasFlag(LogLevel.Sockets))
|
||||||
|
OnLog(LogLevel.Sockets, $"Listening TCP connections on port " + port, null);
|
||||||
|
|
||||||
mMainLoop();
|
mMainLoop();
|
||||||
}
|
}
|
||||||
public void Start(int port)
|
public void Start(int port)
|
||||||
|
@ -143,6 +152,9 @@ namespace BattleBitAPI.Server
|
||||||
}
|
}
|
||||||
catch { }
|
catch { }
|
||||||
|
|
||||||
|
if (this.LogLevel.HasFlag(LogLevel.Sockets))
|
||||||
|
OnLog(LogLevel.Sockets, $"Stopped listening TCP connection.", null);
|
||||||
|
|
||||||
this.mSocket = null;
|
this.mSocket = null;
|
||||||
this.ListeningPort = 0;
|
this.ListeningPort = 0;
|
||||||
this.IsListening = true;
|
this.IsListening = true;
|
||||||
|
@ -161,22 +173,32 @@ namespace BattleBitAPI.Server
|
||||||
{
|
{
|
||||||
var ip = (client.Client.RemoteEndPoint as IPEndPoint).Address;
|
var ip = (client.Client.RemoteEndPoint as IPEndPoint).Address;
|
||||||
|
|
||||||
|
if (this.LogLevel.HasFlag(LogLevel.Sockets))
|
||||||
|
OnLog(LogLevel.Sockets, $"Incoming TCP connection from {ip}", client);
|
||||||
|
|
||||||
|
//Is this IP allowed?
|
||||||
bool allow = true;
|
bool allow = true;
|
||||||
if (OnGameServerConnecting != null)
|
if (OnGameServerConnecting != null)
|
||||||
allow = await OnGameServerConnecting(ip);
|
allow = await OnGameServerConnecting(ip);
|
||||||
|
|
||||||
|
//Close connection if it was not allowed.
|
||||||
if (!allow)
|
if (!allow)
|
||||||
{
|
{
|
||||||
|
if (this.LogLevel.HasFlag(LogLevel.Sockets))
|
||||||
|
OnLog(LogLevel.Sockets, $"Incoming connection from {ip} was denied", client);
|
||||||
|
|
||||||
//Connection is not allowed from this IP.
|
//Connection is not allowed from this IP.
|
||||||
client.SafeClose();
|
client.SafeClose();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
TGameServer server = null;
|
//Read port,token,version
|
||||||
GameServer<TPlayer>.Internal resources;
|
string token;
|
||||||
|
string version;
|
||||||
|
int gamePort;
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
using (CancellationTokenSource source = new CancellationTokenSource(Const.HailConnectTimeout))
|
using (var source = new CancellationTokenSource(2000))
|
||||||
{
|
{
|
||||||
using (var readStream = Common.Serialization.Stream.Get())
|
using (var readStream = Common.Serialization.Stream.Get())
|
||||||
{
|
{
|
||||||
|
@ -187,13 +209,13 @@ namespace BattleBitAPI.Server
|
||||||
readStream.Reset();
|
readStream.Reset();
|
||||||
if (!await networkStream.TryRead(readStream, 1, source.Token))
|
if (!await networkStream.TryRead(readStream, 1, source.Token))
|
||||||
throw new Exception("Unable to read the package type");
|
throw new Exception("Unable to read the package type");
|
||||||
|
|
||||||
NetworkCommuncation type = (NetworkCommuncation)readStream.ReadInt8();
|
NetworkCommuncation type = (NetworkCommuncation)readStream.ReadInt8();
|
||||||
if (type != NetworkCommuncation.Hail)
|
if (type != NetworkCommuncation.Hail)
|
||||||
throw new Exception("Incoming package wasn't hail.");
|
throw new Exception("Incoming package wasn't hail.");
|
||||||
}
|
}
|
||||||
|
|
||||||
//Read the server token
|
//Read the server token
|
||||||
string token;
|
|
||||||
{
|
{
|
||||||
readStream.Reset();
|
readStream.Reset();
|
||||||
if (!await networkStream.TryRead(readStream, 2, source.Token))
|
if (!await networkStream.TryRead(readStream, 2, source.Token))
|
||||||
|
@ -211,7 +233,6 @@ namespace BattleBitAPI.Server
|
||||||
}
|
}
|
||||||
|
|
||||||
//Read the server version
|
//Read the server version
|
||||||
string version;
|
|
||||||
{
|
{
|
||||||
readStream.Reset();
|
readStream.Reset();
|
||||||
if (!await networkStream.TryRead(readStream, 2, source.Token))
|
if (!await networkStream.TryRead(readStream, 2, source.Token))
|
||||||
|
@ -228,23 +249,66 @@ namespace BattleBitAPI.Server
|
||||||
version = readStream.ReadString(stringSize);
|
version = readStream.ReadString(stringSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (version != Const.Version)
|
|
||||||
throw new Exception("Incoming server's version `" + version + "` does not match with current API version `" + Const.Version + "`");
|
|
||||||
|
|
||||||
//Read port
|
//Read port
|
||||||
int gamePort;
|
|
||||||
{
|
{
|
||||||
readStream.Reset();
|
readStream.Reset();
|
||||||
if (!await networkStream.TryRead(readStream, 2, source.Token))
|
if (!await networkStream.TryRead(readStream, 2, source.Token))
|
||||||
throw new Exception("Unable to read the Port");
|
throw new Exception("Unable to read the Port");
|
||||||
gamePort = readStream.ReadUInt16();
|
gamePort = readStream.ReadUInt16();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
if (this.LogLevel.HasFlag(LogLevel.Sockets))
|
||||||
|
OnLog(LogLevel.Sockets, $"{ip} failed to connected because " + e.Message, client);
|
||||||
|
|
||||||
|
client.SafeClose();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var hash = ((ulong)gamePort << 32) | (ulong)ip.ToUInt();
|
||||||
|
TGameServer server = null;
|
||||||
|
GameServer<TPlayer>.Internal resources = null;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
//Does versions match?
|
||||||
|
if (version != Const.Version)
|
||||||
|
throw new Exception("Incoming server's version `" + version + "` does not match with current API version `" + Const.Version + "`");
|
||||||
|
|
||||||
|
//Is valid token?
|
||||||
if (OnValidateGameServerToken != null)
|
if (OnValidateGameServerToken != null)
|
||||||
allow = await OnValidateGameServerToken(ip, (ushort)gamePort, token);
|
{
|
||||||
|
if (!await OnValidateGameServerToken(ip, (ushort)gamePort, token))
|
||||||
if (!allow)
|
|
||||||
throw new Exception("Token was not valid!");
|
throw new Exception("Token was not valid!");
|
||||||
|
}
|
||||||
|
|
||||||
|
//Are there any connections with same IP and port?
|
||||||
|
{
|
||||||
|
bool sessionExist = false;
|
||||||
|
(TGameServer server, GameServer<TPlayer>.Internal resources) oldSession;
|
||||||
|
|
||||||
|
//Any sessions with this IP:Port?
|
||||||
|
lock (this.mActiveConnections)
|
||||||
|
sessionExist = this.mActiveConnections.TryGetValue(hash, out oldSession);
|
||||||
|
|
||||||
|
if (sessionExist)
|
||||||
|
{
|
||||||
|
//Close old session.
|
||||||
|
oldSession.server.CloseConnection("Reconnecting.");
|
||||||
|
|
||||||
|
//Wait until session is fully closed.
|
||||||
|
while (oldSession.resources.HasActiveConnectionSession)
|
||||||
|
await Task.Delay(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
using (var source = new CancellationTokenSource(Const.HailConnectTimeout))
|
||||||
|
{
|
||||||
|
using (var readStream = Common.Serialization.Stream.Get())
|
||||||
|
{
|
||||||
|
var networkStream = client.GetStream();
|
||||||
|
|
||||||
//Read is server protected
|
//Read is server protected
|
||||||
bool isPasswordProtected;
|
bool isPasswordProtected;
|
||||||
|
@ -404,7 +468,28 @@ namespace BattleBitAPI.Server
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var hash = ((ulong)gamePort << 32) | (ulong)ip.ToUInt();
|
//Round index
|
||||||
|
uint roundIndex;
|
||||||
|
{
|
||||||
|
readStream.Reset();
|
||||||
|
if (!await networkStream.TryRead(readStream, 4, source.Token))
|
||||||
|
throw new Exception("Unable to read the Server Round Index");
|
||||||
|
roundIndex = readStream.ReadUInt32();
|
||||||
|
}
|
||||||
|
|
||||||
|
//Round index
|
||||||
|
long sessionID;
|
||||||
|
{
|
||||||
|
readStream.Reset();
|
||||||
|
if (!await networkStream.TryRead(readStream, 8, source.Token))
|
||||||
|
throw new Exception("Unable to read the Server Round ID");
|
||||||
|
sessionID = readStream.ReadInt64();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
server = this.mInstanceDatabase.GetServerInstance(hash, out resources, this.OnCreatingGameServerInstance, ip, (ushort)gamePort);
|
server = this.mInstanceDatabase.GetServerInstance(hash, out resources, this.OnCreatingGameServerInstance, ip, (ushort)gamePort);
|
||||||
resources.Set(
|
resources.Set(
|
||||||
this.mExecutePackage,
|
this.mExecutePackage,
|
||||||
|
@ -422,7 +507,9 @@ namespace BattleBitAPI.Server
|
||||||
queuePlayers,
|
queuePlayers,
|
||||||
maxPlayers,
|
maxPlayers,
|
||||||
loadingScreenText,
|
loadingScreenText,
|
||||||
serverRulesText
|
serverRulesText,
|
||||||
|
roundIndex,
|
||||||
|
sessionID
|
||||||
);
|
);
|
||||||
|
|
||||||
//Room settings
|
//Room settings
|
||||||
|
@ -584,7 +671,6 @@ namespace BattleBitAPI.Server
|
||||||
playerInternal.SteamID = steamid;
|
playerInternal.SteamID = steamid;
|
||||||
playerInternal.Name = username;
|
playerInternal.Name = username;
|
||||||
playerInternal.IP = new IPAddress(ipHash);
|
playerInternal.IP = new IPAddress(ipHash);
|
||||||
playerInternal.GameServer = (GameServer<TPlayer>)server;
|
|
||||||
playerInternal.Team = team;
|
playerInternal.Team = team;
|
||||||
playerInternal.SquadName = squad;
|
playerInternal.SquadName = squad;
|
||||||
playerInternal.Role = role;
|
playerInternal.Role = role;
|
||||||
|
@ -605,6 +691,9 @@ namespace BattleBitAPI.Server
|
||||||
playerInternal._Modifications.Read(readStream);
|
playerInternal._Modifications.Read(readStream);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
playerInternal.GameServer = (GameServer<TPlayer>)server;
|
||||||
|
playerInternal.SessionID = server.SessionID;
|
||||||
|
|
||||||
resources.AddPlayer(player);
|
resources.AddPlayer(player);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -666,54 +755,69 @@ namespace BattleBitAPI.Server
|
||||||
}
|
}
|
||||||
catch { }
|
catch { }
|
||||||
|
|
||||||
|
if (this.LogLevel.HasFlag(LogLevel.Sockets))
|
||||||
|
OnLog(LogLevel.Sockets, $"{ip} failed to connected because " + e.Message, client);
|
||||||
|
|
||||||
client.SafeClose();
|
client.SafeClose();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool connectionExist = false;
|
|
||||||
|
|
||||||
//Track the connection
|
|
||||||
lock (this.mActiveConnections)
|
|
||||||
{
|
|
||||||
//An old connection exist with same IP + Port?
|
|
||||||
if (connectionExist = this.mActiveConnections.TryGetValue(server.ServerHash, out var oldServer))
|
|
||||||
{
|
|
||||||
oldServer.resources.ReconnectFlag = true;
|
|
||||||
this.mActiveConnections.Remove(server.ServerHash);
|
|
||||||
}
|
|
||||||
|
|
||||||
this.mActiveConnections.Add(server.ServerHash, (server, resources));
|
|
||||||
}
|
|
||||||
|
|
||||||
//Call the callback.
|
|
||||||
if (!connectionExist)
|
|
||||||
{
|
|
||||||
//New connection!
|
|
||||||
server.OnConnected();
|
|
||||||
if (this.OnGameServerConnected != null)
|
|
||||||
this.OnGameServerConnected(server);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
//Reconnection
|
|
||||||
server.OnReconnected();
|
|
||||||
if (this.OnGameServerReconnected != null)
|
|
||||||
this.OnGameServerReconnected(server);
|
|
||||||
}
|
|
||||||
|
|
||||||
//Set the buffer sizes.
|
//Set the buffer sizes.
|
||||||
client.ReceiveBufferSize = Const.MaxNetworkPackageSize;
|
client.ReceiveBufferSize = Const.MaxNetworkPackageSize;
|
||||||
client.SendBufferSize = Const.MaxNetworkPackageSize;
|
client.SendBufferSize = Const.MaxNetworkPackageSize;
|
||||||
|
|
||||||
|
if (this.LogLevel.HasFlag(LogLevel.Sockets))
|
||||||
|
OnLog(LogLevel.Sockets, $"Incoming game server from {ip}:{gamePort} accepted.", client);
|
||||||
|
|
||||||
//Join to main server loop.
|
//Join to main server loop.
|
||||||
await mHandleGameServer(server, resources);
|
await mHandleGameServer(server, resources);
|
||||||
}
|
}
|
||||||
private async Task mHandleGameServer(TGameServer server, GameServer<TPlayer>.Internal @internal)
|
private async Task mHandleGameServer(TGameServer server, GameServer<TPlayer>.Internal @internal)
|
||||||
{
|
{
|
||||||
bool isTicking = false;
|
@internal.HasActiveConnectionSession = true;
|
||||||
|
{
|
||||||
|
// ---- Connected ----
|
||||||
|
{
|
||||||
|
lock (this.mActiveConnections)
|
||||||
|
this.mActiveConnections.Replace(server.ServerHash, (server, @internal));
|
||||||
|
|
||||||
|
server.OnConnected();
|
||||||
|
if (this.OnGameServerConnected != null)
|
||||||
|
this.OnGameServerConnected(server);
|
||||||
|
}
|
||||||
|
|
||||||
|
//Update sessions
|
||||||
|
{
|
||||||
|
if (@internal.mPreviousSessionID != @internal.SessionID)
|
||||||
|
{
|
||||||
|
var oldSession = @internal.mPreviousSessionID;
|
||||||
|
@internal.mPreviousSessionID = @internal.SessionID;
|
||||||
|
|
||||||
|
if (oldSession != 0)
|
||||||
|
server.OnSessionChanged(oldSession, @internal.SessionID);
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (var item in @internal.Players)
|
||||||
|
{
|
||||||
|
var @player_internal = mInstanceDatabase.GetPlayerInternals(item.Key);
|
||||||
|
if (@player_internal.PreviousSessionID != @player_internal.SessionID)
|
||||||
|
{
|
||||||
|
var previousID = @player_internal.PreviousSessionID;
|
||||||
|
@player_internal.PreviousSessionID = @player_internal.SessionID;
|
||||||
|
|
||||||
|
if (previousID != 0)
|
||||||
|
item.Value.OnSessionChanged(previousID, @player_internal.SessionID);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.LogLevel.HasFlag(LogLevel.GameServers))
|
||||||
|
OnLog(LogLevel.GameServers, $"{server} has connected", server);
|
||||||
|
|
||||||
|
// ---- Ticking ----
|
||||||
using (server)
|
using (server)
|
||||||
{
|
{
|
||||||
|
var isTicking = false;
|
||||||
async Task mTickAsync()
|
async Task mTickAsync()
|
||||||
{
|
{
|
||||||
isTicking = true;
|
isTicking = true;
|
||||||
|
@ -729,20 +833,24 @@ namespace BattleBitAPI.Server
|
||||||
await server.Tick();
|
await server.Tick();
|
||||||
await Task.Delay(10);
|
await Task.Delay(10);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (!server.ReconnectFlag)
|
// ---- Disconnected ----
|
||||||
{
|
{
|
||||||
server.OnDisconnected();
|
mCleanup(server, @internal);
|
||||||
|
|
||||||
|
lock (this.mActiveConnections)
|
||||||
|
this.mActiveConnections.Remove(server.ServerHash);
|
||||||
|
|
||||||
|
server.OnDisconnected();
|
||||||
if (this.OnGameServerDisconnected != null)
|
if (this.OnGameServerDisconnected != null)
|
||||||
this.OnGameServerDisconnected(server);
|
this.OnGameServerDisconnected(server);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
//Remove from list.
|
if (this.LogLevel.HasFlag(LogLevel.GameServers))
|
||||||
if (!server.ReconnectFlag)
|
OnLog(LogLevel.GameServers, $"{server} has disconnected", server);
|
||||||
lock (this.mActiveConnections)
|
}
|
||||||
this.mActiveConnections.Remove(server.ServerHash);
|
@internal.HasActiveConnectionSession = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// --- Logic Executing ---
|
// --- Logic Executing ---
|
||||||
|
@ -767,8 +875,6 @@ namespace BattleBitAPI.Server
|
||||||
playerInternal.SteamID = steamID;
|
playerInternal.SteamID = steamID;
|
||||||
playerInternal.Name = username;
|
playerInternal.Name = username;
|
||||||
playerInternal.IP = new IPAddress(ip);
|
playerInternal.IP = new IPAddress(ip);
|
||||||
playerInternal.GameServer = (GameServer<TPlayer>)server;
|
|
||||||
|
|
||||||
playerInternal.Team = team;
|
playerInternal.Team = team;
|
||||||
playerInternal.SquadName = squad;
|
playerInternal.SquadName = squad;
|
||||||
playerInternal.Role = role;
|
playerInternal.Role = role;
|
||||||
|
@ -776,9 +882,24 @@ namespace BattleBitAPI.Server
|
||||||
//Start from default.
|
//Start from default.
|
||||||
playerInternal._Modifications.Reset();
|
playerInternal._Modifications.Reset();
|
||||||
|
|
||||||
|
playerInternal.GameServer = (GameServer<TPlayer>)server;
|
||||||
|
playerInternal.SessionID = server.SessionID;
|
||||||
|
|
||||||
resources.AddPlayer(player);
|
resources.AddPlayer(player);
|
||||||
player.OnConnected();
|
player.OnConnected();
|
||||||
server.OnPlayerConnected(player);
|
server.OnPlayerConnected(player);
|
||||||
|
|
||||||
|
if (playerInternal.PreviousSessionID != playerInternal.SessionID)
|
||||||
|
{
|
||||||
|
var previousID = playerInternal.PreviousSessionID;
|
||||||
|
playerInternal.PreviousSessionID = playerInternal.SessionID;
|
||||||
|
|
||||||
|
if (previousID != 0)
|
||||||
|
player.OnSessionChanged(previousID, playerInternal.SessionID);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.LogLevel.HasFlag(LogLevel.Players))
|
||||||
|
OnLog(LogLevel.Players, $"{player} has connected", player);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
@ -817,8 +938,14 @@ namespace BattleBitAPI.Server
|
||||||
server.OnPlayerLeftSquad((TPlayer)player, msquad);
|
server.OnPlayerLeftSquad((TPlayer)player, msquad);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@internal.SessionID = 0;
|
||||||
|
@internal.GameServer = null;
|
||||||
|
|
||||||
player.OnDisconnected();
|
player.OnDisconnected();
|
||||||
server.OnPlayerDisconnected((TPlayer)player);
|
server.OnPlayerDisconnected((TPlayer)player);
|
||||||
|
|
||||||
|
if (this.LogLevel.HasFlag(LogLevel.Players))
|
||||||
|
OnLog(LogLevel.Players, $"{player} has disconnected", player);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
@ -888,6 +1015,9 @@ namespace BattleBitAPI.Server
|
||||||
|
|
||||||
victimClient.OnDowned();
|
victimClient.OnDowned();
|
||||||
server.OnAPlayerDownedAnotherPlayer(args);
|
server.OnAPlayerDownedAnotherPlayer(args);
|
||||||
|
|
||||||
|
if (this.LogLevel.HasFlag(LogLevel.KillsAndSpawns))
|
||||||
|
OnLog(LogLevel.KillsAndSpawns, $"{killer} downed {victim} in {(Vector3.Distance(killerPos, victimPos))} meters", null);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -946,13 +1076,19 @@ namespace BattleBitAPI.Server
|
||||||
ulong steamID = stream.ReadUInt64();
|
ulong steamID = stream.ReadUInt64();
|
||||||
GameRole role = (GameRole)stream.ReadInt8();
|
GameRole role = (GameRole)stream.ReadInt8();
|
||||||
|
|
||||||
if (resources.TryGetPlayer(steamID, out var client))
|
if (resources.TryGetPlayer(steamID, out var player))
|
||||||
{
|
{
|
||||||
async Task mHandle()
|
async Task mHandle()
|
||||||
{
|
{
|
||||||
bool accepted = await server.OnPlayerRequestingToChangeRole((TPlayer)client, role);
|
if (this.LogLevel.HasFlag(LogLevel.Roles))
|
||||||
|
OnLog(LogLevel.Roles, $"{player} asking to change role to {role}", player);
|
||||||
|
|
||||||
|
bool accepted = await server.OnPlayerRequestingToChangeRole((TPlayer)player, role);
|
||||||
if (accepted)
|
if (accepted)
|
||||||
server.SetRoleTo(steamID, role);
|
server.SetRoleTo(steamID, role);
|
||||||
|
|
||||||
|
if (this.LogLevel.HasFlag(LogLevel.Roles))
|
||||||
|
OnLog(LogLevel.Roles, $"{player}'s request to change role to {role} was {(accepted ? "accepted" : "denied")}", player);
|
||||||
}
|
}
|
||||||
|
|
||||||
mHandle();
|
mHandle();
|
||||||
|
@ -967,13 +1103,16 @@ namespace BattleBitAPI.Server
|
||||||
ulong steamID = stream.ReadUInt64();
|
ulong steamID = stream.ReadUInt64();
|
||||||
GameRole role = (GameRole)stream.ReadInt8();
|
GameRole role = (GameRole)stream.ReadInt8();
|
||||||
|
|
||||||
if (resources.TryGetPlayer(steamID, out var client))
|
if (resources.TryGetPlayer(steamID, out var player))
|
||||||
{
|
{
|
||||||
var @internal = mInstanceDatabase.GetPlayerInternals(steamID);
|
var @internal = mInstanceDatabase.GetPlayerInternals(steamID);
|
||||||
@internal.Role = role;
|
@internal.Role = role;
|
||||||
|
|
||||||
client.OnChangedRole(role);
|
player.OnChangedRole(role);
|
||||||
server.OnPlayerChangedRole((TPlayer)client, role);
|
server.OnPlayerChangedRole((TPlayer)player, role);
|
||||||
|
|
||||||
|
if (this.LogLevel.HasFlag(LogLevel.Roles))
|
||||||
|
OnLog(LogLevel.Roles, $"{player} changed role to {role}", player);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
@ -985,18 +1124,21 @@ namespace BattleBitAPI.Server
|
||||||
ulong steamID = stream.ReadUInt64();
|
ulong steamID = stream.ReadUInt64();
|
||||||
Squads squad = (Squads)stream.ReadInt8();
|
Squads squad = (Squads)stream.ReadInt8();
|
||||||
|
|
||||||
if (resources.TryGetPlayer(steamID, out var client))
|
if (resources.TryGetPlayer(steamID, out var player))
|
||||||
{
|
{
|
||||||
var @internal = mInstanceDatabase.GetPlayerInternals(steamID);
|
var @internal = mInstanceDatabase.GetPlayerInternals(steamID);
|
||||||
@internal.SquadName = squad;
|
@internal.SquadName = squad;
|
||||||
|
|
||||||
var msquad = server.GetSquad(client.Team, squad);
|
var msquad = server.GetSquad(player.Team, squad);
|
||||||
var rsquad = resources.GetSquadInternal(msquad);
|
var rsquad = resources.GetSquadInternal(msquad);
|
||||||
lock (rsquad.Members)
|
lock (rsquad.Members)
|
||||||
rsquad.Members.Add((TPlayer)client);
|
rsquad.Members.Add((TPlayer)player);
|
||||||
|
|
||||||
client.OnJoinedSquad(msquad);
|
player.OnJoinedSquad(msquad);
|
||||||
server.OnPlayerJoinedSquad((TPlayer)client, msquad);
|
server.OnPlayerJoinedSquad((TPlayer)player, msquad);
|
||||||
|
|
||||||
|
if (this.LogLevel.HasFlag(LogLevel.Squads))
|
||||||
|
OnLog(LogLevel.Squads, $"{player} has joined to {msquad}", msquad);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
@ -1007,30 +1149,33 @@ namespace BattleBitAPI.Server
|
||||||
{
|
{
|
||||||
ulong steamID = stream.ReadUInt64();
|
ulong steamID = stream.ReadUInt64();
|
||||||
|
|
||||||
if (resources.TryGetPlayer(steamID, out var client))
|
if (resources.TryGetPlayer(steamID, out var player))
|
||||||
{
|
{
|
||||||
var @internal = mInstanceDatabase.GetPlayerInternals(steamID);
|
var @internal = mInstanceDatabase.GetPlayerInternals(steamID);
|
||||||
|
|
||||||
var oldSquad = client.SquadName;
|
var oldSquad = player.SquadName;
|
||||||
var oldRole = client.Role;
|
var oldRole = player.Role;
|
||||||
@internal.SquadName = Squads.NoSquad;
|
@internal.SquadName = Squads.NoSquad;
|
||||||
@internal.Role = GameRole.Assault;
|
@internal.Role = GameRole.Assault;
|
||||||
|
|
||||||
var msquad = server.GetSquad(client.Team, oldSquad);
|
var msquad = server.GetSquad(player.Team, oldSquad);
|
||||||
var rsquad = resources.GetSquadInternal(msquad);
|
var rsquad = resources.GetSquadInternal(msquad);
|
||||||
|
|
||||||
@internal.SquadName = Squads.NoSquad;
|
@internal.SquadName = Squads.NoSquad;
|
||||||
lock (rsquad.Members)
|
lock (rsquad.Members)
|
||||||
rsquad.Members.Remove((TPlayer)client);
|
rsquad.Members.Remove((TPlayer)player);
|
||||||
|
|
||||||
client.OnLeftSquad(msquad);
|
player.OnLeftSquad(msquad);
|
||||||
server.OnPlayerLeftSquad((TPlayer)client, msquad);
|
server.OnPlayerLeftSquad((TPlayer)player, msquad);
|
||||||
|
|
||||||
if (oldRole != GameRole.Assault)
|
if (oldRole != GameRole.Assault)
|
||||||
{
|
{
|
||||||
client.OnChangedRole(GameRole.Assault);
|
player.OnChangedRole(GameRole.Assault);
|
||||||
server.OnPlayerChangedRole((TPlayer)client, GameRole.Assault);
|
server.OnPlayerChangedRole((TPlayer)player, GameRole.Assault);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (this.LogLevel.HasFlag(LogLevel.Squads))
|
||||||
|
OnLog(LogLevel.Squads, $"{player} has left the {msquad}", msquad);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
@ -1064,11 +1209,14 @@ namespace BattleBitAPI.Server
|
||||||
request.Read(stream);
|
request.Read(stream);
|
||||||
ushort vehicleID = stream.ReadUInt16();
|
ushort vehicleID = stream.ReadUInt16();
|
||||||
|
|
||||||
if (resources.TryGetPlayer(steamID, out var client))
|
if (resources.TryGetPlayer(steamID, out var player))
|
||||||
{
|
{
|
||||||
async Task mHandle()
|
async Task mHandle()
|
||||||
{
|
{
|
||||||
var responseSpawn = await server.OnPlayerSpawning((TPlayer)client, request);
|
if (this.LogLevel.HasFlag(LogLevel.KillsAndSpawns))
|
||||||
|
OnLog(LogLevel.KillsAndSpawns, $"{player} asking to spawn at {request.SpawnPosition} ({request.RequestedPoint})", player);
|
||||||
|
|
||||||
|
var responseSpawn = await server.OnPlayerSpawning((TPlayer)player, request);
|
||||||
|
|
||||||
//Respond back.
|
//Respond back.
|
||||||
using (var response = Common.Serialization.Stream.Get())
|
using (var response = Common.Serialization.Stream.Get())
|
||||||
|
@ -1089,6 +1237,15 @@ namespace BattleBitAPI.Server
|
||||||
|
|
||||||
server.WriteToSocket(response);
|
server.WriteToSocket(response);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (this.LogLevel.HasFlag(LogLevel.KillsAndSpawns))
|
||||||
|
{
|
||||||
|
if (responseSpawn == null)
|
||||||
|
OnLog(LogLevel.KillsAndSpawns, $"{player}'s spawn request was denied", player);
|
||||||
|
else
|
||||||
|
OnLog(LogLevel.KillsAndSpawns, $"{player}'s spawn request was accepted at {responseSpawn.Value.SpawnPosition}", player);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
mHandle();
|
mHandle();
|
||||||
|
@ -1120,7 +1277,7 @@ namespace BattleBitAPI.Server
|
||||||
if (stream.CanRead(8 + 2))
|
if (stream.CanRead(8 + 2))
|
||||||
{
|
{
|
||||||
ulong steamID = stream.ReadUInt64();
|
ulong steamID = stream.ReadUInt64();
|
||||||
if (resources.TryGetPlayer(steamID, out var client))
|
if (resources.TryGetPlayer(steamID, out var player))
|
||||||
{
|
{
|
||||||
var @internal = mInstanceDatabase.GetPlayerInternals(steamID);
|
var @internal = mInstanceDatabase.GetPlayerInternals(steamID);
|
||||||
|
|
||||||
|
@ -1132,10 +1289,21 @@ namespace BattleBitAPI.Server
|
||||||
wearings.Read(stream);
|
wearings.Read(stream);
|
||||||
@internal.CurrentWearings = wearings;
|
@internal.CurrentWearings = wearings;
|
||||||
|
|
||||||
|
Vector3 position = new Vector3()
|
||||||
|
{
|
||||||
|
X = stream.ReadFloat(),
|
||||||
|
Y = stream.ReadFloat(),
|
||||||
|
Z = stream.ReadFloat(),
|
||||||
|
};
|
||||||
|
|
||||||
|
@internal.Position = position;
|
||||||
@internal.IsAlive = true;
|
@internal.IsAlive = true;
|
||||||
|
|
||||||
client.OnSpawned();
|
player.OnSpawned();
|
||||||
server.OnPlayerSpawned((TPlayer)client);
|
server.OnPlayerSpawned((TPlayer)player);
|
||||||
|
|
||||||
|
if (this.LogLevel.HasFlag(LogLevel.KillsAndSpawns))
|
||||||
|
OnLog(LogLevel.KillsAndSpawns, $"{player} has spawned at {player.Position}", player);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
@ -1145,13 +1313,16 @@ namespace BattleBitAPI.Server
|
||||||
if (stream.CanRead(8))
|
if (stream.CanRead(8))
|
||||||
{
|
{
|
||||||
ulong steamid = stream.ReadUInt64();
|
ulong steamid = stream.ReadUInt64();
|
||||||
if (resources.TryGetPlayer(steamid, out var client))
|
if (resources.TryGetPlayer(steamid, out var player))
|
||||||
{
|
{
|
||||||
var @internal = mInstanceDatabase.GetPlayerInternals(steamid);
|
var @internal = mInstanceDatabase.GetPlayerInternals(steamid);
|
||||||
@internal.OnDie();
|
@internal.OnDie();
|
||||||
|
|
||||||
client.OnDied();
|
player.OnDied();
|
||||||
server.OnPlayerDied((TPlayer)client);
|
server.OnPlayerDied((TPlayer)player);
|
||||||
|
|
||||||
|
if (this.LogLevel.HasFlag(LogLevel.KillsAndSpawns))
|
||||||
|
OnLog(LogLevel.KillsAndSpawns, $"{player} has died", player);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
@ -1262,19 +1433,38 @@ namespace BattleBitAPI.Server
|
||||||
var @internal = mInstanceDatabase.GetPlayerInternals(steamID);
|
var @internal = mInstanceDatabase.GetPlayerInternals(steamID);
|
||||||
if (@internal.IsAlive)
|
if (@internal.IsAlive)
|
||||||
{
|
{
|
||||||
|
float newHP = (com_healt * 0.5f) - 1f;
|
||||||
|
|
||||||
|
if (this.LogLevel.HasFlag(LogLevel.HealtChanges))
|
||||||
|
{
|
||||||
|
var player = resources.Players[steamID];
|
||||||
|
float dtHP = newHP - @internal.HP;
|
||||||
|
if (dtHP > 0)
|
||||||
|
{
|
||||||
|
//Heal
|
||||||
|
OnLog(LogLevel.HealtChanges, $"{player} was healed by {dtHP} HP (new HP is {newHP} HP)", player);
|
||||||
|
}
|
||||||
|
else if(dtHP < 0)
|
||||||
|
{
|
||||||
|
//Damage
|
||||||
|
OnLog(LogLevel.HealtChanges, $"{player} was damaged by {(-dtHP)} HP (new HP is {newHP} HP)", player);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@internal.Position = new Vector3()
|
@internal.Position = new Vector3()
|
||||||
{
|
{
|
||||||
X = com_posX * decompressX,
|
X = com_posX * decompressX,
|
||||||
Y = com_posY * decompressY,
|
Y = com_posY * decompressY,
|
||||||
Z = com_posZ * decompressZ,
|
Z = com_posZ * decompressZ,
|
||||||
};
|
};
|
||||||
@internal.HP = (com_healt * 0.5f) - 1f;
|
@internal.HP = newHP;
|
||||||
@internal.Standing = standing;
|
@internal.Standing = standing;
|
||||||
@internal.Leaning = side;
|
@internal.Leaning = side;
|
||||||
@internal.CurrentLoadoutIndex = loadoutIndex;
|
@internal.CurrentLoadoutIndex = loadoutIndex;
|
||||||
@internal.InVehicle = inSeat;
|
@internal.InVehicle = inSeat;
|
||||||
@internal.IsBleeding = isBleeding;
|
@internal.IsBleeding = isBleeding;
|
||||||
@internal.PingMs = ping;
|
@internal.PingMs = ping;
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1285,10 +1475,13 @@ namespace BattleBitAPI.Server
|
||||||
if (stream.CanRead(8))
|
if (stream.CanRead(8))
|
||||||
{
|
{
|
||||||
ulong steamID = stream.ReadUInt64();
|
ulong steamID = stream.ReadUInt64();
|
||||||
if (resources.TryGetPlayer(steamID, out var client))
|
if (resources.TryGetPlayer(steamID, out var player))
|
||||||
{
|
{
|
||||||
client.OnGivenUp();
|
player.OnGivenUp();
|
||||||
server.OnPlayerGivenUp((TPlayer)client);
|
server.OnPlayerGivenUp((TPlayer)player);
|
||||||
|
|
||||||
|
if (this.LogLevel.HasFlag(LogLevel.KillsAndSpawns))
|
||||||
|
OnLog(LogLevel.KillsAndSpawns, $"{player} has givenup", player);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
@ -1307,6 +1500,9 @@ namespace BattleBitAPI.Server
|
||||||
{
|
{
|
||||||
fromClient.OnRevivedAnotherPlayer();
|
fromClient.OnRevivedAnotherPlayer();
|
||||||
server.OnAPlayerRevivedAnotherPlayer((TPlayer)fromClient, (TPlayer)toClient);
|
server.OnAPlayerRevivedAnotherPlayer((TPlayer)fromClient, (TPlayer)toClient);
|
||||||
|
|
||||||
|
if (this.LogLevel.HasFlag(LogLevel.KillsAndSpawns))
|
||||||
|
OnLog(LogLevel.KillsAndSpawns, $"{fromClient} revived {toClient}", null);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1328,6 +1524,51 @@ namespace BattleBitAPI.Server
|
||||||
rsquad.SquadPoints = points;
|
rsquad.SquadPoints = points;
|
||||||
server.OnSquadPointsChanged(msquad, points);
|
server.OnSquadPointsChanged(msquad, points);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (this.LogLevel.HasFlag(LogLevel.Squads))
|
||||||
|
OnLog(LogLevel.Squads, $"{msquad} now has {points} points", msquad);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case NetworkCommuncation.NotifyNewRoundID:
|
||||||
|
{
|
||||||
|
if (stream.CanRead(4 + 8))
|
||||||
|
{
|
||||||
|
resources.RoundIndex = stream.ReadUInt32();
|
||||||
|
resources.SessionID = stream.ReadInt64();
|
||||||
|
|
||||||
|
if (resources.mPreviousSessionID != resources.SessionID)
|
||||||
|
{
|
||||||
|
var oldSession = resources.mPreviousSessionID;
|
||||||
|
resources.mPreviousSessionID = resources.SessionID;
|
||||||
|
|
||||||
|
if (oldSession != 0)
|
||||||
|
server.OnSessionChanged(oldSession, resources.SessionID);
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (var item in resources.Players)
|
||||||
|
{
|
||||||
|
var @player_internal = mInstanceDatabase.GetPlayerInternals(item.Key);
|
||||||
|
@player_internal.SessionID = resources.SessionID;
|
||||||
|
|
||||||
|
if (@player_internal.PreviousSessionID != @player_internal.SessionID)
|
||||||
|
{
|
||||||
|
var previousID = @player_internal.PreviousSessionID;
|
||||||
|
@player_internal.PreviousSessionID = @player_internal.SessionID;
|
||||||
|
|
||||||
|
if (previousID != 0)
|
||||||
|
item.Value.OnSessionChanged(previousID, @player_internal.SessionID);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case NetworkCommuncation.Log:
|
||||||
|
{
|
||||||
|
if (this.LogLevel.HasFlag(LogLevel.GameServerErrors))
|
||||||
|
{
|
||||||
|
if (stream.TryReadString(out var log))
|
||||||
|
OnLog(LogLevel.GameServerErrors, log, server);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -1335,6 +1576,18 @@ namespace BattleBitAPI.Server
|
||||||
}
|
}
|
||||||
|
|
||||||
// --- Private ---
|
// --- Private ---
|
||||||
|
private void mCleanup(GameServer<TPlayer> server, GameServer<TPlayer>.Internal @internal)
|
||||||
|
{
|
||||||
|
lock (@internal.Players)
|
||||||
|
{
|
||||||
|
foreach (var item in @internal.Players)
|
||||||
|
{
|
||||||
|
var @player_internal = mInstanceDatabase.GetPlayerInternals(item.Key);
|
||||||
|
@player_internal.SessionID = 0;
|
||||||
|
@player_internal.GameServer = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
private Player<TPlayer>.Internal mGetPlayerInternals(ulong steamID)
|
private Player<TPlayer>.Internal mGetPlayerInternals(ulong steamID)
|
||||||
{
|
{
|
||||||
return mInstanceDatabase.GetPlayerInternals(steamID);
|
return mInstanceDatabase.GetPlayerInternals(steamID);
|
||||||
|
@ -1345,13 +1598,17 @@ namespace BattleBitAPI.Server
|
||||||
{
|
{
|
||||||
get
|
get
|
||||||
{
|
{
|
||||||
var list = new List<TGameServer>(mActiveConnections.Count);
|
using (var list = this.mGameServerPool.Get())
|
||||||
lock (mActiveConnections)
|
|
||||||
{
|
{
|
||||||
|
//Get a copy
|
||||||
|
lock (mActiveConnections)
|
||||||
foreach (var item in mActiveConnections.Values)
|
foreach (var item in mActiveConnections.Values)
|
||||||
list.Add(item.server);
|
list.ListItems.Add(item.server);
|
||||||
|
|
||||||
|
//Iterate
|
||||||
|
for (int i = 0; i < list.ListItems.Count; i++)
|
||||||
|
yield return (TGameServer)list.ListItems[i];
|
||||||
}
|
}
|
||||||
return list;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
public bool TryGetGameServer(IPAddress ip, ushort port, out TGameServer server)
|
public bool TryGetGameServer(IPAddress ip, ushort port, out TGameServer server)
|
||||||
|
|
|
@ -18,6 +18,7 @@
|
||||||
<PackageProjectUrl>https://github.com/MrOkiDoki/BattleBit-Community-Server-API</PackageProjectUrl>
|
<PackageProjectUrl>https://github.com/MrOkiDoki/BattleBit-Community-Server-API</PackageProjectUrl>
|
||||||
<RepositoryUrl>https://github.com/MrOkiDoki/BattleBit-Community-Server-API</RepositoryUrl>
|
<RepositoryUrl>https://github.com/MrOkiDoki/BattleBit-Community-Server-API</RepositoryUrl>
|
||||||
<PackageTags>BattleBit</PackageTags>
|
<PackageTags>BattleBit</PackageTags>
|
||||||
|
<Version>1.0.2</Version>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
|
|
Loading…
Add table
Reference in a new issue