Проблема: VISUAL CONCEPT и STYLE из COVER_STYLES противоречили друг другу
(sculptor on marble + frosted glass style = модель рисовала свой дефолт)
Решение: 3 простых параметра на статью без конфликтов:
SUBJECT: что изображено (из getVisualMetaphor)
SETTING: антураж (12 вариантов по articleId % 12):
oak desk | marble | slate | workbench | velvet | dawn mist |
terracotta | city night | concrete | library | frost | brick
LIGHTING: конкретный свет (golden hour, studio, rim light, etc.)
COLOR TEMPERATURE: warm amber / cool whites / etc.
Убраны STYLE/PALETTE/MOOD/COMPOSITION блоки которые путали модель.
Теперь каждая статья = уникальная физическая сцена с конкретным светом.
COVER_STYLES: 4 → 12 стилей (amber-terrain, violet-gradient, monochrome-sharp,
coral-horizon, neon-circuits, blueprint-tech, glass-morphism, retro-wave,
zen-minimal, data-cosmos, editorial-ink, teal-architecture)
Теперь стиль повторяется раз в 12 статей (было каждые 4)
buildCoverPrompt(): новая структура промта:
- VISUAL CONCEPT: тематическая метафора из getVisualMetaphor()
- STYLE/PALETTE/MOOD/COMPOSITION: стиль из ротации
Промт явно ставит концепцию первой → картинка отражает тему статьи
getVisualMetaphor(): 10 тематических категорий (cybersec, AI, automation,
data, deepfake, code, marketing, email, vector/rag, prompt engineering)
+ 10 универсальных метафор. Детерминированный выбор по хешу заголовка.
covers.js: generateCoverViaRouterAI принимает quality='medium' по умолчанию
postImages.js: quality='low' для постов TG/VK (₽0.25 vs ₽0.84)
Экономия 70% на генерации картинок к постам
- channel_style.image_rubrics (JSONB): 6 рубрик для ZeroPost-блога
(tech-photo, 3d-device, code-screen, data-flow, ai-neural, cinematic-tech)
- selectRubric(): haiku выбирает рубрику по заголовку+тегам статьи
- generateCover(): загружает rubrics из БД, вызывает selectRubric перед генерацией
- buildCoverPrompt(): принимает rubric — рубрика задаёт весь визуальный язык
- Убраны лишние ограничения (no circuit boards, no glowing nodes, no brains)
из базового промпта — теперь только: no text, no logos, no real faces
covers.js lines 156, 220: generateCoverViaResponses и generateCoverViaImagesEndpoint
используют config.ai.baseUrl (aiprimetech) — исправлен ключ imageApiKey → apiKey.
До разделения провайдеров оба ключа были одинаковыми, поэтому не замечалось.
postImages.js line 99: /responses через aiprimetech — аналогичный фикс.
Обложка статьи 50 перегенерирована вручную (была SVG-заглушка).
covers.js:
- buildCoverPrompt() принимает channelStyle: использует image_style из
канала (abstract/3d-render/minimal/etc.) вместо дефолтного ротационного
COVER_STYLES. image_palette и image_custom_colors перекрывают цвета.
image_prompt_instructions добавляется как Channel visual guidelines.
- generateCover() принимает channelId, загружает channel_style из БД.
postImages.js:
- image_prompt_instructions добавляется в промпт постовой картинки.
articles.js:
- generateCover вызывается с channelId=1 (системный блог-канал zeropost.ru).
services/channels.js:
- updateChannel whitelist расширен: добавлены image_enabled, image_style,
image_palette, image_custom_colors, image_prompt_instructions.
Раньше эти поля молча игнорировались при PATCH канала.
DB:
- ALTER TABLE channel_style ADD COLUMN image_prompt_instructions TEXT;
- Системный канал id=1 получил хорошие дефолты: style=abstract,
palette=dark, instructions=Modern tech editorial blog cover...
* config/index.js: добавлен reloadAi() — асинхронно подтягивает значения
из app_settings (категория ai_providers), мутирует config.ai in-place.
Сервисы продолжают использовать config.ai.* синхронно. 3-уровневый fallback:
app_settings.value → process.env (новое имя) → process.env (старое имя) → дефолт.
* index.js: добавлен await config.reloadAi() в startup после migrate().
Добавлен middleware AsyncLocalStorage для проброса service/userId в
AI-сервисы. Сервис определяется по URL-префиксу (zeropost-blog vs
zeropost-tool). Подмонтирован роут /api/usage.
* routes/settings.js: PUT и invalidate вызывают config.reloadAi() после
изменения настройки категории ai_providers — горячая перезагрузка
без рестарта PM2.
* routes/usage.js (новый): GET /api/usage/summary?range&group_by — сводка
расходов с разбивкой по сервису/провайдеру/модели. GET /api/usage/recent
— последние вызовы.
* lib/aiContext.js (новый): обёртка над AsyncLocalStorage для service/userId.
* services/aiUsage.js (новый): log() с расчётом cost_rub по справочнику цен
Anthropic/OpenAI (USD/1M токенов или USD/картинку) × markup × usd_rub.
Никогда не бросает наружу — ошибки логирования не валят генерацию.
* services/ai.js: chat() и image() обёрнуты в try/catch с aiUsage.log().
Все вышестоящие функции (generatePost, transformPost, generateTopics,
generateArticle) идут через chat() — покрытие 100%.
* services/covers.js: 3 call sites хукнуты (generateCoverViaResponses,
generateCoverViaImagesEndpoint, generateCoverViaImageGenerations). Заодно
поправлен process.env.AI_MODEL_IMAGE_VIA_RESPONSES → config.ai.imageModelViaResponses.
Удалён мёртвый код после throw.
* services/postImages.js: 1 call site (/responses + image_generation tool).
Тот же fix env-ref.
* services/articleAutoSeries.js: 1 call site (/messages, Anthropic-формат).
usage.input_tokens / output_tokens корректно парсится.
Verify:
* Все 11 строк app_settings ai_providers заполнены значениями из .env.
* Горячая перезагрузка: PUT /api/settings/admin/AI_TEXT_MODEL_POST -> runtime
обновился без рестарта PM2.
* Live test: ai.chat() через aiprimetech, 118+49 токенов, 0.0414 ₽ — запись
в ai_usage с правильным service/provider/model.
* GET /api/usage/summary возвращает корректные totals + breakdown.
- Новый основной источник: api.aiguoguo199.com (gpt-image-1-mini)
- Убран response_format (не поддерживается aiguoguo)
- Поддержка b64_json и url response форматов
- Цепочка: aiguoguo → aiprimetech /responses → legacy → local SVG
- AI_IMAGE_BASE_URL и AI_MODEL_IMAGE в config
- Персонаж Зеро: 23 позы (zeroCharacter.js), скрипты генерации
- Auto-publish статей в TG: multipart upload, кнопки, режим alternating Zero/cover
- Fallback цепочка обложек: aiprimetech gpt-5.5 → Pollinations → local SVG (6 палитр)
- Auto-series: Claude haiku определяет серию для каждой статьи автоматически
- Channel stats: подписчики, история, delta 24h/7d
- Photo-search: Yandex API, профили доменов, Redis лимиты
- Scheduled posts runner: backfill, preview, queue, cancel
- promptBuilder: author_persona Зеро, голос от первого лица
- Fixes: dollar-placeholder bugs в PATCH channels/autogen, listArticles фильтры
- AI model: gpt-5.5 для image generation
Старый endpoint /v1/images/generations на gpt-image-* возвращает temporarily unavailable
уже несколько часов, а тот же ключ через /v1/responses на GPT-5 успешно генерирует картинки.
- covers.js полностью переписан: generateCoverViaResponses как основной путь
- tool_choice: image_generation — заставляем модель ВСЕГДА вызывать инструмент
- wrappedInput: явная подсказка чтобы GPT не отвечала текстом
- legacy fallback: если /responses упал — пробуем старый /v1/images/generations
- sharp оптимизация: оригинал PNG → WebP 1600w q84 (уменьшение в ~30 раз)
- timeout до 5 минут — GPT-5 с reasoning + image это долго