242 lines
7.4 KiB
JavaScript
242 lines
7.4 KiB
JavaScript
/**
|
|
* Email Service - Handles membership application email notifications
|
|
* Clean Code: Single Responsibility Principle
|
|
*/
|
|
|
|
import nodemailer from 'nodemailer'
|
|
import fs from 'fs/promises'
|
|
import path from 'path'
|
|
import { getServerDataPath } from './paths.js'
|
|
import { isHiddenUser, migrateUserRoles, readUsers } from './auth.js'
|
|
|
|
/**
|
|
* Gets the correct data path for config files
|
|
* @param {string} filename - Config filename
|
|
* @returns {string} Full path to config file
|
|
*/
|
|
// nosemgrep: javascript.lang.security.audit.path-traversal.path-join-resolve-traversal.path-join-resolve-traversal
|
|
// filename is always a hardcoded constant (e.g., 'config.json'), never user input
|
|
function getDataPath(filename) {
|
|
return getServerDataPath(filename)
|
|
}
|
|
|
|
/**
|
|
* Loads configuration from config.json
|
|
* @returns {Promise<Object>} Configuration object
|
|
*/
|
|
async function loadConfig() {
|
|
try {
|
|
const configPath = getDataPath('config.json')
|
|
const configData = await fs.readFile(configPath, 'utf8')
|
|
return JSON.parse(configData)
|
|
} catch (error) {
|
|
console.error('Could not load config.json:', error.message)
|
|
return {}
|
|
}
|
|
}
|
|
|
|
function envFlagEnabled(value) {
|
|
return ['1', 'true', 'yes', 'on'].includes(String(value || '').trim().toLowerCase())
|
|
}
|
|
|
|
function shouldUseDeveloperRecipients() {
|
|
if (process.env.DEBUG !== undefined) return envFlagEnabled(process.env.DEBUG)
|
|
return process.env.NODE_ENV !== 'production' || process.env.APP_ENV === 'test'
|
|
}
|
|
|
|
/**
|
|
* Gets email recipients based on membership type and environment
|
|
* @param {Object} data - Form data
|
|
* @param {Object} config - Configuration
|
|
* @returns {Array<string>} Email addresses
|
|
*/
|
|
async function collectBoardUserRecipients() {
|
|
try {
|
|
const users = await readUsers()
|
|
return users
|
|
.filter(user => user && user.active !== false && !isHiddenUser(user))
|
|
.map(user => migrateUserRoles({ ...user }))
|
|
.filter(user => Array.isArray(user.roles) && user.roles.includes('vorstand'))
|
|
.map(user => String(user.email || '').trim())
|
|
.filter(Boolean)
|
|
} catch (error) {
|
|
console.error('Could not load board recipients from users.json:', error.message || error)
|
|
return []
|
|
}
|
|
}
|
|
|
|
async function getEmailRecipients(data, config) {
|
|
if (shouldUseDeveloperRecipients()) {
|
|
return ['tsschulz@tsschulz.de']
|
|
}
|
|
|
|
const recipients = await collectBoardUserRecipients()
|
|
|
|
// Fallback for legacy installations where Vorstand members are only configured in config.json.
|
|
if (recipients.length === 0 && config.vorstand && typeof config.vorstand === 'object') {
|
|
Object.values(config.vorstand).forEach((member) => {
|
|
if (member && member.email && typeof member.email === 'string' && member.email.trim() !== '') {
|
|
recipients.push(member.email.trim())
|
|
}
|
|
})
|
|
}
|
|
|
|
// For minors, also add first trainer email if configured (trainer is an array)
|
|
if (data.isVolljaehrig === false && Array.isArray(config.trainer) && config.trainer.length > 0 && config.trainer[0].email) {
|
|
recipients.push(config.trainer[0].email)
|
|
}
|
|
|
|
// Fallback if no recipients found
|
|
if (recipients.length === 0) {
|
|
// Prefer website verantwortlicher if set
|
|
if (config.website && config.website.verantwortlicher && config.website.verantwortlicher.email) {
|
|
recipients.push(config.website.verantwortlicher.email)
|
|
} else {
|
|
throw new Error('Keine E-Mail-Empfänger in config.json konfiguriert.')
|
|
}
|
|
}
|
|
|
|
return [...new Set(recipients)]
|
|
}
|
|
|
|
/**
|
|
* Creates email transporter
|
|
* @returns {Object} Nodemailer transporter
|
|
*/
|
|
function createTransporter() {
|
|
const smtpUser = process.env.SMTP_USER
|
|
const smtpPass = process.env.SMTP_PASS
|
|
|
|
if (!smtpUser || !smtpPass) {
|
|
throw new Error(
|
|
'SMTP-Credentials fehlen! Bitte setzen Sie SMTP_USER und SMTP_PASS in der .env Datei.\n' +
|
|
`Aktuell: SMTP_USER=${smtpUser ? 'gesetzt' : 'FEHLT'}, SMTP_PASS=${smtpPass ? 'gesetzt' : 'FEHLT'}`
|
|
)
|
|
}
|
|
|
|
return nodemailer.createTransport({
|
|
host: process.env.SMTP_HOST || 'localhost',
|
|
port: parseInt(process.env.SMTP_PORT) || 587,
|
|
secure: process.env.SMTP_SECURE === 'true',
|
|
auth: {
|
|
user: smtpUser,
|
|
pass: smtpPass
|
|
}
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Sends membership application email
|
|
* @param {Object} data - Form data
|
|
* @param {string} pdfPath - Path to generated PDF
|
|
* @returns {Promise<Object>} Email result
|
|
*/
|
|
export async function sendMembershipEmail(data, pdfPath) {
|
|
try {
|
|
const config = await loadConfig()
|
|
const recipients = await getEmailRecipients(data, config)
|
|
|
|
// Create transporter
|
|
const transporter = createTransporter()
|
|
|
|
// Email content
|
|
const subject = `Neuer Mitgliedschaftsantrag - ${data.vorname} ${data.nachname}`
|
|
const message = `Ein neuer Mitgliedschaftsantrag wurde eingereicht.
|
|
|
|
Antragsteller: ${data.vorname} ${data.nachname}
|
|
Mitgliedschaftsart: ${data.mitgliedschaftsart}
|
|
Volljährig: ${data.isVolljaehrig ? 'Ja' : 'Nein'}
|
|
|
|
Das ausgefüllte Formular ist als Anhang verfügbar.`
|
|
|
|
// Email options
|
|
const mailOptions = {
|
|
from: process.env.SMTP_FROM || 'noreply@harheimertc.de',
|
|
to: recipients.join(', '),
|
|
subject: subject,
|
|
text: message,
|
|
attachments: [
|
|
{
|
|
filename: path.basename(pdfPath),
|
|
path: pdfPath
|
|
}
|
|
]
|
|
}
|
|
|
|
// Send email
|
|
const info = await transporter.sendMail(mailOptions)
|
|
|
|
return {
|
|
success: true,
|
|
message: message,
|
|
recipients: recipients,
|
|
messageId: info.messageId
|
|
}
|
|
|
|
} catch (error) {
|
|
console.error('Email sending failed:', error.message)
|
|
return {
|
|
success: false,
|
|
message: `E-Mail-Versand fehlgeschlagen: ${error.message}`,
|
|
error: error.message
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Sends a simple registration notification to Vorstand/admin and a confirmation to user.
|
|
* @param {Object} data - { name, email, phone }
|
|
*/
|
|
export async function sendRegistrationNotification(data) {
|
|
try {
|
|
const config = await loadConfig()
|
|
const recipients = await getEmailRecipients(data, config)
|
|
|
|
// Create transporter
|
|
const transporter = createTransporter()
|
|
|
|
// Notify Vorstand/admin
|
|
const adminSubject = 'Neue Registrierung - Harheimer TC'
|
|
const adminHtml = `
|
|
<h2>Neue Registrierung</h2>
|
|
<p>Ein neuer Benutzer hat sich registriert und wartet auf Freigabe:</p>
|
|
<ul>
|
|
<li><strong>Name:</strong> ${data.name}</li>
|
|
<li><strong>E-Mail:</strong> ${data.email}</li>
|
|
<li><strong>Telefon:</strong> ${data.phone || 'Nicht angegeben'}</li>
|
|
</ul>
|
|
<p>Bitte prüfen Sie die Registrierung im CMS.</p>
|
|
`
|
|
|
|
await transporter.sendMail({
|
|
from: process.env.SMTP_FROM || 'noreply@harheimertc.de',
|
|
to: recipients.join(', '),
|
|
subject: adminSubject,
|
|
html: adminHtml
|
|
})
|
|
|
|
// Confirmation to user
|
|
const userSubject = 'Registrierung erhalten - Harheimer TC'
|
|
const userHtml = `
|
|
<h2>Registrierung erhalten</h2>
|
|
<p>Hallo ${data.name},</p>
|
|
<p>vielen Dank für Ihre Registrierung beim Harheimer TC!</p>
|
|
<p>Ihre Anfrage wird vom Vorstand geprüft. Sie erhalten eine E-Mail, sobald Ihr Zugang freigeschaltet wurde.</p>
|
|
<br>
|
|
<p>Mit sportlichen Grüßen,<br>Ihr Harheimer TC</p>
|
|
`
|
|
|
|
await transporter.sendMail({
|
|
from: process.env.SMTP_FROM || 'noreply@harheimertc.de',
|
|
to: data.email,
|
|
subject: userSubject,
|
|
html: userHtml
|
|
})
|
|
|
|
return { success: true, recipients }
|
|
} catch (error) {
|
|
console.error('sendRegistrationNotification failed:', error.message || error)
|
|
throw error
|
|
}
|
|
}
|