Commit Graph

80 Commits

Author SHA1 Message Date
Nik (Claude) 0a842476d7 fix: reduce article maxTokens 4000→3000 to avoid 400 on aiprimetech 2026-06-16 22:08:44 +03:00
Nik (Claude) e5965e2804 feat: POST /api/articles/:id/regenerate-cover for any status 2026-06-16 22:02:34 +03:00
Nik (Claude) d9cbbc5fbf fix: button always shows DEFAULT_BUTTON_TEXT when field is null/empty 2026-06-16 21:14:57 +03:00
Nik (Claude) 5852b9f439 feat: draft review flow — autogen→draft, auto-approve 07:00 MSK, /api/drafts routes 2026-06-16 09:17:10 +03:00
Nik (Claude) cd471d67a9 add start script for Coolify nixpacks 2026-06-15 22:44:05 +03:00
Alexey Pavlov bede92a520 feat: post image diversity — style rotation + random scene/concept + expanded AI concepts bank 2026-06-15 10:20:22 +03:00
Nik (Claude) 525870c709 chore: add full schema dump as source of truth (32 tables) 2026-06-15 09:28:13 +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) c40ef90ad1 feat: SMTP, maintenance mode, blog topic bank UI
8. SMTP: emailService.js (nodemailer), templates (welcome/payment/low_credits)
   /api/admin/email/test — тест отправки
   app_settings category=smtp (HOST/PORT/USER/PASS/FROM/ENABLED)
9. Maintenance mode: middleware в index.js, MAINTENANCE_MODE в engine settings
   При true → 503 для всех запросов кроме /uploads и /api/settings
10. Blog topic bank:
   DB: blog_topics(category,topic,is_used,source,priority)
   40 тем мигрированы из хардкода (source=hardcoded)
   autogen.js: getNextTopic берёт из DB, fallback на TOPIC_BANK
   admin API: GET/POST /blog-topics, DELETE /:id, POST /generate (AI +10)
2026-06-13 11:45:23 +03:00
Ник (Claude) 9b40f2cd7a feat: content defaults — applied on channel creation
DB: app_settings category=content (10 keys)
channels.js: createChannel reads DEFAULT_* from settings and applies to new channel
  DEFAULT_POST_LANGUAGE/LENGTH/STYLE/GOAL → channel.goal, language
  DEFAULT_IMAGE_ENABLED → channel.image_enabled
  DEFAULT_AI_STYLE_PROMPT → channel.ai_style_prompt
  DEFAULT_AUTO_DRAFT_COUNT/TIME → channel.auto_draft_count/time
channels: image_enabled BOOLEAN DEFAULT true
2026-06-13 11:22:08 +03:00
Ник (Claude) b5fa77ea01 feat: autogen blog admin API
routes/admin.js:
  GET  /autogen — настройки+статистика+очередь+размеры банков тем
  PATCH /autogen/:category — enabled/per_day/run_hour/run_minute
  POST  /autogen/:category/run — ручной запуск генерации
  POST  /autogen/queue — добавить тему с приоритетом
  DELETE /autogen/queue/:id — удалить тему
2026-06-13 10:35:50 +03:00
Ник (Claude) 6e1cd24b4e feat: error logs API
routes/admin.js: GET /logs — объединённые ошибки из 3 источников:
  generation_jobs (status=failed), ai_usage (!succeeded), scheduled_posts (status=failed)
  Сортировка по времени, топ-5 частых ошибок, группировка по типу
2026-06-13 10:23:24 +03:00
Ник (Claude) 7994b0e73c feat: generation queue admin
routes/admin.js: GET /queue (stats+recent30+stuck), POST /queue/:id/retry, DELETE /queue/stuck
  stuck = processing > 5 min → сбрасываем в failed
  retry = pending + requeue через Bull
2026-06-13 10:13:21 +03:00
Ник (Claude) ce74ac9909 feat: promo codes system
DB: promo_codes, promo_usages tables
routes/admin.js: CRUD /api/admin/promos (GET/POST/PATCH/DELETE)
routes/billing.js: POST /api/billing/apply-promo
  Валидация: exists, active, not expired, not exhausted, not used by this user
  type=credits → начисляет через billing.credit()
2026-06-13 09:36:32 +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) 05fa7644cc feat: user management — detail view, block/unblock, plan change
routes/admin.js: GET /users/:id (profile+channels+balance+transactions)
  PATCH /users/:id (is_blocked, plan_code, name)
  plan change: cancels active sub → creates new → credits reset
generate.js: check is_blocked before generation → 403 ACCOUNT_BLOCKED
DB: users.is_blocked BOOLEAN DEFAULT false
2026-06-13 00:14:11 +03:00
Ник (Claude) f18b83c59b feat: admin dashboard API + separate admin routes file
routes/admin.js: GET /dashboard, /users, POST /credit, PATCH /plans/:id, /credit-costs/:op
index.js: app.use('/api/admin', adminRoutes) — чистый монтаж без хаков
dashboard: users (total/7d/30d), channels by platform, posts (total/today/week),
  revenue (YuKassa), AI costs (30d), registrations chart (14d), pending drafts alert
2026-06-13 00:09:53 +03:00
Ник (Claude) ad9f054701 feat: admin panel — plans editor + credit costs editor
routes/billing.js: PATCH /api/admin/plans/:id, PATCH /api/admin/credit-costs/:operation
index.js: /api/admin/* → billing routes
DB: AI_IMAGE_* → category=legacy (скрыты из UI)
     engine settings: ENGINE_PUBLIC_URL, APP_PUBLIC_URL, AUTO_DRAFT_DEFAULT_*
2026-06-13 00:02:03 +03:00
Ник (Claude) 2b996820d7 fix: routerai cost_rub was 0 — o?.promptTokens → promptTokens in computeCostRub
aiUsage.js: computeCostRub получает распакованные параметры, не объект o
Backfill: UPDATE 8 записей с cost_rub=0 → ₽3.12
2026-06-12 23:56:45 +03:00
Ник (Claude) a8ff295faa feat: post drafts system — batch generation + daily auto-drafts
DB: post_drafts(channel_id, topic, text, image_url, status), channels.auto_draft_*
Engine:
  services/draftService.js: generateOneDraft, generateBatch, generateDailyDrafts,
    approveDraft(→scheduled_post), rejectDraft, updateDraft, listDrafts
  routes/drafts.js: GET/PATCH/DELETE /api/drafts/:id, /approve, /reject
    POST /api/channels/:channelId/drafts/generate?count=N (async, returns immediately)
  index.js: cron каждые 30 мин → generateDailyDrafts() для каналов с auto_draft_enabled
  channels.js: updateChannel сохраняет auto_draft_enabled/count/time
2026-06-12 23:47:27 +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) 7a70f79e61 fix: duplicate article prevention — source_topic deduplication
autogen.js: getNextTopic() теперь проверяет source_topic (exact match) вместо
  сравнения первых 20 символов заголовка (который AI переименовывает)
articles.js: INSERT сохраняет source_topic из topic параметра
DB: articles.source_topic TEXT, articles.topic_hash VARCHAR(64)
Пометили существующие дубли: article 61 → archived, source_topic заполнен
2026-06-12 11:49:33 +03:00
Ник (Claude) bbae6c8832 feat: topic bank + channel limit + onboarding
Topic bank (P6):
- DB: channel_topics(channel_id, topic, is_used)
- services/topicBank.js: nextTopic, refillManual, addManual, listTopics, checkAndRefill
  Авто-пополнение когда <5 тем, пачками по 10 через Claude Haiku
- routes/generate.js: GET/POST /topics-bank/:channelId, /refill, /add, DELETE /item/:id

Channel limit (P7):
- routes/channels.js: POST / → проверяет billing.getBalance().channelsMax перед созданием
  HTTP 402 + CHANNEL_LIMIT_REACHED если лимит исчерпан
- channels/new/page.js: при 402 → ошибка + redirect на /plans через 2 сек

ENGINE_URL fix: 3040 → 3030 (lib/engine.js)
2026-06-11 23:04:45 +03:00
Ник (Claude) 10c138aa33 feat: P6 inbox — TG webhook + AI classify + reply
DB: inbox_messages (text, ai_type, ai_reply, status), channels.tg_webhook_enabled
Engine routes/inbox.js:
  POST /api/tg-webhook/:channelId — получаем комментарии от TG (публичный)
  GET  /api/inbox/:channelId      — список сообщений с фильтром
  GET  /api/inbox/stats/:channelId — статистика по типам
  POST /api/inbox/:id/reply       — отправить ответ в TG
  POST /api/inbox/:id/status      — изменить статус
  POST /api/inbox/:channelId/setup-webhook — зарегистрировать TG webhook
AI: claude haiku → {type, reply} JSON
2026-06-11 20:12:19 +03:00
Ник (Claude) ee63172e08 feat: MAX publisher — platform-api.max.ru with image upload
publishToMax():
- API: platform-api.max.ru (новый домен, Authorization header)
- Text: POST /messages?chat_id={id} с {text, attachments}
- Photo: POST /uploads?type=image → presigned URL → multipart upload → {token}
  Attach: {type:'image', payload:{token}} в теле сообщения
- Graceful fallback: при ошибке фото — пост без картинки
- Убрана заглушка 'не реализована'
2026-06-11 20:01:50 +03:00
Ник (Claude) 6e32241fe8 feat: P7 polls + P8 hashtags
P7 — Опросы в Telegram:
- routes/polls.js: POST /api/channels/:id/poll (sendPoll + schedule support)
- DB: scheduled_posts.post_type + meta для отложенных опросов
- scheduledPostsRunner: обработка post_type='poll' через sendPoll
- index.js: роут /api/channels подключён

P8 — Хештеги:
- ai.js: generateHashtags() через Claude Haiku, JSON-массив тегов
- routes/generate.js: POST /api/generate/hashtags
2026-06-11 19:54:31 +03:00
Ник (Claude) 0a9d886435 fix: VK photo upload — 2-step getWallUploadServer + saveWallPhoto
До: wall.post без attachments → картинка игнорировалась
После:
  1. photos.getWallUploadServer → upload_url
  2. POST upload_url с файлом (local path или download) → server/photo/hash
  3. photos.saveWallPhoto → owner_id + photo_id
  4. wall.post с attachments=photo{owner_id}_{id}

При ошибке загрузки фото — публикуем без картинки (graceful degradation)
Поддерживает как локальные /uploads/ файлы так и внешние URL
2026-06-11 19:45:13 +03:00
Ник (Claude) 4580264de9 feat: yukassa reads keys from app_settings + monthly reset endpoint
- yukassa.js: getConfig() читает YUKASSA_SHOP_ID/SECRET/RETURN_URL из app_settings (fallback env)
- routes/billing.js: POST /monthly-reset — запускает processMonthlyResets()
- app_settings: категория 'payments' с ключами ЮKassa
- cron: 0 6 * * * zeropost-billing-reset.sh
2026-06-11 19:40:10 +03:00
Ник (Claude) 9baa0f0959 feat: YuKassa payment integration
- services/yukassa.js: createPayment, handleWebhook
  createPayment → создаёт платёж + сохраняет в payment_orders
  handleWebhook → activates plan + charges credits on payment.succeeded
- routes/billing.js: POST /checkout, POST /webhook (публичный)
- DB: payment_orders table
- index.js: /api/billing/webhook публичный (до auth middleware)
2026-06-11 18:44:20 +03:00
Ник (Claude) 2e60a6e316 feat: billing system — credits, plans, transactions
DB:
- plans: free/starter/pro/business с ценами и лимитами
- user_subscriptions: подписка пользователя на план
- user_balance: баланс кредитов + monthly reset
- user_transactions: история всех движений кредитов
- credit_costs: image=5, text_post=2, article=5, autopublish=0

Engine:
- services/billing.js: getBalance, check, spend, credit, getTransactions, processMonthlyResets
- routes/billing.js: GET /balance, /transactions, /plans, POST /admin/credit, GET /admin/users
- routes/generate.js: списание кредитов перед генерацией (text_post, article, image)
- index.js: GET /api/billing/plans публично (без auth)
2026-06-11 18:26:38 +03:00
Ник (Claude) eede50bee7 fix: ROUTERAI_API_KEY not loaded from app_settings
config/index.js: добавили ROUTERAI_BASE_URL, ROUTERAI_API_KEY в список keys
который загружается из app_settings через reloadAi()
Без этого ключ был NULL и все генерации падали в local SVG fallback
2026-06-11 16:36:56 +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) 08086650fc feat: customPrompt piped through full generation chain
- generatePost: customPrompt или channel.ai_style_prompt → добавляется к userPrompt
- routes/generate.js: принимает customPrompt, передаёт в очередь
- workers/generation.js: передаёт customPrompt в generatePost и generateArticle
2026-06-11 15:15:22 +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) 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) 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) b4c537487d feat: vk_access_token в updateable fields 2026-06-10 15:22:45 +03:00
Ник (Claude) be2735ea5e feat: delay post if SVG cover, retry up to 3×15min
scheduled_posts.cover_regen_attempts: счётчик попыток регенерации.
Если обложка SVG и рег. не удалась:
  - попытки < 3: откладываем scheduled_at на +15 мин, не публикуем
  - попытки >= 3: публикуем с Zero fallback (не держим пост вечно)
Максимальная задержка = 45 минут после плановой публикации.
2026-06-10 13:32:47 +03:00
Ник (Claude) e79693c29a fix: check cover size before Telegram publish, regen SVG stubs
scheduledPostsRunner.js: перед отправкой обложки в TG проверяем
размер файла. Если < 30KB (SVG-заглушка) — пробуем перегенерировать
через covers.generateCover(). Если регенерация успешна — публикуем
реальную обложку. Если нет — fallback на позу Зеро вместо SVG.
2026-06-10 13:07:15 +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) d8a901131c feat: auto-retry SVG covers every 30 min
coverRetry.js: сканирует articles с cover < 30KB (SVG-заглушки),
перегенерирует через covers.generateCover(). При недоступности
провайдера (timeout/502) прерывает цикл до следующего запуска.
Первый запуск через 5 мин после старта engine, далее каждые 30 мин.
2026-06-10 08:53: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