changemaker.lite/api/dist/scripts/seed-email-templates.js

237 lines
15 KiB
JavaScript

#!/usr/bin/env tsx
"use strict";
/**
* Email Templates Seed Script
*
* Migrates existing email templates from filesystem to database.
* Run with: npx tsx api/src/scripts/seed-email-templates.ts
*/
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
var desc = Object.getOwnPropertyDescriptor(m, k);
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
desc = { enumerable: true, get: function() { return m[k]; } };
}
Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
o["default"] = v;
});
var __importStar = (this && this.__importStar) || (function () {
var ownKeys = function(o) {
ownKeys = Object.getOwnPropertyNames || function (o) {
var ar = [];
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
return ar;
};
return ownKeys(o);
};
return function (mod) {
if (mod && mod.__esModule) return mod;
var result = {};
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
__setModuleDefault(result, mod);
return result;
};
})();
Object.defineProperty(exports, "__esModule", { value: true });
const client_1 = require("@prisma/client");
const fs = __importStar(require("fs"));
const path = __importStar(require("path"));
const prisma = new client_1.PrismaClient();
const TEMPLATES = [
{
key: 'campaign-email',
name: 'Campaign Email to Representative',
description: 'Email sent when a constituent contacts their elected representative through a campaign',
category: 'INFLUENCE',
subjectLine: '{{CAMPAIGN_TITLE}} - Message from {{USER_NAME}}',
isSystem: true,
variables: [
{ key: 'CAMPAIGN_TITLE', label: 'Campaign Title', description: 'Title of the advocacy campaign', isRequired: true, isConditional: false, sampleValue: 'Support Climate Action Bill C-12', sortOrder: 0 },
{ key: 'MESSAGE', label: 'Message Body', description: 'The message content written by the constituent', isRequired: true, isConditional: false, sampleValue: 'I urge you to support this important legislation that will help address climate change...', sortOrder: 1 },
{ key: 'USER_NAME', label: 'Sender Name', description: 'Name of the constituent sending the email', isRequired: true, isConditional: false, sampleValue: 'Jane Doe', sortOrder: 2 },
{ key: 'USER_EMAIL', label: 'Sender Email', description: 'Email address of the constituent', isRequired: true, isConditional: false, sampleValue: 'jane@example.com', sortOrder: 3 },
{ key: 'POSTAL_CODE', label: 'Postal Code', description: 'Postal code of the constituent', isRequired: true, isConditional: false, sampleValue: 'K1A 0B1', sortOrder: 4 },
{ key: 'RECIPIENT_NAME', label: 'Representative Name', description: 'Name of the representative receiving the email', isRequired: false, isConditional: true, sampleValue: 'Hon. John Smith', sortOrder: 5 },
{ key: 'RECIPIENT_LEVEL', label: 'Government Level', description: 'Level of government (Federal, Provincial, Municipal)', isRequired: false, isConditional: true, sampleValue: 'Federal', sortOrder: 6 },
{ key: 'ORGANIZATION_NAME', label: 'Organization Name', description: 'Name of the organization running the campaign', isRequired: true, isConditional: false, sampleValue: 'Changemaker Lite', sortOrder: 7 },
{ key: 'TIMESTAMP', label: 'Timestamp', description: 'Date and time the email was sent', isRequired: true, isConditional: false, sampleValue: '2025-02-11T10:30:00Z', sortOrder: 8 },
],
},
{
key: 'response-verification',
name: 'Response Verification Email',
description: 'Email sent to representatives to verify a submitted response',
category: 'INFLUENCE',
subjectLine: 'Verify Your Response - {{CAMPAIGN_TITLE}}',
isSystem: true,
variables: [
{ key: 'CAMPAIGN_TITLE', label: 'Campaign Title', description: 'Title of the advocacy campaign', isRequired: true, isConditional: false, sampleValue: 'Support Climate Action Bill C-12', sortOrder: 0 },
{ key: 'RESPONSE_TYPE', label: 'Response Type', description: 'Type of response (EMAIL, LETTER, PHONE_CALL, MEETING, etc.)', isRequired: true, isConditional: false, sampleValue: 'SUPPORT', sortOrder: 1 },
{ key: 'RESPONSE_TEXT', label: 'Response Text', description: 'The text of the response submitted', isRequired: true, isConditional: false, sampleValue: 'I fully support this initiative and will be voting in favor...', sortOrder: 2 },
{ key: 'SUBMITTER_NAME', label: 'Submitter Name', description: 'Name of the person who submitted the response', isRequired: true, isConditional: false, sampleValue: 'Jane Doe', sortOrder: 3 },
{ key: 'SUBMITTED_DATE', label: 'Submitted Date', description: 'Date the response was submitted', isRequired: true, isConditional: false, sampleValue: 'February 11, 2025', sortOrder: 4 },
{ key: 'VERIFICATION_URL', label: 'Verification URL', description: 'URL to verify the response', isRequired: true, isConditional: false, sampleValue: 'https://api.cmlite.org/api/responses/123/verify/abc123def456', sortOrder: 5 },
{ key: 'REPORT_URL', label: 'Report URL', description: 'URL to report the response as invalid', isRequired: true, isConditional: false, sampleValue: 'https://api.cmlite.org/api/responses/123/report/abc123def456', sortOrder: 6 },
{ key: 'ORGANIZATION_NAME', label: 'Organization Name', description: 'Name of the organization running the campaign', isRequired: true, isConditional: false, sampleValue: 'Changemaker Lite', sortOrder: 7 },
{ key: 'TIMESTAMP', label: 'Timestamp', description: 'Date and time the verification email was sent', isRequired: true, isConditional: false, sampleValue: '2025-02-11T10:30:00Z', sortOrder: 8 },
],
},
{
key: 'shift-signup-confirmation',
name: 'Shift Signup Confirmation',
description: 'Email sent when a volunteer signs up for a shift',
category: 'MAP',
subjectLine: 'Shift Confirmation - {{SHIFT_TITLE}}',
isSystem: true,
variables: [
{ key: 'ORGANIZATION_NAME', label: 'Organization Name', description: 'Name of the organization', isRequired: true, isConditional: false, sampleValue: 'Changemaker Lite', sortOrder: 0 },
{ key: 'USER_NAME', label: 'Volunteer Name', description: 'Name of the volunteer', isRequired: true, isConditional: false, sampleValue: 'John Smith', sortOrder: 1 },
{ key: 'USER_EMAIL', label: 'Volunteer Email', description: 'Email address of the volunteer', isRequired: true, isConditional: false, sampleValue: 'john@example.com', sortOrder: 2 },
{ key: 'SHIFT_TITLE', label: 'Shift Title', description: 'Title of the volunteer shift', isRequired: true, isConditional: false, sampleValue: 'Weekend Canvassing - Downtown', sortOrder: 3 },
{ key: 'SHIFT_DATE', label: 'Shift Date', description: 'Date of the volunteer shift', isRequired: true, isConditional: false, sampleValue: 'Saturday, February 15, 2025', sortOrder: 4 },
{ key: 'SHIFT_TIME', label: 'Shift Time', description: 'Time of the volunteer shift', isRequired: true, isConditional: false, sampleValue: '10:00 AM — 2:00 PM', sortOrder: 5 },
{ key: 'SHIFT_LOCATION', label: 'Shift Location', description: 'Location of the volunteer shift', isRequired: true, isConditional: false, sampleValue: '123 Main Street', sortOrder: 6 },
{ key: 'IS_NEW_USER', label: 'Is New User', description: 'Whether this is a new user account (used in conditional blocks)', isRequired: false, isConditional: true, sampleValue: 'true', sortOrder: 7 },
{ key: 'TEMP_PASSWORD', label: 'Temporary Password', description: 'Temporary password for new user account', isRequired: false, isConditional: true, sampleValue: 'SwiftEagle42', sortOrder: 8 },
{ key: 'LOGIN_URL', label: 'Login URL', description: 'URL to log in to the system', isRequired: true, isConditional: false, sampleValue: 'https://app.cmlite.org/login', sortOrder: 9 },
],
},
{
key: 'shift-details',
name: 'Shift Details Reminder',
description: 'Reminder email sent to all volunteers signed up for a shift',
category: 'MAP',
subjectLine: 'Shift Reminder - {{SHIFT_TITLE}} on {{SHIFT_DATE}}',
isSystem: true,
variables: [
{ key: 'ORGANIZATION_NAME', label: 'Organization Name', description: 'Name of the organization', isRequired: true, isConditional: false, sampleValue: 'Changemaker Lite', sortOrder: 0 },
{ key: 'USER_NAME', label: 'Volunteer Name', description: 'Name of the volunteer', isRequired: true, isConditional: false, sampleValue: 'John Smith', sortOrder: 1 },
{ key: 'SHIFT_TITLE', label: 'Shift Title', description: 'Title of the volunteer shift', isRequired: true, isConditional: false, sampleValue: 'Weekend Canvassing - Downtown', sortOrder: 2 },
{ key: 'SHIFT_DATE', label: 'Shift Date', description: 'Date of the volunteer shift', isRequired: true, isConditional: false, sampleValue: 'Saturday, February 15, 2025', sortOrder: 3 },
{ key: 'SHIFT_START_TIME', label: 'Start Time', description: 'Start time of the shift', isRequired: true, isConditional: false, sampleValue: '10:00 AM', sortOrder: 4 },
{ key: 'SHIFT_END_TIME', label: 'End Time', description: 'End time of the shift', isRequired: true, isConditional: false, sampleValue: '2:00 PM', sortOrder: 5 },
{ key: 'SHIFT_LOCATION', label: 'Shift Location', description: 'Location of the volunteer shift', isRequired: true, isConditional: false, sampleValue: '123 Main Street', sortOrder: 6 },
{ key: 'SHIFT_DESCRIPTION', label: 'Shift Description', description: 'Description of the shift (used in conditional blocks)', isRequired: false, isConditional: true, sampleValue: 'We will be canvassing the downtown area to connect with voters...', sortOrder: 7 },
{ key: 'CURRENT_VOLUNTEERS', label: 'Current Volunteers', description: 'Number of volunteers currently signed up', isRequired: true, isConditional: false, sampleValue: '8', sortOrder: 8 },
{ key: 'MAX_VOLUNTEERS', label: 'Max Volunteers', description: 'Maximum number of volunteers allowed', isRequired: true, isConditional: false, sampleValue: '10', sortOrder: 9 },
{ key: 'SHIFT_STATUS', label: 'Shift Status', description: 'Status of the shift (OPEN, FULL, CANCELLED)', isRequired: true, isConditional: false, sampleValue: 'OPEN', sortOrder: 10 },
],
},
];
async function seedEmailTemplates() {
console.log('🌱 Starting email template seeding...\n');
// Find or create SUPER_ADMIN user for created_by
let adminUser = await prisma.user.findFirst({
where: { role: 'SUPER_ADMIN' },
});
if (!adminUser) {
console.log('⚠️ No SUPER_ADMIN user found. Creating a system user...');
adminUser = await prisma.user.create({
data: {
email: 'system@cmlite.org',
password: 'not-used-for-login',
name: 'System',
role: 'SUPER_ADMIN',
status: 'ACTIVE',
},
});
console.log(`✓ Created system user: ${adminUser.email}\n`);
}
const templatesDir = path.join(__dirname, '../templates/email');
let seededCount = 0;
let skippedCount = 0;
let errorCount = 0;
for (const templateDef of TEMPLATES) {
try {
console.log(`Processing: ${templateDef.key}`);
// Check if template already exists
const existing = await prisma.emailTemplate.findUnique({
where: { key: templateDef.key },
});
if (existing) {
console.log(` ⏭️ Skipped (already exists)\n`);
skippedCount++;
continue;
}
// Read HTML and TXT files from filesystem
const htmlPath = path.join(templatesDir, `${templateDef.key}.html`);
const txtPath = path.join(templatesDir, `${templateDef.key}.txt`);
if (!fs.existsSync(htmlPath) || !fs.existsSync(txtPath)) {
console.log(` ❌ Error: Template files not found\n`);
errorCount++;
continue;
}
const htmlContent = fs.readFileSync(htmlPath, 'utf-8');
const textContent = fs.readFileSync(txtPath, 'utf-8');
// Create template with variables in a transaction
await prisma.$transaction(async (tx) => {
// Create template
const template = await tx.emailTemplate.create({
data: {
key: templateDef.key,
name: templateDef.name,
description: templateDef.description,
category: templateDef.category,
subjectLine: templateDef.subjectLine,
htmlContent,
textContent,
isSystem: templateDef.isSystem,
isActive: true,
createdByUserId: adminUser.id,
variables: {
create: templateDef.variables.map((v) => ({
key: v.key,
label: v.label,
description: v.description,
isRequired: v.isRequired,
isConditional: v.isConditional,
sampleValue: v.sampleValue,
sortOrder: v.sortOrder,
})),
},
},
});
// Create initial version
await tx.emailTemplateVersion.create({
data: {
templateId: template.id,
versionNumber: 1,
subjectLine: templateDef.subjectLine,
htmlContent,
textContent,
changeNotes: 'Initial migration from filesystem',
createdByUserId: adminUser.id,
},
});
});
console.log(` ✓ Seeded with ${templateDef.variables.length} variables\n`);
seededCount++;
}
catch (error) {
console.error(` ❌ Error: ${error instanceof Error ? error.message : String(error)}\n`);
errorCount++;
}
}
console.log('═'.repeat(60));
console.log(`✨ Seeding complete!`);
console.log(` Seeded: ${seededCount}`);
console.log(` Skipped: ${skippedCount}`);
console.log(` Errors: ${errorCount}`);
console.log('═'.repeat(60));
}
seedEmailTemplates()
.catch((error) => {
console.error('Fatal error:', error);
process.exit(1);
})
.finally(async () => {
await prisma.$disconnect();
});
//# sourceMappingURL=seed-email-templates.js.map