feat: articles — публичный блог zeropost.ru

- БД: таблица articles (slug, title, excerpt, content, cover, tags, SEO)
- services/articles.js: slugify (ru→en транслит), reading_time, генерация со встроенным blog-channel
- routes/articles.js: GET list/tags/:slug, POST /generate
- Универсальный blogChannel со стилем для лонгридов: tone:friendly, structure:headers, без эмодзи и хэштегов
- generateAndSaveArticle: вытаскивает title из H1, генерит excerpt, считает время чтения
This commit is contained in:
Alexey Pavlov
2026-05-31 08:45:34 +03:00
parent 5599de59ce
commit 500bb0299e
4 changed files with 230 additions and 0 deletions
+52
View File
@@ -0,0 +1,52 @@
const express = require('express');
const router = express.Router();
const articlesSvc = require('../services/articles');
// GET /api/articles — список
router.get('/', async (req, res) => {
try {
const limit = Math.min(parseInt(req.query.limit) || 20, 100);
const offset = parseInt(req.query.offset) || 0;
const tag = req.query.tag || null;
const list = await articlesSvc.listArticles({ limit, offset, tag });
res.json(list);
} catch (err) {
res.status(500).json({ error: err.message });
}
});
// GET /api/articles/tags — топ тегов
router.get('/tags', async (_, res) => {
try {
const tags = await articlesSvc.getAllTags();
res.json(tags);
} catch (err) {
res.status(500).json({ error: err.message });
}
});
// GET /api/articles/:slug — одна
router.get('/:slug', async (req, res) => {
try {
const a = await articlesSvc.getArticleBySlug(req.params.slug);
if (!a) return res.status(404).json({ error: 'Not found' });
res.json(a);
} catch (err) {
res.status(500).json({ error: err.message });
}
});
// POST /api/articles/generate — сгенерировать и сохранить (синхронно, для cron)
router.post('/generate', async (req, res) => {
try {
const { topic, keywords = [], tags = [], autoPublish = true } = req.body;
if (!topic) return res.status(400).json({ error: 'topic is required' });
const article = await articlesSvc.generateAndSaveArticle({ topic, keywords, tags, autoPublish });
res.json(article);
} catch (err) {
console.error('[Articles] generate', err);
res.status(500).json({ error: err.message });
}
});
module.exports = router;