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>
This commit is contained in:
Georg Bauer
2026-03-09 14:43:18 +01:00
committed by GitHub
parent f1c9038803
commit b855d61524
116 changed files with 19954 additions and 2094 deletions

View File

@@ -109,7 +109,7 @@ describe('replaceAllMacrosAsync', () => {
expect(result).toContain('<aside class="custom-box">Custom Content</aside>');
});
it('returns empty string for unknown macros without Python renderer', async () => {
it('preserves unknown macros without Python renderer', async () => {
const result = await replaceAllMacrosAsync(
'Before [[unknown_macro]] After',
'post-1',
@@ -120,10 +120,10 @@ describe('replaceAllMacrosAsync', () => {
null,
);
expect(result).toBe('Before After');
expect(result).toBe('Before [[unknown_macro]] After');
});
it('returns empty string for unmatched Python macros', async () => {
it('preserves unmatched Python macros', async () => {
const mockRenderer: PythonMacroRendererContract = {
getEnabledMacroScripts: vi.fn().mockResolvedValue([]),
renderMacro: vi.fn(),
@@ -139,7 +139,7 @@ describe('replaceAllMacrosAsync', () => {
mockRenderer,
);
expect(result).toBe('Before After');
expect(result).toBe('Before [[nonexistent_macro]] After');
expect(mockRenderer.renderMacro).not.toHaveBeenCalled();
});
@@ -186,7 +186,21 @@ describe('replaceAllMacrosAsync', () => {
mockRenderer,
);
expect(result).toBe('Before After');
expect(result).toBe('Before [[my_macro]] After');
});
it('preserves the original unknown macro tag including params', async () => {
const result = await replaceAllMacrosAsync(
'Before [[unknown_macro title="Hello" count="2"]] After',
'post-1',
[],
null,
[],
'en',
null,
);
expect(result).toBe('Before [[unknown_macro title="Hello" count="2"]] After');
});
it('does not look up Python scripts when all macros are built-in', async () => {
@@ -244,6 +258,73 @@ describe('replaceAllMacrosAsync', () => {
expect(call.cacheKey).toBe('ctx-script:2');
});
it('passes languagePrefix and translations in Python macro context', async () => {
const mockRenderer: PythonMacroRendererContract = {
getEnabledMacroScripts: vi.fn().mockResolvedValue([
{
id: 'lang-script',
slug: 'lang_test',
entrypoint: 'render',
content: 'def render(ctx, post): return {"html": "ok"}',
version: 1,
},
] satisfies PythonMacroScript[]),
renderMacro: vi.fn().mockResolvedValue({ html: 'ok' }),
};
await replaceAllMacrosAsync(
'[[lang_test]]',
'post-1',
[],
null,
[],
'fr',
mockRenderer,
null,
'/fr',
);
const call = (mockRenderer.renderMacro as ReturnType<typeof vi.fn>).mock.calls[0][0];
const parsedContext = JSON.parse(call.contextJson);
expect(parsedContext.env.languagePrefix).toBe('/fr');
expect(parsedContext.env.mainLanguage).toBe('fr');
expect(parsedContext.env.translations).toBeDefined();
expect(typeof parsedContext.env.translations).toBe('object');
expect(parsedContext.env.translations['render.archive']).toBe('Archives');
});
it('passes empty languagePrefix when not provided', async () => {
const mockRenderer: PythonMacroRendererContract = {
getEnabledMacroScripts: vi.fn().mockResolvedValue([
{
id: 'no-prefix-script',
slug: 'no_prefix',
entrypoint: 'render',
content: 'def render(ctx, post): return {"html": "ok"}',
version: 1,
},
] satisfies PythonMacroScript[]),
renderMacro: vi.fn().mockResolvedValue({ html: 'ok' }),
};
await replaceAllMacrosAsync(
'[[no_prefix]]',
'post-1',
[],
null,
[],
'en',
mockRenderer,
);
const call = (mockRenderer.renderMacro as ReturnType<typeof vi.fn>).mock.calls[0][0];
const parsedContext = JSON.parse(call.contextJson);
expect(parsedContext.env.languagePrefix).toBe('');
expect(parsedContext.env.translations).toBeDefined();
});
it('returns unchanged text when there are no macros', async () => {
const content = 'Just plain text with no macros';
const result = await replaceAllMacrosAsync(content, '', [], null, [], 'en', null);