mirror of
https://github.com/beilunyang/moemail.git
synced 2026-05-28 19:50:09 +08:00
105 lines
2.9 KiB
TypeScript
105 lines
2.9 KiB
TypeScript
import { NextResponse } from "next/server"
|
|
import { nanoid } from "nanoid"
|
|
import { createDb } from "@/lib/db"
|
|
import { emails } from "@/lib/schema"
|
|
import { eq, and, gt, sql } from "drizzle-orm"
|
|
import { EXPIRY_OPTIONS } from "@/types/email"
|
|
import { EMAIL_CONFIG } from "@/config"
|
|
import { getRequestContext } from "@cloudflare/next-on-pages"
|
|
import { getUserId } from "@/lib/apiKey"
|
|
import { getUserRole } from "@/lib/auth"
|
|
import { ROLES } from "@/lib/permissions"
|
|
|
|
export const runtime = "edge"
|
|
|
|
export async function POST(request: Request) {
|
|
const db = createDb()
|
|
const env = getRequestContext().env
|
|
|
|
const userId = await getUserId()
|
|
const userRole = await getUserRole(userId!)
|
|
|
|
try {
|
|
if (userRole !== ROLES.EMPEROR) {
|
|
const maxEmails = await env.SITE_CONFIG.get("MAX_EMAILS") || EMAIL_CONFIG.MAX_ACTIVE_EMAILS.toString()
|
|
const activeEmailsCount = await db
|
|
.select({ count: sql<number>`count(*)` })
|
|
.from(emails)
|
|
.where(
|
|
and(
|
|
eq(emails.userId, userId!),
|
|
gt(emails.expiresAt, new Date())
|
|
)
|
|
)
|
|
|
|
if (Number(activeEmailsCount[0].count) >= Number(maxEmails)) {
|
|
return NextResponse.json(
|
|
{ error: `已达到最大邮箱数量限制 (${maxEmails})` },
|
|
{ status: 403 }
|
|
)
|
|
}
|
|
}
|
|
|
|
const { name, expiryTime, domain } = await request.json<{
|
|
name: string
|
|
expiryTime: number
|
|
domain: string
|
|
}>()
|
|
|
|
if (!EXPIRY_OPTIONS.some(option => option.value === expiryTime)) {
|
|
return NextResponse.json(
|
|
{ error: "无效的过期时间" },
|
|
{ status: 400 }
|
|
)
|
|
}
|
|
|
|
const domainString = await env.SITE_CONFIG.get("EMAIL_DOMAINS")
|
|
const domains = domainString ? domainString.split(',') : ["moemail.app"]
|
|
|
|
if (!domains || !domains.includes(domain)) {
|
|
return NextResponse.json(
|
|
{ error: "无效的域名" },
|
|
{ status: 400 }
|
|
)
|
|
}
|
|
|
|
const address = `${name || nanoid(8)}@${domain}`
|
|
const existingEmail = await db.query.emails.findFirst({
|
|
where: eq(sql`LOWER(${emails.address})`, address.toLowerCase())
|
|
})
|
|
|
|
if (existingEmail) {
|
|
return NextResponse.json(
|
|
{ error: "该邮箱地址已被使用" },
|
|
{ status: 409 }
|
|
)
|
|
}
|
|
|
|
const now = new Date()
|
|
const expires = expiryTime === 0
|
|
? new Date('9999-01-01T00:00:00.000Z')
|
|
: new Date(now.getTime() + expiryTime)
|
|
|
|
const emailData: typeof emails.$inferInsert = {
|
|
address,
|
|
createdAt: now,
|
|
expiresAt: expires,
|
|
userId: userId!
|
|
}
|
|
|
|
const result = await db.insert(emails)
|
|
.values(emailData)
|
|
.returning({ id: emails.id, address: emails.address })
|
|
|
|
return NextResponse.json({
|
|
id: result[0].id,
|
|
email: result[0].address
|
|
})
|
|
} catch (error) {
|
|
console.error('Failed to generate email:', error)
|
|
return NextResponse.json(
|
|
{ error: "创建邮箱失败" },
|
|
{ status: 500 }
|
|
)
|
|
}
|
|
}
|