- Added a new modelsProxyRouter to handle requests for optimized 3D character models. - Introduced modelsProxyService to manage GLB file optimization using gltf-transform with Draco compression. - Updated app.js to include the new modelsProxyRouter for API access. - Enhanced .gitignore to exclude model cache files. - Added scripts for optimizing GLB models and updated README with optimization instructions. - Integrated DRACOLoader in Character3D.vue for loading compressed models. - Updated FamilyView.vue to streamline character rendering logic.
93 lines
3.7 KiB
JavaScript
93 lines
3.7 KiB
JavaScript
import express from 'express';
|
|
import path from 'path';
|
|
import { fileURLToPath } from 'url';
|
|
import crypto from 'crypto';
|
|
import chatRouter from './routers/chatRouter.js';
|
|
import authRouter from './routers/authRouter.js';
|
|
import navigationRouter from './routers/navigationRouter.js';
|
|
import settingsRouter from './routers/settingsRouter.js';
|
|
import adminRouter from './routers/adminRouter.js';
|
|
import contactRouter from './routers/contactRouter.js';
|
|
import socialnetworkRouter from './routers/socialnetworkRouter.js';
|
|
import forumRouter from './routers/forumRouter.js';
|
|
import falukantRouter from './routers/falukantRouter.js';
|
|
import friendshipRouter from './routers/friendshipRouter.js';
|
|
import modelsProxyRouter from './routers/modelsProxyRouter.js';
|
|
import blogRouter from './routers/blogRouter.js';
|
|
import match3Router from './routers/match3Router.js';
|
|
import taxiRouter from './routers/taxiRouter.js';
|
|
import taxiMapRouter from './routers/taxiMapRouter.js';
|
|
import taxiHighscoreRouter from './routers/taxiHighscoreRouter.js';
|
|
import termineRouter from './routers/termineRouter.js';
|
|
import vocabRouter from './routers/vocabRouter.js';
|
|
import cors from 'cors';
|
|
import './jobs/sessionCleanup.js';
|
|
|
|
const __filename = fileURLToPath(import.meta.url);
|
|
const __dirname = path.dirname(__filename);
|
|
|
|
const app = express();
|
|
|
|
// Request-Timing (aktivierbar per ENV)
|
|
// - LOG_SLOW_REQ_MS=200: Logge Requests, die länger dauern als X ms (Default 500)
|
|
// - LOG_ALL_REQ=1: Logge alle Requests
|
|
const LOG_ALL_REQ = process.env.LOG_ALL_REQ === '1';
|
|
const LOG_SLOW_REQ_MS = Number.parseInt(process.env.LOG_SLOW_REQ_MS || '500', 10);
|
|
app.use((req, res, next) => {
|
|
const reqId = req.headers['x-request-id'] || (crypto.randomUUID ? crypto.randomUUID() : crypto.randomBytes(8).toString('hex'));
|
|
req.reqId = reqId;
|
|
res.setHeader('x-request-id', reqId);
|
|
const t0 = Date.now();
|
|
res.on('finish', () => {
|
|
const ms = Date.now() - t0;
|
|
if (LOG_ALL_REQ || ms >= LOG_SLOW_REQ_MS) {
|
|
console.log(`⏱️ REQ ${ms}ms ${res.statusCode} ${req.method} ${req.originalUrl} rid=${reqId}`);
|
|
}
|
|
});
|
|
next();
|
|
});
|
|
|
|
const corsOptions = {
|
|
origin: ['http://localhost:3000', 'http://localhost:5173', 'http://127.0.0.1:3000', 'http://127.0.0.1:5173'],
|
|
methods: ['GET', 'HEAD', 'PUT', 'PATCH', 'POST', 'DELETE'],
|
|
allowedHeaders: ['Content-Type', 'Authorization', 'userId', 'authCode'],
|
|
credentials: true,
|
|
preflightContinue: false,
|
|
optionsSuccessStatus: 204
|
|
};
|
|
|
|
app.use(cors(corsOptions));
|
|
app.use(express.json()); // To handle JSON request bodies
|
|
|
|
app.use('/api/chat', chatRouter);
|
|
app.use('/api/auth', authRouter);
|
|
app.use('/api/navigation', navigationRouter);
|
|
app.use('/api/settings', settingsRouter);
|
|
app.use('/api/admin', adminRouter);
|
|
app.use('/api/match3', match3Router);
|
|
app.use('/api/taxi', taxiRouter);
|
|
app.use('/api/taxi-maps', taxiMapRouter);
|
|
app.use('/api/taxi/highscores', taxiHighscoreRouter);
|
|
app.use('/images', express.static(path.join(__dirname, '../frontend/public/images')));
|
|
app.use('/api/contact', contactRouter);
|
|
app.use('/api/socialnetwork', socialnetworkRouter);
|
|
app.use('/api/vocab', vocabRouter);
|
|
app.use('/api/forum', forumRouter);
|
|
app.use('/api/falukant', falukantRouter);
|
|
app.use('/api/friendships', friendshipRouter);
|
|
app.use('/api/models', modelsProxyRouter);
|
|
app.use('/api/blog', blogRouter);
|
|
app.use('/api/termine', termineRouter);
|
|
|
|
// Serve frontend SPA for non-API routes to support history mode clean URLs
|
|
const frontendDir = path.join(__dirname, '../frontend');
|
|
app.use(express.static(path.join(frontendDir, 'dist')));
|
|
app.get(/^\/(?!api\/).*/, (req, res) => {
|
|
res.sendFile(path.join(frontendDir, 'dist', 'index.html'));
|
|
});
|
|
|
|
// Fallback 404 for unknown API routes
|
|
app.use('/api/*', (req, res) => res.status(404).send('404 Not Found'));
|
|
|
|
export default app;
|