test / server /server.js
akborana4's picture
Update server/server.js
499ee47 verified
raw
history blame
17.6 kB
import express from 'express';
import http from 'http';
import cors from 'cors';
import { Server as SocketIOServer } from 'socket.io';
import path from 'path';
import { fileURLToPath } from 'url';
import axios from 'axios';
import NodeCache from 'node-cache';
import {
searchUniversal,
getSong,
getAlbum,
getPlaylist,
getLyrics
} from './jiosaavn.js';
const __filename = fileURLToPath(import.meta.url);
const __dirname = path.dirname(__filename);
const app = express();
// CORS for API endpoints (the client is same origin; this also helps external embeds)
app.use(cors({ origin: '*', credentials: true }));
app.use(express.json());
// ---------------- Cache ----------------
const vidflyCache = new NodeCache({ stdTTL: 300, checkperiod: 60 });
// ---------------- Health ----------------
app.get('/healthz', (_req, res) => res.send('OK'));
// ---------------- Media proxy (Range-aware, CORS-safe) ----------------
// Use this for all Vidfly media URLs to bypass CORS.
app.get('/api/proxy', async (req, res) => {
const target = req.query.url;
if (!target) return res.status(400).send('Missing url');
try {
// Forward Range and basic headers for streaming/seeking
const headers = {};
if (req.headers.range) headers.Range = req.headers.range;
if (req.headers['user-agent']) headers['User-Agent'] = req.headers['user-agent'];
if (req.headers['accept']) headers['Accept'] = req.headers['accept'];
if (req.headers['accept-encoding']) headers['Accept-Encoding'] = req.headers['accept-encoding'];
if (req.headers['accept-language']) headers['Accept-Language'] = req.headers['accept-language'];
// Some CDNs check Referer; set a generic one if not present
headers.Referer = req.headers.referer || 'https://www.youtube.com/';
const upstream = await axios.get(target, {
responseType: 'stream',
headers,
// We need to forward non-200 statuses for partial content
validateStatus: () => true,
maxRedirects: 5
});
// CORS headers for the proxied response (so <video> can consume it)
res.setHeader('Access-Control-Allow-Origin', '*');
res.setHeader('Access-Control-Allow-Credentials', 'true');
// Mirror status and key headers (content-range enables seeking)
res.status(upstream.status);
const passthroughHeaders = [
'content-type',
'content-length',
'content-range',
'accept-ranges',
'etag',
'last-modified',
'cache-control',
'expires',
'date',
'server'
];
for (const h of passthroughHeaders) {
const v = upstream.headers[h];
if (v) res.setHeader(h, v);
}
upstream.data.pipe(res);
} catch (e) {
res.status(502).send('Proxy fetch failed');
}
});
// ---------------- YouTube via Vidfly (no yt-dlp) ----------------
// GET /api/yt/source?url=<watchURL or 11-char ID>
app.get('/api/yt/source', async (req, res) => {
try {
const raw = (req.query.url || '').trim();
if (!raw) return res.status(400).json({ error: 'Missing url' });
// Normalize 11-char ID to full watch URL
let watchUrl = raw;
if (/^[A-Za-z0-9_-]{11}$/.test(raw)) {
watchUrl = `https://www.youtube.com/watch?v=${raw}`;
}
// Cache
const cacheKey = `vidfly:${watchUrl}`;
let info = vidflyCache.get(cacheKey);
if (!info) {
// Vidfly API
const resp = await axios.get('https://api.vidfly.ai/api/media/youtube/download', {
params: { url: watchUrl },
timeout: 20000
});
info = resp.data;
vidflyCache.set(cacheKey, info);
}
// Parse Vidfly shape you provided:
// { code: 0, data: { cover, duration, title?, items: [ { ext, type, height, label, url, ... }, ... ] } }
if (info?.code !== 0 || !info?.data?.items?.length) {
return res.status(502).json({ error: 'Vidfly: No playable stream' });
}
const data = info.data;
const items = Array.isArray(data.items) ? data.items : [];
// Prefer video_with_audio mp4 with highest height
const muxed = items
.filter(i => i?.type === 'video_with_audio' && i?.ext === 'mp4' && i?.url)
.sort((a, b) => (b.height || 0) - (a.height || 0))[0];
// Fallback: best "audio" (if provided)
const bestAudio = items
.filter(i => i?.type === 'audio' && i?.url)
.sort((a, b) => {
const ba = parseInt(String(a.bitrate || '').replace(/\D/g, '') || '0', 10);
const bb = parseInt(String(b.bitrate || '').replace(/\D/g, '') || '0', 10);
return bb - ba;
})[0];
const chosen = muxed || bestAudio;
if (!chosen?.url) {
return res.status(502).json({ error: 'Vidfly: No playable stream url' });
}
// Return the raw URL (client will route via /api/proxy automatically on error)
return res.json({
url: chosen.url,
title: data.title || watchUrl,
thumbnail: data.cover || null,
duration: data.duration || null,
kind: muxed ? 'video' : 'audio',
source: 'vidfly'
});
} catch (e) {
return res.status(500).json({ error: e.message || 'Vidfly resolve failed' });
}
});
// ---------------- YouTube search (Data API v3) ----------------
// Set env YT_API_KEY to enable
app.get('/api/ytsearch', async (req, res) => {
try {
const key = process.env.YT_API_KEY;
if (!key) return res.status(501).json({ error: 'YouTube search unavailable: set YT_API_KEY' });
const q = String(req.query.q || '').trim();
if (!q) return res.json({ items: [] });
const resp = await axios.get('https://www.googleapis.com/youtube/v3/search', {
params: { key, q, part: 'snippet', type: 'video', maxResults: 12 },
timeout: 10000
});
const items = (resp.data.items || [])
.map(it => ({
videoId: it.id?.videoId,
title: it.snippet?.title,
channelTitle: it.snippet?.channelTitle,
thumb: it.snippet?.thumbnails?.medium?.url || it.snippet?.thumbnails?.default?.url
}))
.filter(x => x.videoId);
res.json({ items });
} catch (e) {
res.status(500).json({ error: e.message || 'YouTube search failed' });
}
});
// ---------------- JioSaavn proxies ----------------
app.get('/api/result', async (req, res) => {
try {
const q = req.query.q || '';
const data = await searchUniversal(q);
res.json(data);
} catch (e) { res.status(500).json({ error: e.message }); }
});
app.get('/api/song', async (req, res) => {
try {
const q = req.query.q || '';
const data = await getSong(q);
res.json(data);
} catch (e) { res.status(500).json({ error: e.message }); }
});
app.get('/api/album', async (req, res) => {
try {
const q = req.query.q || '';
const data = await getAlbum(q);
res.json(data);
} catch (e) { res.status(500).json({ error: e.message }); }
});
app.get('/api/playlist', async (req, res) => {
try {
const q = req.query.q || '';
const data = await getPlaylist(q);
res.json(data);
} catch (e) { res.status(500).json({ error: e.message }); }
});
app.get('/api/lyrics', async (req, res) => {
try {
const q = req.query.q || '';
const data = await getLyrics(q);
res.json(data);
} catch (e) { res.status(500).json({ error: e.message }); }
});
// ---------------- Rooms, lobby, and sync ----------------
/*
room = {
id, name, hostId,
users: Map<socketId, { name, role: 'host'|'cohost'|'member', muted?: boolean }>,
track: { url, title, meta, kind, thumb? },
isPlaying, anchor, anchorAt,
queue: Array<track>
}
*/
const rooms = new Map();
function ensureRoom(roomId) {
if (!rooms.has(roomId)) {
rooms.set(roomId, {
id: roomId,
name: null,
hostId: null,
users: new Map(),
track: null,
isPlaying: false,
anchor: 0,
anchorAt: 0,
queue: []
});
}
return rooms.get(roomId);
}
function currentState(room) {
return {
track: room.track,
isPlaying: room.isPlaying,
anchor: room.anchor,
anchorAt: room.anchorAt,
queue: room.queue
};
}
function membersPayload(room) {
return [...room.users.entries()].map(([id, u]) => ({
id,
name: u.name || 'Guest',
role: u.role || 'member',
muted: !!u.muted,
isHost: id === room.hostId
}));
}
function broadcastMembers(roomId) {
const room = rooms.get(roomId);
if (!room) return;
const members = membersPayload(room);
io.to(roomId).emit('members', { members, roomName: room.name || room.id });
}
function requireHostOrCohost(room, socketId) {
const u = room.users.get(socket.id);
return (socketId === room.hostId) || (u && u.role === 'cohost');
}
function normalizeThumb(meta) {
return meta?.thumb || meta?.image || meta?.thumbnail || null;
}
function findUserByName(room, targetNameRaw) {
if (!targetNameRaw) return null;
const targetName = String(targetNameRaw).replace(/^@/, '').trim().toLowerCase();
for (const [id, u] of room.users.entries()) {
if ((u.name || '').toLowerCase() === targetName) return { id, u };
}
return null;
}
// Lobby
app.get('/api/rooms', (_req, res) => {
const data = [...rooms.values()]
.filter(r => r.users.size > 0)
.map(r => ({
id: r.id,
name: r.name || r.id,
members: r.users.size,
isPlaying: r.isPlaying
}));
res.json({ rooms: data });
});
app.get('/api/ping', (_req, res) => res.json({ ok: true }));
// Static client
const clientDir = path.resolve(__dirname, '../client/dist');
app.use(express.static(clientDir));
app.get('*', (_req, res) => res.sendFile(path.join(clientDir, 'index.html')));
// Socket.IO realtime
const server = http.createServer(app);
const io = new SocketIOServer(server, {
cors: { origin: '*', methods: ['GET', 'POST'] }
});
io.on('connection', (socket) => {
let joinedRoom = null;
socket.on('join_room', ({ roomId, name, asHost = false, roomName = null }, ack) => {
const room = ensureRoom(roomId);
if (asHost || !room.hostId) room.hostId = socket.id;
if (!room.name && roomName) room.name = String(roomName).trim().slice(0, 60) || null;
const role = socket.id === room.hostId ? 'host' : 'member';
const cleanName = String(name || 'Guest').slice(0, 40);
room.users.set(socket.id, { name: cleanName, role });
socket.join(roomId);
joinedRoom = roomId;
ack?.({
roomId,
isHost: socket.id === room.hostId,
state: currentState(room),
roomName: room.name || room.id
});
io.to(roomId).emit('system', { text: `System: ${cleanName} has joined the chat` });
broadcastMembers(roomId);
});
socket.on('rename', ({ roomId, newName }) => {
const room = rooms.get(roomId);
if (!room) return;
const u = room.users.get(socket.id);
if (!u) return;
u.name = String(newName || 'Guest').slice(0, 40);
broadcastMembers(roomId);
});
socket.on('chat_message', ({ roomId, name, text }) => {
const room = rooms.get(roomId);
if (!room) return;
const u = room.users.get(socket.id);
if (!u) return;
if (u.muted) {
socket.emit('system', { text: 'System: You are muted by the host' });
return;
}
socket.to(roomId).emit('chat_message', { name, text, at: Date.now() });
});
socket.on('song_request', ({ roomId, requester, query }) => {
const room = rooms.get(roomId);
if (!room) return;
const payload = {
requester,
query,
at: Date.now(),
requestId: `${Date.now()}_${Math.random().toString(36).slice(2)}`
};
for (const [id, usr] of room.users.entries()) {
if (id === room.hostId || usr.role === 'cohost') io.to(id).emit('song_request', payload);
}
io.to(roomId).emit('system', { text: `System: ${requester} requested /play ${query}` });
});
socket.on('song_request_action', ({ roomId, action, track }) => {
const room = rooms.get(roomId);
if (!room) return;
const actor = room.users.get(socket.id);
if (!actor || !(socket.id === room.hostId || actor.role === 'cohost')) return;
if ((action === 'accept' || action === 'queue') && track) {
const t = {
url: track.url,
title: track.title || track.url,
meta: track.meta || {},
kind: track.kind || 'audio',
thumb: track.thumb || normalizeThumb(track.meta || {})
};
if (action === 'accept') {
room.track = t;
room.isPlaying = false;
room.anchor = 0;
room.anchorAt = Date.now();
io.to(roomId).emit('set_track', { track: room.track });
io.to(roomId).emit('pause', { anchor: 0, anchorAt: room.anchorAt });
io.to(roomId).emit('system', { text: `System: Now playing ${room.track.title}` });
} else {
room.queue.push(t);
io.to(roomId).emit('queue_update', { queue: room.queue });
io.to(roomId).emit('system', { text: `System: Queued ${t.title}` });
}
}
});
socket.on('set_track', ({ roomId, track }) => {
const room = rooms.get(roomId);
const actor = room.users.get(socket.id);
if (!room || !(socket.id === room.hostId || actor?.role === 'cohost')) return;
const thumb = track.thumb || normalizeThumb(track.meta || {});
room.track = { ...track, thumb };
room.isPlaying = false;
room.anchor = 0;
room.anchorAt = Date.now();
io.to(roomId).emit('set_track', { track: room.track });
io.to(roomId).emit('pause', { anchor: 0, anchorAt: room.anchorAt });
io.to(roomId).emit('system', { text: `System: Selected ${room.track.title || room.track.url}` });
});
socket.on('play', ({ roomId }) => {
const room = rooms.get(roomId);
const actor = room.users.get(socket.id);
if (!room || !(socket.id === room.hostId || actor?.role === 'cohost')) return;
room.isPlaying = true;
room.anchorAt = Date.now();
io.to(roomId).emit('play', { anchor: room.anchor, anchorAt: room.anchorAt });
});
socket.on('pause', ({ roomId }) => {
const room = rooms.get(roomId);
const actor = room.users.get(socket.id);
if (!room || !(socket.id === room.hostId || actor?.role === 'cohost')) return;
const elapsed = Math.max(0, Date.now() - room.anchorAt) / 1000;
if (room.isPlaying) room.anchor += elapsed;
room.isPlaying = false;
room.anchorAt = Date.now();
io.to(roomId).emit('pause', { anchor: room.anchor, anchorAt: room.anchorAt });
});
socket.on('seek', ({ roomId, to }) => {
const room = rooms.get(roomId);
const actor = room.users.get(socket.id);
if (!room || !(socket.id === room.hostId || actor?.role === 'cohost')) return;
room.anchor = to;
room.anchorAt = Date.now();
io.to(roomId).emit('seek', { anchor: room.anchor, anchorAt: room.anchorAt, isPlaying: room.isPlaying });
});
socket.on('ended', ({ roomId }) => {
const room = rooms.get(roomId);
const actor = room.users.get(socket.id);
if (!room || !(socket.id === room.hostId || actor?.role === 'cohost')) return;
const next = room.queue.shift();
if (next) {
const thumb = next.thumb || normalizeThumb(next.meta || {});
room.track = { ...next, thumb };
room.isPlaying = false;
room.anchor = 0;
room.anchorAt = Date.now();
io.to(roomId).emit('set_track', { track: room.track });
io.to(roomId).emit('pause', { anchor: 0, anchorAt: room.anchorAt });
io.to(roomId).emit('queue_update', { queue: room.queue });
io.to(roomId).emit('system', { text: `System: Now playing ${room.track.title}` });
} else {
io.to(roomId).emit('system', { text: 'System: Queue ended' });
}
});
socket.on('admin_command', ({ roomId, cmd, targetName }) => {
const room = rooms.get(roomId);
const actor = room?.users.get(socket.id);
if (!actor || !(socket.id === room.hostId || actor.role === 'cohost')) return;
const found = findUserByName(room, targetName);
if (!found) {
return io.to(socket.id).emit('system', { text: `System: @${targetName} not found` });
}
const { id, u } = found;
if (cmd === 'kick') {
io.to(id).emit('system', { text: 'System: You were kicked' });
io.sockets.sockets.get(id)?.leave(roomId);
room.users.delete(id);
io.to(roomId).emit('system', { text: `System: ${u.name} was kicked` });
broadcastMembers(roomId);
} else if (cmd === 'promote') {
u.role = 'cohost';
io.to(roomId).emit('system', { text: `System: ${u.name} promoted to co-host` });
broadcastMembers(roomId);
} else if (cmd === 'mute') {
u.muted = true;
io.to(roomId).emit('system', { text: `System: ${u.name} was muted` });
broadcastMembers(roomId);
}
});
socket.on('sync_request', ({ roomId }, ack) => {
const room = rooms.get(roomId);
if (room) ack(currentState(room));
});
socket.on('disconnect', () => {
if (!joinedRoom) return;
const room = rooms.get(joinedRoom);
if (!room) return;
const leftUser = room.users.get(socket.id);
room.users.delete(socket.id);
if (socket.id === room.hostId) {
const nextHost = [...room.users.keys()][0] || null;
room.hostId = nextHost;
if (nextHost) {
const u = room.users.get(nextHost);
if (u) u.role = 'host';
}
io.to(joinedRoom).emit('host_changed', { hostId: room.hostId });
}
if (leftUser) {
const nm = leftUser.name || 'User';
io.to(joinedRoom).emit('system', { text: `System: ${nm} has left the chat` });
}
if (room.users.size === 0) rooms.delete(joinedRoom);
else broadcastMembers(joinedRoom);
});
});
const PORT = process.env.PORT || 3000;
server.listen(PORT, () => {
console.log(`Server running on port ${PORT}`);
});