feat(battle-node): IMatchingBridge + MatchingBridge mint battle id + node url
This commit is contained in:
10
SVSim.BattleNode/Bridge/BattleNodeOptions.cs
Normal file
10
SVSim.BattleNode/Bridge/BattleNodeOptions.cs
Normal file
@@ -0,0 +1,10 @@
|
||||
namespace SVSim.BattleNode.Bridge;
|
||||
|
||||
/// <summary>
|
||||
/// DI-injected options for the battle node. The web host populates these — typically
|
||||
/// nodeServerUrl is "ws://localhost:5148" matching ASPNETCORE_URLS.
|
||||
/// </summary>
|
||||
public sealed class BattleNodeOptions
|
||||
{
|
||||
public string NodeServerUrl { get; set; } = "ws://localhost:5148";
|
||||
}
|
||||
12
SVSim.BattleNode/Bridge/IMatchingBridge.cs
Normal file
12
SVSim.BattleNode/Bridge/IMatchingBridge.cs
Normal file
@@ -0,0 +1,12 @@
|
||||
namespace SVSim.BattleNode.Bridge;
|
||||
|
||||
public interface IMatchingBridge
|
||||
{
|
||||
/// <summary>
|
||||
/// Mint a battle id, register a pending session for the given viewer, and return the
|
||||
/// URL the client should open a socket to.
|
||||
/// </summary>
|
||||
PendingMatch RegisterPendingBattle(long viewerId);
|
||||
}
|
||||
|
||||
public sealed record PendingMatch(string BattleId, string NodeServerUrl);
|
||||
23
SVSim.BattleNode/Bridge/MatchingBridge.cs
Normal file
23
SVSim.BattleNode/Bridge/MatchingBridge.cs
Normal file
@@ -0,0 +1,23 @@
|
||||
using SVSim.BattleNode.Sessions;
|
||||
|
||||
namespace SVSim.BattleNode.Bridge;
|
||||
|
||||
public sealed class MatchingBridge : IMatchingBridge
|
||||
{
|
||||
private readonly IBattleSessionStore _store;
|
||||
private readonly BattleNodeOptions _options;
|
||||
|
||||
public MatchingBridge(IBattleSessionStore store, BattleNodeOptions options)
|
||||
{
|
||||
_store = store;
|
||||
_options = options;
|
||||
}
|
||||
|
||||
public PendingMatch RegisterPendingBattle(long viewerId)
|
||||
{
|
||||
// 12-digit decimal battle id mirrors the captures (e.g. "975695075012").
|
||||
var battleId = (Math.Abs(Guid.NewGuid().GetHashCode()) % 1_000_000_000_000L).ToString("D12");
|
||||
_store.RegisterPending(new PendingBattle(battleId, viewerId));
|
||||
return new PendingMatch(battleId, _options.NodeServerUrl);
|
||||
}
|
||||
}
|
||||
35
SVSim.UnitTests/BattleNode/Bridge/MatchingBridgeTests.cs
Normal file
35
SVSim.UnitTests/BattleNode/Bridge/MatchingBridgeTests.cs
Normal file
@@ -0,0 +1,35 @@
|
||||
using NUnit.Framework;
|
||||
using SVSim.BattleNode.Bridge;
|
||||
using SVSim.BattleNode.Sessions;
|
||||
|
||||
namespace SVSim.UnitTests.BattleNode.Bridge;
|
||||
|
||||
[TestFixture]
|
||||
public class MatchingBridgeTests
|
||||
{
|
||||
[Test]
|
||||
public void RegisterPendingBattle_RegistersInStoreAndReturnsNodeUrl()
|
||||
{
|
||||
var store = new InMemoryBattleSessionStore();
|
||||
var bridge = new MatchingBridge(store, new BattleNodeOptions { NodeServerUrl = "ws://localhost:5148" });
|
||||
|
||||
var match = bridge.RegisterPendingBattle(viewerId: 906243102);
|
||||
|
||||
Assert.That(match.NodeServerUrl, Is.EqualTo("ws://localhost:5148"));
|
||||
Assert.That(match.BattleId, Is.Not.Empty);
|
||||
var pending = store.TryGetPending(match.BattleId);
|
||||
Assert.That(pending, Is.Not.Null);
|
||||
Assert.That(pending!.ViewerId, Is.EqualTo(906243102));
|
||||
}
|
||||
|
||||
[Test]
|
||||
public void RegisterPendingBattle_MintsUniqueBattleIds()
|
||||
{
|
||||
var bridge = new MatchingBridge(new InMemoryBattleSessionStore(), new BattleNodeOptions());
|
||||
|
||||
var a = bridge.RegisterPendingBattle(1);
|
||||
var b = bridge.RegisterPendingBattle(2);
|
||||
|
||||
Assert.That(a.BattleId, Is.Not.EqualTo(b.BattleId));
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user