Implement player count per platform and player list API endpoints (#1014)
Some checks failed
Continuous Integration / Build & Test (map[database:true fullName:ubuntu-latest prettyName:Linux webTest:true]) (push) Has been cancelled
Upload Translations to Crowdin / crowdin-sync (push) Has been cancelled
Build Docker Image / Build and Publish (push) Has been cancelled

* Implement player count per platform and player list API endpoints

* Fix inconsistencies in the XML documentation

* Update PlayerListResponse.cs
This commit is contained in:
Josh 2024-08-28 20:17:03 -05:00 committed by GitHub
parent fb2192d37d
commit 0af064ad1e
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
5 changed files with 113 additions and 10 deletions

View file

@ -5,6 +5,7 @@ using LBPUnion.ProjectLighthouse.Helpers;
using LBPUnion.ProjectLighthouse.Servers.API.Responses; using LBPUnion.ProjectLighthouse.Servers.API.Responses;
using LBPUnion.ProjectLighthouse.Types.Users; using LBPUnion.ProjectLighthouse.Types.Users;
using Microsoft.AspNetCore.Mvc; using Microsoft.AspNetCore.Mvc;
using Microsoft.EntityFrameworkCore;
namespace LBPUnion.ProjectLighthouse.Servers.API.Controllers; namespace LBPUnion.ProjectLighthouse.Servers.API.Controllers;
@ -49,24 +50,33 @@ public class StatisticsEndpoints : ApiEndpointController
GameVersion.LittleBigPlanetPSP, GameVersion.LittleBigPlanetPSP,
}; };
private static readonly List<Platform> platforms = new()
{
Platform.PS3,
Platform.RPCS3,
Platform.Vita,
Platform.PSP,
};
/// <summary> /// <summary>
/// Get player counts for each individual title /// Get player counts for each individual title
/// </summary> /// </summary>
/// <returns>An instance of PlayerCountResponse</returns> /// <returns>An instance of PlayerCountByGameResponse</returns>
[HttpGet("playerCount")] [HttpGet("playerCount")]
[ProducesResponseType(typeof(PlayerCountResponse), StatusCodes.Status200OK)] [HttpGet("playerCount/game")]
[ProducesResponseType(typeof(PlayerCountByGameResponse), StatusCodes.Status200OK)]
public async Task<IActionResult> GetPlayerCounts() public async Task<IActionResult> GetPlayerCounts()
{ {
List<PlayerCountObject> gameList = new(); List<PlayerCountObject> gameList = new();
foreach (GameVersion version in gameVersions) foreach (GameVersion version in gameVersions)
{ {
gameList.Add(new PlayerCountObject gameList.Add(new PlayerCountByGameObject
{ {
Game = version.ToString(), Game = version.ToString(),
PlayerCount = await StatisticsHelper.RecentMatchesForGame(this.database, version), PlayerCount = await StatisticsHelper.RecentMatches(this.database, l => l.GameVersion == version),
}); });
} }
PlayerCountResponse response = new() PlayerCountByGameResponse response = new()
{ {
TotalPlayerCount = await StatisticsHelper.RecentMatches(this.database), TotalPlayerCount = await StatisticsHelper.RecentMatches(this.database),
Games = gameList, Games = gameList,
@ -74,4 +84,56 @@ public class StatisticsEndpoints : ApiEndpointController
return this.Ok(response); return this.Ok(response);
} }
/// <summary>
/// Get player counts for each individual platform
/// </summary>
/// <returns>An instance of PlayerCountByPlatformResponse</returns>
[HttpGet("playerCount/platform")]
[ProducesResponseType(typeof(PlayerCountByPlatformResponse), StatusCodes.Status200OK)]
public async Task<IActionResult> GetPlayerCountsByPlatform()
{
List<PlayerCountObject> platformList = new();
foreach (Platform platform in platforms)
{
platformList.Add(new PlayerCountByPlatformObject
{
Platform = platform.ToString(),
PlayerCount = await StatisticsHelper.RecentMatches(this.database, l => l.Platform == platform),
});
}
PlayerCountByPlatformResponse response = new()
{
TotalPlayerCount = await StatisticsHelper.RecentMatches(this.database),
Platforms = platformList,
};
return this.Ok(response);
}
/// <summary>
/// Gets a list of online players
/// </summary>
/// <returns>An instance of PlayerListResponse</returns>
[HttpGet("players")]
[ProducesResponseType(typeof(PlayerListResponse), StatusCodes.Status200OK)]
public async Task<IActionResult> GetPlayerList()
{
List<PlayerListObject> players = await this.database.LastContacts.Where(l => TimeHelper.Timestamp - l.Timestamp < 300)
.Select(l => new PlayerListObject
{
Username = l.User!.Username,
Game = l.GameVersion.ToString(),
Platform = l.Platform.ToString(),
})
.ToListAsync();
PlayerListResponse response = new()
{
Players = players,
};
return this.Ok(response);
}
} }

View file

@ -1,13 +1,37 @@
namespace LBPUnion.ProjectLighthouse.Servers.API.Responses; using System.Text.Json.Serialization;
namespace LBPUnion.ProjectLighthouse.Servers.API.Responses;
[JsonDerivedType(typeof(PlayerCountByGameObject))]
[JsonDerivedType(typeof(PlayerCountByPlatformObject))]
public class PlayerCountObject public class PlayerCountObject
{ {
public string Game { get; set; } = "";
public int PlayerCount { get; set; } public int PlayerCount { get; set; }
} }
public class PlayerCountByGameObject : PlayerCountObject
{
public string Game { get; set; } = "";
}
public class PlayerCountByPlatformObject : PlayerCountObject
{
public string Platform { get; set; } = "";
}
[JsonDerivedType(typeof(PlayerCountByGameResponse))]
[JsonDerivedType(typeof(PlayerCountByPlatformResponse))]
public class PlayerCountResponse public class PlayerCountResponse
{ {
public int TotalPlayerCount { get; set; } public int TotalPlayerCount { get; set; }
}
public class PlayerCountByGameResponse : PlayerCountResponse
{
public List<PlayerCountObject> Games { get; set; } = new(); public List<PlayerCountObject> Games { get; set; } = new();
}
public class PlayerCountByPlatformResponse : PlayerCountResponse
{
public List<PlayerCountObject> Platforms { get; set; } = new();
} }

View file

@ -0,0 +1,13 @@
namespace LBPUnion.ProjectLighthouse.Servers.API.Responses;
public class PlayerListResponse
{
public required List<PlayerListObject> Players { get; set; }
}
public class PlayerListObject
{
public required string Username { get; set; }
public required string Game { get; set; }
public required string Platform { get; set; }
}

View file

@ -27,7 +27,8 @@ public class StatisticsController : ControllerBase
public IActionResult PlayersInPodCount() => this.Ok(StatisticsHelper.RoomCountForPlatform(this.GetToken().Platform).ToString()); public IActionResult PlayersInPodCount() => this.Ok(StatisticsHelper.RoomCountForPlatform(this.GetToken().Platform).ToString());
[HttpGet("totalPlayerCount")] [HttpGet("totalPlayerCount")]
public async Task<IActionResult> TotalPlayerCount() => this.Ok((await StatisticsHelper.RecentMatchesForGame(this.database, this.GetToken().GameVersion)).ToString()); public async Task<IActionResult> TotalPlayerCount() =>
this.Ok((await StatisticsHelper.RecentMatches(this.database, l => l.GameVersion == this.GetToken().GameVersion)).ToString());
[HttpGet("planetStats")] [HttpGet("planetStats")]
[Produces("text/xml")] [Produces("text/xml")]

View file

@ -1,7 +1,10 @@
using System;
using System.Linq; using System.Linq;
using System.Linq.Expressions;
using System.Threading.Tasks; using System.Threading.Tasks;
using LBPUnion.ProjectLighthouse.Database; using LBPUnion.ProjectLighthouse.Database;
using LBPUnion.ProjectLighthouse.Filter; using LBPUnion.ProjectLighthouse.Filter;
using LBPUnion.ProjectLighthouse.Types.Entities.Profile;
using LBPUnion.ProjectLighthouse.Types.Users; using LBPUnion.ProjectLighthouse.Types.Users;
using Microsoft.EntityFrameworkCore; using Microsoft.EntityFrameworkCore;
@ -11,8 +14,8 @@ public static class StatisticsHelper
{ {
public static async Task<int> RecentMatches(DatabaseContext database) => await database.LastContacts.Where(l => TimeHelper.Timestamp - l.Timestamp < 300).CountAsync(); public static async Task<int> RecentMatches(DatabaseContext database) => await database.LastContacts.Where(l => TimeHelper.Timestamp - l.Timestamp < 300).CountAsync();
public static async Task<int> RecentMatchesForGame(DatabaseContext database, GameVersion gameVersion) public static async Task<int> RecentMatches(DatabaseContext database, Expression<Func<LastContactEntity, bool>> contactFilter) =>
=> await database.LastContacts.Where(l => TimeHelper.Timestamp - l.Timestamp < 300 && l.GameVersion == gameVersion).CountAsync(); await database.LastContacts.Where(l => TimeHelper.Timestamp - l.Timestamp < 300).Where(contactFilter).CountAsync();
public static async Task<int> SlotCount(DatabaseContext database, SlotQueryBuilder queryBuilder) => await database.Slots.Where(queryBuilder.Build()).CountAsync(); public static async Task<int> SlotCount(DatabaseContext database, SlotQueryBuilder queryBuilder) => await database.Slots.Where(queryBuilder.Build()).CountAsync();