Phase 1-14 complete: - Unified Express.js API (TypeScript, Prisma ORM, PostgreSQL 16) - React Admin GUI (Vite + Ant Design + Zustand) - JWT auth with refresh tokens - Influence: Campaigns, Representatives, Responses, Email Queue - Map: Locations, Cuts, Shifts, Canvassing System - NAR data import infrastructure (2025 format) - Listmonk newsletter integration - Landing page builder (GrapesJS) - MkDocs + Code Server integration - Volunteer portal with GPS tracking - Monitoring stack (Prometheus, Grafana, Alertmanager) - Pangolin tunnel integration Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
61 lines
2.1 KiB
JavaScript
61 lines
2.1 KiB
JavaScript
const nocodbService = require('../services/nocodb');
|
|
const { sendPasswordRecovery } = require('../services/email');
|
|
const logger = require('../utils/logger');
|
|
|
|
class PasswordRecoveryController {
|
|
async requestPassword(req, res) {
|
|
try {
|
|
const { email } = req.body;
|
|
|
|
if (!email) {
|
|
return res.status(400).json({
|
|
success: false,
|
|
error: 'Email address is required'
|
|
});
|
|
}
|
|
|
|
// Validate email format
|
|
const emailRegex = /^[^\s@]+@[^\s@]+\.[^\s@]+$/;
|
|
if (!emailRegex.test(email)) {
|
|
return res.status(400).json({
|
|
success: false,
|
|
error: 'Invalid email format'
|
|
});
|
|
}
|
|
|
|
logger.info(`Password recovery requested for: ${email}`);
|
|
|
|
// Find user in database
|
|
const user = await nocodbService.getUserByEmail(email);
|
|
|
|
if (!user) {
|
|
// Don't reveal whether the email exists or not for security
|
|
logger.warn(`Password recovery attempted for non-existent email: ${email}`);
|
|
return res.json({
|
|
success: true,
|
|
message: 'If an account exists with this email, you will receive your password shortly.'
|
|
});
|
|
}
|
|
|
|
// Send password email
|
|
await sendPasswordRecovery(user);
|
|
|
|
logger.info(`Password recovery email sent to: ${email}`);
|
|
|
|
res.json({
|
|
success: true,
|
|
message: 'If an account exists with this email, you will receive your password shortly.'
|
|
});
|
|
|
|
} catch (error) {
|
|
logger.error('Password recovery error:', error);
|
|
res.status(500).json({
|
|
success: false,
|
|
error: 'Failed to process password recovery request. Please try again later.'
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
module.exports = new PasswordRecoveryController();
|