Commit Graph

13 Commits

Author SHA1 Message Date
Alexey Pavlov bede92a520 feat: post image diversity — style rotation + random scene/concept + expanded AI concepts bank 2026-06-15 10:20:22 +03:00
Ник (Claude) 31b31b75b8 fix: post images diversity — SUBJECT+SETTING+LIGHTING prompt
Та же проблема что и у обложек статей:
  старый промт 'Topic essence: первые 250 символов' был слишком абстрактным
  → модель рисовала свой дефолт (тёмный фон + геометрия)

Новый промт (3 некофликтующих параметра):
  VISUAL CONCEPT: конкретный предмет из getPostVisualConcept()
  SETTING: физический антураж (8 вариантов по seed из текста поста)
  LIGHTING + COLOR TEMPERATURE: конкретный свет

getPostVisualConcept(): 8 тематических категорий × 4-6 концептов
  AI, automation, cybersec, code, marketing, money, education, health
  + 12 универсальных концептов
Seed = hash первых 80 символов поста → детерминировано но уникально
2026-06-14 15:13:20 +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) 1ef770b5fc feat: custom prompt for articles + HD image quality per channel
- ai.js: generateArticle принимает customPrompt (от юзера) или channel.ai_style_prompt
- articles.js + routes/articles.js: проброс customPrompt через цепочку
- postImages.js: channel.image_quality='hd' → gpt-5.4-image-2+medium, иначе gpt-5-image-mini+low
- aiUsage.js: правильные цены routerai (RUB/token), gpt-5-image-mini и gpt-5.4-image-2
- channels.js: updateChannel сохраняет ai_style_prompt и image_quality
- DB: channels.ai_style_prompt TEXT, channels.image_quality VARCHAR(16) DEFAULT standard
2026-06-11 15:11:18 +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) 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) c7f0b3ed4d fix: getChannel arg order, postImages via Nyxos /images/generations
generate.js: getChannel(userId, channelId) → getChannel(channelId, userId)
channels.js: getChannel alias → getFullChannel
postImages.js: убран /responses + gpt-5.5 (не работал на aiprimetech),
  заменён на Nyxos /images/generations с fallback на aiguoguo
2026-06-10 17:45:18 +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) 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
Alexey Pavlov 2137a92b28 feat: transformPost (7 actions), post image generation with style/palette, topics ideas endpoint 2026-05-31 17:32:38 +03:00