import { updateMember, handleNocoDbError } from '~/server/utils/nocodb'; import { createSessionManager } from '~/server/utils/session'; import type { Member, MembershipStatus } from '~/utils/types'; export default defineEventHandler(async (event) => { const id = getRouterParam(event, 'id'); console.log('[api/members/[id].put] ========================='); console.log('[api/members/[id].put] PUT /api/members/' + id); console.log('[api/members/[id].put] Request from:', getClientIP(event)); if (!id) { throw createError({ statusCode: 400, statusMessage: 'Member ID is required' }); } try { // Validate session and require Board+ privileges const sessionManager = createSessionManager(); const cookieHeader = getCookie(event, 'monacousa-session') ? getHeader(event, 'cookie') : undefined; const session = sessionManager.getSession(cookieHeader); if (!session?.user) { throw createError({ statusCode: 401, statusMessage: 'Authentication required' }); } const userTier = session.user.tier; if (userTier !== 'board' && userTier !== 'admin') { throw createError({ statusCode: 403, statusMessage: 'Board member privileges required to update members' }); } console.log('[api/members/[id].put] Authorized user:', session.user.email, 'Tier:', userTier); // Get and validate request body const body = await readBody(event); console.log('[api/members/[id].put] Request body fields:', Object.keys(body)); // Validate updated fields const validationErrors = validateMemberUpdateData(body); if (validationErrors.length > 0) { console.error('[api/members/[id].put] Validation errors:', validationErrors); throw createError({ statusCode: 400, statusMessage: `Validation failed: ${validationErrors.join(', ')}` }); } // Sanitize and prepare data const memberData = sanitizeMemberUpdateData(body); console.log('[api/members/[id].put] Sanitized data fields:', Object.keys(memberData)); // Update member in NocoDB const updatedMember = await updateMember(id, memberData); console.log('[api/members/[id].put] ✅ Member updated successfully:', id); // Return processed member const processedMember = { ...updatedMember, FullName: `${updatedMember.first_name || ''} ${updatedMember.last_name || ''}`.trim(), FormattedPhone: formatPhoneNumber(updatedMember.phone) }; return { success: true, data: processedMember, message: 'Member updated successfully' }; } catch (error: any) { console.error('[api/members/[id].put] ❌ Error updating member:', error); handleNocoDbError(error, 'updateMember', 'Member'); } }); function validateMemberUpdateData(data: any): string[] { const errors: string[] = []; // Only validate fields that are provided (partial updates allowed) if (data.first_name !== undefined) { if (!data.first_name || typeof data.first_name !== 'string' || data.first_name.trim().length < 2) { errors.push('First Name must be at least 2 characters'); } } if (data.last_name !== undefined) { if (!data.last_name || typeof data.last_name !== 'string' || data.last_name.trim().length < 2) { errors.push('Last Name must be at least 2 characters'); } } if (data.email !== undefined) { if (!data.email || typeof data.email !== 'string' || !isValidEmail(data.email)) { errors.push('Valid email address is required'); } } // Optional field validation if (data.phone !== undefined && data.phone && typeof data.phone === 'string' && data.phone.trim()) { const phoneRegex = /^[\+]?[1-9][\d]{0,15}$/; const cleanPhone = data.phone.replace(/\D/g, ''); if (!phoneRegex.test(cleanPhone)) { errors.push('Phone number format is invalid'); } } if (data.membership_status !== undefined && !['Active', 'Inactive', 'Pending', 'Expired'].includes(data.membership_status)) { errors.push('Invalid membership status'); } return errors; } function sanitizeMemberUpdateData(data: any): Partial { const sanitized: any = {}; // Only include fields that are provided (partial updates) if (data.first_name !== undefined) sanitized.first_name = data.first_name.trim(); if (data.last_name !== undefined) sanitized.last_name = data.last_name.trim(); if (data.email !== undefined) sanitized.email = data.email.trim().toLowerCase(); if (data.phone !== undefined) sanitized.phone = data.phone ? data.phone.trim() : null; if (data.nationality !== undefined) sanitized.nationality = data.nationality ? data.nationality.trim() : null; if (data.address !== undefined) sanitized.address = data.address ? data.address.trim() : null; if (data.date_of_birth !== undefined) sanitized.date_of_birth = data.date_of_birth; if (data.member_since !== undefined) sanitized.member_since = data.member_since; if (data.membership_date_paid !== undefined) sanitized.membership_date_paid = data.membership_date_paid; if (data.payment_due_date !== undefined) sanitized.payment_due_date = data.payment_due_date; // Boolean fields if (data.current_year_dues_paid !== undefined) { sanitized.current_year_dues_paid = Boolean(data.current_year_dues_paid) ? 'true' : 'false'; } // Enum fields if (data.membership_status !== undefined) { sanitized.membership_status = data.membership_status; } return sanitized; } function isValidEmail(email: string): boolean { const emailRegex = /^[^\s@]+@[^\s@]+\.[^\s@]+$/; return emailRegex.test(email); } function formatPhoneNumber(phone: string): string { if (!phone) return ''; const cleaned = phone.replace(/\D/g, ''); if (cleaned.length === 10) { return `(${cleaned.substring(0, 3)}) ${cleaned.substring(3, 6)}-${cleaned.substring(6)}`; } else if (cleaned.length === 11 && cleaned.startsWith('1')) { return `+1 (${cleaned.substring(1, 4)}) ${cleaned.substring(4, 7)}-${cleaned.substring(7)}`; } return phone; }