feat: run batch process as a separate process
This commit is contained in:
@ -8,5 +8,6 @@ docker run \
|
|||||||
--volume "$(pwd)/data:/app/data" \
|
--volume "$(pwd)/data:/app/data" \
|
||||||
--publish 3000:3000 \
|
--publish 3000:3000 \
|
||||||
--name voice-rss-summary \
|
--name voice-rss-summary \
|
||||||
|
-d \
|
||||||
--restart always \
|
--restart always \
|
||||||
voice-rss-summary
|
voice-rss-summary
|
||||||
|
43
server.ts
43
server.ts
@ -3,9 +3,6 @@ import { serve } from "@hono/node-server";
|
|||||||
import fs from "fs";
|
import fs from "fs";
|
||||||
import path from "path";
|
import path from "path";
|
||||||
import { Database } from "bun:sqlite";
|
import { Database } from "bun:sqlite";
|
||||||
import { batchProcess } from "./services/fetch_and_generate";
|
|
||||||
|
|
||||||
import { setInterval } from "timers";
|
|
||||||
|
|
||||||
const projectRoot = import.meta.dirname;
|
const projectRoot = import.meta.dirname;
|
||||||
|
|
||||||
@ -165,19 +162,6 @@ app.get("*", async (c) => {
|
|||||||
return c.notFound();
|
return c.notFound();
|
||||||
});
|
});
|
||||||
|
|
||||||
// サーバー起動
|
|
||||||
serve(
|
|
||||||
{
|
|
||||||
fetch: app.fetch,
|
|
||||||
port: 3000,
|
|
||||||
},
|
|
||||||
(info) => {
|
|
||||||
console.log(`Server is running on http://localhost:${info.port}`);
|
|
||||||
// 初回実行
|
|
||||||
scheduleFirstBatchProcess();
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 初回実行後に1日ごとのバッチ処理をスケジュールする関数
|
* 初回実行後に1日ごとのバッチ処理をスケジュールする関数
|
||||||
*/
|
*/
|
||||||
@ -186,7 +170,7 @@ function scheduleFirstBatchProcess() {
|
|||||||
setTimeout(async () => {
|
setTimeout(async () => {
|
||||||
try {
|
try {
|
||||||
console.log("Running initial batch process...");
|
console.log("Running initial batch process...");
|
||||||
await batchProcess();
|
runBatchProcess();
|
||||||
console.log("Initial batch process completed");
|
console.log("Initial batch process completed");
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Error during initial batch process:", error);
|
console.error("Error during initial batch process:", error);
|
||||||
@ -215,7 +199,7 @@ function scheduleDailyBatchProcess() {
|
|||||||
setTimeout(async () => {
|
setTimeout(async () => {
|
||||||
try {
|
try {
|
||||||
console.log("Running daily batch process...");
|
console.log("Running daily batch process...");
|
||||||
await batchProcess();
|
runBatchProcess();
|
||||||
console.log("Daily batch process completed");
|
console.log("Daily batch process completed");
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Error during daily batch process:", error);
|
console.error("Error during daily batch process:", error);
|
||||||
@ -224,3 +208,26 @@ function scheduleDailyBatchProcess() {
|
|||||||
scheduleDailyBatchProcess();
|
scheduleDailyBatchProcess();
|
||||||
}, delay);
|
}, delay);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const runBatchProcess = () => {
|
||||||
|
try {
|
||||||
|
console.log("Running batch process...");
|
||||||
|
Bun.spawn(["bun", "run", "scripts/fetch_and_generate.ts"]);
|
||||||
|
console.log("Batch process completed");
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error during batch process:", error);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// サーバー起動
|
||||||
|
serve(
|
||||||
|
{
|
||||||
|
fetch: app.fetch,
|
||||||
|
port: 3000,
|
||||||
|
},
|
||||||
|
(info) => {
|
||||||
|
console.log(`Server is running on http://localhost:${info.port}`);
|
||||||
|
// 初回実行
|
||||||
|
scheduleFirstBatchProcess();
|
||||||
|
},
|
||||||
|
);
|
||||||
|
@ -1,137 +0,0 @@
|
|||||||
import Parser from "rss-parser";
|
|
||||||
import { openAI_ClassifyFeed, openAI_GeneratePodcastContent } from "./llm";
|
|
||||||
import { generateTTS } from "./tts";
|
|
||||||
import { saveEpisode, markAsProcessed } from "./database";
|
|
||||||
import { updatePodcastRSS } from "./podcast";
|
|
||||||
import crypto from "crypto";
|
|
||||||
|
|
||||||
interface FeedItem {
|
|
||||||
id: string;
|
|
||||||
title: string;
|
|
||||||
link: string;
|
|
||||||
pubDate: string;
|
|
||||||
contentSnippet?: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
import fs from "fs/promises";
|
|
||||||
import path from "path";
|
|
||||||
import { fileURLToPath } from "url";
|
|
||||||
|
|
||||||
const __filename = fileURLToPath(import.meta.url);
|
|
||||||
const __dirname = path.dirname(__filename);
|
|
||||||
|
|
||||||
export async function batchProcess() {
|
|
||||||
const feedUrlsFile = import.meta.env["FEED_URLS_FILE"] ?? "feed_urls.txt";
|
|
||||||
const feedUrlsPath = path.resolve(__dirname, "..", feedUrlsFile);
|
|
||||||
let feedUrls: string[];
|
|
||||||
try {
|
|
||||||
const data = await fs.readFile(feedUrlsPath, "utf-8");
|
|
||||||
feedUrls = data
|
|
||||||
.split("\n")
|
|
||||||
.map((url) => url.trim())
|
|
||||||
.filter((url) => url.length > 0);
|
|
||||||
} catch (err) {
|
|
||||||
console.warn(`フィードURLファイルの読み込みに失敗: ${feedUrlsFile}`);
|
|
||||||
feedUrls = [];
|
|
||||||
}
|
|
||||||
|
|
||||||
// フィードごとに処理
|
|
||||||
for (const url of feedUrls) {
|
|
||||||
try {
|
|
||||||
await processFeedUrl(url);
|
|
||||||
} finally {
|
|
||||||
await updatePodcastRSS();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log("処理完了:", new Date().toISOString());
|
|
||||||
}
|
|
||||||
|
|
||||||
const processFeedUrl = async (url: string) => {
|
|
||||||
const parser = new Parser<FeedItem>();
|
|
||||||
const feed = await parser.parseURL(url);
|
|
||||||
|
|
||||||
// フィードのカテゴリ分類
|
|
||||||
const feedTitle = feed.title || url;
|
|
||||||
const category = await openAI_ClassifyFeed(feedTitle);
|
|
||||||
console.log(`フィード分類完了: ${feedTitle} - ${category}`);
|
|
||||||
|
|
||||||
// 昨日の記事のみフィルタリング
|
|
||||||
const yesterday = new Date();
|
|
||||||
yesterday.setDate(yesterday.getDate() - 1);
|
|
||||||
|
|
||||||
const yesterdayItems = feed.items.filter((item) => {
|
|
||||||
const pub = new Date(item.pubDate || "");
|
|
||||||
return (
|
|
||||||
pub.getFullYear() === yesterday.getFullYear() &&
|
|
||||||
pub.getMonth() === yesterday.getMonth() &&
|
|
||||||
pub.getDate() === yesterday.getDate()
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
if (yesterdayItems.length === 0) {
|
|
||||||
console.log(`昨日の記事が見つかりません: ${feedTitle}`);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// ポッドキャスト原稿生成
|
|
||||||
console.log(`ポッドキャスト原稿生成開始: ${feedTitle}`);
|
|
||||||
const validItems = yesterdayItems.filter((item): item is FeedItem => {
|
|
||||||
return !!item.title && !!item.link;
|
|
||||||
});
|
|
||||||
const podcastContent = await openAI_GeneratePodcastContent(
|
|
||||||
feedTitle,
|
|
||||||
validItems,
|
|
||||||
);
|
|
||||||
|
|
||||||
// トピックごとの統合音声生成
|
|
||||||
const feedUrlHash = crypto.createHash("md5").update(url).digest("hex");
|
|
||||||
const categoryHash = crypto.createHash("md5").update(category).digest("hex");
|
|
||||||
const uniqueId = `${feedUrlHash}-${categoryHash}`;
|
|
||||||
|
|
||||||
const audioFilePath = await generateTTS(uniqueId, podcastContent);
|
|
||||||
console.log(`音声ファイル生成完了: ${audioFilePath}`);
|
|
||||||
|
|
||||||
// エピソードとして保存(各フィードにつき1つの統合エピソード)
|
|
||||||
const firstItem = yesterdayItems[0];
|
|
||||||
if (!firstItem) {
|
|
||||||
console.warn("アイテムが空です");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const pub = new Date(firstItem.pubDate || "");
|
|
||||||
|
|
||||||
await saveEpisode({
|
|
||||||
id: uniqueId,
|
|
||||||
title: `${category}: ${feedTitle}`,
|
|
||||||
pubDate: pub.toISOString(),
|
|
||||||
audioPath: audioFilePath,
|
|
||||||
sourceLink: url,
|
|
||||||
});
|
|
||||||
|
|
||||||
console.log(`エピソード保存完了: ${category} - ${feedTitle}`);
|
|
||||||
|
|
||||||
// 個別記事の処理記録
|
|
||||||
for (const item of yesterdayItems) {
|
|
||||||
const itemId = item["id"] as string | undefined;
|
|
||||||
const fallbackId = item.link || item.title || JSON.stringify(item);
|
|
||||||
const finalItemId =
|
|
||||||
itemId && typeof itemId === "string" && itemId.trim() !== ""
|
|
||||||
? itemId
|
|
||||||
: `fallback-${Buffer.from(fallbackId).toString("base64")}`;
|
|
||||||
|
|
||||||
if (!finalItemId || finalItemId.trim() === "") {
|
|
||||||
console.warn(`フィードアイテムのIDを生成できませんでした`, {
|
|
||||||
feedUrl: url,
|
|
||||||
itemTitle: item.title,
|
|
||||||
itemLink: item.link,
|
|
||||||
});
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
const already = await markAsProcessed(url, finalItemId);
|
|
||||||
if (already) {
|
|
||||||
console.log(`既に処理済み: ${finalItemId}`);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
Reference in New Issue
Block a user