Files
bDS/tests/engine/ValidationApplyPlannerService.test.ts
Georg Bauer b855d61524 Feature/post media translations (#42)
* chore: updated todo with translation ideas

* feat: first take at the implementation of translations

* fix: small addition for the translation feature

* feat: support language switching in the editor and preview

* feat: better handling of long bodies by not running them through a json envelope

* fix: unknown macros have better fallback

* feat: api for python to get translations

* fix: strip dumb prefix of content in translation

* feat: extend meta diff for translations

* feat: hook up translations to rebuild-from-disk

* feat: generation of the website prefers project language, falling back to canonical language

* fix: crashes during rendering

* feat: translation validation report

* fix: made the translation validation actually work

* chore: reorganization of menu

* fix: some topics cleanup

* chore: updated doc

* feat: translations for media

* feat: more aligned in UI/UX

* feat: edit translations possible

* chore: added full multi-language todo

* chore: updated todo for clarity

* feat: implementation of full multi-linguality

* fix: page creation creates pages

* fix: flags on every page

* fix: better prompt

* feat: made MCP server aware of language content

* feat: python tools for translations

* fix: better fill-in-translations

* fix: better prompt for translation. maybe.

* fix: losing posts from search due to translation process

* fix: translation validation handles in-db content and fill-in of missing translations fixed to flush

* fix: faster scanning for infilling of missing translations

* chore: updated agent instructions

* feat: calendar and tag cloud respect current language now

* fix: retries going up

* fix: got metadata-diff and rebuild into sync

* fix: extended meta-diff for timestamps

* fix: made website validation look at translated content, too

* fix: multi-lingual search

* chore: refactor Editor.tsx into two separate editors

* feat: do language detection when no explicit language given

---------

Co-authored-by: hugo <hugoms@me.com>
2026-03-09 14:43:18 +01:00

141 lines
5.0 KiB
TypeScript

import { describe, expect, it } from 'vitest';
import type { PostData } from '../../src/main/engine/PostEngine';
import {
buildTargetedValidationPlan,
planMissingValidationPaths,
} from '../../src/main/engine/ValidationApplyPlannerService';
function makePost(overrides: Partial<PostData> = {}): PostData {
const createdAt = overrides.createdAt ?? new Date('2025-01-15T10:00:00.000Z');
const updatedAt = overrides.updatedAt ?? createdAt;
return {
id: overrides.id ?? 'post-1',
projectId: overrides.projectId ?? 'default',
title: overrides.title ?? 'Title',
slug: overrides.slug ?? 'title',
excerpt: overrides.excerpt,
content: overrides.content ?? 'Body',
status: overrides.status ?? 'published',
author: overrides.author,
createdAt,
updatedAt,
publishedAt: overrides.publishedAt,
tags: overrides.tags ?? [],
categories: overrides.categories ?? [],
};
}
describe('ValidationApplyPlannerService', () => {
it('classifies missing paths into route request groups', () => {
const plan = planMissingValidationPaths([
'/',
'/page/2',
'/category/news/page/2',
'/tag/dev%20log',
'/2025/01/15/my%20post',
'/2025/page/2',
'/2025/01',
'/2025/01/15',
'/about',
]);
expect(plan.requiresFallbackSectionRender).toBe(false);
expect(plan.requestRootRoutes).toBe(true);
expect(Array.from(plan.requestedCategories)).toEqual(['news']);
expect(Array.from(plan.requestedTags)).toEqual(['dev log']);
expect(plan.requestedPostRoutes).toEqual([
{ year: 2025, month: 1, day: 15, slug: 'my post' },
]);
expect(Array.from(plan.requestedYears)).toContain(2025);
expect(Array.from(plan.requestedYearMonths)).toContain('2025/01');
expect(Array.from(plan.requestedYearMonthDays)).toContain('2025/01/15');
expect(Array.from(plan.requestedPageSlugs)).toEqual(['about']);
});
it('expands targeted rerender plan with single-route lineage and available archives', () => {
const publishedPost = makePost({
id: 'p1',
slug: 'post-one',
categories: ['news'],
tags: ['tag-1'],
createdAt: new Date('2025-01-15T10:00:00.000Z'),
});
const pagePost = makePost({
id: 'p2',
slug: 'about',
categories: ['page'],
tags: [],
createdAt: new Date('2025-01-10T10:00:00.000Z'),
});
const initialPlan = planMissingValidationPaths(['/2025/01/15/post-one', '/2025', '/about', '/category/missing']);
const targeted = buildTargetedValidationPlan({
initialPlan,
publishedPosts: [publishedPost, pagePost],
allCategories: new Set(['news', 'page']),
allTags: new Set(['tag-1']),
availableYearMonths: ['2025/01', '2025/02'],
availableYearMonthDays: ['2025/01/15', '2025/02/20'],
});
expect(targeted.requestedPostIds.has('p1')).toBe(true);
expect(targeted.requestedCategorySet.has('news')).toBe(true);
expect(targeted.requestedCategorySet.has('missing')).toBe(false);
expect(targeted.requestedTagSet.has('tag-1')).toBe(true);
expect(targeted.requestedYears.has(2025)).toBe(true);
expect(targeted.requestedYearMonths.has('2025/01')).toBe(true);
expect(targeted.requestedYearMonths.has('2025/02')).toBe(true);
expect(targeted.requestedYearMonthDays.has('2025/01/15')).toBe(true);
expect(targeted.requestedYearMonthDays.has('2025/02/20')).toBe(true);
expect(targeted.requestedPageSlugs.has('about')).toBe(true);
expect(targeted.requestRootRoutes).toBe(true);
});
it('classifies language-prefixed missing paths into per-language plans', () => {
const plan = planMissingValidationPaths(
[
'/fr/',
'/fr/page/2',
'/fr/category/news',
'/fr/tag/dev',
'/fr/2025/01/15/my-post',
'/fr/2025',
'/fr/2025/01',
'/fr/about',
'/de/',
'/de/category/tech',
],
['fr', 'de'],
);
expect(plan.requiresFallbackSectionRender).toBe(false);
expect(plan.requestRootRoutes).toBe(false);
const frPlan = plan.languagePlans.get('fr');
expect(frPlan).toBeDefined();
expect(frPlan!.requestRootRoutes).toBe(true);
expect(Array.from(frPlan!.requestedCategories)).toEqual(['news']);
expect(Array.from(frPlan!.requestedTags)).toEqual(['dev']);
expect(frPlan!.requestedPostRoutes).toEqual([
{ year: 2025, month: 1, day: 15, slug: 'my-post' },
]);
expect(Array.from(frPlan!.requestedYears)).toContain(2025);
expect(Array.from(frPlan!.requestedYearMonths)).toContain('2025/01');
expect(Array.from(frPlan!.requestedPageSlugs)).toEqual(['about']);
const dePlan = plan.languagePlans.get('de');
expect(dePlan).toBeDefined();
expect(dePlan!.requestRootRoutes).toBe(true);
expect(Array.from(dePlan!.requestedCategories)).toEqual(['tech']);
});
it('treats unknown prefixes as page slugs when no languages specified', () => {
const plan = planMissingValidationPaths(['/fr/category/news', '/fr/']);
expect(plan.languagePlans.size).toBe(0);
expect(plan.requiresFallbackSectionRender).toBe(true);
});
});