2025-02-15 00:37:32 -08:00
|
|
|
import express from "express";
|
|
|
|
|
import expressWs from "express-ws";
|
|
|
|
|
import { MediaPlayer } from "./MediaPlayer";
|
|
|
|
|
|
|
|
|
|
const app = express();
|
|
|
|
|
app.use(express.json());
|
2025-02-15 02:37:17 -08:00
|
|
|
expressWs(app);
|
2025-02-15 00:37:32 -08:00
|
|
|
|
2025-02-15 01:26:10 -08:00
|
|
|
const apiRouter = express.Router();
|
2025-02-15 00:37:32 -08:00
|
|
|
const mediaPlayer = new MediaPlayer();
|
|
|
|
|
|
2025-02-15 02:37:17 -08:00
|
|
|
const withErrorHandling = (func: (req: any, res: any) => Promise<any>) => {
|
|
|
|
|
return async (req: any, res: any) => {
|
|
|
|
|
try {
|
|
|
|
|
await func(req, res);
|
|
|
|
|
} catch (error: any) {
|
|
|
|
|
res.status(500).send(JSON.stringify({ success: false, error: error.message }));
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
apiRouter.get("/playlist", withErrorHandling(async (req, res) => {
|
2025-02-15 00:37:32 -08:00
|
|
|
const playlist = await mediaPlayer.getPlaylist();
|
|
|
|
|
res.send(playlist);
|
2025-02-15 02:37:17 -08:00
|
|
|
}));
|
2025-02-15 00:37:32 -08:00
|
|
|
|
2025-02-15 02:37:17 -08:00
|
|
|
apiRouter.post("/playlist", withErrorHandling(async (req, res) => {
|
|
|
|
|
const { url } = req.body as { url: string };
|
|
|
|
|
await mediaPlayer.append(url);
|
|
|
|
|
res.send(JSON.stringify({ success: true }));
|
|
|
|
|
}));
|
2025-02-15 00:37:32 -08:00
|
|
|
|
2025-02-15 02:37:17 -08:00
|
|
|
apiRouter.delete("/playlist/:index", withErrorHandling(async (req, res) => {
|
2025-02-15 00:37:32 -08:00
|
|
|
const { index } = req.params as { index: string };
|
|
|
|
|
await mediaPlayer.deletePlaylistItem(parseInt(index));
|
|
|
|
|
res.send(JSON.stringify({ success: true }));
|
2025-02-15 02:37:17 -08:00
|
|
|
}));
|
2025-02-15 00:37:32 -08:00
|
|
|
|
2025-02-15 02:37:17 -08:00
|
|
|
apiRouter.post("/play", withErrorHandling(async (req, res) => {
|
|
|
|
|
await mediaPlayer.play();
|
|
|
|
|
res.send(JSON.stringify({ success: true }));
|
|
|
|
|
}));
|
2025-02-15 00:37:32 -08:00
|
|
|
|
2025-02-15 02:37:17 -08:00
|
|
|
apiRouter.post("/pause", withErrorHandling(async (req, res) => {
|
|
|
|
|
await mediaPlayer.pause();
|
|
|
|
|
res.send(JSON.stringify({ success: true }));
|
|
|
|
|
}));
|
|
|
|
|
|
|
|
|
|
apiRouter.post("/skip", withErrorHandling(async (req, res) => {
|
|
|
|
|
await mediaPlayer.skip();
|
|
|
|
|
res.send(JSON.stringify({ success: true }));
|
|
|
|
|
}));
|
2025-02-15 00:37:32 -08:00
|
|
|
|
2025-02-15 16:28:47 -08:00
|
|
|
apiRouter.post("/skip/:index", withErrorHandling(async (req, res) => {
|
|
|
|
|
const { index } = req.params as { index: string };
|
|
|
|
|
await mediaPlayer.skipTo(parseInt(index));
|
|
|
|
|
res.send(JSON.stringify({ success: true }));
|
|
|
|
|
}));
|
|
|
|
|
|
2025-02-15 02:37:17 -08:00
|
|
|
apiRouter.post("/previous", withErrorHandling(async (req, res) => {
|
|
|
|
|
await mediaPlayer.previous();
|
|
|
|
|
res.send(JSON.stringify({ success: true }));
|
|
|
|
|
}));
|
|
|
|
|
|
|
|
|
|
apiRouter.get("/nowplaying", withErrorHandling(async (req, res) => {
|
2025-02-15 00:37:32 -08:00
|
|
|
const nowPlaying = await mediaPlayer.getNowPlaying();
|
2025-02-15 02:37:17 -08:00
|
|
|
const currentFile = await mediaPlayer.getCurrentFile();
|
2025-02-15 00:37:32 -08:00
|
|
|
const pauseState = await mediaPlayer.getPauseState();
|
|
|
|
|
const volume = await mediaPlayer.getVolume();
|
|
|
|
|
const idle = await mediaPlayer.getIdle();
|
|
|
|
|
|
|
|
|
|
res.send(JSON.stringify({
|
|
|
|
|
success: true,
|
|
|
|
|
nowPlaying: nowPlaying,
|
|
|
|
|
isPaused: pauseState,
|
|
|
|
|
volume: volume,
|
2025-02-15 02:37:17 -08:00
|
|
|
isIdle: idle,
|
|
|
|
|
currentFile: currentFile
|
2025-02-15 00:37:32 -08:00
|
|
|
}));
|
2025-02-15 02:37:17 -08:00
|
|
|
}));
|
2025-02-15 00:37:32 -08:00
|
|
|
|
2025-02-15 02:37:17 -08:00
|
|
|
apiRouter.post("/volume", withErrorHandling(async (req, res) => {
|
2025-02-15 00:37:32 -08:00
|
|
|
const { volume } = req.body as { volume: number };
|
|
|
|
|
await mediaPlayer.setVolume(volume);
|
|
|
|
|
res.send(JSON.stringify({ success: true }));
|
2025-02-15 02:37:17 -08:00
|
|
|
}));
|
2025-02-15 00:37:32 -08:00
|
|
|
|
2025-02-15 01:26:10 -08:00
|
|
|
apiRouter.ws("/events", (ws, req) => {
|
2025-02-15 02:19:14 -08:00
|
|
|
console.log("Events client connected");
|
2025-02-15 00:37:32 -08:00
|
|
|
mediaPlayer.subscribe(ws);
|
|
|
|
|
|
|
|
|
|
ws.on("close", () => {
|
2025-02-15 02:19:14 -08:00
|
|
|
console.log("Events client disconnected");
|
2025-02-15 00:37:32 -08:00
|
|
|
mediaPlayer.unsubscribe(ws);
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
|
2025-02-15 01:26:10 -08:00
|
|
|
// Serve static files for React app (after building)
|
|
|
|
|
app.use(express.static("dist/frontend"));
|
|
|
|
|
|
|
|
|
|
// Mount API routes under /api
|
|
|
|
|
app.use("/api", apiRouter);
|
|
|
|
|
|
|
|
|
|
// Serve React app for all other routes (client-side routing)
|
|
|
|
|
app.get("*", (req, res) => {
|
|
|
|
|
res.sendFile("dist/frontend/index.html", { root: "." });
|
|
|
|
|
});
|
2025-02-15 00:37:32 -08:00
|
|
|
|
2025-02-15 16:45:59 -08:00
|
|
|
const port = process.env.PORT || 3000;
|
|
|
|
|
const server = app.listen(port, () => {
|
|
|
|
|
console.log(`Server is running on port ${port}`);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// Add graceful shutdown handling
|
|
|
|
|
const shutdown = async () => {
|
|
|
|
|
console.log('Received shutdown signal. Closing server...');
|
|
|
|
|
|
|
|
|
|
server.close(() => {
|
|
|
|
|
console.log('Server closed');
|
|
|
|
|
process.exit(0);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// Force termination after some timeout (10sec)
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
console.log('Forcing server shutdown');
|
|
|
|
|
process.exit(1);
|
|
|
|
|
}, 10000);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Handle various shutdown signals
|
|
|
|
|
process.on('SIGTERM', shutdown);
|
|
|
|
|
process.on('SIGINT', shutdown);
|