2838885e22
- Replace div-based layouts with table-based HTML throughout (max-width/border-radius/display:inline-block ignored by Outlook) - email.match.body: width:100% table with per-cell borders and fixed 130px label column - email.match.job-search-footer: table-based button with bgcolor attribute - email.search-results.empty: div replaced with full-width table - email.search-results.body: remove div wrapper around items - Add email.search-results.scan-summary and email.search-results.item templates - CvSearchEmailSender: remove all hardcoded HTML; render via IEmailTemplateService Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
257 lines
17 KiB
C#
257 lines
17 KiB
C#
using System;
|
|
using Email.Data;
|
|
using Microsoft.EntityFrameworkCore.Migrations;
|
|
|
|
#nullable disable
|
|
|
|
namespace Email.Data.Migrations
|
|
{
|
|
/// <inheritdoc />
|
|
public partial class InitialSchema : Migration
|
|
{
|
|
/// <inheritdoc />
|
|
protected override void Up(MigrationBuilder migrationBuilder)
|
|
{
|
|
migrationBuilder.EnsureSchema(
|
|
name: MigrationConstants.SchemaName);
|
|
|
|
migrationBuilder.CreateTable(
|
|
name: "Templates",
|
|
schema: MigrationConstants.SchemaName,
|
|
columns: table => new
|
|
{
|
|
Key = table.Column<string>(type: "nvarchar(128)", maxLength: 128, nullable: false),
|
|
Language = table.Column<string>(type: "nvarchar(8)", maxLength: 8, nullable: false),
|
|
Value = table.Column<string>(type: "nvarchar(max)", nullable: false),
|
|
Description = table.Column<string>(type: "nvarchar(500)", maxLength: 500, nullable: false, defaultValue: ""),
|
|
UpdatedAt = table.Column<DateTime>(type: "datetime2", nullable: false, defaultValueSql: "SYSUTCDATETIME()"),
|
|
OperatorCopy = table.Column<string>(type: "nvarchar(256)", maxLength: 256, nullable: false, defaultValue: "")
|
|
},
|
|
constraints: table =>
|
|
{
|
|
table.PrimaryKey("PK_Templates", x => new { x.Key, x.Language });
|
|
});
|
|
|
|
Seed(migrationBuilder);
|
|
}
|
|
|
|
private static void Seed(MigrationBuilder m)
|
|
{
|
|
void Row(string key, string lang, string value, string description = "")
|
|
=> m.InsertData("Templates", ["Key", "Language", "Value", "Description"], [key, lang, value, description], MigrationConstants.SchemaName);
|
|
|
|
// Match result email — subject
|
|
Row("email.match.subject", "en", "MyAi.ro CV Match: {{score}}% - {{jobLabel}}", "Subject for the CV match result email");
|
|
Row("email.match.subject", "ro", "MyAi.ro Potrivire CV: {{score}}% - {{jobLabel}}", "Subiect email rezultat potrivire CV");
|
|
|
|
// Match result email — body (HTML formatted)
|
|
Row("email.match.body", "en",
|
|
@"<h2 style=""color: #2c5282; margin-bottom: 20px; border-bottom: 3px solid #2c5282; padding-bottom: 10px;"">CV Match Report</h2>
|
|
<table width=""100%"" cellpadding=""0"" cellspacing=""0"" border=""0"" style=""width: 100%; margin-bottom: 30px; border-collapse: collapse;"">
|
|
<tr style=""background-color: #2c5282; color: white;"">
|
|
<td width=""130"" style=""width: 130px; padding: 12px 15px; font-weight: bold; border: 1px solid #2c5282;"">CV ID</td>
|
|
<td style=""padding: 12px 15px; border: 1px solid #2c5282;"">{{cvDocumentId}}</td>
|
|
</tr>
|
|
<tr style=""background-color: #f8f9fa;"">
|
|
<td style=""width: 130px; padding: 12px 15px; font-weight: bold; border: 1px solid #ddd; color: #333;"">Job</td>
|
|
<td style=""padding: 12px 15px; border: 1px solid #ddd; color: #333;"">{{jobLabel}}</td>
|
|
</tr>
|
|
<tr>
|
|
<td style=""width: 130px; padding: 12px 15px; font-weight: bold; border: 1px solid #ddd; color: #333;"">URL</td>
|
|
<td style=""padding: 12px 15px; border: 1px solid #ddd;""><a href=""{{jobUrl}}"" style=""color: #2c5282; text-decoration: none;"">{{jobUrl}}</a></td>
|
|
</tr>
|
|
<tr style=""background-color: #f8f9fa;"">
|
|
<td style=""width: 130px; padding: 12px 15px; font-weight: bold; border: 1px solid #ddd; color: #333;"">Score</td>
|
|
<td style=""padding: 12px 15px; border: 1px solid #ddd; color: #27ae60; font-weight: bold; font-size: 18px;"">{{score}}%</td>
|
|
</tr>
|
|
</table>
|
|
<h3 style=""color: #2c5282; margin-top: 25px; margin-bottom: 10px; border-bottom: 2px solid #2c5282; padding-bottom: 5px;"">Summary</h3>
|
|
<p style=""line-height: 1.6; color: #333;"">{{summary}}</p>
|
|
<h3 style=""color: #2c5282; margin-top: 25px; margin-bottom: 10px; border-bottom: 2px solid #2c5282; padding-bottom: 5px;"">Strengths</h3>
|
|
<div style=""line-height: 1.8; color: #333;"">{{strengths}}</div>
|
|
<h3 style=""color: #2c5282; margin-top: 25px; margin-bottom: 10px; border-bottom: 2px solid #2c5282; padding-bottom: 5px;"">Gaps</h3>
|
|
<div style=""line-height: 1.8; color: #333;"">{{gaps}}</div>
|
|
<h3 style=""color: #2c5282; margin-top: 25px; margin-bottom: 10px; border-bottom: 2px solid #2c5282; padding-bottom: 5px;"">Recommendations</h3>
|
|
<div style=""line-height: 1.8; color: #333;"">{{recommendations}}</div>",
|
|
"Body for the CV match result email (HTML formatted)");
|
|
Row("email.match.body", "ro",
|
|
@"<h2 style=""color: #2c5282; margin-bottom: 20px; border-bottom: 3px solid #2c5282; padding-bottom: 10px;"">Report Potrivire CV</h2>
|
|
<table width=""100%"" cellpadding=""0"" cellspacing=""0"" border=""0"" style=""width: 100%; margin-bottom: 30px; border-collapse: collapse;"">
|
|
<tr style=""background-color: #2c5282; color: white;"">
|
|
<td width=""130"" style=""width: 130px; padding: 12px 15px; font-weight: bold; border: 1px solid #2c5282;"">ID Document CV</td>
|
|
<td style=""padding: 12px 15px; border: 1px solid #2c5282;"">{{cvDocumentId}}</td>
|
|
</tr>
|
|
<tr style=""background-color: #f8f9fa;"">
|
|
<td style=""width: 130px; padding: 12px 15px; font-weight: bold; border: 1px solid #ddd; color: #333;"">Job</td>
|
|
<td style=""padding: 12px 15px; border: 1px solid #ddd; color: #333;"">{{jobLabel}}</td>
|
|
</tr>
|
|
<tr>
|
|
<td style=""width: 130px; padding: 12px 15px; font-weight: bold; border: 1px solid #ddd; color: #333;"">URL</td>
|
|
<td style=""padding: 12px 15px; border: 1px solid #ddd;""><a href=""{{jobUrl}}"" style=""color: #2c5282; text-decoration: none;"">{{jobUrl}}</a></td>
|
|
</tr>
|
|
<tr style=""background-color: #f8f9fa;"">
|
|
<td style=""width: 130px; padding: 12px 15px; font-weight: bold; border: 1px solid #ddd; color: #333;"">Scor</td>
|
|
<td style=""padding: 12px 15px; border: 1px solid #ddd; color: #27ae60; font-weight: bold; font-size: 18px;"">{{score}}%</td>
|
|
</tr>
|
|
</table>
|
|
<h3 style=""color: #2c5282; margin-top: 25px; margin-bottom: 10px; border-bottom: 2px solid #2c5282; padding-bottom: 5px;"">Rezumat</h3>
|
|
<p style=""line-height: 1.6; color: #333;"">{{summary}}</p>
|
|
<h3 style=""color: #2c5282; margin-top: 25px; margin-bottom: 10px; border-bottom: 2px solid #2c5282; padding-bottom: 5px;"">Puncte Forte</h3>
|
|
<div style=""line-height: 1.8; color: #333;"">{{strengths}}</div>
|
|
<h3 style=""color: #2c5282; margin-top: 25px; margin-bottom: 10px; border-bottom: 2px solid #2c5282; padding-bottom: 5px;"">Lipsuri</h3>
|
|
<div style=""line-height: 1.8; color: #333;"">{{gaps}}</div>
|
|
<h3 style=""color: #2c5282; margin-top: 25px; margin-bottom: 10px; border-bottom: 2px solid #2c5282; padding-bottom: 5px;"">Recomandări</h3>
|
|
<div style=""line-height: 1.8; color: #333;"">{{recommendations}}</div>",
|
|
"Corpul emailului pentru rezultatul potrivirii CV (format HTML)");
|
|
|
|
// Match result email — job search CTA footer (HTML formatted)
|
|
Row("email.match.job-search-footer", "en",
|
|
@"<hr style=""border: none; border-top: 1px solid #ddd; margin: 30px 0;"">
|
|
<p style=""margin-top: 20px; font-size: 14px; color: #333;"">Want to find more jobs matching your CV?</p>
|
|
<table cellpadding=""0"" cellspacing=""0"" border=""0"" style=""margin-bottom: 20px;"">
|
|
<tr>
|
|
<td align=""center"" bgcolor=""#2c5282"" style=""background-color: #2c5282; padding: 10px 20px;"">
|
|
<a href=""{{jobSearchLink}}"" style=""color: #ffffff; font-weight: bold; text-decoration: none; font-size: 14px; display: block;"">Search Jobs</a>
|
|
</td>
|
|
</tr>
|
|
</table>
|
|
<p style=""font-size: 12px; color: #666;"">(link valid for {{expiryDays}} days)</p>",
|
|
"Job search CTA appended to match result email (HTML formatted)");
|
|
Row("email.match.job-search-footer", "ro",
|
|
@"<hr style=""border: none; border-top: 1px solid #ddd; margin: 30px 0;"">
|
|
<p style=""margin-top: 20px; font-size: 14px; color: #333;"">Vrei să găsești mai multe joburi potrivite CV-ului tău?</p>
|
|
<table cellpadding=""0"" cellspacing=""0"" border=""0"" style=""margin-bottom: 20px;"">
|
|
<tr>
|
|
<td align=""center"" bgcolor=""#2c5282"" style=""background-color: #2c5282; padding: 10px 20px;"">
|
|
<a href=""{{jobSearchLink}}"" style=""color: #ffffff; font-weight: bold; text-decoration: none; font-size: 14px; display: block;"">Caută Joburi</a>
|
|
</td>
|
|
</tr>
|
|
</table>
|
|
<p style=""font-size: 12px; color: #666;"">(link valabil {{expiryDays}} zile)</p>",
|
|
"CTA cautare joburi adaugat la emailul de potrivire CV (format HTML)");
|
|
|
|
// Job search results email — subject
|
|
Row("email.search-results.subject", "en", "MyAi.ro: {{count}} jobs matching your CV", "Subject for job search results email");
|
|
Row("email.search-results.subject", "ro", "MyAi.ro: {{count}} joburi potrivite CV-ului tau", "Subiect email rezultate cautare joburi");
|
|
|
|
// Job search results email — body preamble (items appended in code) - HTML formatted
|
|
Row("email.search-results.body", "en",
|
|
@"<h2 style=""color: #2c5282; margin-bottom: 20px; border-bottom: 3px solid #2c5282; padding-bottom: 10px;"">Job Search Results</h2>
|
|
<p style=""margin-bottom: 20px; color: #333;"">MyAi.ro found <strong>{{count}}</strong> jobs matching your CV:</p>
|
|
{{items}}",
|
|
"Body preamble for job search results email (HTML formatted)");
|
|
Row("email.search-results.body", "ro",
|
|
@"<h2 style=""color: #2c5282; margin-bottom: 20px; border-bottom: 3px solid #2c5282; padding-bottom: 10px;"">Rezultate Căutare Joburi</h2>
|
|
<p style=""margin-bottom: 20px; color: #333;"">MyAi.ro a găsit <strong>{{count}}</strong> joburi potrivite CV-ului tău:</p>
|
|
{{items}}",
|
|
"Corpul emailului de rezultate cautare joburi (format HTML)");
|
|
|
|
// Job search results email — scan summary block (keywords + providers used)
|
|
Row("email.search-results.scan-summary", "en",
|
|
@"<table width=""100%"" cellpadding=""0"" cellspacing=""0"" border=""0"" style=""width: 100%; margin-bottom: 18px; border: 1px solid #dee2e6; border-collapse: collapse;"">
|
|
<tr>
|
|
<td bgcolor=""#f8f9fa"" style=""background-color: #f8f9fa; padding: 14px 16px; font-size: 13px; color: #495057;"">
|
|
<div style=""margin-bottom: 8px;""><strong>Keywords used:</strong> {{keywordsHtml}}</div>
|
|
<div><strong>Providers scanned:</strong> {{providers}}</div>
|
|
</td>
|
|
</tr>
|
|
</table>",
|
|
"Scan summary block prepended to job search results email (HTML formatted)");
|
|
Row("email.search-results.scan-summary", "ro",
|
|
@"<table width=""100%"" cellpadding=""0"" cellspacing=""0"" border=""0"" style=""width: 100%; margin-bottom: 18px; border: 1px solid #dee2e6; border-collapse: collapse;"">
|
|
<tr>
|
|
<td bgcolor=""#f8f9fa"" style=""background-color: #f8f9fa; padding: 14px 16px; font-size: 13px; color: #495057;"">
|
|
<div style=""margin-bottom: 8px;""><strong>Cuvinte cheie folosite:</strong> {{keywordsHtml}}</div>
|
|
<div><strong>Furnizori scanați:</strong> {{providers}}</div>
|
|
</td>
|
|
</tr>
|
|
</table>",
|
|
"Bloc rezumat scanare adaugat la emailul de rezultate cautare joburi (format HTML)");
|
|
|
|
// Job search results email — single job result item card
|
|
Row("email.search-results.item", "en",
|
|
@"<table width=""100%"" cellpadding=""0"" cellspacing=""0"" border=""0"" style=""width: 100%; margin-bottom: 12px; border: 1px solid #dee2e6; border-collapse: collapse;"">
|
|
<tr>
|
|
<td style=""padding: 12px 16px; background-color: #ffffff;"">
|
|
<strong style=""color: #212529;"">{{index}}. {{jobTitle}}</strong>
|
|
<span style=""background: #28a745; color: #fff; padding: 2px 8px; font-size: 12px; margin-left: 8px;"">{{score}}% match</span>
|
|
<span style=""color: #6c757d; font-size: 12px; margin-left: 4px;"">[{{providerName}}]</span><br>
|
|
<a href=""{{jobUrl}}"" style=""color: #2c5282; font-size: 13px;"">{{jobUrl}}</a>
|
|
{{summary}}
|
|
</td>
|
|
</tr>
|
|
</table>",
|
|
"Single job result card in job search results email (HTML formatted)");
|
|
Row("email.search-results.item", "ro",
|
|
@"<table width=""100%"" cellpadding=""0"" cellspacing=""0"" border=""0"" style=""width: 100%; margin-bottom: 12px; border: 1px solid #dee2e6; border-collapse: collapse;"">
|
|
<tr>
|
|
<td style=""padding: 12px 16px; background-color: #ffffff;"">
|
|
<strong style=""color: #212529;"">{{index}}. {{jobTitle}}</strong>
|
|
<span style=""background: #28a745; color: #fff; padding: 2px 8px; font-size: 12px; margin-left: 8px;"">{{score}}% potrivire</span>
|
|
<span style=""color: #6c757d; font-size: 12px; margin-left: 4px;"">[{{providerName}}]</span><br>
|
|
<a href=""{{jobUrl}}"" style=""color: #2c5282; font-size: 13px;"">{{jobUrl}}</a>
|
|
{{summary}}
|
|
</td>
|
|
</tr>
|
|
</table>",
|
|
"Card job individual in emailul de rezultate cautare joburi (format HTML)");
|
|
|
|
// Job search results email — no results found - HTML formatted
|
|
Row("email.search-results.empty", "en",
|
|
@"<table width=""100%"" cellpadding=""0"" cellspacing=""0"" border=""0"" style=""width: 100%; margin: 20px 0;"">
|
|
<tr>
|
|
<td bgcolor=""#fff3cd"" style=""background-color: #fff3cd; border: 1px solid #ffc107; padding: 15px;"">
|
|
<p style=""margin: 0; color: #856404;""><strong>No jobs found</strong><br>
|
|
MyAi.ro found no jobs matching your CV at this moment. Please try again later or update your CV to improve your match results.</p>
|
|
</td>
|
|
</tr>
|
|
</table>",
|
|
"No results message for job search results email (HTML formatted)");
|
|
Row("email.search-results.empty", "ro",
|
|
@"<table width=""100%"" cellpadding=""0"" cellspacing=""0"" border=""0"" style=""width: 100%; margin: 20px 0;"">
|
|
<tr>
|
|
<td bgcolor=""#fff3cd"" style=""background-color: #fff3cd; border: 1px solid #ffc107; padding: 15px;"">
|
|
<p style=""margin: 0; color: #856404;""><strong>Niciun job găsit</strong><br>
|
|
MyAi.ro nu a găsit joburi potrivite CV-ului tău în acest moment. Te rugăm să încerci din nou mai târziu sau să-ți actualizezi CV-ul pentru a obține rezultate mai bune.</p>
|
|
</td>
|
|
</tr>
|
|
</table>",
|
|
"Mesaj fara rezultate pentru emailul de cautare joburi (format HTML)");
|
|
|
|
// HTML job-search start page messages
|
|
Row("html.job-search.started.title", "en", "Job search started", "Title for job search started page");
|
|
Row("html.job-search.started.message", "en", "Your job search has started. Results will be sent to your email shortly.", "Message for job search started page");
|
|
Row("html.job-search.started.title", "ro", "Căutare joburi pornită", "Titlu pagina cautare joburi pornita");
|
|
Row("html.job-search.started.message", "ro", "Căutarea joburilor a început. Rezultatele vor fi trimise pe email în scurt timp.", "Mesaj pagina cautare joburi pornita");
|
|
|
|
Row("html.job-search.already-used.title", "en", "Link already used", "Title for already-used page");
|
|
Row("html.job-search.already-used.message", "en", "This job search link has already been used.", "Message for already-used page");
|
|
Row("html.job-search.already-used.title", "ro", "Link deja folosit", "Titlu pagina link deja folosit");
|
|
Row("html.job-search.already-used.message", "ro", "Acest link de cautare joburi a fost deja folosit.", "Mesaj pagina link deja folosit");
|
|
|
|
Row("html.job-search.expired.title", "en", "Link expired", "Title for expired link page");
|
|
Row("html.job-search.expired.message", "en", "This job search link has expired. Please request a new CV match to get a fresh link.", "Message for expired link page");
|
|
Row("html.job-search.expired.title", "ro", "Link expirat", "Titlu pagina link expirat");
|
|
Row("html.job-search.expired.message", "ro", "Acest link de cautare joburi a expirat. Solicita o noua potrivire CV pentru a primi un link nou.", "Mesaj pagina link expirat");
|
|
|
|
Row("html.job-search.invalid.title", "en", "Invalid link", "Title for invalid link page");
|
|
Row("html.job-search.invalid.message", "en", "This job search link is not valid.", "Message for invalid link page");
|
|
Row("html.job-search.invalid.title", "ro", "Link invalid", "Titlu pagina link invalid");
|
|
Row("html.job-search.invalid.message", "ro", "Acest link de cautare joburi nu este valid.", "Mesaj pagina link invalid");
|
|
|
|
Row("html.job-search.error.title", "en", "Error", "Title for error page");
|
|
Row("html.job-search.error.message", "en", "An error occurred. Please try again later.", "Message for error page");
|
|
Row("html.job-search.error.title", "ro", "Eroare", "Titlu pagina eroare");
|
|
Row("html.job-search.error.message", "ro", "A apărut o eroare. Te rugăm să încerci din nou mai târziu.", "Mesaj pagina eroare");
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
protected override void Down(MigrationBuilder migrationBuilder)
|
|
{
|
|
migrationBuilder.DropTable(
|
|
name: "Templates",
|
|
schema: MigrationConstants.SchemaName);
|
|
}
|
|
}
|
|
}
|