@domainlang/language 0.11.0 → 0.13.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/out/ast-augmentation.d.ts +7 -2
- package/out/diagram/context-map-diagram-generator.d.ts +72 -0
- package/out/diagram/context-map-diagram-generator.js +405 -0
- package/out/diagram/context-map-diagram-generator.js.map +1 -0
- package/out/diagram/context-map-layout-configurator.d.ts +15 -0
- package/out/diagram/context-map-layout-configurator.js +39 -0
- package/out/diagram/context-map-layout-configurator.js.map +1 -0
- package/out/diagram/elk-layout-factory.d.ts +43 -0
- package/out/diagram/elk-layout-factory.js +64 -0
- package/out/diagram/elk-layout-factory.js.map +1 -0
- package/out/domain-lang-module.d.ts +7 -0
- package/out/domain-lang-module.js +11 -2
- package/out/domain-lang-module.js.map +1 -1
- package/out/generated/ast.d.ts +323 -51
- package/out/generated/ast.js +194 -33
- package/out/generated/ast.js.map +1 -1
- package/out/generated/grammar.js +418 -172
- package/out/generated/grammar.js.map +1 -1
- package/out/index.d.ts +3 -0
- package/out/index.js +4 -0
- package/out/index.js.map +1 -1
- package/out/lsp/domain-lang-code-lens-provider.d.ts +8 -0
- package/out/lsp/domain-lang-code-lens-provider.js +48 -0
- package/out/lsp/domain-lang-code-lens-provider.js.map +1 -0
- package/out/lsp/domain-lang-completion.js +39 -15
- package/out/lsp/domain-lang-completion.js.map +1 -1
- package/out/lsp/domain-lang-document-symbol-provider.js +5 -5
- package/out/lsp/domain-lang-document-symbol-provider.js.map +1 -1
- package/out/lsp/domain-lang-formatter.js +32 -0
- package/out/lsp/domain-lang-formatter.js.map +1 -1
- package/out/lsp/domain-lang-index-manager.d.ts +2 -3
- package/out/lsp/domain-lang-index-manager.js +5 -8
- package/out/lsp/domain-lang-index-manager.js.map +1 -1
- package/out/lsp/domain-lang-workspace-manager.d.ts +1 -1
- package/out/lsp/domain-lang-workspace-manager.js +2 -26
- package/out/lsp/domain-lang-workspace-manager.js.map +1 -1
- package/out/lsp/explain.js +9 -3
- package/out/lsp/explain.js.map +1 -1
- package/out/lsp/hover/domain-lang-hover.js +13 -11
- package/out/lsp/hover/domain-lang-hover.js.map +1 -1
- package/out/lsp/hover/domain-lang-keywords.js +29 -26
- package/out/lsp/hover/domain-lang-keywords.js.map +1 -1
- package/out/lsp/tool-handlers.js +63 -57
- package/out/lsp/tool-handlers.js.map +1 -1
- package/out/sdk/ast-augmentation.d.ts +29 -21
- package/out/sdk/ast-augmentation.js +11 -7
- package/out/sdk/ast-augmentation.js.map +1 -1
- package/out/sdk/index.d.ts +2 -2
- package/out/sdk/index.js +1 -1
- package/out/sdk/index.js.map +1 -1
- package/out/sdk/loader-node.js +2 -2
- package/out/sdk/loader-node.js.map +1 -1
- package/out/sdk/patterns.d.ts +50 -61
- package/out/sdk/patterns.js +92 -62
- package/out/sdk/patterns.js.map +1 -1
- package/out/sdk/query.js +54 -43
- package/out/sdk/query.js.map +1 -1
- package/out/sdk/serializers.js +20 -7
- package/out/sdk/serializers.js.map +1 -1
- package/out/sdk/types.d.ts +87 -18
- package/out/sdk/types.js.map +1 -1
- package/out/sdk/validator.js +48 -64
- package/out/sdk/validator.js.map +1 -1
- package/out/services/performance-optimizer.d.ts +3 -3
- package/out/services/performance-optimizer.js +1 -3
- package/out/services/performance-optimizer.js.map +1 -1
- package/out/services/relationship-inference.d.ts +4 -4
- package/out/services/relationship-inference.js +34 -46
- package/out/services/relationship-inference.js.map +1 -1
- package/out/syntaxes/domain-lang.monarch.js +1 -1
- package/out/syntaxes/domain-lang.monarch.js.map +1 -1
- package/out/utils/import-utils.d.ts +6 -20
- package/out/utils/import-utils.js +3 -63
- package/out/utils/import-utils.js.map +1 -1
- package/out/validation/constants.d.ts +23 -6
- package/out/validation/constants.js +24 -7
- package/out/validation/constants.js.map +1 -1
- package/out/validation/maps.js +10 -4
- package/out/validation/maps.js.map +1 -1
- package/out/validation/relationships.d.ts +4 -8
- package/out/validation/relationships.js +96 -48
- package/out/validation/relationships.js.map +1 -1
- package/package.json +5 -2
- package/src/ast-augmentation.ts +7 -2
- package/src/diagram/context-map-diagram-generator.ts +513 -0
- package/src/diagram/context-map-layout-configurator.ts +43 -0
- package/src/diagram/elk-layout-factory.ts +83 -0
- package/src/domain-lang-module.ts +19 -2
- package/src/domain-lang.langium +62 -26
- package/src/generated/ast.ts +413 -63
- package/src/generated/grammar.ts +418 -172
- package/src/index.ts +5 -0
- package/src/lsp/domain-lang-code-lens-provider.ts +54 -0
- package/src/lsp/domain-lang-completion.ts +42 -15
- package/src/lsp/domain-lang-document-symbol-provider.ts +5 -5
- package/src/lsp/domain-lang-formatter.ts +34 -0
- package/src/lsp/domain-lang-index-manager.ts +6 -9
- package/src/lsp/domain-lang-workspace-manager.ts +3 -29
- package/src/lsp/explain.ts +10 -2
- package/src/lsp/hover/domain-lang-hover.ts +10 -8
- package/src/lsp/hover/domain-lang-keywords.ts +27 -24
- package/src/lsp/tool-handlers.ts +61 -47
- package/src/sdk/ast-augmentation.ts +30 -21
- package/src/sdk/index.ts +11 -1
- package/src/sdk/loader-node.ts +2 -2
- package/src/sdk/patterns.ts +114 -76
- package/src/sdk/query.ts +57 -48
- package/src/sdk/serializers.ts +20 -7
- package/src/sdk/types.ts +92 -17
- package/src/sdk/validator.ts +52 -69
- package/src/services/performance-optimizer.ts +4 -6
- package/src/services/relationship-inference.ts +43 -54
- package/src/utils/import-utils.ts +9 -74
- package/src/validation/constants.ts +32 -9
- package/src/validation/maps.ts +12 -4
- package/src/validation/relationships.ts +150 -71
|
@@ -1,6 +1,11 @@
|
|
|
1
1
|
declare module './generated/ast.js' {
|
|
2
|
-
interface
|
|
3
|
-
|
|
2
|
+
interface DirectionalRelationship {
|
|
3
|
+
/** SDK-inferred relationship kind based on patterns and arrow */
|
|
4
|
+
inferredKind?: string;
|
|
5
|
+
}
|
|
6
|
+
interface SymmetricRelationship {
|
|
7
|
+
/** SDK-inferred relationship kind from pattern or >< arrow */
|
|
8
|
+
inferredKind?: string;
|
|
4
9
|
}
|
|
5
10
|
}
|
|
6
11
|
export {};
|
|
@@ -0,0 +1,72 @@
|
|
|
1
|
+
import type { SModelRoot } from 'sprotty-protocol';
|
|
2
|
+
import { LangiumDiagramGenerator, type GeneratorContext } from 'langium-sprotty';
|
|
3
|
+
/**
|
|
4
|
+
* Generates context map diagrams in the **DDD community notation** style.
|
|
5
|
+
*
|
|
6
|
+
* Bounded contexts are rendered as ellipses. Relationships are rendered as edges
|
|
7
|
+
* with U/D (upstream/downstream) annotations and integration pattern labels at
|
|
8
|
+
* each end, matching the notation used in Eric Evans' "Domain-Driven Design" and
|
|
9
|
+
* Vaughn Vernon's "Implementing Domain-Driven Design".
|
|
10
|
+
*
|
|
11
|
+
* Edge label convention:
|
|
12
|
+
* - Position 0.1 (near source): `U [OHS, PL]` or `D [ACL]`
|
|
13
|
+
* - Position 0.9 (near target): `D [CF]` or `U [OHS]`
|
|
14
|
+
* - Position 0.5 (center): relationship type (e.g., "Customer/Supplier")
|
|
15
|
+
*/
|
|
16
|
+
export declare class DomainLangContextMapDiagramGenerator extends LangiumDiagramGenerator {
|
|
17
|
+
protected generateRoot(args: GeneratorContext): SModelRoot;
|
|
18
|
+
/**
|
|
19
|
+
* Creates an edge with DDD community notation labels.
|
|
20
|
+
*
|
|
21
|
+
* For `->`: left = Upstream (U), right = Downstream (D)
|
|
22
|
+
* For `<-`: left = Downstream (D), right = Upstream (U)
|
|
23
|
+
* For `<->`: Partnership (bidirectional)
|
|
24
|
+
* For `><`: Separate Ways
|
|
25
|
+
*
|
|
26
|
+
* U/D labels are rendered as DDD notation badges with optional
|
|
27
|
+
* pattern boxes (e.g., `U [OHS, PL]`).
|
|
28
|
+
*/
|
|
29
|
+
private createRelationshipEdge;
|
|
30
|
+
/**
|
|
31
|
+
* Creates an undirected edge for symmetric relationships (SK, P, SW).
|
|
32
|
+
*
|
|
33
|
+
* Symmetric relationships have no upstream/downstream directionality.
|
|
34
|
+
* The center label shows the relationship kind (e.g. "Shared Kernel").
|
|
35
|
+
*/
|
|
36
|
+
private createSymmetricEdge;
|
|
37
|
+
/**
|
|
38
|
+
* Adds a U/D badge label at the source or target end of an edge.
|
|
39
|
+
*
|
|
40
|
+
* Patterns are normalised to their short abbreviations (e.g. `OpenHostService` →
|
|
41
|
+
* `OHS`) and `BBoM`/`BigBallOfMud` is excluded — BBoM is surfaced visually as a
|
|
42
|
+
* cloud node shape rather than a text annotation.
|
|
43
|
+
*
|
|
44
|
+
* Badge text format: `ROLE|PATTERNS` (e.g. `U|OHS, PL` or `D|ACL`).
|
|
45
|
+
* The webview `UDBadgeLabelView` renders this as a bordered box.
|
|
46
|
+
*/
|
|
47
|
+
private addUDBadge;
|
|
48
|
+
/**
|
|
49
|
+
* Formats the relationship type for the center edge label.
|
|
50
|
+
*
|
|
51
|
+
* Maps DomainLang keywords to DDD community notation display names:
|
|
52
|
+
* CustomerSupplier → Customer/Supplier
|
|
53
|
+
* SharedKernel → Shared Kernel
|
|
54
|
+
* UpstreamDownstream → Upstream/Downstream
|
|
55
|
+
* Partnership → Partnership
|
|
56
|
+
*
|
|
57
|
+
* For `<->` without explicit type, defaults to "Partnership".
|
|
58
|
+
* For `><`, defaults to "Separate Ways".
|
|
59
|
+
*/
|
|
60
|
+
private formatRelationshipKind;
|
|
61
|
+
private displayRelationshipKind;
|
|
62
|
+
private collectContextMapNodes;
|
|
63
|
+
private ensureNodeForContextMapItem;
|
|
64
|
+
private createNode;
|
|
65
|
+
private createNodeLabel;
|
|
66
|
+
private ensureNode;
|
|
67
|
+
private ensureUnresolvedNode;
|
|
68
|
+
private getNodeKey;
|
|
69
|
+
private getSelection;
|
|
70
|
+
private getStringOption;
|
|
71
|
+
private selectContextMap;
|
|
72
|
+
}
|
|
@@ -0,0 +1,405 @@
|
|
|
1
|
+
import { LangiumDiagramGenerator } from 'langium-sprotty';
|
|
2
|
+
import { fromDocument } from '../sdk/query.js';
|
|
3
|
+
import { isBigBallOfMud } from '../generated/ast.js';
|
|
4
|
+
/** Ellipse sizing for bounded context nodes — sized for long names like "CustomerManagementContext" */
|
|
5
|
+
const NODE_WIDTH = 280;
|
|
6
|
+
const NODE_HEIGHT = 100;
|
|
7
|
+
/**
|
|
8
|
+
* Maps long-form DomainLang integration pattern keywords to their standard
|
|
9
|
+
* DDD abbreviations for display in U/D badges.
|
|
10
|
+
*/
|
|
11
|
+
const PATTERN_ABBREVIATIONS = {
|
|
12
|
+
OpenHostService: 'OHS',
|
|
13
|
+
PublishedLanguage: 'PL',
|
|
14
|
+
AntiCorruptionLayer: 'ACL',
|
|
15
|
+
Conformist: 'CF',
|
|
16
|
+
Supplier: 'S',
|
|
17
|
+
Customer: 'C',
|
|
18
|
+
BigBallOfMud: 'BBoM',
|
|
19
|
+
SharedKernel: 'SK',
|
|
20
|
+
Partnership: 'P',
|
|
21
|
+
SeparateWays: 'SW',
|
|
22
|
+
};
|
|
23
|
+
/**
|
|
24
|
+
* Returns the abbreviated form of a side pattern AST node.
|
|
25
|
+
*
|
|
26
|
+
* Maps the `$type` (e.g. `OpenHostService`, `Conformist`) to its standard
|
|
27
|
+
* DDD abbreviation (e.g. `OHS`, `CF`). Unknown types are returned as-is.
|
|
28
|
+
*/
|
|
29
|
+
function normalizePatternNode(pattern) {
|
|
30
|
+
return PATTERN_ABBREVIATIONS[pattern.$type] ?? pattern.$type;
|
|
31
|
+
}
|
|
32
|
+
/**
|
|
33
|
+
* Returns `true` when the side pattern identifies a Big Ball of Mud participant.
|
|
34
|
+
*
|
|
35
|
+
* BBoM is surfaced as a cloud node shape on the bounded context itself, not as
|
|
36
|
+
* a text annotation in the edge badge, so it should be excluded from badge text.
|
|
37
|
+
*/
|
|
38
|
+
function isBBoMSidePattern(pattern) {
|
|
39
|
+
return isBigBallOfMud(pattern);
|
|
40
|
+
}
|
|
41
|
+
/**
|
|
42
|
+
* Generates context map diagrams in the **DDD community notation** style.
|
|
43
|
+
*
|
|
44
|
+
* Bounded contexts are rendered as ellipses. Relationships are rendered as edges
|
|
45
|
+
* with U/D (upstream/downstream) annotations and integration pattern labels at
|
|
46
|
+
* each end, matching the notation used in Eric Evans' "Domain-Driven Design" and
|
|
47
|
+
* Vaughn Vernon's "Implementing Domain-Driven Design".
|
|
48
|
+
*
|
|
49
|
+
* Edge label convention:
|
|
50
|
+
* - Position 0.1 (near source): `U [OHS, PL]` or `D [ACL]`
|
|
51
|
+
* - Position 0.9 (near target): `D [CF]` or `U [OHS]`
|
|
52
|
+
* - Position 0.5 (center): relationship type (e.g., "Customer/Supplier")
|
|
53
|
+
*/
|
|
54
|
+
export class DomainLangContextMapDiagramGenerator extends LangiumDiagramGenerator {
|
|
55
|
+
generateRoot(args) {
|
|
56
|
+
const document = args.document;
|
|
57
|
+
const query = fromDocument(document);
|
|
58
|
+
const selection = this.getSelection(args);
|
|
59
|
+
const contextMaps = query.contextMaps().toArray();
|
|
60
|
+
const selectedMap = this.selectContextMap(contextMaps, query, selection);
|
|
61
|
+
if (!selectedMap) {
|
|
62
|
+
return {
|
|
63
|
+
type: 'graph',
|
|
64
|
+
id: args.idCache.uniqueId('context-map:empty'),
|
|
65
|
+
children: [],
|
|
66
|
+
};
|
|
67
|
+
}
|
|
68
|
+
// Fetch relationships first so the BBoM pre-pass can identify which BCs
|
|
69
|
+
// need the cloud node shape before any nodes are created.
|
|
70
|
+
const relationships = query.relationships()
|
|
71
|
+
.where((relationship) => relationship.source === 'ContextMap' && relationship.astNode.$container === selectedMap)
|
|
72
|
+
.toArray();
|
|
73
|
+
// BBoM pre-pass: collect the node keys of every bounded context that
|
|
74
|
+
// appears on a side annotated with the BigBallOfMud pattern. These get
|
|
75
|
+
// the `node:bbom` Sprotty type so the webview renders them as clouds.
|
|
76
|
+
const bboMNodeKeys = new Set();
|
|
77
|
+
for (const rel of relationships) {
|
|
78
|
+
if (rel.type === 'directional') {
|
|
79
|
+
if (rel.left.patterns.some(isBBoMSidePattern)) {
|
|
80
|
+
bboMNodeKeys.add(this.getNodeKey(query, rel.left.context));
|
|
81
|
+
}
|
|
82
|
+
if (rel.right.patterns.some(isBBoMSidePattern)) {
|
|
83
|
+
bboMNodeKeys.add(this.getNodeKey(query, rel.right.context));
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
const nodeMap = new Map();
|
|
88
|
+
this.collectContextMapNodes(selectedMap, query, nodeMap, bboMNodeKeys, args);
|
|
89
|
+
for (const relationship of relationships) {
|
|
90
|
+
this.ensureNode(nodeMap, query, relationship.left.context, bboMNodeKeys, args);
|
|
91
|
+
this.ensureNode(nodeMap, query, relationship.right.context, bboMNodeKeys, args);
|
|
92
|
+
}
|
|
93
|
+
const edges = relationships.flatMap((relationship) => {
|
|
94
|
+
const leftKey = this.getNodeKey(query, relationship.left.context);
|
|
95
|
+
const rightKey = this.getNodeKey(query, relationship.right.context);
|
|
96
|
+
const leftNode = nodeMap.get(leftKey);
|
|
97
|
+
const rightNode = nodeMap.get(rightKey);
|
|
98
|
+
if (!leftNode || !rightNode) {
|
|
99
|
+
return [];
|
|
100
|
+
}
|
|
101
|
+
return this.createRelationshipEdge({
|
|
102
|
+
leftNode,
|
|
103
|
+
rightNode,
|
|
104
|
+
relationship,
|
|
105
|
+
}, args);
|
|
106
|
+
});
|
|
107
|
+
return {
|
|
108
|
+
type: 'graph',
|
|
109
|
+
id: args.idCache.uniqueId(`context-map:${selectedMap.name}`, selectedMap),
|
|
110
|
+
children: [...nodeMap.values(), ...edges],
|
|
111
|
+
};
|
|
112
|
+
}
|
|
113
|
+
// ── Relationship edges (DDD community notation) ──
|
|
114
|
+
/**
|
|
115
|
+
* Creates an edge with DDD community notation labels.
|
|
116
|
+
*
|
|
117
|
+
* For `->`: left = Upstream (U), right = Downstream (D)
|
|
118
|
+
* For `<-`: left = Downstream (D), right = Upstream (U)
|
|
119
|
+
* For `<->`: Partnership (bidirectional)
|
|
120
|
+
* For `><`: Separate Ways
|
|
121
|
+
*
|
|
122
|
+
* U/D labels are rendered as DDD notation badges with optional
|
|
123
|
+
* pattern boxes (e.g., `U [OHS, PL]`).
|
|
124
|
+
*/
|
|
125
|
+
createRelationshipEdge(params, args) {
|
|
126
|
+
const { leftNode, rightNode, relationship } = params;
|
|
127
|
+
const astNode = relationship.astNode;
|
|
128
|
+
if (relationship.type === 'symmetric') {
|
|
129
|
+
return this.createSymmetricEdge(leftNode, rightNode, relationship.kind, astNode, args);
|
|
130
|
+
}
|
|
131
|
+
// Directional
|
|
132
|
+
const { arrow, kind, left, right } = relationship;
|
|
133
|
+
// Determine source/target nodes based on arrow direction
|
|
134
|
+
const sourceId = arrow === '<-' ? rightNode.id : leftNode.id;
|
|
135
|
+
const targetId = arrow === '<-' ? leftNode.id : rightNode.id;
|
|
136
|
+
const edgeId = args.idCache.uniqueId(`edge:${sourceId}:${targetId}`, astNode);
|
|
137
|
+
const children = [];
|
|
138
|
+
if (kind === 'Bidirectional') {
|
|
139
|
+
// No upstream/downstream roles — show patterns positionally
|
|
140
|
+
this.addUDBadge(children, edgeId, 'source', left.patterns, undefined, args);
|
|
141
|
+
this.addUDBadge(children, edgeId, 'target', right.patterns, undefined, args);
|
|
142
|
+
}
|
|
143
|
+
else {
|
|
144
|
+
// Upstream/downstream or Customer/Supplier
|
|
145
|
+
const sourcePatterns = arrow === '<-' ? right.patterns : left.patterns;
|
|
146
|
+
const targetPatterns = arrow === '<-' ? left.patterns : right.patterns;
|
|
147
|
+
const sourceRole = kind === 'CustomerSupplier' ? 'S' : 'U';
|
|
148
|
+
const targetRole = kind === 'CustomerSupplier' ? 'C' : 'D';
|
|
149
|
+
this.addUDBadge(children, edgeId, 'source', sourcePatterns, sourceRole, args);
|
|
150
|
+
this.addUDBadge(children, edgeId, 'target', targetPatterns, targetRole, args);
|
|
151
|
+
}
|
|
152
|
+
// Center label: relationship kind
|
|
153
|
+
const centerLabel = this.formatRelationshipKind(kind);
|
|
154
|
+
if (centerLabel) {
|
|
155
|
+
children.push({
|
|
156
|
+
type: 'label:edge',
|
|
157
|
+
id: args.idCache.uniqueId(`${edgeId}:type`),
|
|
158
|
+
text: centerLabel,
|
|
159
|
+
edgePlacement: {
|
|
160
|
+
side: 'on',
|
|
161
|
+
position: 0.5,
|
|
162
|
+
rotate: false,
|
|
163
|
+
offset: 10,
|
|
164
|
+
},
|
|
165
|
+
});
|
|
166
|
+
}
|
|
167
|
+
let edgeCssClasses;
|
|
168
|
+
if (arrow === '<->') {
|
|
169
|
+
edgeCssClasses = ['partnership'];
|
|
170
|
+
}
|
|
171
|
+
const edge = {
|
|
172
|
+
type: 'edge',
|
|
173
|
+
id: edgeId,
|
|
174
|
+
sourceId,
|
|
175
|
+
targetId,
|
|
176
|
+
cssClasses: edgeCssClasses,
|
|
177
|
+
children: children.length > 0 ? children : undefined,
|
|
178
|
+
};
|
|
179
|
+
this.traceProvider.trace(edge, astNode);
|
|
180
|
+
return [edge];
|
|
181
|
+
}
|
|
182
|
+
/**
|
|
183
|
+
* Creates an undirected edge for symmetric relationships (SK, P, SW).
|
|
184
|
+
*
|
|
185
|
+
* Symmetric relationships have no upstream/downstream directionality.
|
|
186
|
+
* The center label shows the relationship kind (e.g. "Shared Kernel").
|
|
187
|
+
*/
|
|
188
|
+
createSymmetricEdge(leftNode, rightNode, kind, astNode, args) {
|
|
189
|
+
const edgeId = args.idCache.uniqueId(`edge:${leftNode.id}:${rightNode.id}`, astNode);
|
|
190
|
+
const children = [];
|
|
191
|
+
const centerLabel = this.formatRelationshipKind(kind);
|
|
192
|
+
if (centerLabel) {
|
|
193
|
+
children.push({
|
|
194
|
+
type: 'label:edge',
|
|
195
|
+
id: args.idCache.uniqueId(`${edgeId}:type`),
|
|
196
|
+
text: centerLabel,
|
|
197
|
+
edgePlacement: {
|
|
198
|
+
side: 'on',
|
|
199
|
+
position: 0.5,
|
|
200
|
+
rotate: false,
|
|
201
|
+
offset: 10,
|
|
202
|
+
},
|
|
203
|
+
});
|
|
204
|
+
}
|
|
205
|
+
let edgeCssClasses;
|
|
206
|
+
if (kind === 'SeparateWays') {
|
|
207
|
+
edgeCssClasses = ['separate-ways'];
|
|
208
|
+
}
|
|
209
|
+
else if (kind === 'Partnership') {
|
|
210
|
+
edgeCssClasses = ['partnership'];
|
|
211
|
+
}
|
|
212
|
+
else if (kind === 'SharedKernel') {
|
|
213
|
+
edgeCssClasses = ['shared-kernel'];
|
|
214
|
+
}
|
|
215
|
+
const edge = {
|
|
216
|
+
type: 'edge',
|
|
217
|
+
id: edgeId,
|
|
218
|
+
sourceId: leftNode.id,
|
|
219
|
+
targetId: rightNode.id,
|
|
220
|
+
cssClasses: edgeCssClasses,
|
|
221
|
+
children: children.length > 0 ? children : undefined,
|
|
222
|
+
};
|
|
223
|
+
this.traceProvider.trace(edge, astNode);
|
|
224
|
+
return [edge];
|
|
225
|
+
}
|
|
226
|
+
/**
|
|
227
|
+
* Adds a U/D badge label at the source or target end of an edge.
|
|
228
|
+
*
|
|
229
|
+
* Patterns are normalised to their short abbreviations (e.g. `OpenHostService` →
|
|
230
|
+
* `OHS`) and `BBoM`/`BigBallOfMud` is excluded — BBoM is surfaced visually as a
|
|
231
|
+
* cloud node shape rather than a text annotation.
|
|
232
|
+
*
|
|
233
|
+
* Badge text format: `ROLE|PATTERNS` (e.g. `U|OHS, PL` or `D|ACL`).
|
|
234
|
+
* The webview `UDBadgeLabelView` renders this as a bordered box.
|
|
235
|
+
*/
|
|
236
|
+
addUDBadge(children, edgeId, placement, patterns, role, args) {
|
|
237
|
+
// Normalise pattern names and strip BBoM (shown on node, not in badge).
|
|
238
|
+
// Also strip Supplier/Customer when they are already expressed as the role letter
|
|
239
|
+
// to prevent duplicate annotations like "S S" or "C C".
|
|
240
|
+
const badgePatterns = patterns
|
|
241
|
+
.filter((p) => !isBBoMSidePattern(p))
|
|
242
|
+
.filter((p) => !(role === 'S' && p.$type === 'Supplier'))
|
|
243
|
+
.filter((p) => !(role === 'C' && p.$type === 'Customer'))
|
|
244
|
+
.map(normalizePatternNode);
|
|
245
|
+
if (!role && badgePatterns.length === 0) {
|
|
246
|
+
return;
|
|
247
|
+
}
|
|
248
|
+
// Encode as ROLE|PATTERNS for UDBadgeLabelView parsing
|
|
249
|
+
const rolePart = role ?? '';
|
|
250
|
+
const patternPart = badgePatterns.length > 0 ? badgePatterns.join(', ') : '';
|
|
251
|
+
const badgeText = `${rolePart}|${patternPart}`;
|
|
252
|
+
children.push({
|
|
253
|
+
type: 'label:ud-badge',
|
|
254
|
+
id: args.idCache.uniqueId(`${edgeId}:${placement}`),
|
|
255
|
+
text: badgeText,
|
|
256
|
+
edgePlacement: {
|
|
257
|
+
side: 'on',
|
|
258
|
+
position: placement === 'source' ? 0.1 : 0.9,
|
|
259
|
+
rotate: false,
|
|
260
|
+
offset: 0,
|
|
261
|
+
},
|
|
262
|
+
});
|
|
263
|
+
}
|
|
264
|
+
/**
|
|
265
|
+
* Formats the relationship type for the center edge label.
|
|
266
|
+
*
|
|
267
|
+
* Maps DomainLang keywords to DDD community notation display names:
|
|
268
|
+
* CustomerSupplier → Customer/Supplier
|
|
269
|
+
* SharedKernel → Shared Kernel
|
|
270
|
+
* UpstreamDownstream → Upstream/Downstream
|
|
271
|
+
* Partnership → Partnership
|
|
272
|
+
*
|
|
273
|
+
* For `<->` without explicit type, defaults to "Partnership".
|
|
274
|
+
* For `><`, defaults to "Separate Ways".
|
|
275
|
+
*/
|
|
276
|
+
formatRelationshipKind(kind) {
|
|
277
|
+
if (!kind)
|
|
278
|
+
return undefined;
|
|
279
|
+
return this.displayRelationshipKind(kind);
|
|
280
|
+
}
|
|
281
|
+
displayRelationshipKind(kind) {
|
|
282
|
+
switch (kind) {
|
|
283
|
+
// Directional kinds are already conveyed by U/D and C/S role badges — no center label needed
|
|
284
|
+
case 'CustomerSupplier': return undefined;
|
|
285
|
+
case 'UpstreamDownstream': return undefined;
|
|
286
|
+
// Symmetric and bidirectional kinds have no role badges, so label them explicitly
|
|
287
|
+
case 'SharedKernel': return 'Shared Kernel';
|
|
288
|
+
case 'SeparateWays': return 'Separate Ways';
|
|
289
|
+
case 'Partnership': return 'Partnership';
|
|
290
|
+
default: return kind;
|
|
291
|
+
}
|
|
292
|
+
}
|
|
293
|
+
// ── Node generation ──
|
|
294
|
+
collectContextMapNodes(selectedMap, query, nodeMap, bboMNodeKeys, args) {
|
|
295
|
+
for (const boundedContextRef of selectedMap.boundedContexts) {
|
|
296
|
+
for (const item of boundedContextRef.items) {
|
|
297
|
+
this.ensureNodeForContextMapItem(item, query, nodeMap, bboMNodeKeys, args);
|
|
298
|
+
}
|
|
299
|
+
if (boundedContextRef.items.length === 0 && boundedContextRef.$refText) {
|
|
300
|
+
this.ensureUnresolvedNode(nodeMap, boundedContextRef.$refText, args);
|
|
301
|
+
}
|
|
302
|
+
}
|
|
303
|
+
}
|
|
304
|
+
ensureNodeForContextMapItem(item, query, nodeMap, bboMNodeKeys, args) {
|
|
305
|
+
const boundedContext = item.ref;
|
|
306
|
+
if (!boundedContext) {
|
|
307
|
+
if (item.$refText) {
|
|
308
|
+
this.ensureUnresolvedNode(nodeMap, item.$refText, args);
|
|
309
|
+
}
|
|
310
|
+
return;
|
|
311
|
+
}
|
|
312
|
+
const nodeKey = this.getNodeKey(query, boundedContext);
|
|
313
|
+
if (nodeMap.has(nodeKey)) {
|
|
314
|
+
return;
|
|
315
|
+
}
|
|
316
|
+
const nodeId = args.idCache.uniqueId(`node:${nodeKey}`, boundedContext);
|
|
317
|
+
const node = this.createNode(nodeId, boundedContext.name, bboMNodeKeys.has(nodeKey));
|
|
318
|
+
this.traceProvider.trace(node, boundedContext);
|
|
319
|
+
nodeMap.set(nodeKey, node);
|
|
320
|
+
}
|
|
321
|
+
createNode(id, label, isBBoM = false) {
|
|
322
|
+
return {
|
|
323
|
+
type: isBBoM ? 'node:bbom' : 'node',
|
|
324
|
+
id,
|
|
325
|
+
position: { x: 0, y: 0 },
|
|
326
|
+
size: {
|
|
327
|
+
width: NODE_WIDTH,
|
|
328
|
+
height: NODE_HEIGHT,
|
|
329
|
+
},
|
|
330
|
+
layout: 'vbox',
|
|
331
|
+
layoutOptions: {
|
|
332
|
+
hAlign: 'center',
|
|
333
|
+
vAlign: 'center',
|
|
334
|
+
resizeContainer: false,
|
|
335
|
+
paddingTop: 10,
|
|
336
|
+
paddingBottom: 10,
|
|
337
|
+
paddingLeft: 20,
|
|
338
|
+
paddingRight: 20,
|
|
339
|
+
},
|
|
340
|
+
children: [this.createNodeLabel(id, label)],
|
|
341
|
+
};
|
|
342
|
+
}
|
|
343
|
+
createNodeLabel(nodeId, label) {
|
|
344
|
+
return {
|
|
345
|
+
type: 'label',
|
|
346
|
+
id: `${nodeId}:label`,
|
|
347
|
+
text: label,
|
|
348
|
+
};
|
|
349
|
+
}
|
|
350
|
+
ensureNode(nodeMap, query, boundedContext, bboMNodeKeys, args) {
|
|
351
|
+
const nodeKey = this.getNodeKey(query, boundedContext);
|
|
352
|
+
if (nodeMap.has(nodeKey)) {
|
|
353
|
+
return;
|
|
354
|
+
}
|
|
355
|
+
const nodeId = args.idCache.uniqueId(`node:${nodeKey}`, boundedContext);
|
|
356
|
+
const node = this.createNode(nodeId, boundedContext.name, bboMNodeKeys.has(nodeKey));
|
|
357
|
+
this.traceProvider.trace(node, boundedContext);
|
|
358
|
+
nodeMap.set(nodeKey, node);
|
|
359
|
+
}
|
|
360
|
+
ensureUnresolvedNode(nodeMap, label, args) {
|
|
361
|
+
const key = `unresolved:${label}`;
|
|
362
|
+
if (nodeMap.has(key)) {
|
|
363
|
+
return;
|
|
364
|
+
}
|
|
365
|
+
const nodeId = args.idCache.uniqueId(`node:${key}`);
|
|
366
|
+
nodeMap.set(key, this.createNode(nodeId, label));
|
|
367
|
+
}
|
|
368
|
+
getNodeKey(query, boundedContext) {
|
|
369
|
+
const fqn = query.fqn(boundedContext);
|
|
370
|
+
return fqn ?? boundedContext.name;
|
|
371
|
+
}
|
|
372
|
+
// ── Selection helpers ──
|
|
373
|
+
getSelection(args) {
|
|
374
|
+
const options = args.options;
|
|
375
|
+
if (typeof options !== 'object' || options === null) {
|
|
376
|
+
return {};
|
|
377
|
+
}
|
|
378
|
+
const selectedContextMapFqn = this.getStringOption(options, 'selectedContextMapFqn');
|
|
379
|
+
const selectedContextMapName = this.getStringOption(options, 'selectedContextMapName');
|
|
380
|
+
return {
|
|
381
|
+
selectedContextMapFqn,
|
|
382
|
+
selectedContextMapName,
|
|
383
|
+
};
|
|
384
|
+
}
|
|
385
|
+
getStringOption(options, key) {
|
|
386
|
+
const value = Reflect.get(options, key);
|
|
387
|
+
return typeof value === 'string' && value.trim().length > 0 ? value.trim() : undefined;
|
|
388
|
+
}
|
|
389
|
+
selectContextMap(contextMaps, query, selection) {
|
|
390
|
+
if (selection.selectedContextMapFqn) {
|
|
391
|
+
const byFqn = contextMaps.find((contextMap) => query.fqn(contextMap) === selection.selectedContextMapFqn);
|
|
392
|
+
if (byFqn) {
|
|
393
|
+
return byFqn;
|
|
394
|
+
}
|
|
395
|
+
}
|
|
396
|
+
if (selection.selectedContextMapName) {
|
|
397
|
+
const byName = contextMaps.find((contextMap) => contextMap.name === selection.selectedContextMapName);
|
|
398
|
+
if (byName) {
|
|
399
|
+
return byName;
|
|
400
|
+
}
|
|
401
|
+
}
|
|
402
|
+
return contextMaps[0];
|
|
403
|
+
}
|
|
404
|
+
}
|
|
405
|
+
//# sourceMappingURL=context-map-diagram-generator.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"context-map-diagram-generator.js","sourceRoot":"","sources":["../../src/diagram/context-map-diagram-generator.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import type { LayoutOptions } from 'elkjs/lib/elk-api.js';
|
|
2
|
+
import { DefaultLayoutConfigurator } from 'sprotty-elk/lib/elk-layout.js';
|
|
3
|
+
import type { SGraph, SNode, SModelIndex, SEdge } from 'sprotty-protocol';
|
|
4
|
+
/**
|
|
5
|
+
* ELK layout configurator for DomainLang context map diagrams.
|
|
6
|
+
*
|
|
7
|
+
* Uses the `layered` algorithm with **DOWN** direction (top-to-bottom) so
|
|
8
|
+
* upstream contexts appear above downstream contexts, matching the conventional
|
|
9
|
+
* DDD Context Map layout direction.
|
|
10
|
+
*/
|
|
11
|
+
export declare class ContextMapLayoutConfigurator extends DefaultLayoutConfigurator {
|
|
12
|
+
protected graphOptions(_graph: SGraph, _index: SModelIndex): LayoutOptions;
|
|
13
|
+
protected nodeOptions(_node: SNode, _index: SModelIndex): LayoutOptions;
|
|
14
|
+
protected edgeOptions(_edge: SEdge, _index: SModelIndex): LayoutOptions;
|
|
15
|
+
}
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
import { DefaultLayoutConfigurator } from 'sprotty-elk/lib/elk-layout.js';
|
|
2
|
+
/**
|
|
3
|
+
* ELK layout configurator for DomainLang context map diagrams.
|
|
4
|
+
*
|
|
5
|
+
* Uses the `layered` algorithm with **DOWN** direction (top-to-bottom) so
|
|
6
|
+
* upstream contexts appear above downstream contexts, matching the conventional
|
|
7
|
+
* DDD Context Map layout direction.
|
|
8
|
+
*/
|
|
9
|
+
export class ContextMapLayoutConfigurator extends DefaultLayoutConfigurator {
|
|
10
|
+
graphOptions(_graph, _index) {
|
|
11
|
+
return {
|
|
12
|
+
'elk.algorithm': 'layered',
|
|
13
|
+
'elk.direction': 'DOWN',
|
|
14
|
+
'elk.layered.nodePlacement.strategy': 'BRANDES_KOEPF',
|
|
15
|
+
'elk.spacing.nodeNode': '100',
|
|
16
|
+
'elk.layered.spacing.nodeNodeBetweenLayers': '140',
|
|
17
|
+
'elk.spacing.edgeNode': '60',
|
|
18
|
+
'elk.spacing.edgeEdge': '40',
|
|
19
|
+
// Edge routing mode is irrelevant — the webview SmoothBezierEdgeView
|
|
20
|
+
// ignores ELK routing points and computes dynamic bezier curves
|
|
21
|
+
// anchored directly to node ellipses with obstacle avoidance.
|
|
22
|
+
'elk.edgeRouting': 'POLYLINE',
|
|
23
|
+
'elk.layered.mergeEdges': 'false',
|
|
24
|
+
'elk.layered.crossingMinimization.strategy': 'LAYER_SWEEP',
|
|
25
|
+
'elk.layered.thoroughness': '7',
|
|
26
|
+
};
|
|
27
|
+
}
|
|
28
|
+
nodeOptions(_node, _index) {
|
|
29
|
+
return {
|
|
30
|
+
'elk.layered.nodePlacement.bk.fixedAlignment': 'BALANCED',
|
|
31
|
+
};
|
|
32
|
+
}
|
|
33
|
+
edgeOptions(_edge, _index) {
|
|
34
|
+
return {
|
|
35
|
+
'elk.layered.priority.direction': '1',
|
|
36
|
+
};
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
//# sourceMappingURL=context-map-layout-configurator.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"context-map-layout-configurator.js","sourceRoot":"","sources":["../../src/diagram/context-map-layout-configurator.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,yBAAyB,EAAE,MAAM,+BAA+B,CAAC;AAG1E;;;;;;GAMG;AACH,MAAM,OAAO,4BAA6B,SAAQ,yBAAyB;IACpD,YAAY,CAAC,MAAc,EAAE,MAAmB;QAC/D,OAAO;YACH,eAAe,EAAE,SAAS;YAC1B,eAAe,EAAE,MAAM;YACvB,oCAAoC,EAAE,eAAe;YACrD,sBAAsB,EAAE,KAAK;YAC7B,2CAA2C,EAAE,KAAK;YAClD,sBAAsB,EAAE,IAAI;YAC5B,sBAAsB,EAAE,IAAI;YAC5B,qEAAqE;YACrE,gEAAgE;YAChE,8DAA8D;YAC9D,iBAAiB,EAAE,UAAU;YAC7B,wBAAwB,EAAE,OAAO;YACjC,2CAA2C,EAAE,aAAa;YAC1D,0BAA0B,EAAE,GAAG;SAClC,CAAC;IACN,CAAC;IAEkB,WAAW,CAAC,KAAY,EAAE,MAAmB;QAC5D,OAAO;YACH,6CAA6C,EAAE,UAAU;SAC5D,CAAC;IACN,CAAC;IAEkB,WAAW,CAAC,KAAY,EAAE,MAAmB;QAC5D,OAAO;YACH,gCAAgC,EAAE,GAAG;SACxC,CAAC;IACN,CAAC;CACJ"}
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
import { type ElkFactory, type IElementFilter, type ILayoutConfigurator } from 'sprotty-elk/lib/elk-layout.js';
|
|
2
|
+
import type { IModelLayoutEngine } from 'sprotty-protocol';
|
|
3
|
+
/**
|
|
4
|
+
* Creates an {@link ElkFactory} that produces a new bundled ELK instance
|
|
5
|
+
* pre-configured for the `layered` algorithm.
|
|
6
|
+
*
|
|
7
|
+
* Use this in Langium's DI module (`layout.ElkFactory`) or directly in the
|
|
8
|
+
* CLI when running the layout engine standalone.
|
|
9
|
+
*/
|
|
10
|
+
export declare function createElkFactory(): ElkFactory;
|
|
11
|
+
/**
|
|
12
|
+
* Creates the full set of objects required to run the ELK layout engine for
|
|
13
|
+
* DomainLang context-map diagrams.
|
|
14
|
+
*
|
|
15
|
+
* Returns the three collaborators that together constitute a complete
|
|
16
|
+
* `IModelLayoutEngine`:
|
|
17
|
+
* - `elkFactory` — produces the ELK worker instance
|
|
18
|
+
* - `elementFilter` — controls which model elements are laid out
|
|
19
|
+
* - `layoutConfigurator` — supplies algorithm-specific ELK options
|
|
20
|
+
*
|
|
21
|
+
* Callers that use Langium DI should register these individually via the
|
|
22
|
+
* `layout` service group. Callers that operate outside of DI (e.g. the CLI)
|
|
23
|
+
* can call {@link createElkLayoutEngine} for a fully assembled engine.
|
|
24
|
+
*/
|
|
25
|
+
export declare function createElkLayoutComponents(): {
|
|
26
|
+
elkFactory: ElkFactory;
|
|
27
|
+
elementFilter: IElementFilter;
|
|
28
|
+
layoutConfigurator: ILayoutConfigurator;
|
|
29
|
+
};
|
|
30
|
+
/**
|
|
31
|
+
* Creates a fully assembled {@link IModelLayoutEngine} for DomainLang
|
|
32
|
+
* context-map diagrams using the bundled ELK engine.
|
|
33
|
+
*
|
|
34
|
+
* Intended for use outside of Langium's DI container — for example, in the
|
|
35
|
+
* CLI when generating standalone SVG or image exports.
|
|
36
|
+
*
|
|
37
|
+
* @example
|
|
38
|
+
* ```typescript
|
|
39
|
+
* const layoutEngine = createElkLayoutEngine();
|
|
40
|
+
* const laidOutModel = await layoutEngine.layout(rootModel);
|
|
41
|
+
* ```
|
|
42
|
+
*/
|
|
43
|
+
export declare function createElkLayoutEngine(): IModelLayoutEngine;
|