feat(bp): /battle_pass/item_list — derives product per active season
Adds BattlePassSalesPeriodInfoDto, BattlePassProductDto, BattlePassItemListResponse DTOs, GetItemListAsync on BattlePassService (one product if not premium + CanPurchase, empty if already premium or off-season), and the /battle_pass/item_list controller action. 2 new integration tests; all 408 pass.
This commit is contained in:
@@ -28,4 +28,14 @@ public class BattlePassController : SVSimController
|
||||
if (info is null) return Ok(new { }); // off-season: empty payload
|
||||
return Ok(info);
|
||||
}
|
||||
|
||||
[HttpPost("item_list")]
|
||||
public async Task<IActionResult> ItemList(BaseRequest request, CancellationToken ct)
|
||||
{
|
||||
if (!TryGetViewerId(out long viewerId)) return Unauthorized();
|
||||
|
||||
var list = await _battlePass.GetItemListAsync(viewerId, ct);
|
||||
if (list is null) return Ok(new { });
|
||||
return Ok(list);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -0,0 +1,25 @@
|
||||
using MessagePack;
|
||||
using System.Text.Json.Serialization;
|
||||
|
||||
namespace SVSim.EmulatedEntrypoint.Models.Dtos.BattlePass;
|
||||
|
||||
/// <summary>
|
||||
/// /battle_pass/item_list response (Wizard/BattlePassPurchaseInfoTask.cs:23-44).
|
||||
/// </summary>
|
||||
[MessagePackObject]
|
||||
public class BattlePassItemListResponse
|
||||
{
|
||||
[JsonPropertyName("premium_pass_description")]
|
||||
[Key("premium_pass_description")]
|
||||
[JsonIgnore(Condition = JsonIgnoreCondition.Never)]
|
||||
public string PremiumPassDescription { get; set; } = "";
|
||||
|
||||
[JsonPropertyName("sales_period_info")]
|
||||
[Key("sales_period_info")]
|
||||
public BattlePassSalesPeriodInfoDto? SalesPeriodInfo { get; set; }
|
||||
|
||||
[JsonPropertyName("products")]
|
||||
[Key("products")]
|
||||
[JsonIgnore(Condition = JsonIgnoreCondition.Never)]
|
||||
public List<BattlePassProductDto> Products { get; set; } = new();
|
||||
}
|
||||
@@ -0,0 +1,43 @@
|
||||
using MessagePack;
|
||||
using System.Text.Json.Serialization;
|
||||
|
||||
namespace SVSim.EmulatedEntrypoint.Models.Dtos.BattlePass;
|
||||
|
||||
/// <summary>
|
||||
/// One product in /battle_pass/item_list.products[] (Wizard/BattlePassPurchaseInfoTask.cs:32-43).
|
||||
/// Numerics on this DTO are numeric on the wire (the client uses .ToInt() and the captured
|
||||
/// shape isn't string-typed here).
|
||||
/// </summary>
|
||||
[MessagePackObject]
|
||||
public class BattlePassProductDto
|
||||
{
|
||||
[JsonPropertyName("id")]
|
||||
[Key("id")]
|
||||
[JsonIgnore(Condition = JsonIgnoreCondition.Never)]
|
||||
public int Id { get; set; }
|
||||
|
||||
[JsonPropertyName("season_id")]
|
||||
[Key("season_id")]
|
||||
[JsonIgnore(Condition = JsonIgnoreCondition.Never)]
|
||||
public int SeasonId { get; set; }
|
||||
|
||||
[JsonPropertyName("name")]
|
||||
[Key("name")]
|
||||
[JsonIgnore(Condition = JsonIgnoreCondition.Never)]
|
||||
public string Name { get; set; } = "";
|
||||
|
||||
[JsonPropertyName("price_crystal")]
|
||||
[Key("price_crystal")]
|
||||
[JsonIgnore(Condition = JsonIgnoreCondition.Never)]
|
||||
public int PriceCrystal { get; set; }
|
||||
|
||||
[JsonPropertyName("description")]
|
||||
[Key("description")]
|
||||
[JsonIgnore(Condition = JsonIgnoreCondition.Never)]
|
||||
public string Description { get; set; } = "";
|
||||
|
||||
[JsonPropertyName("sales_period_info")]
|
||||
[Key("sales_period_info")]
|
||||
[JsonIgnore(Condition = JsonIgnoreCondition.Never)]
|
||||
public BattlePassSalesPeriodInfoDto SalesPeriodInfo { get; set; } = new();
|
||||
}
|
||||
@@ -0,0 +1,16 @@
|
||||
using MessagePack;
|
||||
using System.Text.Json.Serialization;
|
||||
|
||||
namespace SVSim.EmulatedEntrypoint.Models.Dtos.BattlePass;
|
||||
|
||||
/// <summary>
|
||||
/// sales_period_info on /battle_pass/item_list (Wizard/BattlePassPurchaseInfoTask.cs:26-29).
|
||||
/// Only sales_period_time is read by the client; other fields are unused but allowed.
|
||||
/// </summary>
|
||||
[MessagePackObject]
|
||||
public class BattlePassSalesPeriodInfoDto
|
||||
{
|
||||
[JsonPropertyName("sales_period_time")]
|
||||
[Key("sales_period_time")]
|
||||
public string? SalesPeriodTime { get; set; }
|
||||
}
|
||||
@@ -94,6 +94,44 @@ public sealed class BattlePassService : IBattlePassService
|
||||
};
|
||||
}
|
||||
|
||||
public async Task<BattlePassItemListResponse?> GetItemListAsync(long viewerId, CancellationToken ct)
|
||||
{
|
||||
var now = _time.GetUtcNow();
|
||||
var season = await _bp.GetActiveSeasonAsync(now, ct);
|
||||
if (season is null) return null;
|
||||
|
||||
var progress = await _viewerBp.GetOrCreateProgressAsync(viewerId, season.Id, ct);
|
||||
|
||||
var response = new BattlePassItemListResponse
|
||||
{
|
||||
PremiumPassDescription = season.Description,
|
||||
SalesPeriodInfo = new BattlePassSalesPeriodInfoDto
|
||||
{
|
||||
SalesPeriodTime = FormatWireDate(season.EndDate),
|
||||
},
|
||||
Products = new List<BattlePassProductDto>(),
|
||||
};
|
||||
|
||||
// One product per active season; empty if viewer is already premium.
|
||||
if (!progress.IsPremium && season.CanPurchase)
|
||||
{
|
||||
response.Products.Add(new BattlePassProductDto
|
||||
{
|
||||
Id = season.Id * 1000,
|
||||
SeasonId = season.Id,
|
||||
Name = $"{season.Name} Premium Pass",
|
||||
PriceCrystal = season.PriceCrystal,
|
||||
Description = season.Description,
|
||||
SalesPeriodInfo = new BattlePassSalesPeriodInfoDto
|
||||
{
|
||||
SalesPeriodTime = FormatWireDate(season.EndDate),
|
||||
},
|
||||
});
|
||||
}
|
||||
|
||||
return response;
|
||||
}
|
||||
|
||||
internal static int ComputeLevel(IReadOnlyList<BattlePassLevelEntry> curve, int point)
|
||||
{
|
||||
if (curve.Count == 0) return 1;
|
||||
|
||||
@@ -13,4 +13,10 @@ public interface IBattlePassService
|
||||
/// (controller emits empty body in that case).
|
||||
/// </summary>
|
||||
Task<BattlePassInfoResponse?> GetInfoAsync(long viewerId, CancellationToken ct);
|
||||
|
||||
/// <summary>
|
||||
/// /battle_pass/item_list payload. Returns one product per active season; empty products
|
||||
/// array if the viewer already owns premium for the active season. Null when no active season.
|
||||
/// </summary>
|
||||
Task<BattlePassItemListResponse?> GetItemListAsync(long viewerId, CancellationToken ct);
|
||||
}
|
||||
|
||||
@@ -0,0 +1,77 @@
|
||||
using System.Net;
|
||||
using System.Text;
|
||||
using System.Text.Json;
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
using Microsoft.Extensions.DependencyInjection;
|
||||
using SVSim.Database;
|
||||
using SVSim.Database.Models;
|
||||
using SVSim.UnitTests.Infrastructure;
|
||||
|
||||
namespace SVSim.UnitTests.Controllers;
|
||||
|
||||
public class BattlePassControllerItemListTests
|
||||
{
|
||||
private static StringContent JsonBody(string json) => new(json, Encoding.UTF8, "application/json");
|
||||
private const string EmptyAuthBody = """{"viewer_id":"0","steam_id":0,"steam_session_ticket":""}""";
|
||||
|
||||
private static async Task SeedSeason23(SVSimTestFactory f)
|
||||
{
|
||||
using var scope = f.Services.CreateScope();
|
||||
var db = scope.ServiceProvider.GetRequiredService<SVSimDbContext>();
|
||||
db.BattlePassSeasons.Add(new BattlePassSeasonEntry
|
||||
{
|
||||
Id = 23, Name = "Season 23", MaxLevel = 100,
|
||||
StartDate = DateTimeOffset.UtcNow.AddDays(-30),
|
||||
EndDate = DateTimeOffset.UtcNow.AddDays(30),
|
||||
CanPurchase = true, PriceCrystal = 980,
|
||||
Description = "Unlock premium track.",
|
||||
});
|
||||
await db.SaveChangesAsync();
|
||||
}
|
||||
|
||||
[Test]
|
||||
public async Task ItemList_returns_one_product_for_active_season()
|
||||
{
|
||||
using var factory = new SVSimTestFactory();
|
||||
long viewerId = await factory.SeedViewerAsync();
|
||||
await SeedSeason23(factory);
|
||||
|
||||
using var client = factory.CreateAuthenticatedClient(viewerId);
|
||||
var response = await client.PostAsync("/battle_pass/item_list", JsonBody(EmptyAuthBody));
|
||||
var body = await response.Content.ReadAsStringAsync();
|
||||
|
||||
Assert.That(response.StatusCode, Is.EqualTo(HttpStatusCode.OK), body);
|
||||
using var doc = JsonDocument.Parse(body);
|
||||
var products = doc.RootElement.GetProperty("products");
|
||||
Assert.That(products.GetArrayLength(), Is.EqualTo(1));
|
||||
var product = products[0];
|
||||
Assert.That(product.GetProperty("id").GetInt32(), Is.EqualTo(23000)); // 23 * 1000
|
||||
Assert.That(product.GetProperty("season_id").GetInt32(), Is.EqualTo(23));
|
||||
Assert.That(product.GetProperty("price_crystal").GetInt32(), Is.EqualTo(980));
|
||||
}
|
||||
|
||||
[Test]
|
||||
public async Task ItemList_returns_empty_products_when_viewer_already_premium()
|
||||
{
|
||||
using var factory = new SVSimTestFactory();
|
||||
long viewerId = await factory.SeedViewerAsync();
|
||||
await SeedSeason23(factory);
|
||||
|
||||
using (var scope = factory.Services.CreateScope())
|
||||
{
|
||||
var db = scope.ServiceProvider.GetRequiredService<SVSimDbContext>();
|
||||
db.ViewerBattlePassProgress.Add(new ViewerBattlePassProgressEntry
|
||||
{
|
||||
ViewerId = viewerId, SeasonId = 23, CurrentPoint = 0,
|
||||
IsPremium = true, WeeklyPoints = 0,
|
||||
});
|
||||
await db.SaveChangesAsync();
|
||||
}
|
||||
|
||||
using var client = factory.CreateAuthenticatedClient(viewerId);
|
||||
var response = await client.PostAsync("/battle_pass/item_list", JsonBody(EmptyAuthBody));
|
||||
var body = await response.Content.ReadAsStringAsync();
|
||||
using var doc = JsonDocument.Parse(body);
|
||||
Assert.That(doc.RootElement.GetProperty("products").GetArrayLength(), Is.EqualTo(0));
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user