2026-02-26 14:14:26 +08:00
|
|
|
|
import type { ServeCommandHandler } from "../types.js";
|
|
|
|
|
|
import { createServer, Server, IncomingMessage, ServerResponse } from "http";
|
|
|
|
|
|
import { readdirSync, statSync, readFileSync, existsSync } from "fs";
|
|
|
|
|
|
import { createReadStream } from "fs";
|
2026-02-26 16:11:56 +08:00
|
|
|
|
import { join, resolve, extname, sep, relative, dirname } from "path";
|
2026-02-26 14:14:26 +08:00
|
|
|
|
import { watch } from "chokidar";
|
2026-02-26 16:11:56 +08:00
|
|
|
|
import { fileURLToPath } from "url";
|
2026-02-26 00:17:23 +08:00
|
|
|
|
|
2026-02-26 13:35:09 +08:00
|
|
|
|
interface ContentIndex {
|
|
|
|
|
|
[path: string]: string;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-26 16:11:56 +08:00
|
|
|
|
/**
|
|
|
|
|
|
* 获取 CLI 脚本文件所在目录路径(用于定位 dist 文件夹)
|
|
|
|
|
|
*/
|
|
|
|
|
|
const __filename = fileURLToPath(import.meta.url);
|
|
|
|
|
|
const __dirname = dirname(__filename);
|
2026-02-26 16:35:57 +08:00
|
|
|
|
const distDir = resolve(__dirname, "..", "..", "..", "dist", "web");
|
2026-02-26 16:11:56 +08:00
|
|
|
|
|
2026-02-26 13:37:10 +08:00
|
|
|
|
/**
|
|
|
|
|
|
* MIME 类型映射
|
|
|
|
|
|
*/
|
|
|
|
|
|
const MIME_TYPES: Record<string, string> = {
|
2026-02-26 14:14:26 +08:00
|
|
|
|
".html": "text/html",
|
|
|
|
|
|
".css": "text/css",
|
|
|
|
|
|
".js": "application/javascript",
|
|
|
|
|
|
".json": "application/json",
|
|
|
|
|
|
".png": "image/png",
|
|
|
|
|
|
".jpg": "image/jpeg",
|
|
|
|
|
|
".jpeg": "image/jpeg",
|
|
|
|
|
|
".gif": "image/gif",
|
|
|
|
|
|
".svg": "image/svg+xml",
|
|
|
|
|
|
".md": "text/markdown",
|
|
|
|
|
|
".csv": "text/csv",
|
|
|
|
|
|
".woff": "font/woff",
|
|
|
|
|
|
".woff2": "font/woff2",
|
2026-02-26 13:37:10 +08:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* 获取文件扩展名对应的 MIME 类型
|
|
|
|
|
|
*/
|
|
|
|
|
|
function getMimeType(filePath: string): string {
|
|
|
|
|
|
const ext = extname(filePath).toLowerCase();
|
2026-02-26 14:14:26 +08:00
|
|
|
|
return MIME_TYPES[ext] || "application/octet-stream";
|
2026-02-26 13:37:10 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-26 13:35:09 +08:00
|
|
|
|
/**
|
2026-02-26 14:14:26 +08:00
|
|
|
|
* 扫描目录内的 .md 文件,生成索引
|
2026-02-26 13:35:09 +08:00
|
|
|
|
*/
|
|
|
|
|
|
function scanDirectory(dir: string): ContentIndex {
|
|
|
|
|
|
const index: ContentIndex = {};
|
2026-02-26 13:37:10 +08:00
|
|
|
|
|
2026-02-26 13:35:09 +08:00
|
|
|
|
function scan(currentPath: string, relativePath: string) {
|
|
|
|
|
|
const entries = readdirSync(currentPath);
|
2026-02-26 13:37:10 +08:00
|
|
|
|
|
2026-02-26 13:35:09 +08:00
|
|
|
|
for (const entry of entries) {
|
2026-02-26 14:14:26 +08:00
|
|
|
|
if (entry.startsWith(".") || entry === "node_modules") continue;
|
2026-02-26 13:37:10 +08:00
|
|
|
|
|
2026-02-26 13:35:09 +08:00
|
|
|
|
const fullPath = join(currentPath, entry);
|
|
|
|
|
|
const relPath = relativePath ? join(relativePath, entry) : entry;
|
2026-02-26 14:14:26 +08:00
|
|
|
|
const normalizedRelPath = "/" + relPath.split(sep).join("/");
|
2026-02-26 13:37:10 +08:00
|
|
|
|
|
2026-02-26 13:35:09 +08:00
|
|
|
|
const stats = statSync(fullPath);
|
|
|
|
|
|
if (stats.isDirectory()) {
|
|
|
|
|
|
scan(fullPath, relPath);
|
2026-03-13 15:52:51 +08:00
|
|
|
|
} else if (entry.endsWith(".md") || entry.endsWith(".csv") || entry.endsWith(".yarn")) {
|
2026-02-26 13:35:09 +08:00
|
|
|
|
try {
|
2026-02-26 14:14:26 +08:00
|
|
|
|
const content = readFileSync(fullPath, "utf-8");
|
2026-02-26 13:35:09 +08:00
|
|
|
|
index[normalizedRelPath] = content;
|
|
|
|
|
|
} catch (e) {
|
|
|
|
|
|
console.error(`读取文件失败:${fullPath}`, e);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2026-02-26 13:37:10 +08:00
|
|
|
|
|
2026-02-26 14:14:26 +08:00
|
|
|
|
scan(dir, "");
|
2026-02-26 13:35:09 +08:00
|
|
|
|
return index;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
2026-02-26 13:37:10 +08:00
|
|
|
|
* 发送文件响应
|
2026-02-26 13:35:09 +08:00
|
|
|
|
*/
|
2026-02-26 13:37:10 +08:00
|
|
|
|
function sendFile(res: ServerResponse, filePath: string) {
|
|
|
|
|
|
res.writeHead(200, {
|
2026-02-26 14:14:26 +08:00
|
|
|
|
"Content-Type": getMimeType(filePath),
|
|
|
|
|
|
"Access-Control-Allow-Origin": "*",
|
2026-02-26 13:37:10 +08:00
|
|
|
|
});
|
|
|
|
|
|
createReadStream(filePath).pipe(res);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* 发送 404 响应
|
|
|
|
|
|
*/
|
|
|
|
|
|
function send404(res: ServerResponse) {
|
2026-02-26 14:14:26 +08:00
|
|
|
|
res.writeHead(404, { "Content-Type": "text/plain" });
|
|
|
|
|
|
res.end("Not Found");
|
2026-02-26 13:37:10 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* 发送 JSON 响应
|
|
|
|
|
|
*/
|
|
|
|
|
|
function sendJson(res: ServerResponse, data: unknown) {
|
|
|
|
|
|
res.writeHead(200, {
|
2026-02-26 14:14:26 +08:00
|
|
|
|
"Content-Type": "application/json",
|
|
|
|
|
|
"Access-Control-Allow-Origin": "*",
|
2026-02-26 13:37:10 +08:00
|
|
|
|
});
|
|
|
|
|
|
res.end(JSON.stringify(data, null, 2));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* 尝试提供静态文件
|
|
|
|
|
|
* @returns 如果文件存在并已成功发送则返回 true
|
|
|
|
|
|
*/
|
2026-02-26 14:14:26 +08:00
|
|
|
|
function tryServeStatic(
|
|
|
|
|
|
res: ServerResponse,
|
|
|
|
|
|
filePath: string,
|
|
|
|
|
|
dir: string,
|
|
|
|
|
|
): boolean {
|
2026-02-26 13:37:10 +08:00
|
|
|
|
const fullPath = join(dir, filePath);
|
2026-02-26 14:14:26 +08:00
|
|
|
|
|
2026-02-26 13:37:10 +08:00
|
|
|
|
if (!existsSync(fullPath)) {
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const stats = statSync(fullPath);
|
|
|
|
|
|
if (!stats.isFile()) {
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
sendFile(res, fullPath);
|
|
|
|
|
|
return true;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* 创建请求处理器
|
|
|
|
|
|
*/
|
|
|
|
|
|
function createRequestHandler(
|
|
|
|
|
|
contentDir: string,
|
|
|
|
|
|
distDir: string,
|
|
|
|
|
|
getIndex: () => ContentIndex,
|
|
|
|
|
|
) {
|
|
|
|
|
|
return (req: IncomingMessage, res: ServerResponse) => {
|
2026-02-26 14:14:26 +08:00
|
|
|
|
const url = req.url || "/";
|
|
|
|
|
|
const filePath = decodeURIComponent(url.split("?")[0]);
|
2026-02-26 13:37:10 +08:00
|
|
|
|
|
|
|
|
|
|
// 1. 处理 /__CONTENT_INDEX.json
|
2026-02-26 14:14:26 +08:00
|
|
|
|
if (filePath === "/__CONTENT_INDEX.json") {
|
2026-02-26 13:37:10 +08:00
|
|
|
|
sendJson(res, getIndex());
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 2. 处理 /static/ 目录(从 dist/web)
|
2026-02-26 14:14:26 +08:00
|
|
|
|
if (filePath.startsWith("/static/")) {
|
2026-02-26 13:37:10 +08:00
|
|
|
|
if (tryServeStatic(res, filePath, distDir)) {
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
send404(res);
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 3. 处理 /content/ 目录
|
|
|
|
|
|
const relativePath = filePath.slice(1); // 去掉开头的 /
|
|
|
|
|
|
if (tryServeStatic(res, relativePath, contentDir)) {
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 4. 处理 SPA 路由:返回 index.html
|
2026-02-26 14:14:26 +08:00
|
|
|
|
if (tryServeStatic(res, "index.html", distDir)) {
|
2026-02-26 13:37:10 +08:00
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 5. 404
|
|
|
|
|
|
send404(res);
|
2026-02-26 13:35:09 +08:00
|
|
|
|
};
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* 启动开发服务器
|
|
|
|
|
|
*/
|
|
|
|
|
|
export const serveCommand: ServeCommandHandler = async (dir, options) => {
|
|
|
|
|
|
const contentDir = resolve(dir);
|
|
|
|
|
|
let contentIndex: ContentIndex = {};
|
2026-02-26 13:37:10 +08:00
|
|
|
|
|
2026-02-26 13:35:09 +08:00
|
|
|
|
// 扫描内容目录生成索引
|
2026-02-26 14:14:26 +08:00
|
|
|
|
console.log("扫描内容目录...");
|
2026-02-26 13:35:09 +08:00
|
|
|
|
contentIndex = scanDirectory(contentDir);
|
|
|
|
|
|
console.log(`已索引 ${Object.keys(contentIndex).length} 个文件`);
|
2026-02-26 13:37:10 +08:00
|
|
|
|
|
2026-02-26 13:35:09 +08:00
|
|
|
|
// 监听文件变化
|
2026-02-26 14:14:26 +08:00
|
|
|
|
console.log("监听文件变化...");
|
2026-02-26 13:35:09 +08:00
|
|
|
|
const watcher = watch(contentDir, {
|
|
|
|
|
|
ignored: /(^|[\/\\])\../,
|
|
|
|
|
|
persistent: true,
|
|
|
|
|
|
ignoreInitial: true,
|
|
|
|
|
|
});
|
2026-02-26 13:37:10 +08:00
|
|
|
|
|
2026-02-26 13:35:09 +08:00
|
|
|
|
watcher
|
2026-02-26 14:14:26 +08:00
|
|
|
|
.on("add", (path) => {
|
|
|
|
|
|
if (path.endsWith(".md")) {
|
2026-02-26 13:35:09 +08:00
|
|
|
|
try {
|
2026-02-26 14:14:26 +08:00
|
|
|
|
const content = readFileSync(path, "utf-8");
|
|
|
|
|
|
const relPath = "/" + relative(contentDir, path).split(sep).join("/");
|
2026-02-26 13:35:09 +08:00
|
|
|
|
contentIndex[relPath] = content;
|
|
|
|
|
|
console.log(`[新增] ${relPath}`);
|
|
|
|
|
|
} catch (e) {
|
|
|
|
|
|
console.error(`读取新增文件失败:${path}`, e);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
2026-02-26 14:14:26 +08:00
|
|
|
|
.on("change", (path) => {
|
|
|
|
|
|
if (path.endsWith(".md")) {
|
2026-02-26 13:35:09 +08:00
|
|
|
|
try {
|
2026-02-26 14:14:26 +08:00
|
|
|
|
const content = readFileSync(path, "utf-8");
|
|
|
|
|
|
const relPath = "/" + relative(contentDir, path).split(sep).join("/");
|
2026-02-26 13:35:09 +08:00
|
|
|
|
contentIndex[relPath] = content;
|
|
|
|
|
|
console.log(`[更新] ${relPath}`);
|
|
|
|
|
|
} catch (e) {
|
|
|
|
|
|
console.error(`读取更新文件失败:${path}`, e);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
2026-02-26 14:14:26 +08:00
|
|
|
|
.on("unlink", (path) => {
|
|
|
|
|
|
if (path.endsWith(".md")) {
|
|
|
|
|
|
const relPath = "/" + relative(contentDir, path).split(sep).join("/");
|
2026-02-26 13:35:09 +08:00
|
|
|
|
delete contentIndex[relPath];
|
|
|
|
|
|
console.log(`[删除] ${relPath}`);
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
2026-02-26 13:37:10 +08:00
|
|
|
|
|
|
|
|
|
|
// 创建请求处理器
|
|
|
|
|
|
const handleRequest = createRequestHandler(
|
|
|
|
|
|
contentDir,
|
|
|
|
|
|
distDir,
|
|
|
|
|
|
() => contentIndex,
|
|
|
|
|
|
);
|
|
|
|
|
|
|
2026-02-26 13:35:09 +08:00
|
|
|
|
// 创建 HTTP 服务器
|
2026-02-26 13:37:10 +08:00
|
|
|
|
const server = createServer(handleRequest);
|
|
|
|
|
|
|
2026-02-26 13:35:09 +08:00
|
|
|
|
const port = parseInt(options.port, 10);
|
2026-02-26 13:37:10 +08:00
|
|
|
|
|
2026-02-26 13:35:09 +08:00
|
|
|
|
server.listen(port, () => {
|
|
|
|
|
|
console.log(`\n开发服务器已启动:http://localhost:${port}`);
|
|
|
|
|
|
console.log(`内容目录:${contentDir}`);
|
2026-02-26 16:11:56 +08:00
|
|
|
|
console.log(`静态资源目录:${distDir}`);
|
2026-02-26 13:35:09 +08:00
|
|
|
|
console.log(`索引文件:http://localhost:${port}/__CONTENT_INDEX.json\n`);
|
|
|
|
|
|
});
|
2026-02-26 00:17:23 +08:00
|
|
|
|
};
|