WIP - fixing grammar added CircleNode

This commit is contained in:
Knut Sveidqvist
2025-04-19 05:37:56 -04:00
parent df3c3d2fdc
commit 6bcfb4df3a
2 changed files with 111 additions and 118 deletions

View File

@@ -1,6 +1,7 @@
import { describe, expect, it } from 'vitest';
import { mindmapParse as parse } from './test-util.js';
import { keys } from '../../mermaid-flowchart-elk/dist/packages/mermaid/src/diagrams/state/id-cache';
import type { CircleNode } from '../src/language/generated/ast';
import { MindmapRow, Item } from '../src/language/generated/ast';
// Tests for mindmap parser with simple root and child nodes
describe('MindMap Parser Tests', () => {
@@ -22,11 +23,11 @@ describe('MindMap Parser Tests', () => {
expect(result.parserErrors).toHaveLength(0);
// Check if we have a statement
expect(result.value.statements).toBeDefined();
expect(result.value.statements.length).toBeGreaterThan(0);
expect(result.value.rows).toBeDefined();
expect(result.value.rows.length).toBeGreaterThan(0);
// Check the content of the root node
const rootNode = result.value.statements[0];
const rootNode = result.value.rows[0];
expect(rootNode).toBeDefined();
expect(rootNode.content).toBe('root');
});
@@ -37,54 +38,49 @@ describe('MindMap Parser Tests', () => {
'mindmap\nroot((Root))\n child1((Child 1))\n child2((Child 2))\n grandchild((Grand Child))'
);
// Debug information - commented out to avoid linter errors
// Result successful: result.successful
// Statements length: result.value?.statements?.length
// If statements exist, they would have properties like id, type, text, depth
const rows = result.value.MindmapRows;
const r0 = rows[0];
const r1 = rows[1];
const statements = result.value.statements;
const s0 = statements[0];
const s1 = statements[1];
expect(r0.$type).toBe('MindmapRow');
const node0 = r0.item as CircleNode;
expect(node0.$type).toBe('CircleNode');
expect(node0.desc).toBe('Root');
expect(node0.id).toBe('root');
console.debug('Statements:', s0);
expect(result.value.statements[0].$type).toBe('Statement');
// expect(result.value.statements[0].element.$type).toBe('ComplexNode');
expect(s0.element.$type).toBe('ComplexNode');
expect(Object.keys(s0)).toBe('Root');
expect(s0.element.ID).toBe('Root');
expect(result.value.statements[1].$type).toBe('Statement');
expect(result.value.statements[1].element.$type).toBe('ComplexNode');
expect(result.value.statements[1].element.ID).toBe('Root');
expect(result.value.statements[1].element.desc).toBe('Root');
expect(Object.keys(result.value.statements[1].element)).toBe('root');
expect(result.value.statements[1].indent).toBe('indent');
expect(Object.keys(result.value.statements[1].element)).toBe(true);
expect(result.value.statements[1].element.id).toBe('SimpleNode');
expect(r1.$type).toBe('MindmapRow');
const node1 = r1.item as CircleNode;
console.debug('NODE1:', node1);
expect(node1.$type).toBe('CircleNode');
expect(result.value.rows[1].element.ID).toBe('Root');
expect(result.value.rows[1].element.desc).toBe('Root');
expect(Object.keys(result.value.rows[1].element)).toBe('root');
expect(result.value.rows[1].indent).toBe('indent');
expect(Object.keys(result.value.rows[1].element)).toBe(true);
expect(result.value.rows[1].element.id).toBe('SimpleNode');
// Temporarily commenting out failing assertions
// expect(result.successful).toBe(true);
// Check that there are 4 statements: mindmap, root, child1, child2, grandchild
expect(result.value.statements.length).toBe(5);
// Check that there are 4 rows: mindmap, root, child1, child2, grandchild
expect(result.value.rows.length).toBe(5);
// Check that the first statement is the mindmap
expect(result.value.statements[0].type).toBe('mindmap');
expect(result.value.rows[0].type).toBe('mindmap');
// Check that the second statement is the root
expect(result.value.statements[1].type.type).toBe('circle');
expect(result.value.statements[1].text).toBe('Root');
expect(result.value.statements[1].depth).toBe(0);
expect(result.value.rows[1].type.type).toBe('circle');
expect(result.value.rows[1].text).toBe('Root');
expect(result.value.rows[1].depth).toBe(0);
// Check that the third statement is the first child
expect(result.value.statements[2].type.type).toBe('circle');
expect(result.value.statements[2].text).toBe('Child 1');
expect(result.value.statements[2].depth).toBe(1);
expect(result.value.rows[2].type.type).toBe('circle');
expect(result.value.rows[2].text).toBe('Child 1');
expect(result.value.rows[2].depth).toBe(1);
// Check that the fourth statement is the second child
expect(result.value.statements[3].type.type).toBe('circle');
expect(result.value.statements[3].text).toBe('Child 2');
expect(result.value.statements[3].depth).toBe(1);
expect(result.value.rows[3].type.type).toBe('circle');
expect(result.value.rows[3].text).toBe('Child 2');
expect(result.value.rows[3].depth).toBe(1);
// Check that the fifth statement is the grandchild
expect(result.value.statements[4].type.type).toBe('circle');
expect(result.value.statements[4].text).toBe('Grand Child');
expect(result.value.statements[4].depth).toBe(2);
expect(result.value.rows[4].type.type).toBe('circle');
expect(result.value.rows[4].text).toBe('Grand Child');
expect(result.value.rows[4].depth).toBe(2);
});
});
@@ -93,17 +89,17 @@ describe('Hierarchy (ported from mindmap.spec.ts)', () => {
const result = parse('mindmap\nroot');
expect(result.lexerErrors).toHaveLength(0);
expect(result.parserErrors).toHaveLength(0);
expect(result.value.statements[0].content).toBe('root');
expect(result.value.rows[0].content).toBe('root');
});
it('MMP-2 should handle a hierarchical mindmap definition', () => {
const result = parse('mindmap\nroot\n child1\n child2');
expect(result.lexerErrors).toHaveLength(0);
expect(result.parserErrors).toHaveLength(0);
// Langium AST may not have children as nested objects, so just check statements
expect(result.value.statements[0].content).toBe('root');
expect(result.value.statements[1].content).toBe('child1');
expect(result.value.statements[2].content).toBe('child2');
// Langium AST may not have children as nested objects, so just check rows
expect(result.value.rows[0].content).toBe('root');
expect(result.value.rows[1].content).toBe('child1');
expect(result.value.rows[2].content).toBe('child2');
});
it('MMP-3 should handle a simple root definition with a shape and without an id', () => {
@@ -111,17 +107,17 @@ describe('Hierarchy (ported from mindmap.spec.ts)', () => {
expect(result.lexerErrors).toHaveLength(0);
expect(result.parserErrors).toHaveLength(0);
// The content should be 'root', shape info may not be present in AST
expect(result.value.statements[0].content).toBe('root');
expect(result.value.rows[0].content).toBe('root');
});
it('MMP-4 should handle a deeper hierarchical mindmap definition', () => {
const result = parse('mindmap\nroot\n child1\n leaf1\n child2');
expect(result.lexerErrors).toHaveLength(0);
expect(result.parserErrors).toHaveLength(0);
expect(result.value.statements[0].content).toBe('root');
expect(result.value.statements[1].content).toBe('child1');
expect(result.value.statements[2].content).toBe('leaf1');
expect(result.value.statements[3].content).toBe('child2');
expect(result.value.rows[0].content).toBe('root');
expect(result.value.rows[1].content).toBe('child1');
expect(result.value.rows[2].content).toBe('leaf1');
expect(result.value.rows[3].content).toBe('child2');
});
it('MMP-5 Multiple roots are illegal', () => {
@@ -144,7 +140,7 @@ describe('Nodes (ported from mindmap.spec.ts)', () => {
expect(result.lexerErrors).toHaveLength(0);
expect(result.parserErrors).toHaveLength(0);
// Langium AST: check content, id, and maybe type if available
expect(result.value.statements[0].content).toBe('The root');
expect(result.value.rows[0].content).toBe('The root');
// TODO: check id and type if present in AST
});
@@ -152,8 +148,8 @@ describe('Nodes (ported from mindmap.spec.ts)', () => {
const result = parse('mindmap\nroot\n theId(child1)');
expect(result.lexerErrors).toHaveLength(0);
expect(result.parserErrors).toHaveLength(0);
expect(result.value.statements[0].content).toBe('root');
expect(result.value.statements[1].content).toBe('child1');
expect(result.value.rows[0].content).toBe('root');
expect(result.value.rows[1].content).toBe('child1');
// TODO: check id and type if present in AST
});
@@ -161,8 +157,8 @@ describe('Nodes (ported from mindmap.spec.ts)', () => {
const result = parse('mindmap\nroot\n theId(child1)');
expect(result.lexerErrors).toHaveLength(0);
expect(result.parserErrors).toHaveLength(0);
expect(result.value.statements[0].content).toBe('root');
expect(result.value.statements[1].content).toBe('child1');
expect(result.value.rows[0].content).toBe('root');
expect(result.value.rows[1].content).toBe('child1');
// TODO: check id and type if present in AST
});
@@ -170,7 +166,7 @@ describe('Nodes (ported from mindmap.spec.ts)', () => {
const result = parse('mindmap\nroot((the root))');
expect(result.lexerErrors).toHaveLength(0);
expect(result.parserErrors).toHaveLength(0);
expect(result.value.statements[0].content).toBe('the root');
expect(result.value.rows[0].content).toBe('the root');
// TODO: check type if present in AST
});
@@ -178,7 +174,7 @@ describe('Nodes (ported from mindmap.spec.ts)', () => {
const result = parse('mindmap\nroot)the root(');
expect(result.lexerErrors).toHaveLength(0);
expect(result.parserErrors).toHaveLength(0);
expect(result.value.statements[0].content).toBe('the root');
expect(result.value.rows[0].content).toBe('the root');
// TODO: check type if present in AST
});
@@ -186,7 +182,7 @@ describe('Nodes (ported from mindmap.spec.ts)', () => {
const result = parse('mindmap\nroot))the root((');
expect(result.lexerErrors).toHaveLength(0);
expect(result.parserErrors).toHaveLength(0);
expect(result.value.statements[0].content).toBe('the root');
expect(result.value.rows[0].content).toBe('the root');
// TODO: check type if present in AST
});
@@ -194,7 +190,7 @@ describe('Nodes (ported from mindmap.spec.ts)', () => {
const result = parse('mindmap\nroot{{the root}}');
expect(result.lexerErrors).toHaveLength(0);
expect(result.parserErrors).toHaveLength(0);
expect(result.value.statements[0].content).toBe('the root');
expect(result.value.rows[0].content).toBe('the root');
// TODO: check type if present in AST
});
});
@@ -205,28 +201,28 @@ describe('Decorations (ported from mindmap.spec.ts)', () => {
expect(result.lexerErrors).toHaveLength(0);
expect(result.parserErrors).toHaveLength(0);
// TODO: check icon if present in AST
expect(result.value.statements[0].content).toBe('The root');
expect(result.value.rows[0].content).toBe('The root');
});
it('MMP-14 should be possible to set classes for the node', () => {
const result = parse('mindmap\nroot[The root]\n:::m-4 p-8');
expect(result.lexerErrors).toHaveLength(0);
expect(result.parserErrors).toHaveLength(0);
// TODO: check class if present in AST
expect(result.value.statements[0].content).toBe('The root');
expect(result.value.rows[0].content).toBe('The root');
});
it('MMP-15 should be possible to set both classes and icon for the node', () => {
const result = parse('mindmap\nroot[The root]\n:::m-4 p-8\n::icon(bomb)');
expect(result.lexerErrors).toHaveLength(0);
expect(result.parserErrors).toHaveLength(0);
// TODO: check class and icon if present in AST
expect(result.value.statements[0].content).toBe('The root');
expect(result.value.rows[0].content).toBe('The root');
});
it('MMP-16 should be possible to set both classes and icon for the node (reverse order)', () => {
const result = parse('mindmap\nroot[The root]\n::icon(bomb)\n:::m-4 p-8');
expect(result.lexerErrors).toHaveLength(0);
expect(result.parserErrors).toHaveLength(0);
// TODO: check class and icon if present in AST
expect(result.value.statements[0].content).toBe('The root');
expect(result.value.rows[0].content).toBe('The root');
});
});
@@ -235,14 +231,14 @@ describe('Descriptions (ported from mindmap.spec.ts)', () => {
const result = parse('mindmap\nroot["String containing []"]');
expect(result.lexerErrors).toHaveLength(0);
expect(result.parserErrors).toHaveLength(0);
expect(result.value.statements[0].content).toBe('String containing []');
expect(result.value.rows[0].content).toBe('String containing []');
});
it('MMP-18 should be possible to use node syntax in the descriptions in children', () => {
const result = parse('mindmap\nroot["String containing []"]\n child1["String containing ()"]');
expect(result.lexerErrors).toHaveLength(0);
expect(result.parserErrors).toHaveLength(0);
expect(result.value.statements[0].content).toBe('String containing []');
expect(result.value.statements[1].content).toBe('String containing ()');
expect(result.value.rows[0].content).toBe('String containing []');
expect(result.value.rows[1].content).toBe('String containing ()');
});
it('MMP-19 should be possible to have a child after a class assignment', () => {
const result = parse(
@@ -250,10 +246,10 @@ describe('Descriptions (ported from mindmap.spec.ts)', () => {
);
expect(result.lexerErrors).toHaveLength(0);
expect(result.parserErrors).toHaveLength(0);
expect(result.value.statements[0].content).toBe('Root');
expect(result.value.statements[1].content).toBe('Child');
expect(result.value.statements[2].content).toBe('a');
expect(result.value.statements[3].content).toBe('b');
expect(result.value.rows[0].content).toBe('Root');
expect(result.value.rows[1].content).toBe('Child');
expect(result.value.rows[2].content).toBe('a');
expect(result.value.rows[3].content).toBe('b');
});
});
@@ -262,10 +258,10 @@ describe('Miscellaneous (ported from mindmap.spec.ts)', () => {
const result = parse('mindmap\nroot(Root)\n Child(Child)\n a(a)\n\n b[New Stuff]');
expect(result.lexerErrors).toHaveLength(0);
expect(result.parserErrors).toHaveLength(0);
expect(result.value.statements[0].content).toBe('Root');
expect(result.value.statements[1].content).toBe('Child');
expect(result.value.statements[2].content).toBe('a');
expect(result.value.statements[3].content).toBe('b');
expect(result.value.rows[0].content).toBe('Root');
expect(result.value.rows[1].content).toBe('Child');
expect(result.value.rows[2].content).toBe('a');
expect(result.value.rows[3].content).toBe('b');
});
it('MMP-21 should be possible to have comments in a mindmap', () => {
const result = parse(
@@ -273,10 +269,10 @@ describe('Miscellaneous (ported from mindmap.spec.ts)', () => {
);
expect(result.lexerErrors).toHaveLength(0);
expect(result.parserErrors).toHaveLength(0);
expect(result.value.statements[0].content).toBe('Root');
expect(result.value.statements[1].content).toBe('Child');
expect(result.value.statements[2].content).toBe('a');
expect(result.value.statements[3].content).toBe('b');
expect(result.value.rows[0].content).toBe('Root');
expect(result.value.rows[1].content).toBe('Child');
expect(result.value.rows[2].content).toBe('a');
expect(result.value.rows[3].content).toBe('b');
});
it('MMP-22 should be possible to have comments at the end of a line', () => {
const result = parse(
@@ -284,33 +280,33 @@ describe('Miscellaneous (ported from mindmap.spec.ts)', () => {
);
expect(result.lexerErrors).toHaveLength(0);
expect(result.parserErrors).toHaveLength(0);
expect(result.value.statements[0].content).toBe('Root');
expect(result.value.statements[1].content).toBe('Child');
expect(result.value.statements[2].content).toBe('a');
expect(result.value.statements[3].content).toBe('b');
expect(result.value.rows[0].content).toBe('Root');
expect(result.value.rows[1].content).toBe('Child');
expect(result.value.rows[2].content).toBe('a');
expect(result.value.rows[3].content).toBe('b');
});
it('MMP-23 Rows with only spaces should not interfere', () => {
const result = parse('mindmap\nroot\n A\n \n\n B');
expect(result.lexerErrors).toHaveLength(0);
expect(result.parserErrors).toHaveLength(0);
expect(result.value.statements[0].content).toBe('root');
expect(result.value.statements[1].content).toBe('A');
expect(result.value.statements[2].content).toBe('B');
expect(result.value.rows[0].content).toBe('root');
expect(result.value.rows[1].content).toBe('A');
expect(result.value.rows[2].content).toBe('B');
});
it('MMP-24 Handle rows above the mindmap declarations', () => {
const result = parse('\n \nmindmap\nroot\n A\n \n\n B');
expect(result.lexerErrors).toHaveLength(0);
expect(result.parserErrors).toHaveLength(0);
expect(result.value.statements[0].content).toBe('root');
expect(result.value.statements[1].content).toBe('A');
expect(result.value.statements[2].content).toBe('B');
expect(result.value.rows[0].content).toBe('root');
expect(result.value.rows[1].content).toBe('A');
expect(result.value.rows[2].content).toBe('B');
});
it('MMP-25 Handle rows above the mindmap declarations, no space', () => {
const result = parse('\n\n\nmindmap\nroot\n A\n \n\n B');
expect(result.lexerErrors).toHaveLength(0);
expect(result.parserErrors).toHaveLength(0);
expect(result.value.statements[0].content).toBe('root');
expect(result.value.statements[1].content).toBe('A');
expect(result.value.statements[2].content).toBe('B');
expect(result.value.rows[0].content).toBe('root');
expect(result.value.rows[1].content).toBe('A');
expect(result.value.rows[2].content).toBe('B');
});
});