Files
Tarokka/server.ts
Gavin McDonald 0b68fd5912 looking at IPs
2025-04-22 08:00:59 -04:00

112 lines
3.3 KiB
TypeScript

import next from 'next';
import { createServer } from 'http';
import { Server as SocketIOServer, type Socket } from 'socket.io';
import GameStore from '@/lib/GameStore';
import omit from '@/tools/omit';
import type { ClientUpdate, GameUpdate } from '@/types';
const dev = process.env.NODE_ENV !== 'production';
const hostname = '0.0.0.0';
const port = 3000;
const app = next({ dev, hostname, port });
const handler = app.getRequestHandler();
const gameStore = new GameStore();
app.prepare().then(() => {
const httpServer = createServer(handler);
const io = new SocketIOServer(httpServer);
const broadcast = (event: string, gameUpdate: GameUpdate) => {
io.to(gameUpdate.dmID).emit(event, gameUpdate);
io.to(gameUpdate.spectatorID).emit(event, omit(gameUpdate, 'dmID'));
};
io.on('connection', (socket: Socket) => {
console.log(Date.now(), `Client connected: ${socket.id}`);
socket.on('start', () => {
const gameUpdate = gameStore.createGame();
console.log(Date.now(), `Socket ${socket.id} started game ${gameUpdate.dmID}`);
socket.emit('new-game', gameUpdate);
});
socket.on('join', (gameID) => {
try {
const gameUpdate = gameStore.joinGame(gameID, socket.id);
const ipAddress = Array.isArray(socket.handshake.headers['x-forwarded-for'])
? socket.handshake.headers['x-forwarded-for'][0]
: socket.handshake.headers['x-forwarded-for']?.split(',')[0];
console.log(Date.now(), `Socket ${socket.id} joined game ${gameID}`);
console.log('x-forwarded-for', socket.handshake.headers['x-forwarded-for']);
console.log('client IP', ipAddress);
console.log('proxy IP', socket.handshake.address);
socket.join(gameID);
if (gameID === gameUpdate.spectatorID) {
socket.emit('init', omit(gameUpdate, 'dmID'));
} else {
socket.emit('init', gameUpdate);
}
} catch (e) {
const error = e instanceof Error ? e.message : e;
console.error(Date.now(), 'Error[join]', error);
socket.emit('join-error', error);
}
});
socket.on('flip-card', ({ gameID, cardIndex }: ClientUpdate) => {
try {
console.log(Date.now(), 'Card flipped:', { gameID, cardIndex });
const gameUpdate = gameStore.flipCard(gameID, cardIndex);
broadcast('game-update', gameUpdate);
} catch (e) {
const error = e instanceof Error ? e.message : e;
console.error(Date.now(), 'Error[flip-card]', error);
socket.emit('flip-error', error);
}
});
socket.on('settings', ({ gameID, gameData }: { gameID: string; gameData: GameUpdate }) => {
try {
const gameUpdate = gameStore.updateSettings(gameID, gameData.settings);
broadcast('game-update', gameUpdate);
} catch (e) {
const error = e instanceof Error ? e.message : e;
console.error(Date.now(), 'Error[settings]', error);
}
});
socket.on('disconnect', () => {
try {
const { dmID } = gameStore.playerExit(socket.id);
console.log(Date.now(), `Client disconnected: ${socket.id} from ${dmID}`);
} catch (e) {
const error = e instanceof Error ? e.message : e;
console.error(Date.now(), 'Error[disconnect]', error);
}
});
});
httpServer
.once('error', (err) => {
console.error(Date.now(), 'Server error:', err);
process.exit(1);
})
.listen(port, () => {
console.log(Date.now(), `> Ready on http://${hostname}:${port}`);
});
});