Improvement of logout. Added Sacrital Service. Added website link for event places and direct link to other websites in worship overview
This commit is contained in:
@@ -13,7 +13,7 @@ const sequelize = new Sequelize('miriamgemeinde', 'miriam_user', 'qTCTTWwpEwy3vP
|
|||||||
/SequelizeInvalidConnectionError/,
|
/SequelizeInvalidConnectionError/,
|
||||||
/SequelizeConnectionTimedOutError/
|
/SequelizeConnectionTimedOutError/
|
||||||
],
|
],
|
||||||
max: 5 // Maximal 5 Versuche
|
max: 5
|
||||||
},
|
},
|
||||||
pool: {
|
pool: {
|
||||||
max: 5,
|
max: 5,
|
||||||
|
|||||||
@@ -1,18 +1,16 @@
|
|||||||
const bcrypt = require('bcryptjs');
|
const bcrypt = require('bcryptjs');
|
||||||
const { User } = require('../models');
|
const { User } = require('../models');
|
||||||
const jwt = require('jsonwebtoken');
|
const jwt = require('jsonwebtoken');
|
||||||
|
const { addTokenToBlacklist } = require('../utils/blacklist');
|
||||||
|
|
||||||
exports.register = async (req, res) => {
|
exports.register = async (req, res) => {
|
||||||
const { name, email, password } = req.body;
|
const { name, email, password } = req.body;
|
||||||
|
|
||||||
if (!name || !email || !password) {
|
if (!name || !email || !password) {
|
||||||
return res.status(400).json({ message: 'Alle Felder sind erforderlich' });
|
return res.status(400).json({ message: 'Alle Felder sind erforderlich' });
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const hashedPassword = await bcrypt.hash(password, 10);
|
const hashedPassword = await bcrypt.hash(password, 10);
|
||||||
const user = await User.create({ name, email, password: hashedPassword, active: true });
|
const user = await User.create({ name, email, password: hashedPassword, active: true });
|
||||||
|
|
||||||
res.status(201).json({ message: 'Benutzer erfolgreich registriert', user });
|
res.status(201).json({ message: 'Benutzer erfolgreich registriert', user });
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
if (error.name === 'SequelizeUniqueConstraintError') {
|
if (error.name === 'SequelizeUniqueConstraintError') {
|
||||||
@@ -27,28 +25,36 @@ exports.login = async (req, res) => {
|
|||||||
if (!email || !password) {
|
if (!email || !password) {
|
||||||
return res.status(400).json({ message: 'Email und Passwort sind erforderlich' });
|
return res.status(400).json({ message: 'Email und Passwort sind erforderlich' });
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const user = await User.findOne({ where: { email } });
|
const user = await User.findOne({ where: { email } });
|
||||||
|
|
||||||
if (!user) {
|
if (!user) {
|
||||||
return res.status(401).json({ message: 'Ungültige Anmeldedaten' });
|
return res.status(401).json({ message: 'Ungültige Anmeldedaten' });
|
||||||
}
|
}
|
||||||
|
|
||||||
const validPassword = await bcrypt.compare(password, user.password);
|
const validPassword = await bcrypt.compare(password, user.password);
|
||||||
|
|
||||||
if (!validPassword) {
|
if (!validPassword) {
|
||||||
return res.status(401).json({ message: 'Ungültige Anmeldedaten' });
|
return res.status(401).json({ message: 'Ungültige Anmeldedaten' });
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!user.active) {
|
if (!user.active) {
|
||||||
return res.status(403).json({ message: 'Benutzerkonto ist nicht aktiv' });
|
return res.status(403).json({ message: 'Benutzerkonto ist nicht aktiv' });
|
||||||
}
|
}
|
||||||
|
|
||||||
const token = jwt.sign({ id: user.id, name: user.name, email: user.email }, 'zTxVgptmPl9!_dr%xxx9999(dd)', { expiresIn: '1h' });
|
const token = jwt.sign({ id: user.id, name: user.name, email: user.email }, 'zTxVgptmPl9!_dr%xxx9999(dd)', { expiresIn: '1h' });
|
||||||
|
|
||||||
res.status(200).json({ message: 'Login erfolgreich', token, 'user': user });
|
res.status(200).json({ message: 'Login erfolgreich', token, 'user': user });
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
res.status(500).json({ message: 'Ein Fehler ist aufgetreten' });
|
res.status(500).json({ message: 'Ein Fehler ist aufgetreten' });
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
exports.logout = async (req, res) => {
|
||||||
|
const authHeader = req.header('Authorization');
|
||||||
|
if (!authHeader) {
|
||||||
|
return res.status(400).json({ message: 'Kein Token bereitgestellt' });
|
||||||
|
}
|
||||||
|
const token = authHeader.replace('Bearer ', '');
|
||||||
|
try {
|
||||||
|
addTokenToBlacklist(token);
|
||||||
|
res.status(200).json({ message: 'Logout erfolgreich' });
|
||||||
|
} catch (error) {
|
||||||
|
console.log(error);
|
||||||
|
res.status(500).json({ message: 'Ein Fehler ist beim Logout aufgetreten' });
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|||||||
@@ -1,14 +1,40 @@
|
|||||||
const { Worship, EventPlace, Sequelize } = require('../models');
|
const { Worship, EventPlace, Sequelize, sequelize } = require('../models');
|
||||||
const { Op, fn, literal } = require('sequelize'); // Importieren Sie die Operatoren von Sequelize
|
const { Op, fn, literal } = require('sequelize');
|
||||||
|
const jwt = require('jsonwebtoken');
|
||||||
|
const { isTokenBlacklisted, addTokenToBlacklist } = require('../utils/blacklist');
|
||||||
|
|
||||||
|
function isAuthorized(req) {
|
||||||
|
const authHeader = req.header('Authorization');
|
||||||
|
if (!authHeader) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
const token = authHeader.replace('Bearer ', '');
|
||||||
|
if (isTokenBlacklisted(token)) {
|
||||||
|
console.log('Token is blacklisted');
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
const decoded = jwt.verify(token, 'zTxVgptmPl9!_dr%xxx9999(dd)');
|
||||||
|
req.user = decoded;
|
||||||
|
return true;
|
||||||
|
} catch (err) {
|
||||||
|
console.log('Token verification failed, adding to blacklist:', err.message);
|
||||||
|
addTokenToBlacklist(token);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
exports.getAllWorships = async (req, res) => {
|
exports.getAllWorships = async (req, res) => {
|
||||||
try {
|
try {
|
||||||
|
const authorized = isAuthorized(req);
|
||||||
const worships = await Worship.findAll({
|
const worships = await Worship.findAll({
|
||||||
where: {
|
where: {
|
||||||
date: {
|
date: {
|
||||||
[Op.gt]: literal("DATE_SUB(NOW(), INTERVAL 4 WEEK)")
|
[Op.gt]: literal("DATE_SUB(NOW(), INTERVAL 4 WEEK)")
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
attributes: authorized ? undefined : { exclude: ['sacristanService'] },
|
||||||
order: [
|
order: [
|
||||||
['date', 'DESC']
|
['date', 'DESC']
|
||||||
],
|
],
|
||||||
@@ -69,14 +95,15 @@ exports.getFilteredWorships = async (req, res) => {
|
|||||||
[Sequelize.Op.in]: locations
|
[Sequelize.Op.in]: locations
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
where.date = {
|
where.date = {
|
||||||
[Op.gte]: fn('CURDATE'),
|
[Op.gte]: fn('CURDATE'),
|
||||||
};
|
};
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
const authorized = isAuthorized(req);
|
||||||
|
console.log(authorized);
|
||||||
const worships = await Worship.findAll({
|
const worships = await Worship.findAll({
|
||||||
where,
|
where,
|
||||||
|
attributes: authorized ? undefined : { exclude: ['sacristanService'] },
|
||||||
include: {
|
include: {
|
||||||
model: EventPlace,
|
model: EventPlace,
|
||||||
as: 'eventPlace',
|
as: 'eventPlace',
|
||||||
|
|||||||
@@ -1,12 +1,15 @@
|
|||||||
const jwt = require('jsonwebtoken');
|
const jwt = require('jsonwebtoken');
|
||||||
|
const { isTokenBlacklisted } = require('../utils/blacklist');
|
||||||
|
|
||||||
const authMiddleware = (req, res, next) => {
|
const authMiddleware = (req, res, next) => {
|
||||||
const authHeader = req.header('Authorization');
|
const authHeader = req.header('Authorization');
|
||||||
if (!authHeader) {
|
if (!authHeader) {
|
||||||
return res.status(401).json({ message: 'Zugriff verweigert. Kein Token vorhanden.' });
|
return res.status(401).json({ message: 'Zugriff verweigert. Kein Token vorhanden.' });
|
||||||
}
|
}
|
||||||
|
|
||||||
const token = authHeader.replace('Bearer ', '');
|
const token = authHeader.replace('Bearer ', '');
|
||||||
|
if (isTokenBlacklisted(token)) {
|
||||||
|
return res.status(401).json({ message: 'Token wurde gesperrt.' });
|
||||||
|
}
|
||||||
try {
|
try {
|
||||||
const decoded = jwt.verify(token, 'zTxVgptmPl9!_dr%xxx9999(dd)');
|
const decoded = jwt.verify(token, 'zTxVgptmPl9!_dr%xxx9999(dd)');
|
||||||
req.user = decoded;
|
req.user = decoded;
|
||||||
|
|||||||
@@ -21,7 +21,11 @@ module.exports = (sequelize) => {
|
|||||||
backgroundColor: {
|
backgroundColor: {
|
||||||
type: DataTypes.STRING,
|
type: DataTypes.STRING,
|
||||||
allowNull: true
|
allowNull: true
|
||||||
}
|
},
|
||||||
|
website: {
|
||||||
|
type: DataTypes.STRING,
|
||||||
|
allowNull: true
|
||||||
|
},
|
||||||
}, {
|
}, {
|
||||||
tableName: 'event_places'
|
tableName: 'event_places'
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -55,7 +55,12 @@ module.exports = (sequelize) => {
|
|||||||
type: DataTypes.STRING,
|
type: DataTypes.STRING,
|
||||||
defaultValue: '',
|
defaultValue: '',
|
||||||
allowNull: false
|
allowNull: false
|
||||||
}
|
},
|
||||||
|
sacristanService: {
|
||||||
|
type: DataTypes.STRING(100),
|
||||||
|
allowNull: true,
|
||||||
|
field: 'sacristan_service'
|
||||||
|
},
|
||||||
}, {
|
}, {
|
||||||
tableName: 'worships',
|
tableName: 'worships',
|
||||||
timestamps: true
|
timestamps: true
|
||||||
|
|||||||
@@ -1,8 +1,10 @@
|
|||||||
const express = require('express');
|
const express = require('express');
|
||||||
const router = express.Router();
|
const router = express.Router();
|
||||||
const authController = require('../controllers/authController');
|
const authController = require('../controllers/authController');
|
||||||
|
const authMiddleware = require('../middleware/authMiddleware');
|
||||||
|
|
||||||
router.post('/register', authController.register);
|
router.post('/register', authController.register);
|
||||||
router.post('/login', authController.login);
|
router.post('/login', authController.login);
|
||||||
|
router.post('/logout', authMiddleware, authController.logout);
|
||||||
|
|
||||||
module.exports = router;
|
module.exports = router;
|
||||||
|
|||||||
@@ -24,7 +24,7 @@ axios.interceptors.response.use(
|
|||||||
},
|
},
|
||||||
error => {
|
error => {
|
||||||
if (error.response && error.response.status === 401) {
|
if (error.response && error.response.status === 401) {
|
||||||
store.commit('logout');
|
store.dispatch('logout');
|
||||||
router.push('/');
|
router.push('/');
|
||||||
}
|
}
|
||||||
return Promise.reject(error);
|
return Promise.reject(error);
|
||||||
|
|||||||
@@ -21,8 +21,12 @@ export default {
|
|||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
...mapActions(['logout']),
|
...mapActions(['logout']),
|
||||||
navigateToLogin() {
|
async handleLogout() {
|
||||||
this.$router.push('/login');
|
try {
|
||||||
|
await this.logout();
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Fehler beim Logout:', error);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,24 +1,32 @@
|
|||||||
<template>
|
<template>
|
||||||
<div>
|
<div>
|
||||||
<table v-if="worships.length" class="worships">
|
<table v-if="worships.length" class="worships">
|
||||||
<tr v-for="worship in worships" :key="worship.id" :style="worship.eventPlace && worship.eventPlace.backgroundColor ? `background-color:${worship.eventPlace.backgroundColor}` : ''">
|
<tr v-for="worship in worships" :key="worship.id"
|
||||||
|
:style="worship.eventPlace && worship.eventPlace.backgroundColor ? `background-color:${worship.eventPlace.backgroundColor}` : ''">
|
||||||
<td>
|
<td>
|
||||||
<div>{{ formatDate(worship.date) }}</div>
|
<div>{{ formatDate(worship.date) }}</div>
|
||||||
<div>{{ worship.dayName }}</div>
|
<div>{{ worship.dayName }}</div>
|
||||||
</td>
|
</td>
|
||||||
<td>
|
<td>
|
||||||
<div v-if="worship.neighborInvitation" class="neighborhood-invitation">Einladung zum Gottesdienst im Nachbarschaftsraum:</div>
|
<div v-if="worship.neighborInvitation" class="neighborhood-invitation">Einladung zum Gottesdienst im
|
||||||
|
Nachbarschaftsraum:</div>
|
||||||
<h3>
|
<h3>
|
||||||
<span :class="worship.highlightTime ? 'highlight-time' : ''">{{ formatTime(worship.time) }}</span> -
|
<span :class="worship.highlightTime ? 'highlight-time' : ''">{{ formatTime(worship.time)
|
||||||
|
}}</span> -
|
||||||
{{ !worship.neighborInvitation ? worship.title : `Gottesdienst in ${worship.eventPlace.name}` }}
|
{{ !worship.neighborInvitation ? worship.title : `Gottesdienst in ${worship.eventPlace.name}` }}
|
||||||
</h3>
|
</h3>
|
||||||
<div v-if="worship.organizer">Gestaltung: {{ worship.organizer }}</div>
|
<div v-if="worship.organizer">Gestaltung: {{ worship.organizer }}</div>
|
||||||
|
<div v-if="worship.sacristanService" class="internal-information">Küsterdienst: {{ worship.sacristanService }}</div>
|
||||||
<div v-if="worship.collection">Kollekte: {{ worship.collection }}</div>
|
<div v-if="worship.collection">Kollekte: {{ worship.collection }}</div>
|
||||||
<div v-if="worship.address">{{ worship.address }}</div>
|
<div v-if="worship.address">{{ worship.address }}</div>
|
||||||
<div v-if="!worship.address && worship.eventPlace.id && worship.eventPlace.id">
|
<div v-if="!worship.address && worship.eventPlace.id && worship.eventPlace.id">
|
||||||
Adresse: {{ worship.eventPlace.name }}, {{ worship.eventPlace.street }}, {{ worship.eventPlace.city }}
|
Adresse: {{ worship.eventPlace.name }}, {{ worship.eventPlace.street }}, {{
|
||||||
|
worship.eventPlace.city }}
|
||||||
|
</div>
|
||||||
|
<div v-if="worship.selfInformation" class="selfinformation">Bitte informieren Sie sich auch auf den
|
||||||
|
<a v-if="worship.eventPlace.website" :href="worship.eventPlace.website" target="_blank">Internetseiten dieser Gemeinde!</a><span
|
||||||
|
v-else>Internetseiten dieser Gemeinde!</span>
|
||||||
</div>
|
</div>
|
||||||
<div v-if="worship.selfInformation" class="selfinformation">Bitte informieren Sie sich auch auf den Internetseiten dieser Gemeinde!</div>
|
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
</table>
|
</table>
|
||||||
@@ -85,4 +93,11 @@ table.worships td div{
|
|||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
color: #0020e0;
|
color: #0020e0;
|
||||||
}
|
}
|
||||||
|
a {
|
||||||
|
color: #0020e0;
|
||||||
|
}
|
||||||
|
.internal-information {
|
||||||
|
color: #e45;
|
||||||
|
font-style: italic;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
@@ -10,6 +10,8 @@
|
|||||||
<input type="text" id="zipcode" v-model="newEventPlace.zipcode" placeholder="PLZ" required>
|
<input type="text" id="zipcode" v-model="newEventPlace.zipcode" placeholder="PLZ" required>
|
||||||
<label for="city">Stadt:</label>
|
<label for="city">Stadt:</label>
|
||||||
<input type="text" id="city" v-model="newEventPlace.city" placeholder="Stadt" required>
|
<input type="text" id="city" v-model="newEventPlace.city" placeholder="Stadt" required>
|
||||||
|
<label for="city">Webseite:</label>
|
||||||
|
<input type="text" id="website" v-model="newEventPlace.website" placeholder="Webseite" required>
|
||||||
<label for="backgroundColor">Hintergrundfarbe:</label>
|
<label for="backgroundColor">Hintergrundfarbe:</label>
|
||||||
<input type="color" id="backgroundColor" v-model="newEventPlace.backgroundColor">
|
<input type="color" id="backgroundColor" v-model="newEventPlace.backgroundColor">
|
||||||
<button type="submit">Speichern</button>
|
<button type="submit">Speichern</button>
|
||||||
@@ -47,7 +49,8 @@ export default {
|
|||||||
street: '',
|
street: '',
|
||||||
zipcode: '',
|
zipcode: '',
|
||||||
city: '',
|
city: '',
|
||||||
backgroundColor: '#ffffff'
|
backgroundColor: '#ffffff',
|
||||||
|
website: '',
|
||||||
},
|
},
|
||||||
editMode: false,
|
editMode: false,
|
||||||
editId: null
|
editId: null
|
||||||
@@ -87,7 +90,8 @@ export default {
|
|||||||
street: '',
|
street: '',
|
||||||
zipcode: '',
|
zipcode: '',
|
||||||
city: '',
|
city: '',
|
||||||
backgroundColor: '#ffffff'
|
backgroundColor: '#ffffff',
|
||||||
|
website: '',
|
||||||
};
|
};
|
||||||
this.editMode = false;
|
this.editMode = false;
|
||||||
this.editId = null;
|
this.editId = null;
|
||||||
|
|||||||
@@ -21,6 +21,9 @@
|
|||||||
<label for="organizer">Gestalter:</label>
|
<label for="organizer">Gestalter:</label>
|
||||||
<input type="text" id="organizer" v-model="worshipData.organizer">
|
<input type="text" id="organizer" v-model="worshipData.organizer">
|
||||||
|
|
||||||
|
<label for="sacristanService">Küsterdienst:</label>
|
||||||
|
<input type="text" id="sacristanService" v-model="worshipData.sacristanService">
|
||||||
|
|
||||||
<label for="collection">Kollekte:</label>
|
<label for="collection">Kollekte:</label>
|
||||||
<input type="text" id="collection" v-model="worshipData.collection">
|
<input type="text" id="collection" v-model="worshipData.collection">
|
||||||
|
|
||||||
@@ -77,7 +80,9 @@ export default {
|
|||||||
selfInformation: false,
|
selfInformation: false,
|
||||||
highlightTime: false,
|
highlightTime: false,
|
||||||
neighborInvitation: false,
|
neighborInvitation: false,
|
||||||
introLine: ''
|
introLine: '',
|
||||||
|
sacristanService: '',
|
||||||
|
website: '',
|
||||||
},
|
},
|
||||||
selectedEventPlace: null,
|
selectedEventPlace: null,
|
||||||
editMode: false,
|
editMode: false,
|
||||||
|
|||||||
@@ -94,8 +94,14 @@ export default createStore({
|
|||||||
login({ commit }, { user, token }) {
|
login({ commit }, { user, token }) {
|
||||||
commit('setLogin', { user, token });
|
commit('setLogin', { user, token });
|
||||||
},
|
},
|
||||||
logout({ commit }) {
|
async logout({ commit }) {
|
||||||
commit('logout');
|
try {
|
||||||
|
await axios.post('/auth/logout');
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Fehler beim Logout:', error);
|
||||||
|
} finally {
|
||||||
|
commit('logout');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
getters: {
|
getters: {
|
||||||
|
|||||||
26
utils/blacklist.js
Normal file
26
utils/blacklist.js
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
const blacklist = new Map();
|
||||||
|
const EXPIRATION_TIME = 24 * 60 * 60 * 1000; // 24 Stunden in Millisekunden
|
||||||
|
|
||||||
|
function cleanupBlacklist() {
|
||||||
|
const now = Date.now();
|
||||||
|
for (const [token, timestamp] of blacklist) {
|
||||||
|
if (now - timestamp > EXPIRATION_TIME) {
|
||||||
|
blacklist.delete(token);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function addTokenToBlacklist(token) {
|
||||||
|
cleanupBlacklist(); // Bereinige alte Einträge
|
||||||
|
blacklist.set(token, Date.now());
|
||||||
|
}
|
||||||
|
|
||||||
|
function isTokenBlacklisted(token) {
|
||||||
|
cleanupBlacklist(); // Bereinige alte Einträge
|
||||||
|
return blacklist.has(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
addTokenToBlacklist,
|
||||||
|
isTokenBlacklisted,
|
||||||
|
};
|
||||||
Reference in New Issue
Block a user