Commit Graph

24 Commits

Author SHA1 Message Date
Nik (Claude) 5b5f703078 fix: avoid repeating last 3 rubrics in cover selection (no more similar covers) 2026-06-19 01:09:52 +03:00
Nik (Claude) f9d1deae58 fix: initialize usedPath in covers.js to prevent ReferenceError 2026-06-16 22:58:28 +03:00
Ник (Claude) 2360e1f7ae fix: cover images — simplified coherent prompts, no style conflicts
Проблема: 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 блоки которые путали модель.
Теперь каждая статья = уникальная физическая сцена с конкретным светом.
2026-06-13 09:29:40 +03:00
Ник (Claude) 170c7b7b16 fix: cover image variety — concrete metaphors + articleId cycling
getVisualMetaphor():
- articleId % array.length как чистый индекс цикличности
- Метафоры конкретные/материальные (ключи, телескопы, часы, книги)
  вместо абстрактных neural nodes которые все выглядят одинаково
- AI категория: 14 метафор (повтор через 14 статей = ~5 дней)
- 11 тематических категорий + 15 универсальных
2026-06-13 09:26:36 +03:00
Ник (Claude) 5a765d27e1 fix: cover image diversity — 12 styles + topic-aware visual metaphors
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 универсальных метафор. Детерминированный выбор по хешу заголовка.
2026-06-12 22:55:25 +03:00
Ник (Claude) fe7af0b3b5 refactor: single image provider — routerai gpt-5-image-mini only
- config: убраны imageBaseUrl/imageFallbackBaseUrl/imageModel (старые провайдеры)
  Остались только routeraiBaseUrl, routeraiApiKey, routeraiModel
- covers.js: единственная цепочка routerai→retry→local SVG
  Убраны generateCoverViaImageGenerations, ViaResponses (aiprimetech), ViaImagesEndpoint
  generateCoverViaRouterAI: убран quality параметр (routerai игнорирует)
- postImages.js: убраны Nyxos/Aiguoguo, убраны isHD/imgQuality/tryNyxos
- aiUsage.js: реальные цены из статистики routerai.ru:
  gpt-5-image-mini ~₽2.72, всегда 4175 image tokens (high quality)
- index.js: лог показывает routerai вместо старого aiguoguo
2026-06-11 15:44:33 +03:00
Ник (Claude) e6c192e806 feat: image quality param — low for posts, medium for article covers
covers.js: generateCoverViaRouterAI принимает quality='medium' по умолчанию
postImages.js: quality='low' для постов TG/VK (₽0.25 vs ₽0.84)
Экономия 70% на генерации картинок к постам
2026-06-11 14:48:10 +03:00
Ник (Claude) 06ab7e0c1d feat: routerai as primary image provider, Nyxos as fallback
- covers.js: RouterAI /responses → Nyxos /images/generations → SVG
- postImages.js: RouterAI /responses → Nyxos /images/generations
- config: imageBaseUrl = routerai, imageFallbackBaseUrl = nyxos
- app_settings: AI_IMAGE_BASE_URL = routerai, MODEL = gpt-5-image-mini
2026-06-11 13:20:52 +03:00
Ник (Claude) 2a61cc08c2 feat: RouterAI as 3rd image fallback via /responses + image_generation
- app_settings: ROUTERAI_BASE_URL, ROUTERAI_API_KEY, ROUTERAI_IMAGE_MODEL
- config/index.js: routeraiBaseUrl, routeraiApiKey, routeraiImageModel
- covers.js: generateCoverViaRouterAI() через /responses endpoint
  Цепочка: aiguoguo → Nyxos → RouterAI → local SVG
2026-06-11 13:13:31 +03:00
Ник (Claude) 8e1b6e8cda fix: reduce image provider timeout 120s→45s for faster fallback
При недоступности primary (aiguoguo) — быстрее переключаемся на
Nyxos fallback вместо ожидания 2 минуты.
2026-06-11 12:57:26 +03:00
Ник (Claude) 80d1885feb feat: multi-style support in covers.js and postImages.js
covers.js: image_style CSV → random pick per generation
postImages.js: image_style CSV → random pick per generation
2026-06-10 15:50:49 +03:00
Ник (Claude) d1e6e2ef4a feat: Nyxos Plus as primary image provider, aiguoguo as fallback
- app_settings: AI_IMAGE_BASE_URL → https://plus.nyxos.workers.dev/v1
- app_settings: AI_IMAGE_FALLBACK_BASE_URL/API_KEY → aiguoguo (резерв)
- config/index.js: загружает imageFallbackBaseUrl + imageFallbackApiKey
- covers.js: generateCoverViaImageGenerations пробует Nyxos, при 5xx/timeout
  автоматически переключается на aiguoguo
2026-06-10 09:55:32 +03:00
Ник (Claude) bcb6583883 feat: upgrade to gpt-image-2, switch to response_format=url
- app_settings AI_IMAGE_MODEL: gpt-image-1-mini → gpt-image-2
- covers.js generateCoverViaImageGenerations: добавлен response_format='url'
  (рекомендация провайдера: url быстрее, b64 весит ~5MB)
  Порядок обработки ответа: url → b64_json (fallback для старых моделей)
- aiUsage.js: добавлена цена gpt-image-2 в IMAGE_PRICES_USD
2026-06-10 08:55:39 +03:00
Ник (Claude) ad133027d0 fix: styleName undefined when rubrics used in generateCover 2026-06-09 11:39:57 +03:00
Ник (Claude) 5576665c02 feat: image rubrics with AI selection for cover variety
- 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
2026-06-09 11:36:19 +03:00
Ник (Claude) b2d20b9646 fix: retry aiguoguo /images/generations once on 5xx (12s delay) 2026-06-09 11:21:57 +03:00
Ник (Claude) 4ddc57c471 fix: wrong API key for aiprimetech image endpoints
covers.js lines 156, 220: generateCoverViaResponses и generateCoverViaImagesEndpoint
используют config.ai.baseUrl (aiprimetech) — исправлен ключ imageApiKey → apiKey.
До разделения провайдеров оба ключа были одинаковыми, поэтому не замечалось.

postImages.js line 99: /responses через aiprimetech — аналогичный фикс.

Обложка статьи 50 перегенерирована вручную (была SVG-заглушка).
2026-06-09 11:17:16 +03:00
Ник (Claude) 95578af261 feat: channel image style settings wired to cover/post generation
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...
2026-06-09 10:48:38 +03:00
Ник (Claude) 449d1fa728 AI config migration to app_settings + ai_usage logging
* 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.
2026-06-08 20:21:04 +03:00
Nik (Claude) 0c01ed7e62 fix: image generation via aiguoguo199.com /images/generations
- Новый основной источник: 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
2026-06-08 16:21:43 +03:00
Nik (Claude) a370b8f7d8 feat: Зеро-персонаж, auto-publish, auto-series, channel-stats, fallback covers
- Персонаж Зеро: 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
2026-06-07 14:03:56 +03:00
Alexey Pavlov 17bc923c59 feat: 6 cover styles, deterministic pick by articleId 2026-05-31 13:43:18 +03:00
Alexey Pavlov 5472603a85 feat: image generation через GPT-5 /v1/responses + image_generation tool
Старый 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 это долго
2026-05-31 11:14:36 +03:00
Alexey Pavlov c7b83147f1 feat: AI-генерация обложек + /api/stats + раздача /uploads
- services/covers.js: gpt-image-1, фиксированный стиль emerald-geometric, fallback на ошибки шлюза
- articles.generateAndSaveArticle: запускает обложку в setImmediate (не блокирует ответ)
- routes/articles: POST /backfill-covers для досгенерации
- routes/stats: статистика блога (статьи, слова, токены, просмотры)
- index.js: express.static на /uploads БЕЗ авторизации (публичные картинки)
2026-05-31 09:17:08 +03:00