open-design/apps/web/tests/artifacts/renderer-registry.test.ts
marco 5dd70b5016
Some checks failed
ci / Validate workspace (push) Successful in 12m32s
landing-page-ci / Validate landing page (push) Successful in 9m41s
landing-page-deploy / Deploy landing page (push) Failing after 5m23s
github-metrics / Generate repository metrics SVG (push) Failing after 2m3s
refresh-contributors-wall / Refresh contributors wall cache bust (push) Failing after 11s
Initial import: open-design source for helix-mind.ai distribution
This repository contains the open-design daemon CLI source code, built
and packaged at https://helix-mind.ai/cli/open-design/latest.tgz for use
by the HelixMind /design slash command.

Licenses: Apache-2.0 (root) + MIT (skills/*)
2026-05-06 20:50:24 +02:00

169 lines
5.0 KiB
TypeScript

import { describe, expect, it } from 'vitest';
import {
DeckHtmlRenderer,
HtmlRenderer,
MarkdownRenderer,
ReactComponentRenderer,
RendererRegistry,
SvgRenderer,
artifactRendererRegistry,
} from '../../src/artifacts/renderer-registry';
import { renderMarkdownToSafeHtml } from '../../src/artifacts/markdown';
import type { ProjectFile } from '../../src/types';
function baseFile(overrides: Partial<ProjectFile> & Pick<ProjectFile, 'name'>): ProjectFile {
return {
path: 'artifact.html',
type: 'file',
size: 1,
mtime: Date.now(),
kind: 'html',
mime: 'text/html; charset=utf-8',
...overrides,
};
}
describe('RendererRegistry', () => {
const registry = new RendererRegistry([
ReactComponentRenderer,
DeckHtmlRenderer,
HtmlRenderer,
MarkdownRenderer,
SvgRenderer,
]);
it('resolves markdown renderer from explicit manifest', () => {
const file = baseFile({
name: 'notes.md',
kind: 'text',
mime: 'text/markdown; charset=utf-8',
artifactManifest: {
version: 1,
kind: 'markdown-document',
title: 'Notes',
entry: 'notes.md',
renderer: 'markdown',
exports: ['md', 'html'],
},
});
const match = registry.resolve({ file, isDeckHint: false });
expect(match?.renderer.id).toBe('markdown');
expect(match?.manifest.renderer).toBe('markdown');
});
it('falls back to inferred markdown manifest for .md files', () => {
const file = baseFile({
name: 'README.md',
kind: 'text',
mime: 'text/markdown; charset=utf-8',
artifactManifest: undefined,
});
const match = registry.resolve({ file, isDeckHint: false });
expect(match?.renderer.id).toBe('markdown');
expect(match?.manifest.kind).toBe('markdown-document');
});
it('resolves svg renderer from explicit manifest', () => {
const file = baseFile({
name: 'diagram.svg',
kind: 'sketch',
mime: 'image/svg+xml',
artifactManifest: {
version: 1,
kind: 'svg',
title: 'Diagram',
entry: 'diagram.svg',
renderer: 'svg',
exports: ['svg'],
},
});
const match = registry.resolve({ file, isDeckHint: false });
expect(match?.renderer.id).toBe('svg');
expect(match?.manifest.renderer).toBe('svg');
});
it('falls back to inferred svg manifest for .svg files', () => {
const file = baseFile({
name: 'logo.svg',
kind: 'sketch',
mime: 'image/svg+xml',
artifactManifest: undefined,
});
const match = registry.resolve({ file, isDeckHint: false });
expect(match?.renderer.id).toBe('svg');
expect(match?.manifest.kind).toBe('svg');
});
it('keeps unknown files on old fallback path', () => {
const file = baseFile({
name: 'archive.bin',
kind: 'binary',
mime: 'application/octet-stream',
artifactManifest: undefined,
});
expect(registry.resolve({ file, isDeckHint: false })).toBeNull();
});
it('exposes conservative streaming contract values', () => {
expect(HtmlRenderer.supportsStreaming).toBe(false);
expect(DeckHtmlRenderer.supportsStreaming).toBe(false);
expect(MarkdownRenderer.supportsStreaming).toBe(true);
expect(MarkdownRenderer.renderPartial).toBe(renderMarkdownToSafeHtml);
expect(SvgRenderer.supportsStreaming).toBe(false);
expect(SvgRenderer.renderPartial).toBeUndefined();
});
it('keeps markdown partial renderer output safe', () => {
const out = MarkdownRenderer.renderPartial?.('[<script>alert(1)</script>](https://example.com/a_b_c)') ?? '';
expect(out).toContain('&lt;script&gt;alert(1)&lt;/script&gt;');
expect(out).toContain('href="https://example.com/a_b_c"');
expect(out).not.toContain('<script>');
});
it('routes JSX and TSX files to the React component renderer', () => {
expect(
artifactRendererRegistry.resolve({
file: baseFile({
name: 'Hero.jsx',
kind: 'code',
mime: 'text/javascript; charset=utf-8',
}),
isDeckHint: false,
})?.renderer.id,
).toBe('react-component');
expect(
artifactRendererRegistry.resolve({
file: baseFile({
name: 'Hero.tsx',
kind: 'code',
mime: 'text/typescript; charset=utf-8',
}),
isDeckHint: false,
})?.renderer.id,
).toBe('react-component');
});
it('prefers an explicit React manifest over the coarse code kind', () => {
expect(
artifactRendererRegistry.resolve({
file: baseFile({
name: 'entry.txt',
kind: 'text',
artifactManifest: {
version: 1,
kind: 'react-component',
title: 'Entry',
entry: 'entry.txt',
renderer: 'react-component',
exports: ['jsx', 'html', 'zip'],
},
}),
isDeckHint: false,
})?.renderer.id,
).toBe('react-component');
});
});