Files
sim/apps/sim/components/emails/render.ts
Emir Karabeg 79be435918 feat(email): welcome email; improvement(emails): ui/ux (#2658)
* feat(email): welcome email; improvement(emails): ui/ux

* improvement(emails): links, accounts, preview

* refactor(emails): file structure and wrapper components

* added envvar for personal emails sent, added isHosted gate

* fixed failing tests, added env mock

* fix: removed comment

---------

Co-authored-by: waleed <walif6@gmail.com>
2026-01-02 00:16:27 -08:00

242 lines
5.7 KiB
TypeScript

import { render } from '@react-email/components'
import { OTPVerificationEmail, ResetPasswordEmail, WelcomeEmail } from '@/components/emails/auth'
import {
CreditPurchaseEmail,
EnterpriseSubscriptionEmail,
FreeTierUpgradeEmail,
PaymentFailedEmail,
PlanWelcomeEmail,
UsageThresholdEmail,
} from '@/components/emails/billing'
import { CareersConfirmationEmail, CareersSubmissionEmail } from '@/components/emails/careers'
import {
BatchInvitationEmail,
InvitationEmail,
WorkspaceInvitationEmail,
} from '@/components/emails/invitations'
import { HelpConfirmationEmail } from '@/components/emails/support'
import { getBaseUrl } from '@/lib/core/utils/urls'
export type { EmailSubjectType } from './subjects'
export { getEmailSubject } from './subjects'
interface WorkspaceInvitation {
workspaceId: string
workspaceName: string
permission: 'admin' | 'write' | 'read'
}
export async function renderOTPEmail(
otp: string,
email: string,
type: 'sign-in' | 'email-verification' | 'forget-password' = 'email-verification',
chatTitle?: string
): Promise<string> {
return await render(OTPVerificationEmail({ otp, email, type, chatTitle }))
}
export async function renderPasswordResetEmail(
username: string,
resetLink: string
): Promise<string> {
return await render(ResetPasswordEmail({ username, resetLink }))
}
export async function renderInvitationEmail(
inviterName: string,
organizationName: string,
invitationUrl: string
): Promise<string> {
return await render(
InvitationEmail({
inviterName,
organizationName,
inviteLink: invitationUrl,
})
)
}
export async function renderBatchInvitationEmail(
inviterName: string,
organizationName: string,
organizationRole: 'admin' | 'member',
workspaceInvitations: WorkspaceInvitation[],
acceptUrl: string
): Promise<string> {
return await render(
BatchInvitationEmail({
inviterName,
organizationName,
organizationRole,
workspaceInvitations,
acceptUrl,
})
)
}
export async function renderHelpConfirmationEmail(
type: 'bug' | 'feedback' | 'feature_request' | 'other',
attachmentCount = 0
): Promise<string> {
return await render(
HelpConfirmationEmail({
type,
attachmentCount,
submittedDate: new Date(),
})
)
}
export async function renderEnterpriseSubscriptionEmail(userName: string): Promise<string> {
const baseUrl = getBaseUrl()
const loginLink = `${baseUrl}/login`
return await render(
EnterpriseSubscriptionEmail({
userName,
loginLink,
})
)
}
export async function renderUsageThresholdEmail(params: {
userName?: string
planName: string
percentUsed: number
currentUsage: number
limit: number
ctaLink: string
}): Promise<string> {
return await render(
UsageThresholdEmail({
userName: params.userName,
planName: params.planName,
percentUsed: params.percentUsed,
currentUsage: params.currentUsage,
limit: params.limit,
ctaLink: params.ctaLink,
})
)
}
export async function renderFreeTierUpgradeEmail(params: {
userName?: string
percentUsed: number
currentUsage: number
limit: number
upgradeLink: string
}): Promise<string> {
return await render(
FreeTierUpgradeEmail({
userName: params.userName,
percentUsed: params.percentUsed,
currentUsage: params.currentUsage,
limit: params.limit,
upgradeLink: params.upgradeLink,
})
)
}
export async function renderPlanWelcomeEmail(params: {
planName: 'Pro' | 'Team'
userName?: string
loginLink?: string
}): Promise<string> {
return await render(
PlanWelcomeEmail({
planName: params.planName,
userName: params.userName,
loginLink: params.loginLink,
})
)
}
export async function renderWelcomeEmail(userName?: string): Promise<string> {
return await render(WelcomeEmail({ userName }))
}
export async function renderCreditPurchaseEmail(params: {
userName?: string
amount: number
newBalance: number
}): Promise<string> {
return await render(
CreditPurchaseEmail({
userName: params.userName,
amount: params.amount,
newBalance: params.newBalance,
purchaseDate: new Date(),
})
)
}
export async function renderWorkspaceInvitationEmail(
inviterName: string,
workspaceName: string,
invitationLink: string
): Promise<string> {
return await render(
WorkspaceInvitationEmail({
inviterName,
workspaceName,
invitationLink,
})
)
}
export async function renderPaymentFailedEmail(params: {
userName?: string
amountDue: number
lastFourDigits?: string
billingPortalUrl: string
failureReason?: string
}): Promise<string> {
return await render(
PaymentFailedEmail({
userName: params.userName,
amountDue: params.amountDue,
lastFourDigits: params.lastFourDigits,
billingPortalUrl: params.billingPortalUrl,
failureReason: params.failureReason,
})
)
}
export async function renderCareersConfirmationEmail(
name: string,
position: string
): Promise<string> {
return await render(
CareersConfirmationEmail({
name,
position,
})
)
}
export async function renderCareersSubmissionEmail(params: {
name: string
email: string
phone?: string
position: string
linkedin?: string
portfolio?: string
experience: string
location: string
message: string
}): Promise<string> {
return await render(
CareersSubmissionEmail({
name: params.name,
email: params.email,
phone: params.phone,
position: params.position,
linkedin: params.linkedin,
portfolio: params.portfolio,
experience: params.experience,
location: params.location,
message: params.message,
})
)
}