1240 lines
38 KiB
JavaScript
1240 lines
38 KiB
JavaScript
const express = require("express");
|
|
const session = require("express-session");
|
|
const bcrypt = require("bcrypt");
|
|
const crypto = require("crypto");
|
|
const nodemailer = require("nodemailer");
|
|
const { Pool } = require("pg");
|
|
const path = require("path");
|
|
const moment = require("moment");
|
|
require("dotenv").config();
|
|
const log = require("node-file-logger");
|
|
|
|
const app = express();
|
|
const port = process.env.PORT;
|
|
|
|
/* const TeleBot = require('telebot');
|
|
const bot = new TeleBot(process.env.TELEBOT);
|
|
telebotChatID=process.env.TELECHAT;
|
|
|
|
bot.on('text', (msg) => {
|
|
if (msg.from.id==telebotChatID) {
|
|
msg.reply.text(msg.text);
|
|
} else {
|
|
msg.reply.text("Entschuldige, " + msg.from.username + "\nIch darf nicht mit fremden reden.");
|
|
log.Info("Telebot-nachricht: " + msg.from.username + " - " + msg.text);
|
|
}
|
|
});
|
|
|
|
bot.start(); */
|
|
|
|
const options = {
|
|
timeZone: "europe/Berlin",
|
|
folderPath: "./logs/",
|
|
dateBasedFileNaming: true,
|
|
fileNamePrefix: "DailyLogs_",
|
|
fileNameExtension: ".log",
|
|
dateFormat: "YYYY_MM_DD",
|
|
timeFormat: "H:mm:ss",
|
|
};
|
|
log.SetUserOptions(options);
|
|
|
|
// Middleware
|
|
app.use(express.static(path.join(__dirname, "public")));
|
|
app.use(express.json());
|
|
app.set("view engine", "ejs");
|
|
app.use(express.urlencoded({ extended: false }));
|
|
|
|
// Session-Konfiguration
|
|
app.use(
|
|
session({
|
|
secret: process.env.SESSIONSECRET,
|
|
resave: false,
|
|
saveUninitialized: true,
|
|
cookie: { maxAge: 1000 * 60 * 60 * 24 * 2 },
|
|
})
|
|
);
|
|
|
|
// Authentifizierungs-Middleware
|
|
const requireAuth = (req, res, next) => {
|
|
if (!req.session.userId) {
|
|
return res.redirect("/login");
|
|
}
|
|
next();
|
|
};
|
|
|
|
const requireAdmin = (req, res, next) => {
|
|
if (req.session.role !== "admin") {
|
|
return res.status(403).send("Access denied");
|
|
}
|
|
next();
|
|
};
|
|
|
|
// Email-Konfiguration
|
|
const transporter = nodemailer.createTransport({
|
|
host: process.env.MAILHOST,
|
|
port: 465,
|
|
secure: true,
|
|
auth: {
|
|
user: process.env.MAILUSER,
|
|
pass: process.env.MAILPASS,
|
|
},
|
|
});
|
|
|
|
// Datenbankverbindung
|
|
const pool = new Pool({
|
|
connectionString: process.env.DATABASE_URL,
|
|
});
|
|
|
|
// Altersberechnung
|
|
const calculateAge = (birthdate) => {
|
|
const today = new Date();
|
|
const birthDate = new Date(birthdate);
|
|
let age = today.getFullYear() - birthDate.getFullYear();
|
|
const m = today.getMonth() - birthDate.getMonth();
|
|
if (m < 0 || (m === 0 && today.getDate() < birthDate.getDate())) {
|
|
age--;
|
|
}
|
|
return age;
|
|
};
|
|
|
|
const isBirthday = (birthdate, date, previousTraining) => {
|
|
let adjustedBirthdate = new Date(birthdate);
|
|
adjustedBirthdate.setFullYear(date.getFullYear());
|
|
|
|
if (adjustedBirthdate.getTime() === date.getTime()) {
|
|
return 2;
|
|
}
|
|
if (
|
|
adjustedBirthdate.getTime() >= previousTraining.getTime() &&
|
|
adjustedBirthdate.getTime() <= date.getTime()
|
|
) {
|
|
return 1;
|
|
}
|
|
return 0;
|
|
};
|
|
|
|
// Datumsformatierung
|
|
const formatDate = (date) => {
|
|
const d = new Date(date);
|
|
const day = String(d.getDate()).padStart(2, "0");
|
|
const month = String(d.getMonth() + 1).padStart(2, "0");
|
|
const year = d.getFullYear();
|
|
return `${day}.${month}.${year}`;
|
|
};
|
|
|
|
// Funktion zum Laden des aktuellen oder letzten Trainings
|
|
const getTraining = async (kw, jahr) => {
|
|
const result = await pool.query(
|
|
`
|
|
SELECT tr.*, g1.name AS geraet_riege_1_name, g2.name AS geraet_riege_2_name,
|
|
g3.name AS geraet_riege_3_name, g4.name AS geraet_riege_4_name,
|
|
g5.name AS geraet_riege_5_name,
|
|
t1.name AS aufwaermleiter_name, t2.name AS spielleiter_name,
|
|
sp1.name AS aufwaermen_name, sp2.name AS spiel_name
|
|
FROM trainings tr
|
|
LEFT JOIN geraete g1 ON tr.geraet_riege_1 = g1.id
|
|
LEFT JOIN geraete g2 ON tr.geraet_riege_2 = g2.id
|
|
LEFT JOIN geraete g3 ON tr.geraet_riege_3 = g3.id
|
|
LEFT JOIN geraete g4 ON tr.geraet_riege_4 = g4.id
|
|
LEFT JOIN geraete g5 ON tr.geraet_riege_5 = g5.id
|
|
LEFT JOIN teilnehmende t1 ON tr.aufwaermleiter = t1.id
|
|
LEFT JOIN teilnehmende t2 ON tr.spielleiter = t2.id
|
|
LEFT JOIN spiele sp1 ON tr.aufwaermen = sp1.id
|
|
LEFT JOIN spiele sp2 ON tr.spiel = sp2.id
|
|
WHERE tr.kw <= $1 AND tr.jahr = $2
|
|
ORDER BY tr.kw DESC
|
|
LIMIT 1
|
|
`,
|
|
[kw, jahr]
|
|
);
|
|
return result.rows[0];
|
|
};
|
|
|
|
const getTrainingbyID = async (id) => {
|
|
const result = await pool.query(
|
|
`
|
|
SELECT tr.*, g1.name AS geraet_riege_1_name, g2.name AS geraet_riege_2_name,
|
|
g3.name AS geraet_riege_3_name, g4.name AS geraet_riege_4_name,
|
|
g5.name AS geraet_riege_5_name,
|
|
t1.name AS aufwaermleiter_name, t2.name AS spielleiter_name,
|
|
sp1.name AS aufwaermen_name, sp2.name AS spiel_name
|
|
FROM trainings tr
|
|
LEFT JOIN geraete g1 ON tr.geraet_riege_1 = g1.id
|
|
LEFT JOIN geraete g2 ON tr.geraet_riege_2 = g2.id
|
|
LEFT JOIN geraete g3 ON tr.geraet_riege_3 = g3.id
|
|
LEFT JOIN geraete g4 ON tr.geraet_riege_4 = g4.id
|
|
LEFT JOIN geraete g5 ON tr.geraet_riege_5 = g5.id
|
|
LEFT JOIN teilnehmende t1 ON tr.aufwaermleiter = t1.id
|
|
LEFT JOIN teilnehmende t2 ON tr.spielleiter = t2.id
|
|
LEFT JOIN spiele sp1 ON tr.aufwaermen = sp1.id
|
|
LEFT JOIN spiele sp2 ON tr.spiel = sp2.id
|
|
WHERE tr.id=$1
|
|
`,
|
|
[id]
|
|
);
|
|
return result.rows[0];
|
|
};
|
|
|
|
// Funktion zum Laden aller Spiele
|
|
const getAllSpiele = async () => {
|
|
const result = await pool.query(`
|
|
SELECT * FROM spiele ORDER BY name ASC;
|
|
`);
|
|
return result.rows;
|
|
};
|
|
|
|
// Funktion zum Hinzufügen eines neuen Spiels, falls es noch nicht existiert
|
|
const addNewSpiel = async (spielName) => {
|
|
try {
|
|
const result = await pool.query(
|
|
"INSERT INTO spiele (name) VALUES ($1) RETURNING id",
|
|
[spielName]
|
|
);
|
|
return result.rows[0].id;
|
|
} catch (error) {
|
|
console.error("Error adding new spiel:", error);
|
|
throw error;
|
|
}
|
|
};
|
|
|
|
function getdayOfWeek(week, year) {
|
|
const weekday = parseInt(process.env.WEEKDAY, 10);
|
|
const firstDayOfYear = new Date(year, 0, 1);
|
|
const dayOfWeek = firstDayOfYear.getDay();
|
|
const daysUntilFirstThursday =
|
|
dayOfWeek <= weekday ? weekday - dayOfWeek : 11 - dayOfWeek;
|
|
const daysUntilThursdayOfWeek = (week - 1) * 7 + daysUntilFirstThursday;
|
|
const thursdayOfWeek = new Date(year, 0, 1 + daysUntilThursdayOfWeek);
|
|
return thursdayOfWeek;
|
|
}
|
|
|
|
// Funktion zum Laden der vier Leute, die am längsten nicht Aufwärmen oder Spiel geleitet haben
|
|
const getCandidatesForAufwaermleiter = async () => {
|
|
const result = await pool.query(`
|
|
SELECT t.id, t.name,
|
|
COALESCE(EXTRACT(EPOCH FROM (NOW() - MAX(tr.datum))) / 604800, EXTRACT(EPOCH FROM (NOW() - '1970-01-01'::date)) / 604800) AS weeks_since_last
|
|
FROM teilnehmende t
|
|
LEFT JOIN trainings tr ON t.id = tr.aufwaermleiter
|
|
WHERE t.helfer = true
|
|
GROUP BY t.id
|
|
ORDER BY weeks_since_last DESC;
|
|
`);
|
|
return result.rows;
|
|
};
|
|
|
|
const putInRiege = async (riege, teilnehmerID) => {
|
|
try {
|
|
if (riege == 0) {
|
|
const resultRiege = await pool.query(
|
|
"DELETE FROM riegen WHERE fremdid_teilnehmende = $1;",
|
|
[teilnehmerID]
|
|
);
|
|
} else {
|
|
const resultRiege = await pool.query(
|
|
"SELECT * FROM riegen WHERE fremdid_teilnehmende = $1;",
|
|
[teilnehmerID]
|
|
);
|
|
|
|
if (resultRiege.rows.length > 0) {
|
|
const resultRiege = await pool.query(
|
|
"UPDATE riegen SET riegennummer = $1 WHERE fremdid_teilnehmende = $2",
|
|
[riege, teilnehmerID]
|
|
);
|
|
log.Info("Mitglied " + teilnehmerID + " ist jetzt in Riege " + riege);
|
|
} else {
|
|
const resultRiege = await pool.query(
|
|
"INSERT INTO riegen (fremdid_teilnehmende, riegennummer) VALUES ($1, $2)",
|
|
[teilnehmerID, riege]
|
|
);
|
|
log.Info(
|
|
"Mitglied " + teilnehmerID + " neu in Riege " + riege + " eingefügt."
|
|
);
|
|
}
|
|
}
|
|
} catch (error) {
|
|
console.error("Error adding new spiel:", error);
|
|
throw error;
|
|
}
|
|
};
|
|
|
|
// Funktion zum Laden der vier Leute, die am längsten nicht Spiel geleitet haben
|
|
const getCandidatesForSpielleiter = async () => {
|
|
const result = await pool.query(`
|
|
SELECT t.id, t.name,
|
|
COALESCE(EXTRACT(EPOCH FROM (NOW() - MAX(tr.datum))) / 604800, EXTRACT(EPOCH FROM (NOW() - '1970-01-01'::date)) / 604800) AS weeks_since_last
|
|
FROM teilnehmende t
|
|
LEFT JOIN trainings tr ON t.id = tr.spielleiter
|
|
WHERE t.helfer = true
|
|
GROUP BY t.id
|
|
ORDER BY weeks_since_last DESC;
|
|
`);
|
|
return result.rows;
|
|
};
|
|
|
|
// Funktion zum Laden der Riegenteilnehmer absteigend nach Alter sortiert
|
|
const getRiegenMitgliederSortedByAge = async () => {
|
|
const result = await pool.query(`
|
|
SELECT r.riegennummer, t.name, t.alter
|
|
FROM riegen r
|
|
JOIN teilnehmende t ON r.fremdid_teilnehmende = t.id
|
|
ORDER BY r.riegennummer, t.alter DESC
|
|
`);
|
|
return result.rows;
|
|
};
|
|
|
|
const getAnwesenheit = function (id, anwesend) {
|
|
for (let item of anwesend) {
|
|
if (item.fid_teilnehmer === id) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
};
|
|
|
|
// Registrierung
|
|
app.post("/register", async (req, res) => {
|
|
const { username, email, password } = req.body;
|
|
try {
|
|
const hashedPassword = await bcrypt.hash(password, 10);
|
|
await pool.query(
|
|
"INSERT INTO users (username, email, password) VALUES ($1, $2, $3)",
|
|
[username, email, hashedPassword]
|
|
);
|
|
const message =
|
|
"Registrierung erfolgreich. Ein Admin wird dich in kürze freischalten";
|
|
const mailOptions = {
|
|
to: "admin@boergmann.it",
|
|
from: "admin@boergmann.it",
|
|
subject: "Neue Registrierung",
|
|
text: `${username} hat sich registriert`,
|
|
};
|
|
|
|
transporter.sendMail(mailOptions, (error) => {
|
|
if (error) {
|
|
console.error("Error sending email:", error);
|
|
const message = "Error sending Mail:" + error;
|
|
res.render("error", { session: req.session, message });
|
|
}
|
|
});
|
|
|
|
res.render("error", { session: req.session, message });
|
|
} catch (error) {
|
|
console.error("Error registering user:", error);
|
|
const message = "Error registering user:" + error;
|
|
res.render("error", { session: req.session, message });
|
|
}
|
|
});
|
|
|
|
// Login
|
|
app.post("/login", async (req, res) => {
|
|
const { username, password } = req.body;
|
|
|
|
try {
|
|
const userResult = await pool.query(
|
|
"SELECT *, CASE WHEN admin_temp IS NOT NULL AND (now() - admin_temp) > interval '22 hours' THEN 'expired' ELSE 'valid' END AS admin_status FROM users WHERE username = $1",
|
|
[username]
|
|
);
|
|
if (userResult.rows.length > 0) {
|
|
const user = userResult.rows[0];
|
|
const match = await bcrypt.compare(password, user.password);
|
|
if (match) {
|
|
if (user.is_active) {
|
|
req.session.userId = user.id;
|
|
req.session.userName = user.username;
|
|
req.session.activeRiege = 1;
|
|
req.session.activeTab = "geraete";
|
|
req.session.message = [(title = ""), (message = ""), (type = "none")];
|
|
if (user.admin_status === "expired") {
|
|
await pool.query(
|
|
"UPDATE users SET role = $1, admin_temp = NULL WHERE id = $2",
|
|
["user", user.id]
|
|
);
|
|
req.session.role = "user";
|
|
} else {
|
|
req.session.role = user.role;
|
|
}
|
|
res.redirect("/training");
|
|
} else {
|
|
res.redirect("/freischaltung");
|
|
}
|
|
} else {
|
|
const message = "Falsches Passwort";
|
|
res.render("login", { session: req.session, username, message });
|
|
}
|
|
} else {
|
|
const message = "Unbekannter Benutzer";
|
|
res.render("login", { session: req.session, username, message });
|
|
}
|
|
} catch (error) {
|
|
console.error("Error logging in:", error);
|
|
const message = "Error logging in:" + error;
|
|
res.render("error", { session: req.session, message });
|
|
}
|
|
});
|
|
|
|
//Wird angezeigt, wenn ein nicht freigeschalteter User sich anmelden will.
|
|
app.get("/freischaltung", async (req, res) => {
|
|
res.render("freischaltung", { session: req.session });
|
|
});
|
|
|
|
// Logout
|
|
app.get("/logout", (req, res) => {
|
|
req.session.destroy((err) => {
|
|
if (err) {
|
|
return res.status(500).send("Internal Server Error");
|
|
}
|
|
res.redirect("/");
|
|
});
|
|
});
|
|
|
|
// Benutzer freischalten (nur Admin)
|
|
app.post("/userrights", requireAuth, requireAdmin, async (req, res) => {
|
|
const { userId, type } = req.body;
|
|
try {
|
|
if (type === "activate") {
|
|
await pool.query("UPDATE users SET is_active = TRUE WHERE id = $1", [
|
|
userId,
|
|
]);
|
|
const userResult = await pool.query("SELECT * FROM users WHERE id = $1", [
|
|
userId,
|
|
]);
|
|
if (userResult.rows.length > 0) {
|
|
if (userResult.rows[0].email) {
|
|
const mailOptions = {
|
|
to: userResult.rows[0].email,
|
|
from: "admin@boergmann.it",
|
|
subject: "Freischaltung",
|
|
text: `Hallo ${userResult.rows[0].username}, du wurdest soeben freigeschaltet.`,
|
|
};
|
|
|
|
transporter.sendMail(mailOptions, (error) => {
|
|
if (error) {
|
|
console.error("Error sending email:", error);
|
|
const message = "Error sending Mail:" + error;
|
|
res.render("error", { session: req.session, message });
|
|
}
|
|
});
|
|
}
|
|
}
|
|
} else if (type === "admin") {
|
|
await pool.query("UPDATE users SET role = $1 WHERE id = $2", [
|
|
"admin",
|
|
userId,
|
|
]);
|
|
} else if (type === "admint") {
|
|
await pool.query(
|
|
"UPDATE users SET role = $1, admin_temp = $2 WHERE id = $3",
|
|
["admin", moment().toDate(), userId]
|
|
);
|
|
} else if (type === "delete") {
|
|
await pool.query("DELETE FROM users WHERE id = $1", [userId]);
|
|
}
|
|
res.redirect("/admin");
|
|
} catch (error) {
|
|
console.error("Error activating user:", error);
|
|
const message = "Error activating user:" + error;
|
|
res.render("error", { session: req.session, message });
|
|
}
|
|
});
|
|
|
|
// Passwort-Zurücksetzung anfordern
|
|
app.post("/send-password", async (req, res) => {
|
|
const { email } = req.body;
|
|
try {
|
|
const userResult = await pool.query(
|
|
"SELECT * FROM users WHERE email = $1",
|
|
[email]
|
|
);
|
|
if (userResult.rows.length > 0) {
|
|
const user = userResult.rows[0];
|
|
const token = crypto.randomBytes(20).toString("hex");
|
|
const resetLink = `http://tkd.boergmann.it/reset-password/${token}`;
|
|
await pool.query(
|
|
"UPDATE users SET reset_password_token = $1, reset_password_expires = $2 WHERE id = $3",
|
|
[token, (selectedDate = moment().add(1, "d").toDate()), user.id]
|
|
);
|
|
|
|
const mailOptions = {
|
|
to: user.email,
|
|
from: "admin@boergmann.it",
|
|
subject: "Password Reset",
|
|
text: `Click the following link to reset your password: ${resetLink}`,
|
|
};
|
|
|
|
transporter.sendMail(mailOptions, (err) => {
|
|
if (err) {
|
|
console.error("Error sending email:", err);
|
|
const message = "Error sending Mail:" + error;
|
|
res.render("error", { session: req.session, message });
|
|
} else {
|
|
const message = "Password reset link sent";
|
|
res.render("error", { session: req.session, message });
|
|
}
|
|
});
|
|
} else {
|
|
const message = "Email not found";
|
|
res.render("error", { session: req.session, message });
|
|
}
|
|
} catch (error) {
|
|
console.error("Error in forgot-password:", error);
|
|
const message = "Error in forgot-password";
|
|
res.render("error", { session: req.session, message });
|
|
}
|
|
});
|
|
|
|
app.get("/forgot-password", async (req, res) => {
|
|
res.render("forgot-password", { session: req.session });
|
|
});
|
|
|
|
// Passwort zurücksetzen
|
|
app.get("/reset-password/:token", async (req, res) => {
|
|
const { token } = req.params;
|
|
try {
|
|
// const userResult = await pool.query('SELECT * FROM users WHERE reset_password_token = $1 AND reset_password_expires > $2', [token, Date.now()]);
|
|
const userResult = await pool.query(
|
|
"SELECT *, CASE WHEN reset_password_token IS NOT NULL AND (now() - reset_password_expires) > interval '22 hours' THEN 'expired' ELSE 'valid' END AS reset_status FROM users WHERE reset_password_token = $1",
|
|
[token]
|
|
);
|
|
if (userResult.rows.length > 0) {
|
|
res.render("reset-password", { session: req.session, token }); // Stelle sicher, dass es eine reset-password.ejs gibt
|
|
} else {
|
|
const message = "Token ungültig oder abgelaufen";
|
|
res.render("error", { session: req.session, message });
|
|
}
|
|
} catch (error) {
|
|
console.error("Error in reset-password:", error);
|
|
const message = "Error in reset-password";
|
|
res.render("error", { session: req.session, message });
|
|
}
|
|
});
|
|
|
|
app.post("/reset-password/:token", async (req, res) => {
|
|
const { token } = req.params;
|
|
const { password } = req.body;
|
|
try {
|
|
const userResult = await pool.query(
|
|
"SELECT *, CASE WHEN reset_password_token IS NOT NULL AND (now() - reset_password_expires) > interval '22 hours' THEN 'expired' ELSE 'valid' END AS reset_status FROM users WHERE reset_password_token = $1",
|
|
[token]
|
|
);
|
|
if (userResult.rows.length > 0) {
|
|
const user = userResult.rows[0];
|
|
const hashedPassword = await bcrypt.hash(password, 10);
|
|
await pool.query(
|
|
"UPDATE users SET password = $1, reset_password_token = NULL, reset_password_expires = NULL WHERE id = $2",
|
|
[hashedPassword, user.id]
|
|
);
|
|
res.redirect("/login");
|
|
} else {
|
|
const message = "Token ungültig oder abgelaufen";
|
|
res.render("error", { session: req.session, message });
|
|
}
|
|
} catch (error) {
|
|
console.error("Error in reset-password:", error);
|
|
const message = "Error in reset-password";
|
|
res.render("error", { session: req.session, message });
|
|
}
|
|
});
|
|
|
|
// Profilseite
|
|
app.get("/profile", requireAuth, (req, res) => {
|
|
res.render("profile", { session: req.session }); // Stelle sicher, dass es eine profile.ejs gibt
|
|
});
|
|
|
|
app.post("/profile", requireAuth, async (req, res) => {
|
|
const { email, password } = req.body;
|
|
try {
|
|
if (email) {
|
|
await pool.query("UPDATE users SET email = $1 WHERE id = $2", [
|
|
email,
|
|
req.session.userId,
|
|
]);
|
|
}
|
|
if (password) {
|
|
const hashedPassword = await bcrypt.hash(password, 10);
|
|
await pool.query("UPDATE users SET password = $1 WHERE id = $2", [
|
|
hashedPassword,
|
|
req.session.userId,
|
|
]);
|
|
}
|
|
res.redirect("/profile");
|
|
} catch (error) {
|
|
console.error("Error updating profile:", error);
|
|
const message = "Error updating profile:" + error;
|
|
res.render("error", { session: req.session, message });
|
|
}
|
|
});
|
|
|
|
app.post("/update-training", requireAuth, async (req, res) => {
|
|
const { trainingId, type, spielName } = req.body;
|
|
req.session.activeTab = "spiel";
|
|
let spielId;
|
|
|
|
const spiel = await pool.query(`SELECT * FROM spiele WHERE name = $1`, [
|
|
spielName,
|
|
]);
|
|
|
|
if (spiel.rows.length > 0) {
|
|
spielId = spiel.rows[0].id;
|
|
} else {
|
|
const newSpiel = await pool.query(
|
|
"INSERT INTO spiele (name) VALUES ($1) RETURNING id",
|
|
[spielName]
|
|
);
|
|
spielId = newSpiel.rows[0].id;
|
|
req.session.message = [
|
|
(title = "Neues Spiel"),
|
|
(body = "Das Spiel ${ spielName } wurde angelegt."),
|
|
];
|
|
}
|
|
|
|
try {
|
|
if (type === "spiel") {
|
|
await pool.query("UPDATE trainings SET spiel = $1 WHERE id = $2", [
|
|
spielId,
|
|
trainingId,
|
|
]);
|
|
} else if (type === "aufwaermen") {
|
|
await pool.query("UPDATE trainings SET aufwaermen = $1 WHERE id = $2", [
|
|
spielId,
|
|
trainingId,
|
|
]);
|
|
}
|
|
|
|
res.redirect("/training");
|
|
} catch (error) {
|
|
console.error(error);
|
|
const message = "Error:" + error;
|
|
res.render("error", { session: req.session, message });
|
|
}
|
|
});
|
|
|
|
app.post("/update-leader", requireAuth, async (req, res) => {
|
|
const { trainingId, type, leaderId } = req.body;
|
|
req.session.activeTab = "spiel";
|
|
try {
|
|
if (type === "aufwaermleiter") {
|
|
await pool.query(
|
|
"UPDATE trainings SET aufwaermleiter = $1 WHERE id = $2",
|
|
[leaderId, trainingId]
|
|
);
|
|
} else if (type === "spielleiter") {
|
|
await pool.query("UPDATE trainings SET spielleiter = $1 WHERE id = $2", [
|
|
leaderId,
|
|
trainingId,
|
|
]);
|
|
}
|
|
res.redirect("/training");
|
|
} catch (error) {
|
|
console.error(error);
|
|
const message = "Error:" + error;
|
|
res.render("error", { session: req.session, message });
|
|
}
|
|
});
|
|
|
|
// Admin-Seite
|
|
app.get("/admin", requireAuth, requireAdmin, async (req, res) => {
|
|
const usersResult = await pool.query("SELECT * FROM users");
|
|
res.render("admin", { users: usersResult.rows, session: req.session }); // Stelle sicher, dass es eine admin.ejs gibt
|
|
});
|
|
|
|
app.post("/new-member", requireAuth, requireAdmin, async (req, res) => {
|
|
var {
|
|
from,
|
|
vorname,
|
|
nachname,
|
|
geburt,
|
|
riege,
|
|
adresse,
|
|
probe,
|
|
from,
|
|
training,
|
|
} = req.body;
|
|
var geb = "01.01.2024";
|
|
const name = vorname + " " + nachname;
|
|
try {
|
|
if (geburt) {
|
|
geb = geburt;
|
|
}
|
|
const teilnehmerID = await pool.query(
|
|
"INSERT INTO teilnehmende (name, geburtsdatum, adresse, vorname, nachname, probe) VALUES ($1, $2, $3, $4, $5, $6) RETURNING id",
|
|
[name, geb, adresse, vorname, nachname, probe ? true : false]
|
|
);
|
|
if (from === "admin") {
|
|
selectedKW = moment().isoWeek();
|
|
selectedYear = moment().year();
|
|
const trainingresult = await getTraining(selectedKW, selectedYear);
|
|
training = trainingresult.id;
|
|
}
|
|
await pool.query(
|
|
"INSERT INTO anwesend (fid_teilnehmer, fid_training) VALUES ($1, $2)",
|
|
[teilnehmerID.rows[0].id, training]
|
|
);
|
|
|
|
log.Info(
|
|
"Mitglied " + name + " durch " + req.session.userName + " angelegt"
|
|
);
|
|
putInRiege(riege, teilnehmerID.rows[0].id);
|
|
req.session.activeRiege = riege;
|
|
req.session.activeTab = "anwesend";
|
|
res.redirect("/training");
|
|
} catch (error) {
|
|
const message = "Error:" + error;
|
|
res.render("error", { session: req.session, message });
|
|
}
|
|
});
|
|
|
|
// Teilnehmer_innen
|
|
app.get("/training", requireAuth, async (req, res) => {
|
|
try {
|
|
let dateParam = req.query.date;
|
|
let kwParam = req.query.kw;
|
|
let jahrParam = req.query.jahr;
|
|
let idParam = req.query.id;
|
|
let selectedDate;
|
|
let selectedKW;
|
|
let selectetYear;
|
|
let training;
|
|
|
|
if (idParam) {
|
|
training = await getTrainingbyID(idParam);
|
|
} else if (kwParam) {
|
|
selectedKW = kwParam;
|
|
if (jahrParam) {
|
|
selectedYear = jahrParam;
|
|
} else {
|
|
selectedYear = moment().year();
|
|
}
|
|
training = await getTraining(selectedKW, selectedYear);
|
|
} else {
|
|
selectedKW = moment().isoWeek();
|
|
selectedYear = moment().year();
|
|
training = await getTraining(selectedKW, selectedYear);
|
|
}
|
|
|
|
const anwesendResult = await pool.query(
|
|
"SELECT * FROM anwesend WHERE fid_training = $1",
|
|
[training.id]
|
|
);
|
|
anwesend = anwesendResult.rows;
|
|
const anzahl = anwesend.length;
|
|
const trainingsResult = await pool.query(
|
|
"SELECT id, kw, jahr FROM trainings ORDER BY jahr DESC, kw DESC"
|
|
);
|
|
const trainingsDates = trainingsResult.rows.map((tr) => ({
|
|
kw: tr.kw,
|
|
id: tr.id,
|
|
jahr: tr.jahr,
|
|
datum: formatDate(getdayOfWeek(tr.kw, tr.jahr)),
|
|
}));
|
|
|
|
// Vorheriges Training ermitteln
|
|
const previousTrainingResult = await pool.query(
|
|
"SELECT * FROM trainings WHERE id < $1 ORDER BY id DESC LIMIT 1",
|
|
[training.id]
|
|
);
|
|
const nextTrainingResult = await pool.query(
|
|
"SELECT * FROM trainings WHERE id > $1 ORDER BY id ASC LIMIT 1",
|
|
[training.id]
|
|
);
|
|
const previousTraining =
|
|
previousTrainingResult.rows.length > 0
|
|
? previousTrainingResult.rows[0]
|
|
: null;
|
|
const nextTraining =
|
|
nextTrainingResult.rows.length > 0 ? nextTrainingResult.rows[0] : null;
|
|
// Abrufen der Riegendaten einschließlich der Teilnehmer und deren Altersberechnung
|
|
const result = await pool.query(`
|
|
SELECT r.riegennummer,
|
|
t.id,
|
|
t.name,
|
|
t.vorname,
|
|
t.nachname,
|
|
t.geburtsdatum,
|
|
t.probe,
|
|
r.helfer,
|
|
COUNT(a.fid_teilnehmer) AS anwesenheit
|
|
FROM riegen r
|
|
JOIN teilnehmende t ON r.fremdID_Teilnehmende = t.id
|
|
LEFT JOIN anwesend a ON t.id = a.fid_teilnehmer
|
|
GROUP BY r.riegennummer, t.id, t.name, t.vorname, t.nachname, t.geburtsdatum, r.helfer
|
|
ORDER BY r.riegennummer, t.geburtsdatum ASC;
|
|
`);
|
|
|
|
// Gruppieren der Riegenteilnehmer nach Riegennummer
|
|
const riegen = {};
|
|
result.rows.forEach((row) => {
|
|
const age = calculateAge(row.geburtsdatum);
|
|
const hasBirthday = isBirthday(
|
|
row.geburtsdatum,
|
|
training.datum,
|
|
previousTraining.datum
|
|
);
|
|
const tnAnwesend = getAnwesenheit(row.id, anwesend);
|
|
if (!riegen[row.riegennummer]) {
|
|
riegen[row.riegennummer] = [];
|
|
}
|
|
riegen[row.riegennummer].push({
|
|
id: row.id,
|
|
name: row.name,
|
|
vorname: row.vorname,
|
|
nachname: row.nachname,
|
|
geb: row.geburtsdatum,
|
|
hasBirthday: hasBirthday,
|
|
age: age,
|
|
helfer: row.helfer,
|
|
probe: row.probe,
|
|
anwesenheit: row.anwesenheit,
|
|
anwesend: tnAnwesend,
|
|
anzahl,
|
|
});
|
|
});
|
|
if (training) {
|
|
training.datum = getdayOfWeek(training.kw, training.jahr);
|
|
}
|
|
|
|
const aufwaermleiterCandidates = await getCandidatesForAufwaermleiter();
|
|
const spielleiterCandidates = await getCandidatesForSpielleiter();
|
|
const spielCandidates = await getAllSpiele();
|
|
const aufwaermenCandidates = await getAllSpiele();
|
|
|
|
const abteilung = process.env.ABTEILUNG;
|
|
|
|
let view;
|
|
|
|
if (abteilung == "KiTu") {
|
|
view = "trainings_riegen";
|
|
} else {
|
|
view = "trainings";
|
|
}
|
|
|
|
res.render(view, {
|
|
training,
|
|
trainingsDates,
|
|
selectedKW: training.kw,
|
|
aufwaermleiterCandidates,
|
|
spielleiterCandidates,
|
|
aufwaermenCandidates, // Übergeben der Kandidaten für Aufwärmleiter
|
|
spielCandidates, // Übergeben der Spiele
|
|
session: req.session,
|
|
previousTraining,
|
|
nextTraining,
|
|
riegen,
|
|
activeRiege: req.session.activeRiege,
|
|
activeTab: req.session.activeTab,
|
|
moment,
|
|
});
|
|
} catch (error) {
|
|
console.error(error);
|
|
const message = "Error:" + error;
|
|
res.render("error", { session: req.session, message });
|
|
}
|
|
});
|
|
|
|
app.get("/riege", requireAuth, async (req, res) => {
|
|
try {
|
|
// Abrufen der Riegendaten einschließlich der Teilnehmer und deren Altersberechnung
|
|
const result = await pool.query(`
|
|
SELECT r.riegennummer,
|
|
t.id,
|
|
t.name,
|
|
t.vorname,
|
|
t.nachname,
|
|
t.geburtsdatum,
|
|
t.probe,
|
|
r.helfer,
|
|
COUNT(a.fid_teilnehmer) AS anwesenheit
|
|
FROM riegen r
|
|
JOIN teilnehmende t ON r.fremdID_Teilnehmende = t.id
|
|
LEFT JOIN anwesend a ON t.id = a.fid_teilnehmer
|
|
GROUP BY r.riegennummer, t.id, t.name, t.vorname, t.nachname, t.geburtsdatum, r.helfer
|
|
ORDER BY r.riegennummer, t.geburtsdatum ASC;
|
|
|
|
`);
|
|
// Gruppieren der Riegenteilnehmer nach Riegennummer
|
|
const date = new Date();
|
|
let previous = new Date(date);
|
|
previous.setDate(date.getDate() - 7);
|
|
const riegen = {};
|
|
result.rows.forEach((row) => {
|
|
const age = calculateAge(row.geburtsdatum);
|
|
if (!riegen[row.riegennummer]) {
|
|
riegen[row.riegennummer] = [];
|
|
}
|
|
riegen[row.riegennummer].push({
|
|
id: row.id,
|
|
name: row.name,
|
|
vorname: row.vorname,
|
|
nachname: row.nachname,
|
|
hasBirthday: isBirthday(row.geburtsdatum, date, previous),
|
|
age: age,
|
|
helfer: row.helfer,
|
|
probe: row.probe,
|
|
anwesenheit: row.anwesenheit,
|
|
});
|
|
});
|
|
|
|
res.render("riegen", { riegen: riegen, session: req.session });
|
|
} catch (error) {
|
|
console.error("Error fetching riegen:", error);
|
|
req.session.message = ["Error", error, "error"];
|
|
res.render("error", { session: req.session });
|
|
}
|
|
});
|
|
|
|
app.get("/teilnehmer", requireAuth, requireAdmin, async (req, res) => {
|
|
try {
|
|
const teilnehmendeResult = await pool.query(
|
|
"SELECT m.*, COUNT(a.fid_teilnehmer) AS anwesenheit FROM teilnehmende m LEFT JOIN anwesend a ON m.id = a.fid_teilnehmer GROUP BY m.id ORDER BY helfer DESC, vorname ASC"
|
|
);
|
|
const date = new Date();
|
|
let previous = new Date(date);
|
|
previous.setDate(date.getDate() - 7);
|
|
const teilnehmende = teilnehmendeResult.rows.map((t) => ({
|
|
...t,
|
|
age: calculateAge(t.geburtsdatum),
|
|
hasBirthday: isBirthday(t.geburtsdatum, date, previous),
|
|
}));
|
|
|
|
res.render("teilnehmer", { teilnehmende, session: req.session });
|
|
} catch (error) {
|
|
console.error(error);
|
|
req.session.message = ["Error", error, "error"];
|
|
res.render("error", { session: req.session });
|
|
}
|
|
});
|
|
|
|
app.get("/mitglied/:id", requireAuth, requireAdmin, async (req, res) => {
|
|
const { id } = req.params;
|
|
try {
|
|
var riege = 0;
|
|
const userResult = await pool.query(
|
|
"SELECT * FROM teilnehmende WHERE id = $1",
|
|
[id]
|
|
);
|
|
const riegeResult = await pool.query(
|
|
"SELECT * FROM riegen WHERE fremdid_teilnehmende = $1",
|
|
[id]
|
|
);
|
|
if (riegeResult.rows.length > 0) {
|
|
riege = riegeResult.rows[0].riegennummer;
|
|
}
|
|
req.session.activeRiege = riege;
|
|
const anwesendResult = await pool.query(
|
|
"SELECT t.id, TO_CHAR(t.datum, 'DD.MM.YYYY') as date, t.* FROM trainings t JOIN anwesend a ON t.id = a.fid_training WHERE a.fid_teilnehmer = $1;",
|
|
[id]
|
|
);
|
|
console.log(anwesendResult);
|
|
const anwesenheiten = anwesendResult.rows;
|
|
const anwesend = anwesendResult.rows.length;
|
|
const numberResult = await pool.query(
|
|
"SELECT telefonnummern.*, telverbindung.bezeichnung FROM telefonnummern JOIN telverbindung ON telefonnummern.verbindung = telverbindung.id WHERE telefonnummern.fid_teilnehmer = $1",
|
|
[id]
|
|
);
|
|
const numbers = numberResult.rows;
|
|
if (userResult.rows.length > 0) {
|
|
const mitglied = userResult.rows.map((t) => ({
|
|
...t,
|
|
age: calculateAge(t.geburtsdatum),
|
|
}));
|
|
res.render("mitglied", {
|
|
id,
|
|
mitglied: mitglied[0],
|
|
numbers,
|
|
riege,
|
|
anwesenheiten,
|
|
anwesend,
|
|
session: req.session,
|
|
});
|
|
} else {
|
|
req.session.message = ["Error", "Mitglied existiert nicht", "error"];
|
|
res.redirect("/teilnehmer");
|
|
}
|
|
} catch (error) {
|
|
console.error("Error in Mitglied:", error);
|
|
req.session.message = ["Error", "Mitglied existiert nicht", "error"];
|
|
res.redirect("/teilnehmer");
|
|
}
|
|
});
|
|
|
|
app.post("/set-riege", requireAuth, async (req, res) => {
|
|
const { riege, id } = req.body;
|
|
putInRiege(riege, id);
|
|
res.redirect("/riege");
|
|
});
|
|
|
|
app.post("/update-mitglied", requireAdmin, async (req, res) => {
|
|
const {
|
|
type,
|
|
id,
|
|
geburt,
|
|
telid,
|
|
adresse,
|
|
name,
|
|
verbindung,
|
|
nummer,
|
|
vorname,
|
|
nachname,
|
|
probe,
|
|
helfer,
|
|
} = req.body;
|
|
try {
|
|
if (type == "tel") {
|
|
await pool.query(
|
|
"INSERT INTO telefonnummern (fid_teilnehmer, name, verbindung, nummer, stand) VALUES ($1, $2, $3, $4, $5)",
|
|
[id, name, verbindung, nummer, moment().toDate()]
|
|
);
|
|
} else if (type == "tel-delete") {
|
|
await pool.query("DELETE FROM telefonnummern WHERE id = $1", [telid]);
|
|
} else if (type == "adresse") {
|
|
await pool.query(
|
|
"UPDATE teilnehmende SET vorname = $1, nachname = $2, geburtsdatum = $3, adresse = $4, probe = $5, helfer = $6 WHERE id = $7",
|
|
[
|
|
vorname,
|
|
nachname,
|
|
geburt,
|
|
adresse,
|
|
probe ? true : false,
|
|
helfer ? true : false,
|
|
id,
|
|
]
|
|
);
|
|
}
|
|
|
|
res.redirect("/mitglied/" + id);
|
|
} catch (error) {
|
|
console.error(error);
|
|
req.session.message = ["Error", error, "error"];
|
|
res.redirect("/mitglied/" + id);
|
|
}
|
|
});
|
|
|
|
// Login und Registrierung anzeigen
|
|
app.get("/login", (req, res) => {
|
|
req.session.message = ["", "", "none"];
|
|
res.render("login", { session: req.session }); // Stelle sicher, dass es eine login.ejs gibt
|
|
});
|
|
|
|
// Registrierung
|
|
app.get("/register", (req, res) => {
|
|
res.render("register", { session: req.session }); // Stelle sicher, dass es eine register.ejs gibt
|
|
});
|
|
|
|
// Spieleliste
|
|
app.get("/spiele", async (req, res) => {
|
|
try {
|
|
const spieleResult = await pool.query(
|
|
"SELECT * FROM spiele ORDER BY name ASC;"
|
|
);
|
|
const spiele = spieleResult.rows;
|
|
res.render("spiele", { spiele, session: req.session });
|
|
} catch (error) {
|
|
console.error("Error in Mitglied:", error);
|
|
req.session.message = ["Error", error, "error"];
|
|
res.render("spiele", { spiele, session: req.session });
|
|
}
|
|
});
|
|
|
|
app.post("/delete-spiel", requireAdmin, async (req, res) => {
|
|
const { spielId } = req.body;
|
|
await pool.query("DELETE FROM spiele WHERE id = $1", [spielId]);
|
|
res.redirect("/spiele");
|
|
});
|
|
|
|
// Gerenderte Seite für gewähltes Spiel
|
|
app.get("/spiel/:id", async (req, res) => {
|
|
const { id } = req.params;
|
|
try {
|
|
const spieleResult = await pool.query(
|
|
"SELECT * FROM spiele WHERE id = $1",
|
|
[id]
|
|
);
|
|
if (spieleResult.rows.length > 0) {
|
|
const spiel = spieleResult.rows[0];
|
|
res.render("spiel", { spiel, session: req.session });
|
|
} else {
|
|
req.session.message = ["Error", "Spiel existiert nicht", "error"];
|
|
res.redirect("/spiele");
|
|
}
|
|
} catch (error) {
|
|
console.error("Error in Spiele:", error);
|
|
req.session.message = ["Error", "Spiel existiert nicht", "error"];
|
|
res.redirect("/spiele");
|
|
}
|
|
});
|
|
|
|
// Postseite für Änderungen
|
|
app.post("/edit-spiel", requireAuth, async (req, res) => {
|
|
const { material, dauer, regeln, variationen, type, id } = req.body;
|
|
try {
|
|
const queryResult = await pool.query(
|
|
"UPDATE spiele set material = $1, regeln = $2, dauer = $3, variationen = $4, type = $5 Where id = $6 ",
|
|
[material, regeln, dauer, variationen, type, id]
|
|
);
|
|
req.session.message = ["Erfolg", "Das Spiel wurde gespeichert", "success"];
|
|
res.redirect("/spiel/" + id);
|
|
} catch (error) {
|
|
console.error("Error in edit Spiel:", error);
|
|
req.session.message = ["Error", error, "error"];
|
|
res.redirect("/spiel/" + id);
|
|
}
|
|
});
|
|
|
|
// Startseite
|
|
app.get("/", (req, res) => {
|
|
req.session.message = ["", "", "none"];
|
|
res.render("index", { session: req.session });
|
|
});
|
|
|
|
// Changelog
|
|
app.get("/changelog", async (req, res) => {
|
|
try {
|
|
const changeResult = await pool.query(
|
|
"SELECT * FROM changelog ORDER BY datetime DESC;"
|
|
);
|
|
const changes = changeResult.rows;
|
|
req.session.message = ["Erfolg", "Log gespeichert", "success"];
|
|
res.render("changelog", { changes, session: req.session });
|
|
} catch (error) {
|
|
console.error("Error:", error);
|
|
req.session.message = ["Error", error, "error"];
|
|
res.redirect("/changelog");
|
|
}
|
|
});
|
|
|
|
app.post("/changelog", requireAdmin, async (req, res) => {
|
|
const { title, body } = req.body;
|
|
try {
|
|
await pool.query("INSERT INTO changelog (title, body) VALUES ($1, $2);", [
|
|
title,
|
|
body,
|
|
]);
|
|
|
|
const changeResult = await pool.query(
|
|
"SELECT * FROM changelog ORDER BY datetime DESC;"
|
|
);
|
|
const changes = changeResult.rows;
|
|
res.render("changelog", { changes, session: req.session });
|
|
} catch (error) {
|
|
console.error("Error:", error);
|
|
req.session.message = ["Error", error, "error"];
|
|
res.render("changelog", { changes, session: req.session });
|
|
}
|
|
});
|
|
|
|
app.post("/anwesend", requireAuth, async (req, res) => {
|
|
var { anw, inriege, trainingId, riege } = req.body;
|
|
req.session.activeTab = "anwesend";
|
|
try {
|
|
for (const mitgliedId of inriege) {
|
|
const Idresult = await pool.query(
|
|
"SELECT *From anwesend WHERE fid_teilnehmer = $1 AND fid_training=$2",
|
|
[mitgliedId, trainingId]
|
|
);
|
|
if (Idresult.rows.length > 0) {
|
|
if (!anw.includes(mitgliedId)) {
|
|
await pool.query(
|
|
"Delete FROM anwesend WHERE fid_teilnehmer = $1 AND fid_training=$2",
|
|
[mitgliedId, trainingId]
|
|
);
|
|
}
|
|
}
|
|
}
|
|
if (!Array.isArray(anw)) {
|
|
anw = [anw];
|
|
}
|
|
for (const teilnehmer of anw) {
|
|
const Tnresult = await pool.query(
|
|
"SELECT *From anwesend WHERE fid_teilnehmer = $1 AND fid_training=$2",
|
|
[teilnehmer, trainingId]
|
|
);
|
|
if (Tnresult.rows.length === 0) {
|
|
await pool.query(
|
|
"INSERT INTO anwesend (fid_teilnehmer, fid_training) VALUES ($1, $2)",
|
|
[teilnehmer, trainingId]
|
|
);
|
|
}
|
|
}
|
|
req.session.activeRiege = riege;
|
|
req.session.activeTab = "anwesend";
|
|
req.session.message = [
|
|
(title = "Erfolg"),
|
|
(body = "Die Anwesenheit wurde gespeichert"),
|
|
(type = "success"),
|
|
];
|
|
res.redirect("/training");
|
|
} catch (error) {
|
|
console.error("Error:", error);
|
|
req.session.message = ["Error", error, "error"];
|
|
res.redirect("/training");
|
|
}
|
|
});
|
|
|
|
app.post("/deleteMember", requireAuth, async (req, res) => {
|
|
const { id } = req.body;
|
|
try {
|
|
await pool.query("DELETE from teilnehmende WHERE id = $1", [id]);
|
|
} catch (error) {
|
|
console.error("Error:", error);
|
|
}
|
|
res.redirect("/teilnehmer");
|
|
});
|
|
|
|
app.get("/feature", requireAuth, async (req, res) => {
|
|
try {
|
|
const featureResult = await pool.query(
|
|
"SELECT * FROM features ORDER BY datetime DESC;"
|
|
);
|
|
const features = featureResult.rows;
|
|
res.render("feature", { features, session: req.session });
|
|
} catch (error) {
|
|
console.error("Error:", error);
|
|
req.session.message = ["Error", error, "error"];
|
|
res.redirect("/feature");
|
|
}
|
|
});
|
|
|
|
app.post("/featuredone", requireAdmin, async (req, res) => {
|
|
const { id } = req.body;
|
|
try {
|
|
const featureResult = await pool.query(
|
|
"UPDATE features set done = true WHERE id = $1",
|
|
[id]
|
|
);
|
|
res.redirect("/feature");
|
|
} catch (error) {
|
|
console.error("Error:", error);
|
|
req.session.message = ["Error", error, "error"];
|
|
res.render("feature", { features, session: req.session });
|
|
}
|
|
});
|
|
|
|
app.post("/feature", requireAdmin, async (req, res) => {
|
|
const { title, body, type, urgency, user } = req.body;
|
|
const userResult = await pool.query(
|
|
"SELECT id FROM users WHERE username = $1",
|
|
[user]
|
|
);
|
|
try {
|
|
await pool.query(
|
|
"INSERT INTO features (title, body, type, urgency, fid_user) VALUES ($1, $2, $3, $4, $5);",
|
|
[title, body, type, urgency, userResult.rows[0].id]
|
|
);
|
|
|
|
const featureResult = await pool.query(
|
|
"SELECT * FROM features ORDER BY datetime DESC;"
|
|
);
|
|
const features = featureResult.rows;
|
|
req.session.message = ["Erfolg", "Feature-request gespeichert", "success"];
|
|
res.render("feature", { features, session: req.session });
|
|
} catch (error) {
|
|
console.error("Error:", error);
|
|
req.session.message = ["Error", error, "error"];
|
|
res.render("feature", { features, session: req.session });
|
|
}
|
|
});
|
|
|
|
const server = app.listen(port, "0.0.0.0", () => {
|
|
console.log(`Server is running on ${process.env.HOST}:${port}/`);
|
|
log.Info(`Server is running on ${process.env.HOST}:${port}/`);
|
|
});
|