Files
ifb-schulapp/index.js
T
Simon adc3ac828f feat: stundenplan page with timetable, exams, and quicklinks
- Static timetable for all 5 classes (Info1/2, Freko1/2, MR)
- Click-to-highlight subject across week grid
- Cell tooltip with full subject, teacher, room, time
- Today list with remaining lesson count and Freistunde gaps
- Klausurenplan with Bayern Ferien dividers
- iCal export button for weekly schedule
- Quicklinks panel (Notenportal)
2026-04-30 08:14:18 +02:00

60 lines
1.9 KiB
JavaScript

require('dotenv').config();
const express = require('express');
const cookieParser = require('cookie-parser');
const helmet = require('helmet');
const path = require('path');
const routes = require('./src/routes');
const { router: filesRouter } = require('./src/files');
const teacherRouter = require('./src/teacher');
const { startNotificationScheduler } = require('./src/push');
if (!process.env.JWT_SECRET) {
console.error('FATAL: JWT_SECRET environment variable is not set.');
process.exit(1);
}
const app = express();
const PORT = 3010;
app.use(helmet({
contentSecurityPolicy: {
directives: {
defaultSrc: ["'self'"],
scriptSrc: ["'self'", "'unsafe-inline'", 'unpkg.com'],
scriptSrcAttr: ["'unsafe-inline'"],
styleSrc: ["'self'", "'unsafe-inline'", 'https://fonts.googleapis.com'],
fontSrc: ["'self'", 'https://fonts.gstatic.com'],
imgSrc: ["'self'", 'data:'],
connectSrc: ["'self'", 'https://api.open-meteo.com'],
workerSrc: ["'self'"],
frameAncestors: ["'none'"],
objectSrc: ["'none'"],
},
},
}));
app.use(express.json());
app.use(cookieParser());
app.get('/sw.js', (req, res, next) => {
res.setHeader('Cache-Control', 'no-cache');
next();
});
app.use(express.static(path.join(__dirname, 'public')));
app.use('/api', routes);
app.use('/api/files', filesRouter);
app.use('/api/teacher', teacherRouter);
const html = f => (req, res) => res.sendFile(path.join(__dirname, 'public', f));
app.get('/login', html('login.html'));
app.get('/admin', html('admin.html'));
app.get('/datenschutz', html('datenschutz.html'));
app.get('/app', html('app.html'));
app.get('/reset-password', html('reset-password.html'));
app.get('/stundenplan', html('stundenplan.html'));
app.get('/{*path}', html('index.html'));
app.listen(PORT, '127.0.0.1', () => {
console.log(`info1 läuft auf :${PORT}`);
startNotificationScheduler();
});