Files
QZMusic-Web/server.cjs

120 lines
4.1 KiB
JavaScript
Raw Permalink Normal View History

2026-06-04 15:15:17 +00:00
/**
* QZMusic Web 静态文件服务器生产用
* 监听 0.0.0.0:1219支持公网访问
2026-06-04 15:15:17 +00:00
*/
const http = require('http');
const fs = require('fs');
const path = require('path');
const url = require('url');
2026-06-04 15:15:17 +00:00
const PORT = parseInt(process.env.QZMUSIC_PORT || '1219', 10);
const HOST = process.env.QZMUSIC_HOST || '0.0.0.0';
2026-06-04 15:15:17 +00:00
const ROOT = path.join(__dirname, 'dist');
const mimeTypes = {
'.html': 'text/html; charset=utf-8',
'.htm': 'text/html; charset=utf-8',
'.js': 'application/javascript; charset=utf-8',
'.mjs': 'application/javascript; charset=utf-8',
'.css': 'text/css; charset=utf-8',
'.json': 'application/json; charset=utf-8',
'.png': 'image/png',
'.jpg': 'image/jpeg',
'.jpeg': 'image/jpeg',
'.gif': 'image/gif',
'.webp': 'image/webp',
'.svg': 'image/svg+xml',
'.ico': 'image/x-icon',
'.ttf': 'font/ttf',
2026-06-04 15:15:17 +00:00
'.woff': 'font/woff',
'.woff2':'font/woff2',
'.map': 'application/json; charset=utf-8',
'.txt': 'text/plain; charset=utf-8'
2026-06-04 15:15:17 +00:00
};
function log(...args) {
const ts = new Date().toISOString().replace('T', ' ').substring(0, 19);
console.log(`[${ts}]`, ...args);
}
2026-06-04 15:15:17 +00:00
const server = http.createServer((req, res) => {
const parsed = url.parse(req.url);
let pathname = decodeURIComponent(parsed.pathname || '/');
if (pathname === '/') pathname = '/index.html';
let filePath = path.join(ROOT, pathname);
// 防止路径穿越
if (!filePath.startsWith(ROOT)) {
res.writeHead(403);
res.end('Forbidden');
return;
}
2026-06-04 15:15:17 +00:00
const extname = String(path.extname(filePath)).toLowerCase();
const contentType = mimeTypes[extname] || 'application/octet-stream';
// SPA 路由:如果不是带扩展名的文件且文件不存在,则回退到 index.html
const tryStatic = (cb) => {
fs.stat(filePath, (err, stat) => {
if (!err && stat.isDirectory()) {
filePath = path.join(filePath, 'index.html');
return fs.stat(filePath, (e2, s2) => {
if (!e2 && s2.isFile()) return cb(true);
return cb(false);
2026-06-04 15:15:17 +00:00
});
}
if (!err && stat.isFile()) return cb(true);
return cb(false);
});
};
tryStatic((found) => {
if (!found) {
filePath = path.join(ROOT, 'index.html');
2026-06-04 15:15:17 +00:00
}
fs.readFile(filePath, (error, content) => {
if (error) {
log(`${req.method} ${req.url} -> 500 (${error.code})`);
res.writeHead(500, { 'Content-Type': 'text/plain; charset=utf-8' });
res.end('Server Error: ' + error.code);
} else {
const finalType = found ? contentType : 'text/html; charset=utf-8';
res.writeHead(200, {
'Content-Type': finalType,
'Cache-Control': extname === '.html' || extname === '.htm'
? 'no-cache, no-store, must-revalidate'
: 'public, max-age=3600'
});
res.end(content);
log(`${req.method} ${req.url} -> 200 (${pathname})`);
}
});
2026-06-04 15:15:17 +00:00
});
});
server.listen(PORT, HOST, () => {
const banner = `
QZMusic Web Server 启动成功
监听地址 http://${HOST}:${PORT} ║
本地访问 http://localhost:${PORT} ║
公网访问 http://[你的公网IP]:${PORT} ║
`;
console.log(banner);
});
process.on('SIGINT', () => {
log('收到 SIGINT正在关闭服务器...');
server.close(() => process.exit(0));
});
process.on('SIGTERM', () => {
log('收到 SIGTERM正在关闭服务器...');
server.close(() => process.exit(0));
2026-06-04 15:15:17 +00:00
});