72 lines
4.7 KiB
Markdown
72 lines
4.7 KiB
Markdown
# Спринт 4 (частичный) — POS Sync API + observability
|
||
|
||
Автономная работа. WPF/POS-UI (`food-market.pos`, net8.0-windows) на Linux
|
||
не собирается — пропускаем. Здесь только API-сторона синхронизации
|
||
и метрики.
|
||
|
||
После каждого пункта: `dotnet build` (SDK 8.0.126), unit + integration
|
||
тесты, коммит порцией, отметка `[x]`, коммит прогресса.
|
||
|
||
## Чек-лист
|
||
|
||
1. [x] **P1-12a Контракты POS в food-market.shared** — DTO `ProductSyncDto`,
|
||
`PriceSyncDto`, `StockSyncDto`, `CounterpartySyncDto`, `PosSaleBatchDto`
|
||
(с idempotency-key). Версионирование через namespace v1.
|
||
✅ `src/food-market.shared/Pos/V1/SyncDtos.cs` — все DTO как record'ы
|
||
с required-полями, конверты `PosSyncResponse` и `PosSaleBatchResponse`,
|
||
двойная идемпотентность (batch IdempotencyKey + per-sale ClientSaleId).
|
||
3 unit-теста на round-trip сериализации.
|
||
2. [x] **P1-12b POS Sync API** — `GET /api/pos/sync?since={iso8601}` возвращает
|
||
изменения после ts (товары, цены, остатки, контрагенты). `POST /api/pos/sales`
|
||
принимает батч продаж с idempotency-key (повторный запрос возвращает прежний
|
||
результат без дублей). Multi-tenant: POS-токен ↔ один магазин.
|
||
✅ `/api/pos/v1/sync` и `/api/pos/v1/sales` (URL-versioned). Двойная
|
||
идемпотентность: `PosBatchAck` (unique idx OrgId+Key, race ловится 23505)
|
||
+ ClientSaleId через маркер `pos:GUID32` в `RetailSale.Notes`. Pre-flight
|
||
на остатки — недостающие позиции попадают в `Failed`, остальные проводятся.
|
||
7 интеграционных тестов.
|
||
3. [x] **P1-17 Метрики Prometheus** — `prometheus-net.AspNetCore`, `/metrics`.
|
||
`http_requests_total`, `http_request_duration_seconds`,
|
||
`db_query_duration_seconds` (через EF interceptor), бизнес:
|
||
`sales_posted_total`, `supplies_posted_total`, `documents_error_total`.
|
||
`docs/observability.md` с образцом Grafana dashboard.
|
||
✅ `prometheus-net.AspNetCore@8.2.1` + `DbMetricsInterceptor` (singleton,
|
||
EF Core `DbCommandInterceptor`). Бизнес-счётчики в `AppMetrics` —
|
||
инкрементятся в успешных Post'ах Sales/Supplies/POS. Endpoint
|
||
`/metrics` без auth. `docs/observability.md` с scrape-конфигом,
|
||
образцом Grafana и alert-rules (HighErrorRate / DbSerializationContention /
|
||
NoSalesIn30Min). 3 интеграционных теста.
|
||
|
||
## Итог
|
||
|
||
**Все 3 пункта выполнены.** Спринт 4 (POS Sync API + observability) завершён 2026-05-28.
|
||
|
||
Сводка:
|
||
- **P1-12a Контракты POS** — `food-market.shared/Pos/V1/SyncDtos.cs`, версионирование
|
||
через namespace, двойная идемпотентность, 3 unit-теста.
|
||
- **P1-12b POS Sync API** — `/api/pos/v1/sync` + `/api/pos/v1/sales` с batch-
|
||
идемпотентностью (`PosBatchAck`) + per-sale (`ClientSaleId` через `Notes`).
|
||
7 интеграционных тестов.
|
||
- **P1-17 Prometheus** — `/metrics`, EF interceptor, бизнес-счётчики,
|
||
`docs/observability.md` с Grafana/alerts. 3 интеграционных теста.
|
||
|
||
**Сборка:** зелёная.
|
||
**Тесты:** 27 unit + 59 integration (49 sprint3 + 7 POS + 3 Metrics) **зелёные**.
|
||
**Web:** не трогали в этом спринте.
|
||
|
||
### POS-side notes
|
||
WPF/POS UI (`food-market.pos`, net8.0-windows) собирается только на Windows
|
||
с Windows SDK — оставлен как отдельный шаг на Windows-машине. Сервер
|
||
готов принимать кассу через `/api/pos/v1/*`.
|
||
|
||
### Что не сделано в спринте (требует внешних участников)
|
||
- ОФД-интеграция (нужен оператор данных — `Транском`/`Касса24`);
|
||
- MoySklad webhook-токены (production-grade приёмка нотификаций);
|
||
- Прод-деплой stage→prod (нужен пользователь для проверки stage сначала);
|
||
- WPF/POS UI (Windows-машина с SDK).
|
||
|
||
## Лог
|
||
|
||
- Каждый пункт: build + тесты + коммит порцией + отметка [x] + коммит прогресса.
|
||
- Все правки на `main` (origin Forgejo), без коммита `global.json`.
|