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>
This commit is contained in:
2026-05-16 17:45:42 +03:00
parent 292223174c
commit 1ad896b07e
36 changed files with 3315 additions and 0 deletions
@@ -360,4 +360,67 @@
<data name="Insights.Action.Refresh"><value>Обновить</value></data>
<data name="Insights.Action.OpenAnomaly"><value>Открыть</value></data>
<data name="Insights.Bucket.NotApplicable"><value>—</value></data>
<data name="Nav.MyBets"><value>Мои ставки</value></data>
<data name="Journal.Kicker"><value>Журнал</value></data>
<data name="Journal.Title"><value>Ваши ставки и CLV</value></data>
<data name="Journal.Lede"><value>Каждая зафиксированная ставка с автоматическим расчётом результата и оценкой против линии закрытия. Положительный CLV — главный долгосрочный индикатор того, что вы стабильно обыгрываете рынок.</value></data>
<data name="Journal.Stat.Roi"><value>ROI</value></data>
<data name="Journal.Stat.Roi.Hint"><value>Чистая прибыль ÷ сумма ставок.</value></data>
<data name="Journal.Stat.StrikeRate"><value>Strike rate</value></data>
<data name="Journal.Stat.StrikeRate.Hint"><value>Победы ÷ (победы + поражения).</value></data>
<data name="Journal.Stat.AvgClv"><value>Средний CLV</value></data>
<data name="Journal.Stat.AvgClv.Hint"><value>Средний прирост вероятности к линии закрытия.</value></data>
<data name="Journal.Stat.NetProfit"><value>Чистая прибыль</value></data>
<data name="Journal.Stat.NetProfit.Hint"><value>Возвраты минус ставки (учтены сыгравшие).</value></data>
<data name="Journal.Stat.TotalBets"><value>Всего ставок</value></data>
<data name="Journal.Stat.Pending"><value>В ожидании</value></data>
<data name="Journal.Stat.Won"><value>Победа</value></data>
<data name="Journal.Stat.Lost"><value>Проигрыш</value></data>
<data name="Journal.Stat.Void"><value>Возврат</value></data>
<data name="Journal.Section.Add"><value>Записать ставку</value></data>
<data name="Journal.Section.List"><value>Журнал ставок</value></data>
<data name="Journal.Action.Refresh"><value>Обновить</value></data>
<data name="Journal.Action.Resolve"><value>Рассчитать ожидающие</value></data>
<data name="Journal.Action.Submit"><value>Записать</value></data>
<data name="Journal.Action.Delete"><value>Удалить</value></data>
<data name="Journal.Action.Confirm"><value>Подтвердить</value></data>
<data name="Journal.Action.Cancel"><value>Отмена</value></data>
<data name="Journal.Field.EventId"><value>ID события</value></data>
<data name="Journal.Field.EventId.Hint"><value>Числовой ID из URL детальной страницы.</value></data>
<data name="Journal.Field.Type"><value>Тип ставки</value></data>
<data name="Journal.Field.Side"><value>Сторона</value></data>
<data name="Journal.Field.Value"><value>Порог</value></data>
<data name="Journal.Field.Value.Hint"><value>Гандикап или тотал (например 1.5, 2.5).</value></data>
<data name="Journal.Field.Rate"><value>Кэф на момент ставки</value></data>
<data name="Journal.Field.Stake"><value>Сумма ставки</value></data>
<data name="Journal.Field.Notes"><value>Заметки</value></data>
<data name="Journal.Field.Notes.Placeholder"><value>Тег стратегии, букмекер или что угодно для памяти…</value></data>
<data name="Journal.BetType.Win"><value>Победа</value></data>
<data name="Journal.BetType.Draw"><value>Ничья</value></data>
<data name="Journal.BetType.WinFora"><value>Фора</value></data>
<data name="Journal.BetType.Total"><value>Тотал</value></data>
<data name="Journal.Side.Side1"><value>Сторона 1</value></data>
<data name="Journal.Side.Side2"><value>Сторона 2</value></data>
<data name="Journal.Side.Draw"><value>Ничья</value></data>
<data name="Journal.Side.Less"><value>Меньше</value></data>
<data name="Journal.Side.More"><value>Больше</value></data>
<data name="Journal.Outcome.Pending"><value>Ожидает</value></data>
<data name="Journal.Outcome.Won"><value>Победа</value></data>
<data name="Journal.Outcome.Lost"><value>Проигрыш</value></data>
<data name="Journal.Outcome.Void"><value>Возврат</value></data>
<data name="Journal.Column.PlacedAt"><value>Размещено</value></data>
<data name="Journal.Column.Match"><value>Матч</value></data>
<data name="Journal.Column.Selection"><value>Выбор</value></data>
<data name="Journal.Column.Stake"><value>Ставка</value></data>
<data name="Journal.Column.Rate"><value>Кэф</value></data>
<data name="Journal.Column.Profit"><value>P&amp;L</value></data>
<data name="Journal.Column.Clv"><value>CLV</value></data>
<data name="Journal.Column.Outcome"><value>Итог</value></data>
<data name="Journal.Empty.None"><value>Ставок пока нет. Запишите свою ставку через форму выше — после окончания матча журнал авто-проставит результат и посчитает CLV против последнего пре-матч снимка.</value></data>
<data name="Journal.Empty.NotApplicable"><value>—</value></data>
<data name="Journal.Error.Generic"><value>Не удалось сохранить ставку — проверьте ID события и повторите.</value></data>
<data name="Journal.Resolve.None"><value>Ожидающих ставок к расчёту нет.</value></data>
<data name="Journal.Resolve.Done"><value>Рассчитано ожидающих: {0}.</value></data>
<data name="Journal.Confirm.Delete"><value>Удалить эту ставку безвозвратно?</value></data>
</root>