-
Notifications
You must be signed in to change notification settings - Fork 7
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Hopefully optimize the DB and split the YAML schema out #27
Changes from all commits
489881c
aac7156
2f422c1
6d4f144
8bdfa28
d672821
b10a099
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Large diffs are not rendered by default.
Large diffs are not rendered by default.
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,88 @@ | ||
using Microsoft.EntityFrameworkCore.Migrations; | ||
using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata; | ||
|
||
#nullable disable | ||
|
||
namespace Server.Migrations | ||
{ | ||
/// <inheritdoc /> | ||
public partial class JobDepartmentMap : Migration | ||
{ | ||
/// <inheritdoc /> | ||
protected override void Up(MigrationBuilder migrationBuilder) | ||
{ | ||
migrationBuilder.CreateTable( | ||
name: "JobDepartments", | ||
columns: table => new | ||
{ | ||
Id = table.Column<int>(type: "integer", nullable: false) | ||
.Annotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn), | ||
Job = table.Column<string>(type: "text", nullable: false), | ||
Department = table.Column<string>(type: "text", nullable: false) | ||
}, | ||
constraints: table => | ||
{ | ||
table.PrimaryKey("PK_JobDepartments", x => x.Id); | ||
}); | ||
|
||
migrationBuilder.CreateIndex( | ||
name: "IX_JobDepartments_Job", | ||
table: "JobDepartments", | ||
column: "Job", | ||
unique: true); | ||
|
||
migrationBuilder.Sql( | ||
""" | ||
INSERT INTO "JobDepartments" | ||
( "Job", "Department" ) | ||
VALUES | ||
( 'Captain', 'Command' ), | ||
( 'HeadOfPersonnel', 'Command' ), | ||
( 'ChiefMedicalOfficer', 'Command' ), | ||
( 'ResearchDirector', 'Command' ), | ||
( 'HeadOfSecurity', 'Command' ), | ||
( 'ChiefEngineer', 'Command' ), | ||
( 'Quartermaster', 'Command' ), | ||
( 'Borg', 'Science' ), | ||
( 'Scientist', 'Science' ), | ||
( 'ResearchAssistant', 'Science' ), | ||
( 'Warden', 'Security' ), | ||
( 'Detective', 'Security' ), | ||
( 'SecurityOfficer', 'Security' ), | ||
( 'SecurityCadet', 'Security' ), | ||
( 'MedicalDoctor', 'Medical' ), | ||
( 'Chemist', 'Medical' ), | ||
( 'Paramedic', 'Medical' ), | ||
( 'Psychologist', 'Medical' ), | ||
( 'MedicalIntern', 'Medical' ), | ||
( 'StationEngineer', 'Engineering' ), | ||
( 'AtmosphericTechnician', 'Engineering' ), | ||
( 'TechnicalAssistant', 'Engineering' ), | ||
( 'Janitor', 'Service' ), | ||
( 'Chef', 'Service' ), | ||
( 'Botanist', 'Service' ), | ||
( 'Bartender', 'Service' ), | ||
( 'Chaplain', 'Service' ), | ||
( 'Lawyer', 'Service' ), | ||
( 'Musician', 'Service' ), | ||
( 'Reporter', 'Service' ), | ||
( 'Zookeeper', 'Service' ), | ||
( 'Librarian', 'Service' ), | ||
( 'ServiceWorker', 'Service' ), | ||
( 'Clown', 'Service' ), | ||
( 'Mime', 'Service' ), | ||
( 'CargoTechnician', 'Cargo' ), | ||
( 'SalvageSpecialist', 'Cargo' ), | ||
( 'Passenger', 'The tide' ); | ||
""" | ||
); | ||
} | ||
|
||
/// <inheritdoc /> | ||
protected override void Down(MigrationBuilder migrationBuilder) | ||
{ | ||
migrationBuilder.DropTable( | ||
name: "JobDepartments"); | ||
} | ||
} | ||
} |
Large diffs are not rendered by default.
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,105 @@ | ||
using System; | ||
using Microsoft.EntityFrameworkCore.Migrations; | ||
using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata; | ||
|
||
#nullable disable | ||
|
||
namespace Server.Migrations | ||
{ | ||
/// <inheritdoc /> | ||
public partial class ReplayParticipant : Migration | ||
{ | ||
/// <inheritdoc /> | ||
protected override void Up(MigrationBuilder migrationBuilder) | ||
{ | ||
migrationBuilder.DropForeignKey( | ||
name: "FK_Players_Replays_ReplayId", | ||
table: "Players"); | ||
|
||
migrationBuilder.AlterColumn<int>( | ||
name: "ReplayId", | ||
table: "Players", | ||
type: "integer", | ||
nullable: false, | ||
defaultValue: 0, | ||
oldClrType: typeof(int), | ||
oldType: "integer", | ||
oldNullable: true); | ||
|
||
migrationBuilder.CreateTable( | ||
name: "ReplayParticipants", | ||
columns: table => new | ||
{ | ||
Id = table.Column<int>(type: "integer", nullable: false) | ||
.Annotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn), | ||
PlayerGuid = table.Column<Guid>(type: "uuid", nullable: false), | ||
ReplayId = table.Column<int>(type: "integer", nullable: false) | ||
}, | ||
constraints: table => | ||
{ | ||
table.PrimaryKey("PK_ReplayParticipants", x => x.Id); | ||
table.ForeignKey( | ||
name: "FK_ReplayParticipants_Replays_ReplayId", | ||
column: x => x.ReplayId, | ||
principalTable: "Replays", | ||
principalColumn: "Id", | ||
onDelete: ReferentialAction.Cascade); | ||
}); | ||
|
||
migrationBuilder.CreateIndex( | ||
name: "IX_ReplayParticipants_PlayerGuid_ReplayId", | ||
table: "ReplayParticipants", | ||
columns: new[] { "PlayerGuid", "ReplayId" }, | ||
unique: true); | ||
|
||
migrationBuilder.CreateIndex( | ||
name: "IX_ReplayParticipants_ReplayId", | ||
table: "ReplayParticipants", | ||
column: "ReplayId"); | ||
|
||
migrationBuilder.AddForeignKey( | ||
name: "FK_Players_Replays_ReplayId", | ||
table: "Players", | ||
column: "ReplayId", | ||
principalTable: "Replays", | ||
principalColumn: "Id", | ||
onDelete: ReferentialAction.Cascade); | ||
|
||
migrationBuilder.Sql( | ||
""" | ||
insert into "ReplayParticipants" ("PlayerGuid", "ReplayId") ( | ||
select DISTINCT p."PlayerGuid", p."ReplayId" | ||
from "Players" p | ||
GROUP BY p."PlayerGuid", p."ReplayId" | ||
); | ||
""" | ||
); | ||
} | ||
|
||
/// <inheritdoc /> | ||
protected override void Down(MigrationBuilder migrationBuilder) | ||
{ | ||
migrationBuilder.DropForeignKey( | ||
name: "FK_Players_Replays_ReplayId", | ||
table: "Players"); | ||
|
||
migrationBuilder.DropTable( | ||
name: "ReplayParticipants"); | ||
|
||
migrationBuilder.AlterColumn<int>( | ||
name: "ReplayId", | ||
table: "Players", | ||
type: "integer", | ||
nullable: true, | ||
oldClrType: typeof(int), | ||
oldType: "integer"); | ||
|
||
migrationBuilder.AddForeignKey( | ||
name: "FK_Players_Replays_ReplayId", | ||
table: "Players", | ||
column: "ReplayId", | ||
principalTable: "Replays", | ||
principalColumn: "Id"); | ||
} | ||
} | ||
} |
Large diffs are not rendered by default.
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,185 @@ | ||
using System; | ||
using Microsoft.EntityFrameworkCore.Migrations; | ||
|
||
#nullable disable | ||
|
||
namespace Server.Migrations | ||
{ | ||
/// <inheritdoc /> | ||
public partial class PlayerTrim : Migration | ||
{ | ||
/// <inheritdoc /> | ||
protected override void Up(MigrationBuilder migrationBuilder) | ||
{ | ||
// FIXME: Merge with previous migration | ||
|
||
migrationBuilder.Sql( | ||
""" | ||
TRUNCATE "ReplayParticipants"; | ||
""" | ||
); | ||
|
||
migrationBuilder.AlterColumn<string>( | ||
name: "Link", | ||
table: "Replays", | ||
type: "text", | ||
nullable: false, | ||
defaultValue: "", | ||
oldClrType: typeof(string), | ||
oldType: "text", | ||
oldNullable: true); | ||
|
||
migrationBuilder.AddColumn<int>( | ||
name: "ParticipantId", | ||
table: "Players", | ||
type: "integer", | ||
nullable: false, | ||
defaultValue: 0); | ||
|
||
migrationBuilder.CreateIndex( | ||
name: "IX_Players_ParticipantId", | ||
table: "Players", | ||
column: "ParticipantId"); | ||
|
||
migrationBuilder.AddColumn<string>( | ||
name: "Username", | ||
table: "ReplayParticipants", | ||
type: "text", | ||
nullable: false, | ||
defaultValue: ""); | ||
|
||
migrationBuilder.CreateIndex( | ||
name: "IX_ReplayParticipants_Username", | ||
table: "ReplayParticipants", | ||
column: "Username"); | ||
|
||
// SQL magic here | ||
|
||
migrationBuilder.Sql( | ||
""" | ||
insert into "ReplayParticipants" ("PlayerGuid", "ReplayId", "Username") ( | ||
select DISTINCT p."PlayerGuid", p."ReplayId", p."PlayerOocName" | ||
from "Players" p | ||
GROUP BY p."PlayerGuid", p."ReplayId", p."PlayerOocName" | ||
); | ||
""" | ||
); | ||
|
||
migrationBuilder.Sql( | ||
""" | ||
UPDATE "Players" AS pl | ||
SET "ParticipantId" = p."Id" | ||
FROM "ReplayParticipants" AS p | ||
WHERE p."ReplayId" = pl."ReplayId" | ||
AND p."PlayerGuid" = pl."PlayerGuid"; | ||
""" | ||
); | ||
|
||
// Data must be valid by this point | ||
|
||
migrationBuilder.AddForeignKey( | ||
name: "FK_Players_ReplayParticipants_ParticipantId", | ||
table: "Players", | ||
column: "ParticipantId", | ||
principalTable: "ReplayParticipants", | ||
principalColumn: "Id", | ||
onDelete: ReferentialAction.Cascade); | ||
|
||
migrationBuilder.DropForeignKey( | ||
name: "FK_Players_Replays_ReplayId", | ||
table: "Players"); | ||
|
||
// Data lossy operations | ||
|
||
migrationBuilder.DropIndex( | ||
name: "IX_Players_PlayerGuid", | ||
table: "Players"); | ||
|
||
migrationBuilder.DropIndex( | ||
name: "IX_Players_PlayerOocName", | ||
table: "Players"); | ||
|
||
migrationBuilder.DropIndex( | ||
name: "IX_Players_ReplayId", | ||
table: "Players"); | ||
|
||
migrationBuilder.DropColumn( | ||
name: "PlayerGuid", | ||
table: "Players"); | ||
|
||
migrationBuilder.DropColumn( | ||
name: "PlayerOocName", | ||
table: "Players"); | ||
|
||
migrationBuilder.DropColumn( | ||
name: "ReplayId", | ||
table: "Players"); | ||
} | ||
|
||
/// <inheritdoc /> | ||
protected override void Down(MigrationBuilder migrationBuilder) | ||
{ | ||
migrationBuilder.DropForeignKey( | ||
name: "FK_Players_ReplayParticipants_ParticipantId", | ||
table: "Players"); | ||
|
||
migrationBuilder.DropIndex( | ||
name: "IX_ReplayParticipants_Username", | ||
table: "ReplayParticipants"); | ||
|
||
migrationBuilder.DropColumn( | ||
name: "Username", | ||
table: "ReplayParticipants"); | ||
|
||
migrationBuilder.RenameColumn( | ||
name: "ParticipantId", | ||
table: "Players", | ||
newName: "ReplayId"); | ||
|
||
migrationBuilder.RenameIndex( | ||
name: "IX_Players_ParticipantId", | ||
table: "Players", | ||
newName: "IX_Players_ReplayId"); | ||
|
||
migrationBuilder.AlterColumn<string>( | ||
name: "Link", | ||
table: "Replays", | ||
type: "text", | ||
nullable: true, | ||
oldClrType: typeof(string), | ||
oldType: "text"); | ||
|
||
migrationBuilder.AddColumn<Guid>( | ||
name: "PlayerGuid", | ||
table: "Players", | ||
type: "uuid", | ||
nullable: false, | ||
defaultValue: new Guid("00000000-0000-0000-0000-000000000000")); | ||
|
||
migrationBuilder.AddColumn<string>( | ||
name: "PlayerOocName", | ||
table: "Players", | ||
type: "text", | ||
nullable: false, | ||
defaultValue: ""); | ||
|
||
migrationBuilder.CreateIndex( | ||
name: "IX_Players_PlayerGuid", | ||
table: "Players", | ||
column: "PlayerGuid"); | ||
|
||
migrationBuilder.CreateIndex( | ||
name: "IX_Players_PlayerOocName", | ||
table: "Players", | ||
column: "PlayerOocName"); | ||
|
||
migrationBuilder.AddForeignKey( | ||
name: "FK_Players_Replays_ReplayId", | ||
table: "Players", | ||
column: "ReplayId", | ||
principalTable: "Replays", | ||
principalColumn: "Id", | ||
onDelete: ReferentialAction.Cascade); | ||
} | ||
} | ||
} |
Large diffs are not rendered by default.
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,59 @@ | ||
using Microsoft.EntityFrameworkCore.Migrations; | ||
|
||
#nullable disable | ||
|
||
namespace Server.Migrations | ||
{ | ||
/// <inheritdoc /> | ||
public partial class JobForeignKey : Migration | ||
{ | ||
/// <inheritdoc /> | ||
protected override void Up(MigrationBuilder migrationBuilder) | ||
{ | ||
migrationBuilder.AddColumn<int>( | ||
name: "EffectiveJobId", | ||
table: "Players", | ||
type: "integer", | ||
nullable: true); | ||
|
||
migrationBuilder.CreateIndex( | ||
name: "IX_Players_EffectiveJobId", | ||
table: "Players", | ||
column: "EffectiveJobId"); | ||
|
||
migrationBuilder.AddForeignKey( | ||
name: "FK_Players_JobDepartments_EffectiveJobId", | ||
table: "Players", | ||
column: "EffectiveJobId", | ||
principalTable: "JobDepartments", | ||
principalColumn: "Id"); | ||
|
||
migrationBuilder.Sql( | ||
""" | ||
UPDATE "Players" | ||
SET "EffectiveJobId" = ( | ||
SELECT j."Id" FROM "JobDepartments" j | ||
WHERE "Players"."JobPrototypes"[1] = j."Job" | ||
) | ||
WHERE "Players"."JobPrototypes" != '{}' | ||
""" | ||
); | ||
} | ||
|
||
/// <inheritdoc /> | ||
protected override void Down(MigrationBuilder migrationBuilder) | ||
{ | ||
migrationBuilder.DropForeignKey( | ||
name: "FK_Players_JobDepartments_EffectiveJobId", | ||
table: "Players"); | ||
|
||
migrationBuilder.DropIndex( | ||
name: "IX_Players_EffectiveJobId", | ||
table: "Players"); | ||
|
||
migrationBuilder.DropColumn( | ||
name: "EffectiveJobId", | ||
table: "Players"); | ||
} | ||
} | ||
} |
Large diffs are not rendered by default.
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,20 @@ | ||
using Microsoft.EntityFrameworkCore; | ||
using Microsoft.EntityFrameworkCore.Metadata.Builders; | ||
|
||
namespace ReplayBrowser.Data.Models; | ||
|
||
/// <summary> | ||
/// Maps a given job ID to a department | ||
/// </summary> | ||
public class JobDepartment : IEntityTypeConfiguration<JobDepartment> | ||
{ | ||
public int Id { get; set; } | ||
|
||
public required string Job { get; set; } | ||
public required string Department { get; set; } | ||
|
||
public void Configure(EntityTypeBuilder<JobDepartment> builder) | ||
{ | ||
builder.HasIndex(j => j.Job).IsUnique(); | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,52 +1,51 @@ | ||
using Microsoft.EntityFrameworkCore; | ||
using Microsoft.EntityFrameworkCore.Metadata.Builders; | ||
using YamlDotNet.Serialization; | ||
using ReplayBrowser.Models.Ingested; | ||
|
||
namespace ReplayBrowser.Data.Models; | ||
|
||
public class Player : IEntityTypeConfiguration<Player> | ||
{ | ||
public int Id { get; set; } | ||
|
||
[YamlMember(Alias = "antagPrototypes")] | ||
public List<string> AntagPrototypes { get; set; } | ||
[YamlMember(Alias = "jobPrototypes")] | ||
public List<string> JobPrototypes { get; set; } | ||
[YamlMember(Alias = "playerGuid")] | ||
public Guid PlayerGuid { get; set; } | ||
[YamlMember(Alias = "playerICName")] | ||
public string PlayerIcName { get; set; } | ||
[YamlMember(Alias = "playerOOCName")] | ||
public string PlayerOocName { get; set; } | ||
[YamlMember(Alias = "antag")] | ||
public List<string> AntagPrototypes { get; set; } = null!; | ||
public List<string> JobPrototypes { get; set; } = null!; | ||
public required string PlayerIcName { get; set; } | ||
public bool Antag { get; set; } | ||
|
||
// Foreign key | ||
|
||
public int? ReplayId { get; set; } | ||
public Replay? Replay { get; set; } | ||
public ReplayParticipant Participant { get; set; } = null!; | ||
public int ParticipantId { get; set; } | ||
|
||
public JobDepartment? EffectiveJob { get; set; } | ||
public int? EffectiveJobId { get; set; } | ||
|
||
public void Configure(EntityTypeBuilder<Player> builder) | ||
{ | ||
builder.HasIndex(p => p.PlayerGuid); | ||
builder.HasIndex(p => p.PlayerIcName); | ||
builder.HasIndex(p => p.PlayerOocName); | ||
builder.HasIndex(p => p.ReplayId); | ||
builder.HasIndex(p => p.ParticipantId); | ||
} | ||
|
||
public static Player FromYaml(YamlPlayer player) | ||
{ | ||
return new Player { | ||
PlayerIcName = player.PlayerIcName, | ||
|
||
JobPrototypes = player.JobPrototypes, | ||
AntagPrototypes = player.AntagPrototypes, | ||
|
||
Antag = player.Antag | ||
}; | ||
} | ||
|
||
public void RedactInformation(bool wasGdpr = false) | ||
{ | ||
if (wasGdpr) | ||
{ | ||
PlayerIcName = "Removed by GDPR request"; | ||
PlayerOocName = "Removed by GDPR request"; | ||
} | ||
else | ||
{ | ||
PlayerIcName = "Redacted"; | ||
PlayerOocName = "Redacted"; | ||
} | ||
PlayerGuid = Guid.Empty; | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,25 @@ | ||
using Microsoft.EntityFrameworkCore; | ||
using Microsoft.EntityFrameworkCore.Metadata.Builders; | ||
|
||
namespace ReplayBrowser.Data.Models; | ||
|
||
/// <summary> | ||
/// Basic entry that says some player was in a given round | ||
/// </summary> | ||
public class ReplayParticipant : IEntityTypeConfiguration<ReplayParticipant> | ||
{ | ||
public int Id { get; set; } | ||
public Guid PlayerGuid { get; set; } | ||
public string Username { get; set; } = null!; | ||
|
||
public Replay Replay { get; set; } = null!; | ||
public int ReplayId { get; set; } | ||
|
||
public List<Player>? Players { get; set; } | ||
|
||
public void Configure(EntityTypeBuilder<ReplayParticipant> builder) | ||
{ | ||
builder.HasIndex(p => new { p.PlayerGuid, p.ReplayId }).IsUnique(); | ||
builder.HasIndex(p => p.Username); | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,15 @@ | ||
using YamlDotNet.Serialization; | ||
|
||
namespace ReplayBrowser.Models.Ingested; | ||
|
||
public class YamlPlayer | ||
{ | ||
public required List<string> AntagPrototypes { get; set; } | ||
public required List<string> JobPrototypes { get; set; } | ||
public Guid PlayerGuid { get; set; } | ||
[YamlMember(Alias = "playerICName", ApplyNamingConventions = false)] | ||
public required string PlayerIcName { get; set; } | ||
[YamlMember(Alias = "playerOOCName", ApplyNamingConventions = false)] | ||
public required string PlayerOocName { get; set; } | ||
public bool Antag { get; set; } | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,26 @@ | ||
using YamlDotNet.Serialization; | ||
|
||
namespace ReplayBrowser.Models.Ingested; | ||
|
||
public class YamlReplay { | ||
public int? RoundId { get; set; } | ||
|
||
[YamlMember(Alias = "server_id", ApplyNamingConventions = false)] | ||
public string ServerId { get; set; } | ||
[YamlMember(Alias = "server_name", ApplyNamingConventions = false)] | ||
public string? ServerName { get; set; } | ||
|
||
public string Gamemode { get; set; } | ||
public string? Map { get; set; } | ||
public List<string>? Maps { get; set; } | ||
|
||
public List<YamlPlayer>? RoundEndPlayers { get; set; } | ||
public string? RoundEndText { get; set; } | ||
|
||
public int EndTick { get; set; } | ||
public string Duration { get; set; } | ||
public int FileCount { get; set; } | ||
public int Size { get; set; } | ||
public int UncompressedSize { get; set; } | ||
public string EndTime { get; set; } | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,3 +1,5 @@ | ||
using ReplayBrowser.Services; | ||
using ReplayBrowser.Services.ReplayParser; | ||
using Serilog; | ||
using Serilog.Events; | ||
|
||
|
@@ -11,7 +13,7 @@ public static void Main(string[] args) | |
.WriteTo.Console() | ||
.WriteTo.File("logs/log.txt", rollingInterval: RollingInterval.Hour, fileSizeLimitBytes: null) | ||
.CreateLogger(); | ||
|
||
try | ||
{ | ||
Log.Information("Starting up"); | ||
|
@@ -26,7 +28,7 @@ public static void Main(string[] args) | |
Log.CloseAndFlush(); | ||
} | ||
} | ||
|
||
public static IHostBuilder CreateHostBuilder(string[] args) => | ||
Host.CreateDefaultBuilder(args) | ||
.ConfigureAppConfiguration((context, builder) => | ||
|
@@ -50,5 +52,11 @@ public static IHostBuilder CreateHostBuilder(string[] args) => | |
{ | ||
webBuilder.UseKestrel(); | ||
webBuilder.UseStartup<Startup>(); | ||
}) | ||
.ConfigureServices(s => { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Whats the reason for pulling this out of the startup? Not an issue, just curious. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. They depend on DbContext, but need setting up somewhere, and so they ended up outside. And put after dbcontext has migrated, so they don't error out as they rush to do things. And migration has to be in other part of Startup because otherwise the EF dotnet tool just breaks in weird ways |
||
s.AddHostedService<BackgroundServiceStarter<ReplayParserService>>(); | ||
s.AddHostedService<BackgroundServiceStarter<AccountService>>(); | ||
s.AddHostedService<BackgroundServiceStarter<LeaderboardService>>(); | ||
s.AddHostedService<BackgroundServiceStarter<AnalyticsService>>(); | ||
}); | ||
} |
Large diffs are not rendered by default.
Unchanged files with check annotations Beta
@using Microsoft.AspNetCore.Components.Authorization | ||
@using ReplayBrowser.Data.Models.Account | ||
@using ReplayBrowser.Services | ||
@using Microsoft.AspNetCore.Components.Web | ||
@using ReplayBrowser.Data.Models | ||
@using ReplayBrowser.Helpers | ||
@inject AuthenticationStateProvider AuthenticationStateProvider |
protected override async Task OnInitializedAsync() | ||
{ | ||
var authstate = await AuthenticationStateProvider.GetAuthenticationStateAsync(); | ||
Replay = await ReplayHelper.GetReplay(Convert.ToInt32(Id), authstate)!; | ||
IsLoading = false; | ||
} | ||
} |
<h5 class="card-title">@_nameFormatted</h5> | ||
@if (ReplayData.Map == null) | ||
{ | ||
<p>Maps: @string.Join(", ", ReplayData.Maps)</p> | ||
} | ||
else | ||
{ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
A bit nitpick-y: This doesn't include any of the RMC14 jobs. It would be nice to have them also count (for example) to command. Think of CMCMO or something. But I don't really know any of the roles so /shrug/
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Should they be added into existing departments?
Could also make an extra category for Xenos because... apparently those are jobs
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Maybe? Unsure. It would certainly look fancy.