feat: Add a large collection of emoji and other frontend assets, including a sound file, and a backend package.json.

This commit is contained in:
Bryan1029384756
2026-01-06 17:58:56 -06:00
parent f531301863
commit abedd78893
3795 changed files with 10981 additions and 229 deletions

View File

@@ -2,6 +2,7 @@ const express = require('express');
const http = require('http');
const { Server } = require('socket.io');
const cors = require('cors');
const axios = require('axios');
require('dotenv').config();
const app = express();
@@ -15,36 +16,104 @@ const io = new Server(server, {
const authRoutes = require('./routes/auth');
const channelRoutes = require('./routes/channels');
const uploadRoutes = require('./routes/upload');
const inviteRoutes = require('./routes/invites');
const rolesRoutes = require('./routes/roles');
app.use(cors());
app.use(express.json());
// Attach IO to request
app.use((req, res, next) => {
req.io = io;
next();
});
// Serve uploads folder based on relative path from server.js
const path = require('path');
app.use('/uploads', express.static(path.join(__dirname, 'uploads')));
app.use('/api/auth', authRoutes);
app.use('/api/channels', channelRoutes);
app.use('/api/upload', uploadRoutes);
app.use('/api/invites', inviteRoutes);
app.use('/api/invites', inviteRoutes);
app.use('/api/roles', rolesRoutes);
app.use('/api/voice', require('./routes/voice'));
app.get('/', (req, res) => {
res.send('Secure Chat Backend Running');
});
// GIF Routes
const gifCategories = require('./data/gif_categories.json');
app.get('/api/gifs/categories', (req, res) => {
res.json(gifCategories);
});
app.get('/api/gifs/search', async (req, res) => {
const { q, limit = 8 } = req.query;
const apiKey = process.env.TENOR_API_KEY;
if (!apiKey) {
// Return mock data or error if no key
console.warn('TENOR_API_KEY missing in .env');
// Return dummy response to prevent crash
return res.json({ results: [] });
}
try {
const response = await axios.get(`https://tenor.googleapis.com/v2/search`, {
params: {
q,
key: apiKey,
limit
}
});
res.json(response.data);
} catch (err) {
console.error('Tenor API Error:', err.message);
res.status(500).json({ error: 'Failed to fetch GIFs' });
}
});
const redisClient = require('./redis');
const db = require('./db');
io.on('connection', (socket) => {
console.log('User connected:', socket.id);
socket.on('join_channel', async (channelId) => {
socket.on('join_channel', async (data) => {
// Handle both simple string (legacy) and object payload
const channelId = typeof data === 'object' ? data.channelId : data;
const userId = typeof data === 'object' ? data.userId : null;
socket.join(channelId);
console.log(`User ${socket.id} joined channel ${channelId}`);
// Load recent messages
console.log(`User ${socket.id} (ID: ${userId}) joined channel ${channelId}`);
// Load recent messages with reactions
try {
const result = await db.query(
`SELECT m.*, u.username, u.public_signing_key
FROM messages m
JOIN users u ON m.sender_id = u.id
WHERE m.channel_id = $1
ORDER BY m.created_at DESC LIMIT 50`,
[channelId]
);
const query = `
SELECT m.*, u.username, u.public_signing_key,
(
SELECT json_object_agg(res.emoji, res.data)
FROM (
SELECT r.emoji, json_build_object(
'count', COUNT(*)::int,
'me', BOOL_OR(r.user_id = $2)
) as data
FROM message_reactions r
WHERE r.message_id = m.id
GROUP BY r.emoji
) res
) as reactions
FROM messages m
JOIN users u ON m.sender_id = u.id
WHERE m.channel_id = $1
ORDER BY m.created_at DESC LIMIT 50
`;
const result = await db.query(query, [channelId, userId]);
socket.emit('recent_messages', result.rows.reverse());
} catch (err) {
console.error('Error fetching messages:', err);
@@ -83,12 +152,114 @@ io.on('connection', (socket) => {
}
});
socket.on('typing', (data) => {
socket.to(data.channelId).emit('user_typing', { username: data.username });
socket.on('typing_start', (data) => {
io.to(data.channelId).emit('typing_start', data);
});
socket.on('typing_stop', (data) => {
io.to(data.channelId).emit('typing_stop', data);
});
// ... Message handling ...
// Voice State Handling
// Stores: channelId -> Set(serializedUser: {userId, username})
// For simplicity: channelId -> [{userId, username}]
// We need a global variable outside the connection loop, but for this file structure 'socket.on' is inside io.on
// So we need to define it outside.
// See defining it at top of file logic.
// Reactions
socket.on('add_reaction', async ({ channelId, messageId, userId, emoji }) => {
try {
if (!messageId || !userId || !emoji) return;
const result = await db.query(
`INSERT INTO message_reactions (message_id, user_id, emoji)
VALUES ($1, $2, $3)
ON CONFLICT (message_id, user_id, emoji) DO NOTHING`,
[messageId, userId, emoji]
);
if (result.rowCount > 0) {
io.to(channelId).emit('reaction_added', { messageId, userId, emoji });
}
} catch (err) {
console.error('Error adding reaction:', err);
}
});
socket.on('remove_reaction', async ({ channelId, messageId, userId, emoji }) => {
try {
if (!messageId || !userId || !emoji) return;
const result = await db.query(
`DELETE FROM message_reactions
WHERE message_id = $1 AND user_id = $2 AND emoji = $3`,
[messageId, userId, emoji]
);
if (result.rowCount > 0) {
io.to(channelId).emit('reaction_removed', { messageId, userId, emoji });
}
} catch (err) {
console.error('Error removing reaction:', err);
}
});
socket.on('join_voice', (channelId) => { }); // Deprecated/Unused placeholder
socket.on('voice_state_change', (data) => {
// data: { channelId, userId, username, action: 'joined' | 'left' | 'state_update', isMuted, isDeafened }
console.log(`Voice State Update: ${data.username} (${data.userId}) ${data.action} ${data.channelId}`);
// Update Server State
if (!global.voiceStates) global.voiceStates = {};
const currentUsers = global.voiceStates[data.channelId] || [];
if (data.action === 'joined') {
const existingUser = currentUsers.find(u => u.userId === data.userId);
if (!existingUser) {
currentUsers.push({
userId: data.userId,
username: data.username,
isMuted: data.isMuted || false,
isDeafened: data.isDeafened || false
});
} else {
// Update existing on re-join (or just state sync)
existingUser.isMuted = data.isMuted || false;
existingUser.isDeafened = data.isDeafened || false;
}
global.voiceStates[data.channelId] = currentUsers;
} else if (data.action === 'left') {
const index = currentUsers.findIndex(u => u.userId === data.userId);
if (index !== -1) {
currentUsers.splice(index, 1);
global.voiceStates[data.channelId] = currentUsers;
}
} else if (data.action === 'state_update') {
const user = currentUsers.find(u => u.userId === data.userId);
if (user) {
if (data.isMuted !== undefined) user.isMuted = data.isMuted;
if (data.isDeafened !== undefined) user.isDeafened = data.isDeafened;
global.voiceStates[data.channelId] = currentUsers;
}
}
io.emit('voice_state_update', data);
});
socket.on('request_voice_state', () => {
socket.emit('full_voice_state', global.voiceStates || {});
});
socket.on('disconnect', () => {
console.log('User disconnected:', socket.id);
// TODO: Auto-leave logic would require mapping socketId -> userId/channelId
});
});