NeahFront9/app/api/mail/route.ts

231 lines
5.7 KiB
TypeScript

import { NextResponse } from 'next/server';
import Imap from 'imap';
import { parseEmailHeaders } from '@/lib/email-parser';
interface StoredCredentials {
email: string;
password: string;
host: string;
port: number;
}
interface Email {
id: string;
from: string;
subject: string;
date: Date;
read: boolean;
starred: boolean;
body: string;
}
interface ImapBox {
messages: {
total: number;
};
}
interface ImapMessage {
on: (event: string, callback: (data: any) => void) => void;
once: (event: string, callback: (data: any) => void) => void;
attributes: {
uid: number;
flags: string[];
size: number;
};
body: {
[key: string]: {
on: (event: string, callback: (data: any) => void) => void;
};
};
}
interface ImapConfig {
user: string;
password: string;
host: string;
port: number;
tls: boolean;
authTimeout: number;
connTimeout: number;
debug?: (info: string) => void;
}
function getStoredCredentials(): StoredCredentials | null {
if (typeof window === 'undefined') {
// Server-side
const email = process.env.IMAP_USER;
const password = process.env.IMAP_PASSWORD;
const host = process.env.IMAP_HOST;
const port = process.env.IMAP_PORT ? parseInt(process.env.IMAP_PORT) : 993;
if (!email || !password || !host) {
return null;
}
return { email, password, host, port };
} else {
// Client-side
const stored = localStorage.getItem('imapCredentials');
return stored ? JSON.parse(stored) : null;
}
}
export async function GET() {
try {
const credentials = getStoredCredentials();
if (!credentials) {
return NextResponse.json(
{ error: 'No stored credentials found' },
{ status: 401 }
);
}
const imapConfig: ImapConfig = {
user: credentials.email,
password: credentials.password,
host: credentials.host,
port: credentials.port,
tls: true,
authTimeout: 10000,
connTimeout: 10000,
debug: (info: string) => console.log('IMAP Debug:', info)
};
console.log('IMAP Config:', {
...imapConfig,
password: '***'
});
const imap = new Imap(imapConfig);
const connectPromise = new Promise((resolve, reject) => {
imap.once('ready', () => resolve(true));
imap.once('error', (err: Error) => reject(err));
imap.connect();
});
await connectPromise;
const openBoxPromise = new Promise<ImapBox>((resolve, reject) => {
imap.openBox('INBOX', false, (err: Error | null, box: ImapBox) => {
if (err) reject(err);
else resolve(box);
});
});
const box = await openBoxPromise;
const fetchPromise = (imap: Imap, seqno: number): Promise<ImapMessage> => {
return new Promise((resolve, reject) => {
const f = imap.fetch(seqno, { bodies: ['HEADER', 'TEXT'] });
f.on('message', (msg: ImapMessage) => {
resolve(msg);
});
f.once('error', reject);
});
};
const processMessage = (msg: ImapMessage): Promise<any> => {
return new Promise((resolve, reject) => {
let body = '';
msg.body['TEXT'].on('data', (chunk: Buffer) => {
body += chunk.toString('utf8');
});
msg.body['TEXT'].on('end', () => {
const headers = parseEmailHeaders(body);
resolve({
uid: msg.attributes.uid,
flags: msg.attributes.flags,
size: msg.attributes.size,
...headers
});
});
msg.body['TEXT'].on('error', reject);
});
};
const messages: any[] = [];
for (let seqno = 1; seqno <= 10; seqno++) {
const msg = await fetchPromise(imap, seqno);
const processedMsg = await processMessage(msg);
messages.push(processedMsg);
}
imap.end();
const emails: Email[] = messages.map((msg) => {
return {
id: msg.uid.toString(),
from: msg.from,
subject: msg.subject,
date: msg.date,
read: !msg.flags?.includes('\\Unseen'),
starred: msg.flags?.includes('\\Flagged') || false,
body: msg.body
};
});
return NextResponse.json(emails);
} catch (error) {
console.error('Error fetching emails:', error);
const status = error instanceof Error && error.message.includes('Invalid login')
? 401
: 500;
return NextResponse.json(
{ error: error instanceof Error ? error.message : 'Failed to fetch emails' },
{ status }
);
}
}
export async function POST(request: Request) {
try {
const credentials = getStoredCredentials();
if (!credentials) {
return NextResponse.json(
{ error: 'No stored credentials found' },
{ status: 401 }
);
}
const { to, subject, body, attachments } = await request.json();
if (!to || !subject || !body) {
return NextResponse.json(
{ error: 'Missing required fields' },
{ status: 400 }
);
}
const nodemailer = require('nodemailer');
const transporter = nodemailer.createTransport({
host: credentials.host,
port: credentials.port,
secure: true,
auth: {
user: credentials.email,
pass: credentials.password,
},
});
const mailOptions = {
from: credentials.email,
to,
subject,
text: body,
attachments: attachments || [],
};
const info = await transporter.sendMail(mailOptions);
console.log('Email sent:', info.messageId);
return NextResponse.json({ success: true, messageId: info.messageId });
} catch (error) {
console.error('Error sending email:', error);
return NextResponse.json(
{ error: error instanceof Error ? error.message : 'Failed to send email' },
{ status: 500 }
);
}
}