Files
maraphon-app/tests/Marathon.Application.Tests/UseCases/RecordPlacedBetUseCaseTests.cs
T
alexei.dolgolyov 1ad896b07e feat(my-bets): personal bet journal with CLV tracking
Adds a manual bet-tracking journal that turns the analyzer into an actual
bet tracker. Users record wagers; the journal auto-grades them when event
results land and computes per-bet Closing-Line-Value against the latest
pre-match snapshot — the strongest long-run indicator of betting skill.

Domain:
- PlacedBet entity (reuses Bet vocabulary for Scope/Type/Side/Value/Rate)
  with stake, placed-at, outcome, and notes. Derived GrossReturn / NetProfit.
- BetOutcome enum (Pending / Won / Lost / Void).
- BetOutcomeResolver: pure function grading any Match-scope bet against an
  EventResult. Handles 1X2, draws, handicap (incl. push), and totals.
  Period-scope bets stay manual since EventResult only carries full-time.

Application:
- IPlacedBetRepository abstraction.
- ClosingLineValueCalculator: pure CLV math (implied-probability delta) +
  snapshot-matching predicate by Scope/Type/Side/Value.
- BetJournalReport + BetJournalStats records.
- Four use cases: Record / ResolvePending / BuildReport / Delete.
- New ISnapshotRepository.GetLatestPreMatchAsync pushes the closing-line
  pick into a single SQLite query rather than materialising the 30-day
  window in memory per event.
- ROI turnover excludes Void stakes — pushes are not real turnover and
  including them would dilute the user's edge.

Infrastructure:
- PlacedBetEntity / Configuration / Repository / Mapping helpers.
- 20260516 migration adding the PlacedBets table with EventCode and
  Outcome indices. Intentionally NO foreign key to Events — the journal
  is user data and must survive snapshot-retention pruning. Covered by an
  explicit round-trip test.

UI:
- Pages/MyBets/Journal.razor: hero header, 4-card KPI strip (ROI / strike
  rate / avg CLV / net profit, tinted by tone), inline add-bet form with
  the same invariants as the Bet record, drill-down table with per-row
  outcome pills, CLV percentage-points column, P&L, notes underline, and
  inline-confirm delete. RU + EN i18n.
- Nav entry under Analysis.

Tests: +55 across Domain / Application / Infrastructure (resolver math
including handicap push and total push boundaries, PlacedBet invariants
and derived properties, CLV math + null-handling, four use cases under
NSubstitute, EF round-trip including survives-event-deletion). All 379
tests pass.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-16 17:45:42 +03:00

76 lines
3.0 KiB
C#

using FluentAssertions;
using Marathon.Application.Abstractions;
using Marathon.Application.UseCases;
using Marathon.Domain.Entities;
using Marathon.Domain.Enums;
using Marathon.Domain.ValueObjects;
using Microsoft.Extensions.Logging.Abstractions;
using NSubstitute;
namespace Marathon.Application.Tests.UseCases;
public sealed class RecordPlacedBetUseCaseTests
{
private static readonly TimeSpan MoscowOffset = TimeSpan.FromHours(3);
private readonly IPlacedBetRepository _bets = Substitute.For<IPlacedBetRepository>();
private readonly IEventRepository _events = Substitute.For<IEventRepository>();
private readonly IResultRepository _results = Substitute.For<IResultRepository>();
private RecordPlacedBetUseCase CreateSut() =>
new(_bets, _events, _results, NullLogger<RecordPlacedBetUseCase>.Instance);
private static PlacedBet MakePending(EventId id, Side side = Side.Side1) =>
new(
Id: Guid.NewGuid(),
EventId: id,
Selection: new Bet(MatchScope.Instance, BetType.Win, side, null, new OddsRate(2.10m)),
Stake: 100m,
PlacedAt: new DateTimeOffset(2026, 5, 16, 12, 0, 0, MoscowOffset),
Outcome: BetOutcome.Pending,
Notes: null);
[Fact]
public async Task Should_Throw_When_EventDoesNotExist()
{
var id = new EventId("missing");
_events.GetAsync(id, Arg.Any<CancellationToken>()).Returns((Event?)null);
var act = async () => await CreateSut().ExecuteAsync(MakePending(id), CancellationToken.None);
await act.Should().ThrowAsync<InvalidOperationException>().WithMessage("*unknown event*");
await _bets.DidNotReceive().AddAsync(Arg.Any<PlacedBet>(), Arg.Any<CancellationToken>());
}
[Fact]
public async Task Should_PersistPending_When_NoResultYet()
{
var id = new EventId("event001");
_events.GetAsync(id, Arg.Any<CancellationToken>()).Returns(TestFixtures.MakeEvent(id.Value));
_results.GetAsync(id, Arg.Any<CancellationToken>()).Returns((EventResult?)null);
var bet = MakePending(id);
var stored = await CreateSut().ExecuteAsync(bet, CancellationToken.None);
stored.Outcome.Should().Be(BetOutcome.Pending, "no result yet — should remain pending");
await _bets.Received(1).AddAsync(Arg.Any<PlacedBet>(), Arg.Any<CancellationToken>());
await _bets.Received(1).SaveChangesAsync(Arg.Any<CancellationToken>());
}
[Fact]
public async Task Should_AutoGrade_When_ResultAlreadyAvailable()
{
var id = new EventId("event002");
_events.GetAsync(id, Arg.Any<CancellationToken>()).Returns(TestFixtures.MakeEvent(id.Value));
_results.GetAsync(id, Arg.Any<CancellationToken>())
.Returns(new EventResult(id, 2, 1, Side.Side1, DateTimeOffset.UtcNow));
var bet = MakePending(id, side: Side.Side1);
var stored = await CreateSut().ExecuteAsync(bet, CancellationToken.None);
stored.Outcome.Should().Be(BetOutcome.Won, "Side1 was selected and Side1 won");
}
}