From 823cbecb842475f1e21f37a8c67aa3812f8bfdfc Mon Sep 17 00:00:00 2001 From: claude Date: Mon, 1 Jun 2026 16:50:23 +0300 Subject: [PATCH] Use raw SQL for template seeding in email InitialSchema migration EF Core migration scaffolding doesn't recognize InsertData calls made through local functions in manually-edited migrations. Changed to use raw SQL INSERT statements with migrationBuilder.Sql() to directly populate the Templates table with all required email.* and html.job-search.* templates (en+ro). This ensures templates are present when EmailTemplateService loads the cache, preventing 'Email template not found' warnings and enabling proper email rendering for CV match results and job search pages. Co-Authored-By: Claude Haiku 4.5 --- .../20260601133043_InitialSchema.cs | 101 +++++++----------- 1 file changed, 37 insertions(+), 64 deletions(-) diff --git a/Apis/email-data/Migrations/20260601133043_InitialSchema.cs b/Apis/email-data/Migrations/20260601133043_InitialSchema.cs index cc3b1f3..8beea0c 100644 --- a/Apis/email-data/Migrations/20260601133043_InitialSchema.cs +++ b/Apis/email-data/Migrations/20260601133043_InitialSchema.cs @@ -31,72 +31,45 @@ namespace Email.Data.Migrations table.PrimaryKey("PK_Templates", x => new { x.Key, x.Language }); }); - SeedTemplates(migrationBuilder); + // Seed email templates using raw SQL + migrationBuilder.Sql($$""" + INSERT INTO [email].Templates ([Key], [Language], [Value], [Description]) + VALUES + ('email.match.subject', 'en', 'MyAi.ro CV Match: {{score}}% - {{jobLabel}}', 'Subject for the CV match result email'), + ('email.match.subject', 'ro', 'MyAi.ro Potrivire CV: {{score}}% - {{jobLabel}}', 'Subiect email rezultat potrivire CV'), + ('email.match.body', 'en', 'CV Matcher result\n\nCV Document ID: {{cvDocumentId}}\nJob: {{jobLabel}}\nJob URL: {{jobUrl}}\nScore: {{score}}%\n\nSummary:\n{{summary}}\n\nStrengths:\n{{strengths}}\n\nGaps:\n{{gaps}}\n\nRecommendations:\n{{recommendations}}', 'Body for the CV match result email'), + ('email.match.body', 'ro', 'Rezultat potrivire CV\n\nID document CV: {{cvDocumentId}}\nJob: {{jobLabel}}\nURL job: {{jobUrl}}\nScor: {{score}}%\n\nRezumat:\n{{summary}}\n\nPuncte forte:\n{{strengths}}\n\nLipsuri:\n{{gaps}}\n\nRecomandări:\n{{recommendations}}', 'Corpul emailului pentru rezultatul potrivirii CV'), + ('email.match.job-search-footer', 'en', '\n\n---\nWant to find more jobs matching your CV?\nClick: {{jobSearchLink}}\n(link valid for {{expiryDays}} days)', 'Job search CTA appended to match result email'), + ('email.match.job-search-footer', 'ro', '\n\n---\nVrei sa gasesti mai multe joburi potrivite CV-ului tau?\nClick: {{jobSearchLink}}\n(link valabil {{expiryDays}} zile)', 'CTA cautare joburi adaugat la emailul de potrivire CV'), + ('email.search-results.subject', 'en', 'MyAi.ro: {{count}} jobs matching your CV', 'Subject for job search results email'), + ('email.search-results.subject', 'ro', 'MyAi.ro: {{count}} joburi potrivite CV-ului tau', 'Subiect email rezultate cautare joburi'), + ('email.search-results.body', 'en', 'MyAi.ro found {{count}} jobs matching your CV:\n\n{{items}}', 'Body preamble for job search results email'), + ('email.search-results.body', 'ro', 'MyAi.ro a gasit {{count}} joburi potrivite CV-ului tau:\n\n{{items}}', 'Corpul emailului de rezultate cautare joburi'), + ('email.search-results.empty', 'en', 'MyAi.ro found no jobs matching your CV. Try again later or update your CV.', 'No results message for job search results email'), + ('email.search-results.empty', 'ro', 'MyAi.ro nu a gasit joburi care sa corespunda CV-ului tau. Incercati mai tarziu sau ajustati CV-ul.', 'Mesaj fara rezultate pentru emailul de cautare joburi'), + ('html.job-search.started.title', 'en', 'Job search started', 'Title for job search started page'), + ('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'), + ('html.job-search.started.title', 'ro', 'Căutare joburi pornită', 'Titlu pagina cautare joburi pornita'), + ('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'), + ('html.job-search.already-used.title', 'en', 'Link already used', 'Title for already-used page'), + ('html.job-search.already-used.message', 'en', 'This job search link has already been used.', 'Message for already-used page'), + ('html.job-search.already-used.title', 'ro', 'Link deja folosit', 'Titlu pagina link deja folosit'), + ('html.job-search.already-used.message', 'ro', 'Acest link de cautare joburi a fost deja folosit.', 'Mesaj pagina link deja folosit'), + ('html.job-search.expired.title', 'en', 'Link expired', 'Title for expired link page'), + ('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'), + ('html.job-search.expired.title', 'ro', 'Link expirat', 'Titlu pagina link expirat'), + ('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'), + ('html.job-search.invalid.title', 'en', 'Invalid link', 'Title for invalid link page'), + ('html.job-search.invalid.message', 'en', 'This job search link is not valid.', 'Message for invalid link page'), + ('html.job-search.invalid.title', 'ro', 'Link invalid', 'Titlu pagina link invalid'), + ('html.job-search.invalid.message', 'ro', 'Acest link de cautare joburi nu este valid.', 'Mesaj pagina link invalid'), + ('html.job-search.error.title', 'en', 'Error', 'Title for error page'), + ('html.job-search.error.message', 'en', 'An error occurred. Please try again later.', 'Message for error page'), + ('html.job-search.error.title', 'ro', 'Eroare', 'Titlu pagina eroare'), + ('html.job-search.error.message', 'ro', 'A apărut o eroare. Te rugăm să încerci din nou mai târziu.', 'Mesaj pagina eroare'); + """); } - private static void SeedTemplates(MigrationBuilder m) - { - void Row(string key, string lang, string value, string description = "") - => m.InsertData("Templates", ["Key", "Language", "Value", "Description"], [key, lang, value, description], "email"); - - // 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 - Row("email.match.body", "en", - "CV Matcher result\n\nCV Document ID: {{cvDocumentId}}\nJob: {{jobLabel}}\nJob URL: {{jobUrl}}\nScore: {{score}}%\n\nSummary:\n{{summary}}\n\nStrengths:\n{{strengths}}\n\nGaps:\n{{gaps}}\n\nRecommendations:\n{{recommendations}}", - "Body for the CV match result email"); - Row("email.match.body", "ro", - "Rezultat potrivire CV\n\nID document CV: {{cvDocumentId}}\nJob: {{jobLabel}}\nURL job: {{jobUrl}}\nScor: {{score}}%\n\nRezumat:\n{{summary}}\n\nPuncte forte:\n{{strengths}}\n\nLipsuri:\n{{gaps}}\n\nRecomandări:\n{{recommendations}}", - "Corpul emailului pentru rezultatul potrivirii CV"); - - // Match result email — job search CTA footer - Row("email.match.job-search-footer", "en", - "\n\n---\nWant to find more jobs matching your CV?\nClick: {{jobSearchLink}}\n(link valid for {{expiryDays}} days)", - "Job search CTA appended to match result email"); - Row("email.match.job-search-footer", "ro", - "\n\n---\nVrei sa gasesti mai multe joburi potrivite CV-ului tau?\nClick: {{jobSearchLink}}\n(link valabil {{expiryDays}} zile)", - "CTA cautare joburi adaugat la emailul de potrivire CV"); - - // 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) - Row("email.search-results.body", "en", "MyAi.ro found {{count}} jobs matching your CV:\n\n{{items}}", "Body preamble for job search results email"); - Row("email.search-results.body", "ro", "MyAi.ro a gasit {{count}} joburi potrivite CV-ului tau:\n\n{{items}}", "Corpul emailului de rezultate cautare joburi"); - - // Job search results email — no results found - Row("email.search-results.empty", "en", "MyAi.ro found no jobs matching your CV. Try again later or update your CV.", "No results message for job search results email"); - Row("email.search-results.empty", "ro", "MyAi.ro nu a gasit joburi care sa corespunda CV-ului tau. Incercati mai tarziu sau ajustati CV-ul.", "Mesaj fara rezultate pentru emailul de cautare joburi"); - - // 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"); - } /// protected override void Down(MigrationBuilder migrationBuilder)