/** * 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} 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} 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} 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 = `

Neue Registrierung

Ein neuer Benutzer hat sich registriert und wartet auf Freigabe:

  • Name: ${data.name}
  • E-Mail: ${data.email}
  • Telefon: ${data.phone || 'Nicht angegeben'}

Bitte prüfen Sie die Registrierung im CMS.

` 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 = `

Registrierung erhalten

Hallo ${data.name},

vielen Dank für Ihre Registrierung beim Harheimer TC!

Ihre Anfrage wird vom Vorstand geprüft. Sie erhalten eine E-Mail, sobald Ihr Zugang freigeschaltet wurde.


Mit sportlichen Grüßen,
Ihr Harheimer TC

` 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 } }