feat: トップページの最新記事を自動生成に切り替え
All checks were successful
Deploy Docusaurus Site / deploy (push) Successful in 29s

This commit is contained in:
koide 2026-02-28 02:16:29 +00:00
parent 7a509d5dc5
commit 8fe2e55d81
4 changed files with 161 additions and 35 deletions

View File

@ -22,6 +22,9 @@ jobs:
- name: Install npm dependencies
run: npm ci
- name: Generate recent posts
run: node scripts/generate-recent-posts.js
- name: Build static site
run: npm run build

View File

@ -0,0 +1,77 @@
const fs = require('fs');
const path = require('path');
const ROOT = path.resolve(__dirname, '..');
function parseFrontmatter(content) {
const m = content.match(/^---\n([\s\S]*?)\n---/);
if (!m) return {};
const fm = {};
for (const line of m[1].split('\n')) {
const idx = line.indexOf(':');
if (idx === -1) continue;
const key = line.slice(0, idx).trim();
let val = line.slice(idx + 1).trim();
// Remove quotes
if ((val.startsWith('"') && val.endsWith('"')) || (val.startsWith("'") && val.endsWith("'")))
val = val.slice(1, -1);
fm[key] = val;
}
return fm;
}
function guessTag(folderName) {
if (folderName.startsWith('dgx-spark')) return 'DGX Spark';
if (folderName.startsWith('gitea')) return 'Gitea';
if (folderName.startsWith('searxng')) return 'SearXNG';
if (folderName.startsWith('ollama')) return 'Ollama';
if (folderName.startsWith('browser')) return 'Browser';
if (folderName.startsWith('game')) return 'Game';
return 'Tech';
}
function getTag(fm, folderName, defaultTag) {
if (fm.tag) return fm.tag;
if (fm.tags) return fm.tags.replace(/[\[\]]/g, '').split(',')[0].trim();
if (defaultTag === 'blog') return 'AI';
return guessTag(folderName);
}
function scanDir(dir, urlPrefix, isBlog) {
if (!fs.existsSync(dir)) return [];
const entries = fs.readdirSync(dir, { withFileTypes: true });
const posts = [];
for (const e of entries) {
if (!e.isDirectory()) continue;
const indexPath = path.join(dir, e.name, 'index.md');
if (!fs.existsSync(indexPath)) continue;
const content = fs.readFileSync(indexPath, 'utf8');
const fm = parseFrontmatter(content);
const title = fm.title || e.name;
const tag = getTag(fm, e.name, isBlog ? 'blog' : 'tech');
// Extract date
let date = null;
const dateMatch = e.name.match(/^(\d{4}-\d{2}-\d{2})/);
if (dateMatch) {
date = dateMatch[1];
} else if (fm.date) {
date = fm.date.slice(0, 10);
} else {
// Use file mtime
const stat = fs.statSync(indexPath);
date = stat.mtime.toISOString().slice(0, 10);
}
posts.push({ title, date, tag, url: `${urlPrefix}${e.name}` });
}
posts.sort((a, b) => b.date.localeCompare(a.date));
return posts.slice(0, 5);
}
const tech = scanDir(path.join(ROOT, 'docs-tech'), '/tech/', false);
const blog = scanDir(path.join(ROOT, 'docs'), '/blog/', true);
const outPath = path.join(ROOT, 'src', 'data', 'recent-posts.json');
fs.writeFileSync(outPath, JSON.stringify({ tech, blog }, null, 2));
console.log(`Generated ${outPath} (tech: ${tech.length}, blog: ${blog.length})`);

View File

@ -0,0 +1,66 @@
{
"tech": [
{
"title": "ローカルサーバーでマイク・カメラを使う方法",
"date": "2026-02-28",
"tag": "Browser",
"url": "/tech/browser-secure-context"
},
{
"title": "DGX SparkにAnythingLLMを導入してローカルLLMエージェントを構築",
"date": "2026-02-28",
"tag": "DGX Spark",
"url": "/tech/dgx-spark-anythingllm"
},
{
"title": "DGX SparkでClaude Code + Qwen3-Coder-Nextをローカル実行する",
"date": "2026-02-28",
"tag": "DGX Spark",
"url": "/tech/dgx-spark-claude-code-local"
},
{
"title": "ローカルClaude Code + Playwright CLIでブラウザ自動化エージェントを作る",
"date": "2026-02-28",
"tag": "DGX Spark",
"url": "/tech/dgx-spark-claude-code-playwright"
},
{
"title": "DGX Spark デュアル構成ガイド",
"date": "2026-02-28",
"tag": "DGX Spark",
"url": "/tech/dgx-spark-dual"
}
],
"blog": [
{
"title": "02/28 AIヘッドライン朝刊",
"date": "2026-02-28",
"tag": "AI",
"url": "/blog/2026-02-28-morning-headline"
},
{
"title": "Ollama がローカルAIのハブとしてめちゃ最強な件",
"date": "2026-02-28",
"tag": "AI",
"url": "/blog/ollama-local-ai-hub"
},
{
"title": "02/27 AIヘッドライン夕刊",
"date": "2026-02-27",
"tag": "AI",
"url": "/blog/2026-02-27-evening-headline"
},
{
"title": "02/27 AIヘッドライン朝刊",
"date": "2026-02-27",
"tag": "AI",
"url": "/blog/2026-02-27-morning-headline"
},
{
"title": "02/26 AIヘッドライン夕刊",
"date": "2026-02-26",
"tag": "AI",
"url": "/blog/2026-02-26-evening-headline"
}
]
}

View File

@ -2,6 +2,7 @@ import React from 'react';
import Layout from '@theme/Layout';
import Link from '@docusaurus/Link';
import styles from './index.module.css';
import recentPosts from '../data/recent-posts.json';
function HomepageHeader() {
return (
@ -25,39 +26,14 @@ function HomepageHeader() {
}
function RecentTech() {
const posts = [
{
title: "ローカルGitea × Webhook連携でAI自動コードレビュー",
date: "2026-02-26",
tag: "Gitea",
url: "/tech/gitea-webhook-ai-review",
},
{
title: "DGX SparkにAnythingLLMでローカルLLMエージェント構築",
date: "2026-02-20",
tag: "DGX Spark",
url: "/tech/dgx-spark-anythingllm",
},
{
title: 'DGX Spark デュアル構成ガイド',
date: '2026-02-19',
tag: 'DGX Spark',
url: '/tech/dgx-spark-dual',
},
{
title: 'DGX SparkでMiniMax-M2.5-REAP-172Bを動かす',
date: '2026-02-18',
tag: 'DGX Spark',
url: '/tech/dgx-spark-minimax',
},
];
const posts = recentPosts.tech;
return (
<section className={styles.section}>
<h2 className={styles.sectionTitle}>// Tech</h2>
<p className={styles.sectionDesc}></p>
<div className={styles.posts}>
{posts.map((post, idx) => (
{posts.length > 0 ? posts.map((post, idx) => (
<Link key={idx} to={post.url} className={styles.postCard}>
<h3 className={styles.postTitle}>{post.title}</h3>
<div className={styles.postMeta}>
@ -65,7 +41,7 @@ function RecentTech() {
<span className={styles.postTag}>{post.tag}</span>
</div>
</Link>
))}
)) : <p></p>}
</div>
<Link to="/tech" className={styles.moreLink}>
@ -75,18 +51,22 @@ function RecentTech() {
}
function RecentBlog() {
const posts = recentPosts.blog;
return (
<section className={styles.section}>
<h2 className={styles.sectionTitle}>// Blog</h2>
<p className={styles.sectionDesc}></p>
<div className={styles.posts}>
<Link to="/blog/2026-02-19-headline" className={styles.postCard}>
<h3 className={styles.postTitle}>02/19 AIヘッドライン</h3>
{posts.length > 0 ? posts.map((post, idx) => (
<Link key={idx} to={post.url} className={styles.postCard}>
<h3 className={styles.postTitle}>{post.title}</h3>
<div className={styles.postMeta}>
<span className={styles.postDate}>2026-02-19</span>
<span className={styles.postTag}>AI</span>
<span className={styles.postDate}>{post.date}</span>
<span className={styles.postTag}>{post.tag}</span>
</div>
</Link>
)) : <p></p>}
</div>
<Link to="/blog" className={styles.moreLink}>