mirror of
https://github.com/danny-avila/LibreChat.git
synced 2025-12-25 12:48:53 +01:00
* feat: Add CodeArtifacts component to Beta settings tab * chore: Update npm dependency to @codesandbox/sandpack-react@2.18.2 * WIP: artifacts first pass * WIP first pass remark-directive * chore: revert markdown to original component + new artifacts rendering * refactor: first pass rewrite * refactor: add throttling * first pass styling * style: Add Radix Tabs, more styling changes * feat: second pass * style: code styling * fix: package markdown fixes * feat: Add useEffect hook to Artifacts component for visibility control, slide in animation * fix: only set artifact if there is content * refactor: typing and make latest artifact active if the number of artifacts changed * feat: artifacts + shadcnui * feat: Add Copy Code button to Artifacts component * feat: first pass streaming updates * refactor: optimize ordering of artifacts in Artifacts component * refactor: optimize ordering of artifacts and add latest artifact activation in Artifacts component * refactor: add order prop to Artifact * feat: update to latest, use update time for ordering * refactor: optimize ordering of artifacts and activate latest artifact in Artifacts component * wip: remove thinking text and artifact formatting if empty * refactor: optimize Markdown rendering and add support for code artifacts * feat: global state for current artifact Id and set on artifact preview click * refactor: Rename CodePreview component to ArtifactButton * refactor: apply growth to artifact frame so artifact preview can take full space * refactor: remove artifactIdsState * refactor: nullify artifact state and reset on empty conversation * feat: reset artifact state on conversation change * feat: artifacts system prompt in backend * refactor: update UI artifact toggle label to match localization key * style: remove ArtifactButton inline-block styling * feat: memoize ArtifactPreview, add html support * refactor: abstract out components * chore: bump react-resizable-panel * refactor: resizable panel order props * fix: side panel resizing crashes * style: temporarily remove scrolling, add better styling * chore: remove thinking for now * chore: preprocess artifacts for now * feat: Add auto scrolling to CodeMarkdown (artifacts) * feat: autoswitch to preview * feat: auto switch to code, adjust prompt, remove unused code * feat: refresh button * feat: open/close artifacts * wip: mermaid * refactor: w-fit Artifact button * chore: organize code * feat: first pass mermaid * refactor: improve panning logic in MermaidDiagram component * feat: center/zoom on first render * refactor: add centering with reset button * style: mermaid styling * refactor: add back MermaidDiagram * fix: static/html template * fix: mermaid * add examples to artifacts prompt * refactor: fix CodeBar plugin prop logic * refactor: remove unnecessary mention of artifacts when not requested * fix: remove preprocessCodeArtifacts function and fix imports * feat: improve artifacts guidelines and remove unnecessary mentions * refactor: improve artifacts guidelines and remove unnecessary mentions * chore: uninstall unused packages * chore: bump vite * chore: update three dependency to version 0.167.1 * refactor: move beta settings, add additional artifacts toggles * feat: artifacts mode toggles * refactor: adjust prompt * feat: shadcnui instructions * feat: code artifacts custom prompt mode * chore: Update artifacts UI labels and instructions localizations * refactor: Remove unused code in Markdown component
94 lines
3.4 KiB
TypeScript
94 lines
3.4 KiB
TypeScript
import { preprocessCodeArtifacts } from './artifacts';
|
|
|
|
describe('preprocessCodeArtifacts', () => {
|
|
test('should return non-string inputs unchanged', () => {
|
|
expect(preprocessCodeArtifacts(123 as unknown as string)).toBe('');
|
|
expect(preprocessCodeArtifacts(null as unknown as string)).toBe('');
|
|
expect(preprocessCodeArtifacts(undefined)).toBe('');
|
|
expect(preprocessCodeArtifacts({} as unknown as string)).toEqual('');
|
|
});
|
|
|
|
test('should remove <thinking> tags and their content', () => {
|
|
const input = '<thinking>This should be removed</thinking>Some content';
|
|
const expected = 'Some content';
|
|
expect(preprocessCodeArtifacts(input)).toBe(expected);
|
|
});
|
|
|
|
test('should remove unclosed <thinking> tags and their content', () => {
|
|
const input = '<thinking>This should be removed\nSome content';
|
|
const expected = '';
|
|
expect(preprocessCodeArtifacts(input)).toBe(expected);
|
|
});
|
|
|
|
test('should remove artifact headers up to and including empty code block', () => {
|
|
const input = ':::artifact{identifier="test"}\n```\n```\nSome content';
|
|
const expected = ':::artifact{identifier="test"}\n```\n```\nSome content';
|
|
expect(preprocessCodeArtifacts(input)).toBe(expected);
|
|
});
|
|
|
|
test('should keep artifact headers when followed by empty code block and content', () => {
|
|
const input = ':::artifact{identifier="test"}\n```\n```\nSome content';
|
|
const expected = ':::artifact{identifier="test"}\n```\n```\nSome content';
|
|
expect(preprocessCodeArtifacts(input)).toBe(expected);
|
|
});
|
|
|
|
test('should handle multiple artifact headers correctly', () => {
|
|
const input = ':::artifact{id="1"}\n```\n```\n:::artifact{id="2"}\n```\ncode\n```\nContent';
|
|
const expected = ':::artifact{id="1"}\n```\n```\n:::artifact{id="2"}\n```\ncode\n```\nContent';
|
|
expect(preprocessCodeArtifacts(input)).toBe(expected);
|
|
});
|
|
|
|
test('should handle complex input with multiple patterns', () => {
|
|
const input = `
|
|
<thinking>Remove this</thinking>
|
|
Some text
|
|
:::artifact{id="1"}
|
|
\`\`\`
|
|
\`\`\`
|
|
<thinking>And this</thinking>
|
|
:::artifact{id="2"}
|
|
\`\`\`
|
|
keep this code
|
|
\`\`\`
|
|
More text
|
|
`;
|
|
const expected = `
|
|
|
|
Some text
|
|
:::artifact{id="1"}
|
|
\`\`\`
|
|
\`\`\`
|
|
|
|
:::artifact{id="2"}
|
|
\`\`\`
|
|
keep this code
|
|
\`\`\`
|
|
More text
|
|
`;
|
|
expect(preprocessCodeArtifacts(input)).toBe(expected);
|
|
});
|
|
|
|
test('should remove artifact headers without code blocks', () => {
|
|
const input = ':::artifact{identifier="test"}\nSome content without code block';
|
|
const expected = '';
|
|
expect(preprocessCodeArtifacts(input)).toBe(expected);
|
|
});
|
|
|
|
test('should remove artifact headers up to incomplete code block', () => {
|
|
const input = ':::artifact{identifier="react-cal';
|
|
const expected = '';
|
|
expect(preprocessCodeArtifacts(input)).toBe(expected);
|
|
});
|
|
|
|
test('should keep artifact headers when any character follows code block', () => {
|
|
const input = ':::artifact{identifier="react-calculator"}\n```t';
|
|
const expected = ':::artifact{identifier="react-calculator"}\n```t';
|
|
expect(preprocessCodeArtifacts(input)).toBe(expected);
|
|
});
|
|
|
|
test('should keep artifact headers when whitespace follows code block', () => {
|
|
const input = ':::artifact{identifier="react-calculator"}\n``` ';
|
|
const expected = ':::artifact{identifier="react-calculator"}\n``` ';
|
|
expect(preprocessCodeArtifacts(input)).toBe(expected);
|
|
});
|
|
});
|