dvr_admin/dvr_admin_docker
deeily 8b47442251 dvr_admin v2.0 (testing): H.265+ Smart Codec со Stream Encryption
Первая волна: live preview / main view / playback archive для Hikvision
DVR с user-заданным Encryption Key (Live View Parameters). Работает на
нативной libPlayCtrl.so без wine в runtime.

### Добавлено
- Live 16-канальный грид (2 fps), main view (12 fps), playback archive
  через `/SDK/play` / `/SDK/playback` HTTP protocol + libPlayCtrl.so
- Pull-модель PlayM4_GetJPEG вместо SetDecCallBack (callback'и крашили
  под нагрузкой из-за GIL race в 16 native decoder threads)
- Session reconnect loop (DVR рвёт /SDK/play ~30 сек) + heartbeat
  /ISAPI/Security/sessionHeartbeat + auto-recovery на 401
- UI sync часов к видео-PTS через PlayM4_GetSystemTime + /pos endpoint,
  _pbWall re-peg каждые 2 сек. Оба template'а (fullscreen + modal)
- Pure Python KDF `_hik_stream_kdf_128(user_key)` — реверс NetStream.dll
  через Ghidra headless. K1=6a68a361bf6eb567, K2=cd7afe68ca6fde75,
  cipher D(K1)-swap-E(K2)-swap-D(K1) на 16-байт блоки. Финальная формула
  `user_key_ascii + derived16 × 7` (константа для любого ключа)
- Spy DLL под wine + Ghidra reverse engineering workflow

### Исправлено
- UI часы в архиве тикают на всех скоростях (убрано `if(_pbRate!==1)`
  в _startSeekTimer + добавлено обновление m-status в модальном template)
- Сервер больше не падает в 16-канальном гриде encrypted (убраны
  per-frame ffmpeg forks, pull-модель JPEG)

### TODO (остаются в testing до фикса)
- Playback speed ×2/×4/×8 — протокол захвачен, код применён
  (PlayM4_Fast + dynamic GETJPEG_INTERVAL), ожидает теста
- Clip download encrypted — NET_DVR_SetPlayBackSecretKey returns FALSE,
  план B: post-factum decrypt через pure Python cipher

### Docker
- gunicorn --workers 1 --threads 16 (single worker обязательно для
  libPlayCtrl thread safety)
- _DVR_LD_SET=1 env чтобы skip self-exec в контейнере
2026-04-16 03:33:50 +03:00
..
2026-04-05 21:44:40 +03:00
2026-04-07 12:26:38 +03:00