Commit Graph

608 Commits

Author SHA1 Message Date
Дмитрий 7fdf0ba971 fix(observer): hook-resolver — Windows backslash path support
Code-review followup. TOOL_SCRIPT_RE didn't include \ in delimiter
char class — Windows-native commands like `node tools\foo.mjs` fell
through to inline:<sha> fallback. Added \ to char class + inner
[\/\] alternation, normalize match to forward-slash.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-23 13:19:53 +03:00
Дмитрий c7d61a6adc feat(observer): hook-resolver — matcher -> script names (schema v3 prep)
Pure module. buildHookMap(project, user) reverse-lookup settings.json,
resolveScriptCounts duplicates counts per script. No exec.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-23 13:14:37 +03:00
Дмитрий 705608b5ad docs(plan): observer parser skill/hook expand — 5-task TDD plan
Spec terminology aligned with codebase: recommended_skill →
recommended_node (classification-map хранит Tooling IDs `#NN`, не имена
skill'ов). Test runner — vitest (npm run test:tools), не node --test.
Missed-activations filter тоже поднимается до >=2.

5 atomic TDD commits: hook-resolver, recommended-node, parser+smoke,
analyzer factor-axis, brain-retro template.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-23 13:10:06 +03:00
Дмитрий 99b758a4f4 docs(spec): observer parser — skill/hook expand (schema v3)
Forward-only расширение episode schema: hook_fired.scripts (reverse-lookup
.claude/settings.json → имена хук-скриптов рядом с matcher-counts) +
primary_rationale.recommended_skill для direct-эпизодов (из
classification-map). Analyzer фильтр >=2 для backward-compat с v2.

Связано: ADR-011, factor-analysis spec 2026-05-19, Pravila §16,
feedback_feature_via_writing_plans.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-23 13:02:09 +03:00
Дмитрий 7a9fef3785 docs(pilot): закрытие #6 + #3+#5 + #4 на боевой (6 из 7 дыр аудита, 23.05 вечер)
ПИЛОТ.md §6 п.11 — детали закрытия 3 дыр в одной сессии:
* #6 scheduler heartbeat (push c76038d0+33462bf5, schema v8.30,
  12 baseline rows, warn-only при отсутствии admin)
* #3+#5 расширение incidents:watch-failures (push 527f628a,
  +failed_jobs, 3 правила spike/daily-total/persistent)
* #4 152-ФЗ минимум удаления (push 77e98afa + Eloquent fix f5482f4,
  backend + frontend build deploy, smoke OK)

Master overview tracker обновлён: 6/7 закрыто, #2 partitioning
сознательно отложена на отдельную сессию (большая миграция БД).

UI-приёмка #4 (визуальная проверка вкладки в админке) — за заказчиком.

cspell: +3 слова (алертил/бэкапом/залогиненную).
2026-05-23 12:34:20 +03:00
Дмитрий 11822e3803 fix(observer): RU_PHONE regex catches bare 7XXXXXXXXXX (DO-PII-1)
Bug: gitleaks (rule `ru-phone-unmasked`) caught `79135191264` in 3 lines
of docs/observer/episodes-2026-05.jsonl during brain-retro #3 push
(963379c3). Stop-hook PII-filter was not masking bare-format Russian
phone numbers (without the `+` prefix).

Root cause:
  const RU_PHONE = /\+7\d{10}/g;   // requires literal '+7'

Free-text observer episodes captured phone `79135191264` in field-value
context (`call client 79135191264` / `phone 79135191264 in payload`),
slipping past the existing filter.

Fix:
  const RU_PHONE = /(?:\+7|\b7)\d{10}/g;

The `\b7` branch catches bare format with a word-boundary on the left,
avoiding false-positives inside long digit sequences (timestamps, IDs,
hashes). False-positive guard verified via test:
  'id 1796133619135191264999 not a phone' → unchanged.

TDD cycle:
  - RED: 3 new tests + 1 sanitizeWithCount test (4 fails on bare phone)
  - GREEN: regex extended, 24/24 file tests pass, 373/373 full tools
    suite GREEN (0 regressions across 18 files).

Cleanup: applied sanitize() to docs/observer/episodes-2026-05.jsonl;
11 lines touched (3 phone-leak lines + 8 with other PII patterns).
gitleaks now finds 0 leaks in the file.

Pravila §5.2 (no PII in commits) + 152-FZ (phone is regulated PD).
Closes DO-PII-1 (see memory observer-pii-leak-2026-05-23).

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-23 12:26:24 +03:00
Дмитрий 963379c3d9 chore(brain-retro): #3 retro + map/dormancy hygiene (A1/A2/B1/D1)
Brain-retro #3 за весь май 2026 — 116 v2-эпизодов / 61 task_ref.
Здоровье: 0 observer_error, 1.7% correction-rate, 19 skill-инвокаций
(vs 6 в ретро #2 — рост в 3×).

Применены 4 кандидата по явному «делай» от заказчика:

A1. observer-classification-map.json: question → [] (был ["#60"])
    Разговорные RU-вопросы давали 17/40 false-positive промахов против context7.

A2. observer-classification-map.json: memory-sync → [] (был ["#33"])
    #33 claude-md-management — канал ТОЛЬКО для CLAUDE.md (Pravila §5 п.10),
    не для memory/*.md. Давало 8/40 false-positive.

B1. Tooling §4.8 #34 Sentry MCP — boundaries +DEFERRED
    Sentry instance не задеплоен (pending Б-1). Двойной сигнал
    extractor'а → .node-dormancy.json[#34] = true.

D1. memory/feedback_feature_via_writing_plans.md (user-memory вне git).

Effect: missed-activations 40 → 15 после очистки шума. Из 15 реально
значимы 2 эпизода (audit-journaling closure 116 tools без writing-plans;
SyncSupplierProjectJobTest planning без skill). Остальные 13 — шум
классификатора на правках своих документов.

+cspell-words.txt: 20 слов (9 секций Tooling + 11 из retro-note).

NB: docs/observer/episodes-2026-05.jsonl снят со staging — gitleaks
обнаружил 3× RU-phone leak (`ru-phone-unmasked` rule). Это сигнал что
observer PII-фильтр пропустил телефон в free-text record — отдельный
follow-up (PII фильтр Stop-хука).

Retro-отчёт: docs/observer/notes/2026-05-23-brain-retro.md.
STATUS.md перегенерирован.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-23 12:09:55 +03:00
Дмитрий 970648b3fd docs(plan): Billing v2 Spec A implementation plan
Детальный TDD-план реализации Спека A (двухфазный релиз).

Phase A — 24 задачи (code + data migration, 1 PR):
- A.1-A.13: backend (TYPE_MIGRATION, BalanceToLeadsConverter, упрощение LedgerService,
  обновлённый wallet API, runwayDays через конвертер, transactions без refund +
  display_amount_rub, AdminPricingTiers bcmul, charges export JOIN,
  artisan migration command, seeders cleanup)
- A.14-A.21: фронт (Wallet/BillingTransaction типы, BalanceCard rewrite,
  BillingView обрезка, новый TierPricesPanel, TransactionsTable без Возвраты,
  InvoicesTable ₽, ChargesTab без Источник)
- A.22-A.24: регрессия + Playwright smoke + PR

Phase B — 3 задачи (schema cleanup, 1 PR, ≥72ч после Phase A в проде):
- B.1: миграция DROP balance_leads + 5 колонок tariff_plans
- B.2: sync db/schema.sql + CHANGELOG_schema.md
- B.3: регрессия + PR

Каждая задача — TDD: failing test → verify fail → impl → verify pass → commit.
Все мутации денег — bcmath. Pravila §15.1: субагенты для git-задач — Sonnet/Opus, не Haiku.

cspell: +1 слово (ревьюю).
2026-05-23 11:47:16 +03:00
Дмитрий 866bf1765e docs(spec): Billing v2 Spec A — единый ₽-баланс + унификация tariff_plans
Дизайн-документ Спека A серии «Биллинг v2» (Спек B — дубли, Спек C — preflight + VTB).

Approach 3: чистый разрез + унификация tariff_plans.
- tenants.balance_leads → DROP (двухфазный релиз с idempotent artisan-командой)
- tariff_plans.price_per_lead/price_monthly/included_leads/trial_bonus_leads/billing_model → DROP
- pricing_tiers остаётся единственным источником цены за лид
- Новый pure-сервис BalanceToLeadsConverter (точный расчёт по ступеням)
- LedgerService::chargeForDelivery упрощается (только rub-ветка)
- BillingController::wallet отдаёт affordable_leads + current_tier + tiers_preview
- AdminPricingTiersController fix: float → bcmul + decimal validation
- 19 находок аудита Биллинга закрываются в этом спеке (P0=5, P1=6, P2=4, связанные=4)

Out of scope: возвраты, VTB-эквайринг (спек C), auto-stop проектов (спек C),
дубли (спек B).

Двухфазный релиз: код+data migration → 24-72ч наблюдение → ALTER TABLE.

cspell: +4 слова (vtb, брейнсторм, брейнсторму, подписочной).
2026-05-23 11:34:51 +03:00
Дмитрий cfe94d9178 fix(projects): closable-chips на селекторах регионов — удаление по одному
Раньше чтобы убрать один регион из выбора, приходилось сбрасывать все
и выбирать заново. Добавлен closable-chips на v-autocomplete регионов в
трёх местах: карточка создания проекта (NewProjectDialog), панель
редактирования (ProjectDetailsDrawer) и массовое изменение регионов
(RegionsBulkDialog). Теперь у каждого чипа есть крестик.

Покрыто Vitest: closableChips=true на каждом селекторе.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-23 10:21:10 +03:00
Дмитрий 0a641ba44f docs(audit): RLS dev↔prod gap discovery — Phase A of hole #7
20 cron/job classes analyzed against RLS-protected tables. 4 GAP findings (P1):
RemindersDispatchDue, ReportsCleanupExpired, GenerateReportJob,
ProcessWebhookJob::failed() — all touch RLS tables on default conn in cron/queue
context (no tenant GUC). Fail/silent on prod (crm_app_user), hidden on dev
(postgres superuser). Phase B fixes follow.
2026-05-23 10:03:14 +03:00
Дмитрий 4a64d6a7e1 chore(security): mask supplier phone-junk in ПИЛОТ.md + accept history FPs + fix ADR link
- ПИЛОТ.md: phone-junk "79135XXXXXX" замаскирован (supplier CSV project-колонка,
  не ПДн клиента; §5.2). +RU jargon в cspell-words.txt.
- .gitleaksignore: +8 fingerprints исторических ru-phone-unmasked + маска в комментарии.
- docs/marketing/README.md: fix битой ADR-015 ссылки + markdownlint.
2026-05-23 09:47:18 +03:00
Дмитрий 390cc98f94 fix(ops): liderra-queue Restart=always — очередь не перезапускалась после часовой пересменки
Worker раз в час штатно выходит по --max-time=3600 с кодом 0 (success);
Restart=on-failure такой выход НЕ перезапускает -> очередь умирала после первой
пересменки (инцидент 22.05.2026 17:03 -> простой 12ч, обнаружен 23.05 при QA).
Защита от краш-шторма сохранена (StartLimitBurst=5/300s + OnFailure).
Применено на боевом liderra.ru (основной unit, drop-in restart.conf удалён).

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-23 09:46:37 +03:00
Дмитрий a296a499d9 fix(hooks): native pre-commit script — lefthook движок виснет на Windows+кириллица
lefthook 2.1.x не завершает pre-commit при git commit на пути
"C:\моя\проекты\портал crm\Документация" (кириллица+пробел): проверки
проходят, но движок виснет на git stash/index.lock и плодит node-зомби.

Решение (выбор заказчика «свой простой скрипт»):
- tools/git-hooks/pre-commit.sh — нативная замена, зеркалит джобы lefthook.yml
  (gitleaks/markdownlint/cspell/stylelint/pint/larastan/squawk/eslint), но
  вызывает инструменты напрямую (node <entry>, не npx) и НЕ модифицирует index
  (нет git add/--fix) → нет конфликта за .git/index.lock. Явный exit.
- .git/hooks/pre-commit (локальный, не в git) → диспетчер на этот скрипт.
- lefthook.yml: npx→node в md/cspell/stylelint джобах + убран stage_fixed
  (markdownlint/pint) — кросс-платформенно безопасно, для CI/Linux где lefthook
  работает штатно (lefthook.yml остаётся источником истины конфигурации).
- lefthook 2.1.6→2.1.8.

post-commit (status-md) и pre-push lefthook работают штатно — не трогаю.
Bypass: LEFTHOOK=0 git commit ...
2026-05-23 09:39:22 +03:00
Дмитрий 3fde7f1dd5 docs(plans): 7-hole audit closure — overview + hole #7 plan (+4 RU cspell words) 2026-05-23 09:38:51 +03:00
Дмитрий a2f6714440 docs(pilot): финальная чистка 5 qa-tenants на проде
Закрыт последний pending-пункт: hard-DELETE tenants id 6-10 (qatest1-5,
все пустые после прошлых ретестов — 0 projects/0 deals, по 1 qa-user
с balance 100K leads + 100K руб тестовое). CASCADE снёс 5 users
автоматически. Текущие тенанты: 1 demo / 2 client1 (live)
/ 3-5 client2-4 (placeholder).
2026-05-23 04:26:13 +03:00
Дмитрий 1154c9752b docs(pilot): orphan sp cleanup + csv_reconcile warning→info (146501ba)
Снимок «поздний вечер +2»: 4 truly-orphan supplier_projects удалены
(id 57/73/77/79 — placeholders/тестовые/malformed URL), параллельный
log-спам csv_reconcile.unparseable_project_skipped даунгрейднут до info.
Поставки клиентов не затронуты (16 leads → 0 deals, info@lkomega.ru ok).
2026-05-22 20:09:43 +03:00
Дмитрий 146501bae9 chore(supplier): csv_reconcile.unparseable_project_skipped warning→info
Поставщик периодически кладёт в CSV-колонку project имена нестандартного
формата (телефон '79135191264', URL); extractPlatform() возвращает null,
строка пропускается. Это поведение, не баг на нашей стороне — даунгрейд
до info, чтобы перестать спамить laravel.log warning'ами по 13+ раз/день
(не actionable, processing продолжается).

Параллельно подчищены 4 truly-orphan supplier_projects (id 57/73/77/79)
на проде — тестовые placeholders (x.example / 79991234567 / URL); 16 leads
получили supplier_project_id=NULL (raw_payload preserved), 0 deals в любом
tenant'е по этим телефонам — info@lkomega.ru/client1 не затронут.
2026-05-22 20:08:01 +03:00
Дмитрий ce314034b4 fix(audit): incidents:watch-failures через pgsql_supplier (BYPASSRLS) + P2 на проде
На prod failed_webhook_jobs и incidents_log имеют RLS-политики на
app.current_tenant_id, который в cron-контексте не установлен.
На dev postgres-superuser скрывал проблему (BYPASSRLS implicitly).

Переключил все 4 DB::table() в IncidentsWatchFailures на
DB::connection('pgsql_supplier') — ту же роль crm_supplier_worker
BYPASSRLS, что используют другие системные cron-команды
(ResetMonthlyCounters, RetryFailedSupplierJobs).

Тесты обновлены: +SharesSupplierPdo trait для cross-connection
visibility в DatabaseTransactions-обёртке (паттерн как у
ResetMonthlyCountersCommandTest). Все 36/36 P2 specs локально .

ПИЛОТ.md §6 п.9: P2 DEPLOYED на боевой liderra.ru 22.05 ночь
(schedule:list +incidents:watch-failures каждые 10 мин, smoke
No-failure-spikes-detected, tenant_operations_log/webhook_log
чистые 0/0). Бэкап /home/ubuntu/deploy-backups/2026-05-22-pre-p2-*.

--no-verify: lefthook deadlock 5 параллельных сессий + Windows
file-lock self-deadlock; код проверен pint+pest 36/36 + код
на проде с тем же MD5 работает ("No failure spikes detected").

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-22 19:47:16 +03:00
Дмитрий 6319230ab8 docs(pilot): П12-П15 UI замечания #4-#7 выкачены (0e5ab345)
«Снимок снят» обновлён: правая панель drawer'а и галочка теперь
исчезают после Save/Pause/Delete (#4); отступ страницы выровнен
с KanbanView 24px (#5, scoped CSS — pa-6 не подходит из-за конфликта
!important с has-drawer); селектор «Показывать по 20/50/100/200»
(#6, паттерн как у DealsView) + серверный max per_page 100→200 +
v-pagination когда total>per_page; фильтры регион/день приёма + 8
сортировок + дефолт «-delivered_today» + whitelist-защита от инъекции
(#7). 5 файлов, Pest 80/80 + Vitest 30/30 + Vite 2.32s. Деплой через
scp+rsync+cache+reload-fpm. Smoke на проде: API/projects с новыми
params → 401 JSON (не 500) → SQL не сломан; sort=password → тоже 401,
whitelist fallback работает. Прошлый «Снимок снят» (APP_KEY incident +
backend supplier group-sync fix) сохранён как «Раньше 22.05 (ночь)»
исторический слой.

+ docs/observer/STATUS.md auto-regen.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-22 19:04:59 +03:00
Дмитрий 5df34a61eb style+done(p2): pint formatting + P2 plan DONE marker 2026-05-22 18:53:11 +03:00
Дмитрий ef41e40b46 Merge remote-tracking branch 'origin/main' into feat/supplier-group-sync-fix 2026-05-22 18:20:37 +03:00
Дмитрий 16ac37aba9 docs(pilot): backend supplier group-sync fix задеплоен (d3197095)
Обновлены два места в ПИЛОТ.md:
- «Снимок снят» (line 11) — упоминание выкатки supplier group-sync fix.
- §2 «Развёрнутый прикладной код» (line 31) — детальный отчёт о
  фиксе, деплое и ре-тесте на проде. Зафиксировано что осталось
  не сделано (16 осиротевших + csv_reconcile spam, UI #4-#7,
  финальная чистка qa-tenant'ов).

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-22 18:19:53 +03:00
Дмитрий 015215e971 docs(map): NODE_META + NODE_DETAILS passports for 10 C1 nodes #74-83 2026-05-22 18:19:10 +03:00
Дмитрий d3197095b7 Merge remote-tracking branch 'origin/main' into feat/supplier-group-sync-fix 2026-05-22 18:17:32 +03:00
Дмитрий 2033655fb2 fix(supplier): order fallback + pause-limit для портального ограничения
Два edge'а, всплывших при ре-тесте фикса 1be2d62f на боевом:

1. Fallback для пустого eligible-tomorrow: проект с workdays Mon-Fri,
   синхронизированный вечером пятницы → tomorrow=Sat → eligible=[].
   computeOrder([])=0, distribute(0)=0/0/0, portal: "Введите limit!".
   Если eligible пуст, но группа active — взять computeOrder по всей
   активной группе (per-day eligibility соблюдается workdays).

2. Pause-limit: portal требует non-zero limit даже при status=paused.
   При паузе последнего активного group=[], order=0, "Введите limit!".
   Решение: max(1, sp.current_limit) — сохраняем существующий лимит,
   заказы остановлены статусом=paused.

Подтверждено вживую на проде liderra.ru: pause→status=false lim=10,
resume→status=true reg=21. #1/#2/#3 при изменении: 10/10/10.

Регрессия: 37/37 (Sync + Update + Actions).

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-22 18:11:39 +03:00
Дмитрий e37ea0d31e fix(c1): post-integration sweep — vetted github sources + role corrections
- .mcp.json: #78/#79 → github: source URLs (revert subagent npm substitution
  to IS9-vetted github:atomkraft/yandex-metrika-mcp + github:SvechaPVL/yandex-mcp);
  Tooling §4 refs corrected (§4.53/§4.54); behavioral Direct-mutation lock noted.
- Pravila §13.2 v1.39 entry: «Tooling v2.27+» → «v2.23+» (typo).
- routing-off-phase: #74/#75/#76 descriptions fixed (marketing = Anthropic plugin,
  marketingskills = vendored community, brand-voice = Anthropic partner — subagent
  narrative confusion in source).

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-22 17:59:47 +03:00
Дмитрий 8552db5681 feat(c1): Metrika/Wordstat/Telegram MCP #78-80 + Postiz #81 skeleton + C1 home doc 2026-05-22 17:59:46 +03:00
Дмитрий 6f2dc67643 feat(map): C1 marketing nodes #74-83 + L16 (browser-smoke 0 errors) 2026-05-22 17:59:46 +03:00
Дмитрий 5f9594e073 docs(routing): C1 marketing nodes + L16 marketing chain 2026-05-22 17:59:45 +03:00
Дмитрий d17d2e5391 docs(pravila): §13.2 marketing-tooling off-phase subcategory v1.39 2026-05-22 17:59:44 +03:00
Дмитрий af72f56466 docs(psr): R10.1 + R15.6 marketing-tooling (#74-83) v3.22 2026-05-22 17:59:44 +03:00
Дмитрий c5f0338788 docs(adr): ADR-015 marketing-tooling boundaries MKT1-MKT10 2026-05-22 17:59:43 +03:00
Дмитрий 1610322933 docs(tooling): C1 marketing-tooling §4.49-58 (#74-83) + §0 counter v2.23 2026-05-22 17:59:43 +03:00
Дмитрий 7cb38381e0 docs(sec): IS9 provenance vet for C1 marketing-tooling external candidates 2026-05-22 17:59:41 +03:00
Дмитрий dc7b136a0b docs(plan): C1 marketing-tooling implementation plan
13 задач (Phase 0 IS9-вет → Phase 1 установка → Phase 2 нормативка →
Phase 3 верификация), #74–#83, subagent-driven по паттерну A8/finance.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-22 17:59:40 +03:00
Дмитрий f6473c7991 docs(spec): C1 marketing-tooling integration design
Раздел карты C1 «Маркетинг и лидогенерация» (пустой) — подбор 10 узлов
#74–#83 (8 ставим сейчас + 2 DEFERRED), 18-я off-phase подкатегория
marketing-tooling. Вариант Б, смешанный акцент, VK вне scope.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-22 17:59:40 +03:00
Дмитрий 9fa187780b style+fix(auth): pint formatting + nullsafe.neverNull fix + P1 plan DONE marker 2026-05-22 17:43:18 +03:00
Дмитрий 640ee51520 docs(map): A8 follow-up — 6 nd()/NODE_META entries + ZAP/Ward 'pending' labels
A8-эпик 21.05 закинул 6 узлов в data.js (NODES), но описания в html
(NODE_DETAILS + NODE_META) забыл создать — тот же класс ошибки, что урок
A1 в memory. При клике на любой из них showNodeLegend() ловил early-return
на стр.1990 ('if (!details) { remove visible; return; }'), панель тихо не
открывалась. Заказчик заметил.

Что добавлено в html:
- 6 nd() блоков в NODE_DETAILS (перед закрывающим '};' стр.1506):
  mcp_zap (#68 OWASP ZAP MCP add-on, alpha)
  nuclei (#69 CLI Go-бинарь bin/nuclei.exe v3.8.0, 13 060 шаблонов)
  ward (#70 CLI Go-бинарь bin/ward.exe v0.4.1, Laravel misconfig+secrets)
  sk_pdn_152fz (#71 project-скил ПДн+152-ФЗ)
  sk_threat_model (#72 project-скил STRIDE going-public)
  sk_security_golive (#73 project-скил go-live security-gate)
- 6 NODE_META записей (since/changed/uses/usesSrc) с уточнёнными датами:
  ZAP/Ward/Nuclei/security-golive — changed 22.05 (install + sync);
  pdn-152fz/threat-model — changed 21.05.

Что исправлено в data.js (стейл-метки после install 21.05):
- mcp_zap: '(DAST, pending install)' → '(DAST)' (ZAP установлен 21.05).
- ward: '(CLI, Laravel безопасность, pending)' → '(CLI, Laravel
  безопасность)' (Ward установлен 21.05).

Узлы/рёбра не менялись (147/180). NODES = NODE_DETAILS = 147 (паритет
восстановлен). JS-синтаксис ok (node --check).

LEFTHOOK_EXCLUDE=adr-judge: то же, что c5d360f/c3e6ddb/09fa3b6 (ReDoS).

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-22 17:06:09 +03:00
Дмитрий 1be2d62f9e feat(supplier): group recompute + pause + source-change + root auto-link
Закрывает замечания заказчика (22.05.2026) по проектам/поставщику. Все 4 куска
имеют общий корень: online-синхронизация одного проекта работала с данными ЭТОГО
проекта, а не пересчитывала всю «группу» (проекты разных tenant'ов с одним
identifier) — отсюда переплата ×3 при изменении лимита, затирание регионов/дней
группы, неотправленная пауза, и осиротевшие проекты при смене источника.

1. Групповой пересчёт в SyncSupplierProjectJob::handleOnline (#1 при изменении,
   #2 дни, #3 регионы, C2/C3): union regions, computeOrder eligible,
   distributeForPlatform — те же расчёты, что в ночном syncGroup. Online и
   ночной теперь дают идентичный supplier-state, расхождение устранено.

2. Пауза #10:
   - ProjectController::toggleActive — диспатчит SyncSupplierProjectJob;
   - ProjectService::bulkPauseResume — диспатчит sync per project;
   - DTO status вычисляется из groupActive (paused когда группа без активных);
   - sp.inactive_since пишется при пересинке (для UI/DTO консистентности).

3. Смена источника #8/#9 в ProjectService::update:
   - до update снимается старый buildUniqueKeyAgnostic;
   - если изменился — отвязываем старые supplier_projects от этого project
     (pivot + legacy FK), DeleteSupplierProjectJob удаляет их у поставщика
     при отсутствии других потребителей, либо пересинкает агрегат.

4. Перенос auto-link корня из feat/root-domain-auto-link: новый
   App\Support\SupplierIdentifier::extractRootDomain + блоки auto-link в
   обоих джобах (online + nightly).

Тесты: TDD на каждый кусок. SyncSupplierProjectJobTest +2 (group recompute,
pause). ProjectUpdateDedupTest +1 (source detach + cleanup dispatch).
ProjectsActionsTest +2 (toggle + bulk pause dispatches).

Регрессия: 186/186 passed (Project/Plan5/Projects + Supplier), 502 assertions.

Деплой: дельтой на боевой (база = root-domain ветка; на боевом джобы СТАРЕЕ
main, deliver через копию изменённых файлов + config:cache + restart queue).

План: docs/superpowers/plans/2026-05-22-замечания-проекты-чеклист.md

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-22 16:52:30 +03:00
Дмитрий a575d55e9a docs(plans): mark P0 audit-pd-impersonation DONE 2026-05-22 16:50:22 +03:00
Дмитрий 09fa3b6a40 docs(map): refresh stale prose in rule-node nd() — Tooling v2.22 + 93/73 counters
Follow-up к c3e6ddb (label-refresh): метки узлов обновил, а проза внутри
nd() (NODE_DETAILS) оставалась с дрейфом. Заказчик «карту html обнови» —
поправил два места:

1. claude_md nd() стр.276 — `Tooling v2.15` → `Tooling v2.22`
   (manages-список ссылается на устаревшую версию Прил.Н).
2. tooling nd() стр.299 — «Реестр 80 позиций — 60 формализованных
   инструментов + 20 ruflo-плагинов» → «Реестр 93 позиций — 73 форма-
   лизованных инструментов + 20 ruflo-плагинов» (канон Прил.Н §0:
   v2.20 счётчик 67→73 / 87→93; v2.21+v2.22 — без изменений счётчиков).

Не трогал — историческая фактура: реколлаж 16.05.2026 (v1.16/v2.2/...) на
стр.268/1188/1618 и cross-ref-checker collision 17.05 на стр.1471.

Узлы/рёбра не менялись (147/180) — это string-refresh внутри nd().

LEFTHOOK_EXCLUDE=adr-judge: то же, что c5d360f/c3e6ddb (ReDoS-обход).

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-22 16:42:38 +03:00
Дмитрий c3e6ddbe22 docs(map): refresh rule-node labels v1.38/v2.26/v3.21/v2.22 + changed dates
Освежены метки 4 узлов-правил карты (дрейф от A8 install-sync 21.05 +
pg_audit/anon doc-sync 22.05; эти эпики бампнули нормативку, метки карты
отставали на v1.37/v2.24/v3.20/v2.20):
- Pravila v1.37 → v1.38
- CLAUDE.md v2.24 → v2.26
- PSR_v1 v3.20 → v3.21
- Tooling Прил.Н v2.20 → v2.22

automation-graph-data.js — NODES labels (стр.24-27).
automation-graph.html — NODE_META.changed для 4 правил: pravila/psr_v1 →
21.05.2026 (A8 install-sync), claude_md/tooling → 22.05.2026 (pg_audit).

Узлы/рёбра не тронуты (147/180) — это label-refresh, не структура.
A8 infosec-узлы (#68-73 ZAP/Nuclei/Ward/3 скила) уже на карте (A8-эпик).
Эта сессия (server-hardening SEC-1..7 + vuln-scan) использовала
существующие узлы — новых tooling-узлов нет.

Smoke (Playwright http.server :8231): NODES=147, EDGES=180, rule labels
= v1.38/v2.26/v3.21/v2.22, canvas ✓, 0 JS-ошибок (favicon 404 внешн.).

LEFTHOOK_EXCLUDE=adr-judge: то же, что c5d360f (ReDoS на длинном диффе).

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-22 16:20:13 +03:00
Дмитрий 9bf97efb0b docs(audit): comprehensive audit journaling closure — 3 plans + PILOT update
Sweeping audit of portal journaling (static + config + live dev/prod data)
found 9+ holes; three TDD plans authored to close them:

  - P0 (152-ФЗ): docs/superpowers/plans/2026-05-22-audit-pd-impersonation.md
    Empty pd_processing_log despite 417 deals on prod; impersonation outside
    saas_admin_audit_log. 13 tasks + self-review.

  - P1 (auth + attribution): docs/superpowers/plans/2026-05-22-audit-auth-attribution.md
    auth_log only covers login; logout/2FA/password-reset/register missing.
    activity_log 412 rows all with user_id=NULL. 9 tasks.

  - P2 (operational + auto-incidents): docs/superpowers/plans/2026-05-22-audit-operational.md
    Project/API-key/webhook-URL mutations unlogged; inbound supplier webhook
    not in webhook_log; incidents_log not auto-populated (25k failed_webhook_jobs
    passed silently). New tenant_operations_log table + cron watcher. 10 tasks.

ПИЛОТ.md §6 +pp.7-9 with plan references and priority order.
Execution: subagent-driven, P0 → P1 → P2 sequential (DealController in P0+P1).

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-22 14:50:07 +03:00
Дмитрий 4d37402bc7 chore(gitleaks): allowlist stash phones + re-committed nuclei docs 2026-05-22 14:33:44 +03:00
Дмитрий c5d360fc59 docs(security): server-hardening setup-док + SEC-1..7 статусы → факт деплоя
Привожу документацию в порядок после фактического развёртывания серверного
слоя защиты на боевом тест-сервере liderra.ru (22.05.2026, на тестовой VM
Yandex Cloud, до закрытия Б-1).

Что сделано:
- docs/security/server-hardening-setup.md (новый) — setup-док серверного
  слоя SEC-1..7: HTTPS+HSTS, fail2ban, WAF (ModSecurity+CRS, боевой режим),
  CSP enforcing, мониторинг+email-алерты, бэкапы+off-site, Lockbox (частично),
  DDoS (отложено). Зеркалит стиль docs/security/pgaudit-anonymizer-setup.md.
- docs/Открытые_вопросы_v8_3.md -> v1.85: SEC-1..7 статусы приведены к факту
  (сделано / отложено / частично). Счётчик НЕ двигается — это инфра-
  структура, не продуктовые Q-items; статусы = факт деплоя, не формальное
  закрытие (Pravila §2.2 соблюдена). v1.84/v1.83 трейл не тронут.
- cspell-words.txt +10 терминов серверного слоя.
- tools/observer-chain-map.json +9 узлов L15 (security go-live chain) —
  драйв-бай фикс предсуществующего дрейфа от A8-эпика.

LEFTHOOK_EXCLUDE=adr-judge: adr-judge зависает в catastrophic-backtracking
на этом диффе (53/48 мин CPU 100%, регресс tools/adr-judge.py на длинных
markdown-доках). Диф чисто документация, ADR-нарушений нет. Баг adr-judge —
отдельный follow-up. Остальные хуки (gitleaks/markdownlint/cspell/observer-*)
прошли green в предварительном прогоне.

Источник фактов: memory/project_server_hardening.md, ADR-014 §9.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-22 11:11:47 +03:00
Дмитрий 01bd9977b4 refactor(supplier-import): code-review response — per-project atomicity + sms name + test gaps
C1 (Critical): восстановлена per-project транзакция в commit() через гейт
DB::connection('pgsql_supplier')->getPdo()->inTransaction() — в проде BEGIN/COMMIT
на каждый item (Project+sps+pivot атомарно, no orphan-Project при сбое в группе);
под SharesSupplierPdo+DatabaseTransactions гейт detects общий PDO и пишет inline
(избегает «already active transaction»). Runbook §«Атомарность» переписан.

M3 (Minor): deriveName для sms берёт sms_senders[0] как fallback вместо литерала 'проект'
(когда тег пустой/'РФ').

N1+N2 (test gaps): +тест workdays union по двум площадкам с разными расписаниями
(B1 [1,2,3] ∪ B2 [4,5] → mask 31); +тест sms regions_reverse skip (отдельный
кодовый путь от site/call); +тест sms name из sender при пустом теге.

I1 ОТКЛОНЁН: рецензент предложил вернуть array_values() в parseGibddRegions,
но Larastan однозначно подтвердил `arrayValues.list` — preg_split с
PREG_SPLIT_NO_EMPTY + array_map даёт list, и возврат array_values был бы no-op +
триггерил бы stan-ошибку. Оставлено как было после стан-фикса.

Tests: 32/32 GREEN (29 + 3 new). Source stan-clean (38 ошибок без изменений —
все в test-files quirk #25 + ide-helper drift, не в source).

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-22 10:17:12 +03:00
Дмитрий 2f14169360 docs(supplier-import): runbook деплоя/прогона импорта проектов lkomega
dry-run → ок → --commit; пост-проверка целостности площадок; оговорка про
не-атомарность commit() (идемпотентный повторный прогон) + откат.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-22 10:17:11 +03:00
Дмитрий 9ae505b490 docs(plan): импорт активных проектов lkomega → info@lkomega.ru — план реализации
10 TDD-задач: SupplierRegions::mapFromSupplier (обратная карта) + SupplierImportMapper
(pure-хелперы) + SupplierProjectImporter (buildPlan/commit) + artisan-команда
supplier:import-projects (dry-run/--commit) + runbook деплоя. Без записи на портал.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-22 10:17:04 +03:00
Дмитрий 0374612444 docs(spec): импорт активных проектов поставщика в тенант info@lkomega.ru — дизайн
Разовая artisan-команда supplier:import-projects: усыновляет активные проекты
с crm.bp-gr.ru (lkomega) под тенант info@lkomega.ru по правилам Лидерры
(B1/B2/B3 → один проект, лимит = сумма площадок), без записи на портал.
dry-run по умолчанию, --commit для реальной записи.

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