@likec4/language-server 1.43.0 → 1.44.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.
Files changed (299) hide show
  1. package/browser/package.json +1 -1
  2. package/browser-worker/package.json +1 -1
  3. package/dist/LikeC4LanguageServices.d.ts +12 -19
  4. package/dist/LikeC4LanguageServices.js +182 -0
  5. package/dist/Rpc.js +245 -0
  6. package/dist/ast.d.ts +5 -1
  7. package/dist/ast.js +253 -0
  8. package/dist/browser-worker.js +4 -0
  9. package/dist/browser.js +35 -0
  10. package/dist/bundled.js +42 -0
  11. package/dist/bundled.mjs +3542 -3896
  12. package/dist/documentation/documentation-provider.js +51 -0
  13. package/dist/documentation/index.js +1 -0
  14. package/dist/empty.js +2 -0
  15. package/dist/filesystem/ChokidarWatcher.js +97 -0
  16. package/dist/filesystem/FileSystemWatcher.js +14 -0
  17. package/dist/filesystem/LikeC4FileSystem.d.ts +1 -2
  18. package/dist/filesystem/LikeC4FileSystem.js +126 -0
  19. package/dist/filesystem/index.d.ts +26 -0
  20. package/dist/filesystem/index.js +29 -0
  21. package/dist/formatting/LikeC4Formatter.js +637 -0
  22. package/dist/formatting/utils.js +18 -0
  23. package/dist/generated/ast.js +2155 -0
  24. package/dist/generated/{grammar.mjs → grammar.js} +6 -2
  25. package/dist/generated/module.d.ts +6 -1
  26. package/dist/generated/module.js +27 -0
  27. package/dist/generated-lib/{icons.mjs → icons.js} +11 -7
  28. package/dist/index.d.ts +7 -0
  29. package/dist/index.js +53 -0
  30. package/dist/{likec4lib.mjs → likec4lib.js} +3 -3
  31. package/dist/logger.js +81 -0
  32. package/dist/lsp/CodeActionProvider.d.ts +14 -0
  33. package/dist/lsp/CodeActionProvider.js +33 -0
  34. package/dist/lsp/CodeLensProvider.js +44 -0
  35. package/dist/lsp/CompletionProvider.d.ts +3 -1
  36. package/dist/lsp/CompletionProvider.js +200 -0
  37. package/dist/lsp/DocumentHighlightProvider.js +10 -0
  38. package/dist/lsp/DocumentLinkProvider.js +58 -0
  39. package/dist/lsp/DocumentSymbolProvider.js +306 -0
  40. package/dist/lsp/HoverProvider.js +106 -0
  41. package/dist/lsp/RenameProvider.js +6 -0
  42. package/dist/lsp/SemanticTokenProvider.js +257 -0
  43. package/dist/lsp/index.d.ts +1 -0
  44. package/dist/lsp/index.js +9 -0
  45. package/dist/mcp/MCPServerFactory.js +73 -0
  46. package/dist/mcp/NoopLikeC4MCPServer.js +17 -0
  47. package/dist/mcp/interfaces.js +5 -0
  48. package/dist/mcp/server/StdioLikeC4MCPServer.js +47 -0
  49. package/dist/mcp/server/StreamableLikeC4MCPServer.js +145 -0
  50. package/dist/mcp/server/WithMCPServer.js +56 -0
  51. package/dist/mcp/tools/_common.d.ts +8 -7
  52. package/dist/mcp/tools/_common.js +49 -0
  53. package/dist/mcp/tools/find-relationships.d.ts +7 -8
  54. package/dist/mcp/tools/find-relationships.js +150 -0
  55. package/dist/mcp/tools/list-projects.d.ts +3 -3
  56. package/dist/mcp/tools/list-projects.js +62 -0
  57. package/dist/mcp/tools/open-view.d.ts +6 -7
  58. package/dist/mcp/tools/open-view.js +52 -0
  59. package/dist/mcp/tools/read-deployment.d.ts +6 -7
  60. package/dist/mcp/tools/read-deployment.js +132 -0
  61. package/dist/mcp/tools/read-element.d.ts +6 -7
  62. package/dist/mcp/tools/read-element.js +194 -0
  63. package/dist/mcp/tools/read-project-summary.d.ts +5 -6
  64. package/dist/mcp/tools/read-project-summary.js +176 -0
  65. package/dist/mcp/tools/read-view.d.ts +6 -7
  66. package/dist/mcp/tools/read-view.js +203 -0
  67. package/dist/mcp/tools/search-element.d.ts +3 -3
  68. package/dist/mcp/tools/search-element.js +177 -0
  69. package/dist/mcp/utils.d.ts +2 -2
  70. package/dist/mcp/utils.js +48 -0
  71. package/dist/model/builder/MergedExtends.js +74 -0
  72. package/dist/model/builder/MergedSpecification.js +175 -0
  73. package/dist/model/builder/buildModel.js +176 -0
  74. package/dist/model/deployments-index.js +102 -0
  75. package/dist/model/fqn-index.js +250 -0
  76. package/dist/model/index.js +6 -0
  77. package/dist/model/model-builder.d.ts +13 -11
  78. package/dist/model/model-builder.js +234 -0
  79. package/dist/model/model-locator.d.ts +6 -5
  80. package/dist/model/model-locator.js +240 -0
  81. package/dist/model/model-parser-where.js +81 -0
  82. package/dist/model/model-parser.d.ts +309 -304
  83. package/dist/model/model-parser.js +119 -0
  84. package/dist/model/parser/Base.d.ts +2 -2
  85. package/dist/model/parser/Base.js +367 -0
  86. package/dist/model/parser/DeploymentModelParser.d.ts +2 -2
  87. package/dist/model/parser/DeploymentModelParser.js +176 -0
  88. package/dist/model/parser/DeploymentViewParser.d.ts +3 -3
  89. package/dist/model/parser/DeploymentViewParser.js +86 -0
  90. package/dist/model/parser/FqnRefParser.d.ts +2 -2
  91. package/dist/model/parser/FqnRefParser.js +382 -0
  92. package/dist/model/parser/GlobalsParser.d.ts +6 -6
  93. package/dist/model/parser/GlobalsParser.js +84 -0
  94. package/dist/model/parser/ImportsParser.d.ts +11 -12
  95. package/dist/model/parser/ImportsParser.js +24 -0
  96. package/dist/model/parser/ModelParser.d.ts +2 -2
  97. package/dist/model/parser/ModelParser.js +165 -0
  98. package/dist/model/parser/PredicatesParser.d.ts +2 -2
  99. package/dist/model/parser/PredicatesParser.js +45 -0
  100. package/dist/model/parser/SpecificationParser.d.ts +2 -2
  101. package/dist/model/parser/SpecificationParser.js +109 -0
  102. package/dist/model/parser/ValueConverter.js +12 -0
  103. package/dist/model/parser/ViewsParser.d.ts +3 -3
  104. package/dist/model/parser/ViewsParser.js +477 -0
  105. package/dist/model-change/ModelChanges.d.ts +6 -3
  106. package/dist/model-change/ModelChanges.js +102 -0
  107. package/dist/model-change/changeElementStyle.js +134 -0
  108. package/dist/model-change/changeViewLayout.d.ts +2 -2
  109. package/dist/model-change/changeViewLayout.js +28 -0
  110. package/dist/model-change/removeManualLayoutV1.d.ts +7 -0
  111. package/dist/model-change/removeManualLayoutV1.js +27 -0
  112. package/dist/module.d.ts +10 -5
  113. package/dist/module.js +143 -0
  114. package/dist/protocol.d.ts +1 -17
  115. package/dist/protocol.js +114 -0
  116. package/dist/references/index.js +3 -0
  117. package/dist/references/name-provider.js +37 -0
  118. package/dist/references/scope-computation.js +288 -0
  119. package/dist/references/scope-provider.d.ts +3 -3
  120. package/dist/references/scope-provider.js +242 -0
  121. package/dist/shared/NodeKindProvider.js +57 -0
  122. package/dist/shared/{WorkspaceSymbolProvider.mjs → WorkspaceSymbolProvider.js} +1 -1
  123. package/dist/shared/index.js +2 -0
  124. package/dist/test/index.js +1 -0
  125. package/dist/test/testServices.d.ts +16 -16
  126. package/dist/test/testServices.js +210 -0
  127. package/dist/utils/disposable.js +26 -0
  128. package/dist/utils/elementRef.d.ts +1 -1
  129. package/dist/utils/elementRef.js +27 -0
  130. package/dist/utils/fqnRef.js +63 -0
  131. package/dist/utils/index.js +35 -0
  132. package/dist/utils/printDocs.js +1 -0
  133. package/dist/utils/projectId.js +16 -0
  134. package/dist/utils/stringHash.js +5 -0
  135. package/dist/validation/DocumentValidator.js +17 -0
  136. package/dist/validation/_shared.js +26 -0
  137. package/dist/validation/deployment-checks.js +140 -0
  138. package/dist/validation/dynamic-view.js +67 -0
  139. package/dist/validation/element-ref.js +12 -0
  140. package/dist/validation/element.js +49 -0
  141. package/dist/validation/imports.js +46 -0
  142. package/dist/validation/index.d.ts +1 -1
  143. package/dist/validation/index.js +157 -0
  144. package/dist/validation/property-checks.js +108 -0
  145. package/dist/validation/relation.js +55 -0
  146. package/dist/validation/specification.js +190 -0
  147. package/dist/validation/view-predicates/fqn-expr-with.js +43 -0
  148. package/dist/validation/view-predicates/fqn-ref-expr.js +51 -0
  149. package/dist/validation/view-predicates/incoming.js +16 -0
  150. package/dist/validation/view-predicates/index.js +6 -0
  151. package/dist/validation/view-predicates/outgoing.js +20 -0
  152. package/dist/validation/view-predicates/relation-expr.js +46 -0
  153. package/dist/validation/view-predicates/relation-with.js +16 -0
  154. package/dist/validation/view.d.ts +1 -1
  155. package/dist/validation/view.js +42 -0
  156. package/dist/view-utils/assignNavigateTo.js +27 -0
  157. package/dist/view-utils/index.d.ts +1 -0
  158. package/dist/view-utils/index.js +2 -0
  159. package/dist/view-utils/manual-layout.d.ts +6 -0
  160. package/dist/view-utils/manual-layout.js +151 -0
  161. package/dist/views/ConfigurableLayouter.js +51 -0
  162. package/dist/views/LikeC4ManualLayouts.d.ts +28 -0
  163. package/dist/views/LikeC4ManualLayouts.js +132 -0
  164. package/dist/views/{likec4-views.d.ts → LikeC4Views.d.ts} +9 -8
  165. package/dist/views/LikeC4Views.js +200 -0
  166. package/dist/views/index.d.ts +4 -1
  167. package/dist/views/index.js +11 -0
  168. package/dist/workspace/AstNodeDescriptionProvider.js +15 -0
  169. package/dist/workspace/IndexManager.js +21 -0
  170. package/dist/workspace/LangiumDocuments.d.ts +1 -1
  171. package/dist/workspace/LangiumDocuments.js +58 -0
  172. package/dist/workspace/ProjectsManager.d.ts +8 -3
  173. package/dist/workspace/ProjectsManager.js +373 -0
  174. package/dist/workspace/WorkspaceManager.d.ts +3 -2
  175. package/dist/workspace/WorkspaceManager.js +93 -0
  176. package/dist/workspace/index.js +5 -0
  177. package/likec4lib/package.json +1 -1
  178. package/package.json +25 -24
  179. package/protocol/package.json +1 -1
  180. package/dist/LikeC4LanguageServices.mjs +0 -197
  181. package/dist/Rpc.mjs +0 -296
  182. package/dist/ast.mjs +0 -221
  183. package/dist/browser-worker.mjs +0 -2
  184. package/dist/browser.mjs +0 -32
  185. package/dist/documentation/documentation-provider.mjs +0 -48
  186. package/dist/documentation/index.mjs +0 -1
  187. package/dist/empty.mjs +0 -1
  188. package/dist/filesystem/ChokidarWatcher.mjs +0 -68
  189. package/dist/filesystem/FileSystemWatcher.mjs +0 -11
  190. package/dist/filesystem/LikeC4FileSystem.mjs +0 -64
  191. package/dist/filesystem/index.mjs +0 -19
  192. package/dist/formatting/LikeC4Formatter.mjs +0 -511
  193. package/dist/formatting/utils.mjs +0 -15
  194. package/dist/generated/ast.mjs +0 -2150
  195. package/dist/generated/module.mjs +0 -23
  196. package/dist/index.mjs +0 -50
  197. package/dist/logger.mjs +0 -82
  198. package/dist/lsp/CodeLensProvider.mjs +0 -42
  199. package/dist/lsp/CompletionProvider.mjs +0 -208
  200. package/dist/lsp/DocumentHighlightProvider.mjs +0 -10
  201. package/dist/lsp/DocumentLinkProvider.mjs +0 -53
  202. package/dist/lsp/DocumentSymbolProvider.mjs +0 -287
  203. package/dist/lsp/HoverProvider.mjs +0 -104
  204. package/dist/lsp/RenameProvider.mjs +0 -6
  205. package/dist/lsp/SemanticTokenProvider.mjs +0 -276
  206. package/dist/lsp/index.mjs +0 -7
  207. package/dist/mcp/MCPServerFactory.mjs +0 -70
  208. package/dist/mcp/NoopLikeC4MCPServer.mjs +0 -17
  209. package/dist/mcp/interfaces.mjs +0 -4
  210. package/dist/mcp/server/StdioLikeC4MCPServer.mjs +0 -46
  211. package/dist/mcp/server/StreamableLikeC4MCPServer.mjs +0 -153
  212. package/dist/mcp/server/WithMCPServer.mjs +0 -58
  213. package/dist/mcp/tools/_common.mjs +0 -42
  214. package/dist/mcp/tools/find-relationships.mjs +0 -151
  215. package/dist/mcp/tools/list-projects.mjs +0 -62
  216. package/dist/mcp/tools/open-view.mjs +0 -52
  217. package/dist/mcp/tools/read-deployment.mjs +0 -130
  218. package/dist/mcp/tools/read-element.mjs +0 -198
  219. package/dist/mcp/tools/read-project-summary.mjs +0 -178
  220. package/dist/mcp/tools/read-view.mjs +0 -205
  221. package/dist/mcp/tools/search-element.mjs +0 -171
  222. package/dist/mcp/utils.mjs +0 -47
  223. package/dist/model/builder/MergedExtends.mjs +0 -76
  224. package/dist/model/builder/MergedSpecification.mjs +0 -205
  225. package/dist/model/builder/assignTagColors.d.ts +0 -7
  226. package/dist/model/builder/assignTagColors.mjs +0 -51
  227. package/dist/model/builder/buildModel.mjs +0 -226
  228. package/dist/model/deployments-index.mjs +0 -100
  229. package/dist/model/fqn-index.mjs +0 -243
  230. package/dist/model/index.mjs +0 -6
  231. package/dist/model/model-builder.mjs +0 -285
  232. package/dist/model/model-locator.mjs +0 -239
  233. package/dist/model/model-parser-where.mjs +0 -81
  234. package/dist/model/model-parser.mjs +0 -127
  235. package/dist/model/parser/Base.mjs +0 -376
  236. package/dist/model/parser/DeploymentModelParser.mjs +0 -212
  237. package/dist/model/parser/DeploymentViewParser.mjs +0 -95
  238. package/dist/model/parser/FqnRefParser.mjs +0 -398
  239. package/dist/model/parser/GlobalsParser.mjs +0 -82
  240. package/dist/model/parser/ImportsParser.mjs +0 -28
  241. package/dist/model/parser/ModelParser.mjs +0 -190
  242. package/dist/model/parser/PredicatesParser.mjs +0 -45
  243. package/dist/model/parser/SpecificationParser.mjs +0 -120
  244. package/dist/model/parser/ValueConverter.mjs +0 -12
  245. package/dist/model/parser/ViewsParser.mjs +0 -490
  246. package/dist/model-change/ModelChanges.mjs +0 -89
  247. package/dist/model-change/changeElementStyle.mjs +0 -143
  248. package/dist/model-change/changeViewLayout.mjs +0 -32
  249. package/dist/model-change/saveManualLayout.d.ts +0 -11
  250. package/dist/model-change/saveManualLayout.mjs +0 -27
  251. package/dist/module.mjs +0 -180
  252. package/dist/protocol.mjs +0 -65
  253. package/dist/references/index.mjs +0 -3
  254. package/dist/references/name-provider.mjs +0 -39
  255. package/dist/references/scope-computation.mjs +0 -312
  256. package/dist/references/scope-provider.mjs +0 -239
  257. package/dist/shared/NodeKindProvider.mjs +0 -110
  258. package/dist/shared/index.mjs +0 -2
  259. package/dist/test/index.mjs +0 -1
  260. package/dist/test/testServices.mjs +0 -200
  261. package/dist/utils/disposable.mjs +0 -25
  262. package/dist/utils/elementRef.mjs +0 -20
  263. package/dist/utils/fqnRef.mjs +0 -57
  264. package/dist/utils/index.mjs +0 -33
  265. package/dist/utils/printDocs.mjs +0 -1
  266. package/dist/utils/projectId.mjs +0 -16
  267. package/dist/utils/stringHash.mjs +0 -5
  268. package/dist/validation/DocumentValidator.mjs +0 -16
  269. package/dist/validation/_shared.mjs +0 -25
  270. package/dist/validation/deployment-checks.mjs +0 -146
  271. package/dist/validation/dynamic-view.mjs +0 -67
  272. package/dist/validation/element-ref.mjs +0 -12
  273. package/dist/validation/element.mjs +0 -50
  274. package/dist/validation/imports.mjs +0 -25
  275. package/dist/validation/index.mjs +0 -180
  276. package/dist/validation/property-checks.mjs +0 -107
  277. package/dist/validation/relation.mjs +0 -53
  278. package/dist/validation/specification.mjs +0 -173
  279. package/dist/validation/view-predicates/fqn-expr-with.mjs +0 -43
  280. package/dist/validation/view-predicates/fqn-ref-expr.mjs +0 -53
  281. package/dist/validation/view-predicates/incoming.mjs +0 -16
  282. package/dist/validation/view-predicates/index.mjs +0 -6
  283. package/dist/validation/view-predicates/outgoing.mjs +0 -20
  284. package/dist/validation/view-predicates/relation-expr.mjs +0 -39
  285. package/dist/validation/view-predicates/relation-with.mjs +0 -16
  286. package/dist/validation/view.mjs +0 -25
  287. package/dist/view-utils/assignNavigateTo.mjs +0 -25
  288. package/dist/view-utils/index.mjs +0 -1
  289. package/dist/view-utils/manual-layout.mjs +0 -99
  290. package/dist/views/configurable-layouter.mjs +0 -51
  291. package/dist/views/index.mjs +0 -1
  292. package/dist/views/likec4-views.mjs +0 -166
  293. package/dist/workspace/AstNodeDescriptionProvider.mjs +0 -17
  294. package/dist/workspace/IndexManager.mjs +0 -17
  295. package/dist/workspace/LangiumDocuments.mjs +0 -53
  296. package/dist/workspace/ProjectsManager.mjs +0 -360
  297. package/dist/workspace/WorkspaceManager.mjs +0 -83
  298. package/dist/workspace/index.mjs +0 -5
  299. /package/dist/views/{configurable-layouter.d.ts → ConfigurableLayouter.d.ts} +0 -0
@@ -0,0 +1,177 @@
1
+ import { ifilter } from '@likec4/core/utils';
2
+ import * as z from 'zod/v3';
3
+ import { likec4Tool, logger } from '../utils';
4
+ import { includedInViews, includedInViewsSchema } from './_common';
5
+ const searchResultSchema = z.array(z.discriminatedUnion('type', [
6
+ z.object({
7
+ type: z.literal('element'),
8
+ project: z.string().describe('Project ID'),
9
+ id: z.string().describe('Element ID (FQN)'),
10
+ name: z.string().describe('Element name'),
11
+ kind: z.string(),
12
+ title: z.string(),
13
+ technology: z.string().nullable(),
14
+ shape: z.string(),
15
+ includedInViews: includedInViewsSchema,
16
+ metadata: z.record(z.union([z.string(), z.array(z.string())])),
17
+ tags: z.array(z.string()),
18
+ }),
19
+ z.object({
20
+ type: z.literal('deployment-node'),
21
+ project: z.string().describe('Project ID'),
22
+ id: z.string().describe('Deployment ID (FQN)'),
23
+ name: z.string().describe('Deployment name'),
24
+ kind: z.string(),
25
+ title: z.string(),
26
+ technology: z.string().nullable(),
27
+ shape: z.string(),
28
+ includedInViews: includedInViewsSchema,
29
+ metadata: z.record(z.union([z.string(), z.array(z.string())])),
30
+ tags: z.array(z.string()),
31
+ }),
32
+ ]));
33
+ export const searchElement = likec4Tool({
34
+ name: 'search-element',
35
+ annotations: {
36
+ readOnlyHint: true,
37
+ idempotentHint: true,
38
+ title: 'Search elements',
39
+ },
40
+ description: `
41
+ Search LikeC4 elements and deployment nodes across all projects.
42
+
43
+ Query syntax (case-insensitive):
44
+ - kind:<value>: filters by kind
45
+ - shape:<value>: filters by shape
46
+ - meta:<key>: filters by having metadata with the given key
47
+ - #<value>: matches assigned tags
48
+ - Free text: matches id (FQN) or title
49
+
50
+ Request:
51
+ - search: string — at least 2 characters
52
+
53
+ Response (JSON object):
54
+ - total: number - total number of results
55
+ - found: Result[] - returns top 20 results
56
+
57
+ Result (discriminated union by "type"):
58
+ - type = "element": { id: string, name: string, kind: string, title: string, technology: string|null, shape: string, project: string, includedInViews: View[], tags: string[], metadata: Record<string, string> }
59
+ - type = "deployment-node": { id: string, name: string, kind: string, title: string, technology: string|null, shape: string, project: string, includedInViews: View[], tags: string[], metadata: Record<string, string> }
60
+
61
+ View (object) fields:
62
+ - id: string — view identifier
63
+ - title: string — view title
64
+ - type: "element" | "deployment" | "dynamic"
65
+
66
+ Notes:
67
+ - Read-only, idempotent.
68
+ - Use results as input to other tools (e.g., read-element, read-view).
69
+
70
+ Example response:
71
+ {
72
+ "total": 1,
73
+ "found": [
74
+ {
75
+ "type": "logical",
76
+ "project": "default",
77
+ "id": "shop.frontend",
78
+ "name": "frontend",
79
+ "kind": "container",
80
+ "title": "Frontend",
81
+ "technology": "React",
82
+ "shape": "rectangle",
83
+ "includedInViews": [
84
+ {
85
+ "id": "system-overview",
86
+ "title": "System Overview",
87
+ "type": "element"
88
+ }
89
+ ],
90
+ "tags": ["public"],
91
+ "metadata": {}
92
+ }
93
+ ]
94
+ }
95
+ `,
96
+ inputSchema: {
97
+ search: z.string().min(2, 'Search must be at least 2 characters long'),
98
+ },
99
+ outputSchema: {
100
+ total: z.number(),
101
+ found: searchResultSchema,
102
+ },
103
+ }, async (languageServices, args) => {
104
+ const projects = languageServices.projects();
105
+ const found = [];
106
+ let search = args.search.toLowerCase();
107
+ let predicate;
108
+ if (search.startsWith('kind:')) {
109
+ search = search.slice(5);
110
+ logger.debug('search by kind: {search}', { search });
111
+ predicate = (el) => el.kind.toLowerCase() === search;
112
+ }
113
+ else if (search.startsWith('shape:')) {
114
+ search = search.slice(6);
115
+ logger.debug('search by shape: {search}', { search });
116
+ predicate = (el) => el.shape.toLowerCase() === search;
117
+ }
118
+ else if (search.startsWith('meta:')) {
119
+ search = search.slice(5);
120
+ logger.debug('search by metadata: {search}', { search });
121
+ predicate = (el) => !!el.getMetadata(search);
122
+ }
123
+ else if (search.startsWith('#')) {
124
+ search = search.slice(1);
125
+ logger.debug('search by tag: {search}', { search });
126
+ predicate = (el) => el.tags.some(tag => tag.toLowerCase().includes(search));
127
+ }
128
+ else {
129
+ logger.debug('search by id/title: {search}', { search });
130
+ predicate = (el) => el.id.toLowerCase().includes(search)
131
+ || el.title.toLowerCase().includes(search);
132
+ }
133
+ for (const project of projects) {
134
+ try {
135
+ const model = await languageServices.computedModel(project.id);
136
+ // filter elements
137
+ for (const el of ifilter(model.elements(), e => !e.imported && predicate(e))) {
138
+ found.push({
139
+ type: 'element',
140
+ project: project.id,
141
+ id: el.id,
142
+ name: el.name,
143
+ kind: el.kind,
144
+ title: el.title,
145
+ technology: el.technology,
146
+ shape: el.shape,
147
+ tags: [...el.tags],
148
+ metadata: el.getMetadata(),
149
+ includedInViews: includedInViews(el.views()),
150
+ });
151
+ }
152
+ // filter deployment nodes
153
+ for (const el of ifilter(model.deployment.nodes(), predicate)) {
154
+ found.push({
155
+ type: 'deployment-node',
156
+ project: project.id,
157
+ id: el.id,
158
+ name: el.name,
159
+ kind: el.kind,
160
+ title: el.title,
161
+ technology: el.technology,
162
+ shape: el.shape,
163
+ tags: [...el.tags],
164
+ metadata: el.getMetadata(),
165
+ includedInViews: includedInViews(el.views()),
166
+ });
167
+ }
168
+ }
169
+ catch (error) {
170
+ logger.error(`Error searching in project ${project.id}:`, { error });
171
+ }
172
+ }
173
+ return {
174
+ total: found.length,
175
+ found: found.slice(0, 20),
176
+ };
177
+ });
@@ -1,9 +1,9 @@
1
1
  import type { RequestHandlerExtra } from '@modelcontextprotocol/sdk/shared/protocol.js';
2
2
  import type { ServerNotification, ServerRequest, ToolAnnotations } from '@modelcontextprotocol/sdk/types.js';
3
- import type { z, ZodRawShape, ZodTypeAny } from 'zod';
3
+ import type { z, ZodRawShape, ZodTypeAny } from 'zod/v3';
4
4
  import type { ToolCallback } from '@modelcontextprotocol/sdk/server/mcp.js';
5
5
  import type { LikeC4LanguageServices } from '../LikeC4LanguageServices';
6
- export declare const logger: any;
6
+ export declare const logger: import("@logtape/logtape").Logger;
7
7
  type ToolResult<Out extends undefined | ZodRawShape = undefined> = Out extends ZodRawShape ? z.objectOutputType<Out, ZodTypeAny> : string;
8
8
  type LikeC4ToolCallback<Args extends undefined | ZodRawShape, Out extends undefined | ZodRawShape> = Args extends ZodRawShape ? (languageServices: LikeC4LanguageServices, args: z.objectOutputType<Args, ZodTypeAny>, extra: RequestHandlerExtra<ServerRequest, ServerNotification>) => Promise<ToolResult<Out>> : (languageServices: LikeC4LanguageServices, extra: RequestHandlerExtra<ServerRequest, ServerNotification>) => Promise<ToolResult<Out>>;
9
9
  export declare function likec4Tool<InputArgs extends ZodRawShape, OutputArgs extends ZodRawShape, Cb extends LikeC4ToolCallback<InputArgs, OutputArgs>>(config: {
@@ -0,0 +1,48 @@
1
+ import { loggable } from '@likec4/log';
2
+ import { logger as mainLogger } from '../logger';
3
+ export const logger = mainLogger.getChild('mcp');
4
+ export function likec4Tool(config, cb) {
5
+ const { name, description, ...rest } = config;
6
+ return (languageServices) => [
7
+ name,
8
+ {
9
+ description: description?.trim() ?? '',
10
+ ...rest,
11
+ },
12
+ mkcallTool(name, languageServices, cb),
13
+ ];
14
+ }
15
+ function mkcallTool(name, languageServices, cb) {
16
+ const tool = cb.bind(null, languageServices);
17
+ return (async function callTool(args, extra) {
18
+ logger.debug('Calling tool {name}, args: {args}', { name, args });
19
+ try {
20
+ const result = await tool.call(null, args, extra);
21
+ if (typeof result === 'string') {
22
+ return {
23
+ content: [{
24
+ type: 'text',
25
+ text: result,
26
+ }],
27
+ };
28
+ }
29
+ return {
30
+ content: [{
31
+ type: 'text',
32
+ text: JSON.stringify(result),
33
+ }],
34
+ structuredContent: result,
35
+ };
36
+ }
37
+ catch (err) {
38
+ logger.error(`Tool ${name} failed`, { err });
39
+ return {
40
+ content: [{
41
+ type: 'text',
42
+ text: err instanceof Error ? err.message : loggable(err),
43
+ }],
44
+ isError: true,
45
+ };
46
+ }
47
+ });
48
+ }
@@ -0,0 +1,74 @@
1
+ import { hasAtLeast, isEmpty, unique, } from 'remeda';
2
+ export class MergedExtends {
3
+ mergedData = new Map();
4
+ mergeMetadata(existing, incoming) {
5
+ const result = { ...existing };
6
+ for (const [key, incomingValue] of Object.entries(incoming)) {
7
+ const existingValue = result[key];
8
+ if (existingValue === undefined) {
9
+ result[key] = incomingValue;
10
+ continue;
11
+ }
12
+ // Convert both values to arrays to make merging easier.
13
+ const existingArray = Array.isArray(existingValue) ? existingValue : [existingValue];
14
+ const incomingArray = Array.isArray(incomingValue) ? incomingValue : [incomingValue];
15
+ // Merge and deduplicate based on value.
16
+ const merged = unique([...existingArray, ...incomingArray]);
17
+ result[key] = merged.length === 1 ? merged[0] : merged;
18
+ }
19
+ return result;
20
+ }
21
+ merge(parsedExtends) {
22
+ for (const parsedExtend of parsedExtends) {
23
+ const { id, links, tags, metadata } = parsedExtend;
24
+ const existing = this.mergedData.get(id) ?? {
25
+ links: [],
26
+ tags: [],
27
+ metadata: {},
28
+ };
29
+ if (links) {
30
+ existing.links.push(...links);
31
+ }
32
+ if (tags) {
33
+ existing.tags = unique([
34
+ ...existing.tags,
35
+ ...tags,
36
+ ]);
37
+ }
38
+ if (metadata) {
39
+ existing.metadata = this.mergeMetadata(existing.metadata, metadata);
40
+ }
41
+ this.mergedData.set(id, existing);
42
+ }
43
+ }
44
+ applyExtended(el) {
45
+ const extendData = this.mergedData.get(el.id);
46
+ if (!extendData) {
47
+ return el;
48
+ }
49
+ let links = extendData.links;
50
+ if (el.links && el.links.length > 0) {
51
+ links = [
52
+ ...el.links,
53
+ ...links,
54
+ ];
55
+ }
56
+ let tags = extendData.tags;
57
+ if (el.tags && el.tags.length > 0) {
58
+ tags = unique([
59
+ ...el.tags,
60
+ ...tags,
61
+ ]);
62
+ }
63
+ let metadata = extendData.metadata;
64
+ if (el.metadata) {
65
+ metadata = this.mergeMetadata(el.metadata, extendData.metadata);
66
+ }
67
+ return {
68
+ ...el,
69
+ tags: hasAtLeast(tags, 1) ? tags : null,
70
+ links: hasAtLeast(links, 1) ? links : null,
71
+ ...(!isEmpty(metadata) && { metadata }),
72
+ };
73
+ }
74
+ }
@@ -0,0 +1,175 @@
1
+ import { assignTagColors } from '@likec4/core/styles';
2
+ import { exact, FqnRef } from '@likec4/core/types';
3
+ import { isNonEmptyArray, MultiMap, nameFromFqn } from '@likec4/core/utils';
4
+ import { isEmpty, isNonNullish, unique, } from 'remeda';
5
+ import { logger, logWarnError } from '../../logger';
6
+ /**
7
+ * The `MergedSpecification` class is responsible for merging multiple parsed
8
+ * LikeC4Langium documents into a single specification. It consolidates tags,
9
+ * elements, deployments, relationships, and colors from the provided documents
10
+ * and provides methods to convert parsed models into C4 model elements and relations.
11
+ */
12
+ export class MergedSpecification {
13
+ specs = {
14
+ elements: {},
15
+ deployments: {},
16
+ relationships: {},
17
+ colors: {},
18
+ };
19
+ tags;
20
+ globals = {
21
+ predicates: {},
22
+ dynamicPredicates: {},
23
+ styles: {},
24
+ };
25
+ imports = new MultiMap(Set);
26
+ constructor(docs) {
27
+ const tags = {};
28
+ for (const doc of docs) {
29
+ const { c4Specification: spec, c4Globals, c4Imports, } = doc;
30
+ Object.assign(tags, spec.tags);
31
+ Object.assign(this.specs.elements, spec.elements);
32
+ Object.assign(this.specs.relationships, spec.relationships);
33
+ Object.assign(this.specs.colors, spec.colors);
34
+ Object.assign(this.specs.deployments, spec.deployments);
35
+ Object.assign(this.globals.predicates, c4Globals.predicates);
36
+ Object.assign(this.globals.dynamicPredicates, c4Globals.dynamicPredicates);
37
+ Object.assign(this.globals.styles, c4Globals.styles);
38
+ for (const [projectId, fqn] of c4Imports) {
39
+ this.imports.set(projectId, fqn);
40
+ }
41
+ }
42
+ this.tags = assignTagColors(tags);
43
+ }
44
+ /**
45
+ * Converts a parsed model into a C4 model element.
46
+ */
47
+ toModelElement = ({ tags, links, style, id, kind, title, description, technology, summary, metadata, }) => {
48
+ try {
49
+ const __kind = this.specs.elements[kind];
50
+ if (!__kind) {
51
+ logger.warn `No kind '${kind}' found for ${id}`;
52
+ return null;
53
+ }
54
+ technology ??= __kind.technology;
55
+ description ??= __kind.description;
56
+ summary ??= __kind.summary;
57
+ links ??= __kind.links;
58
+ title = title === nameFromFqn(id) && __kind.title ? __kind.title : title;
59
+ if (__kind.tags && isNonEmptyArray(__kind.tags)) {
60
+ tags = tags
61
+ ? unique([
62
+ ...__kind.tags,
63
+ ...tags,
64
+ ])
65
+ : __kind.tags;
66
+ }
67
+ return exact({
68
+ metadata: metadata && !isEmpty(metadata) ? metadata : undefined,
69
+ notation: __kind.notation,
70
+ style: exact({
71
+ ...__kind.style,
72
+ ...style,
73
+ }),
74
+ links,
75
+ tags,
76
+ summary,
77
+ technology,
78
+ description,
79
+ title,
80
+ kind,
81
+ id,
82
+ });
83
+ }
84
+ catch (e) {
85
+ logWarnError(e);
86
+ }
87
+ return null;
88
+ };
89
+ /**
90
+ * Converts a parsed model into a C4 model relation.
91
+ */
92
+ toModelRelation = ({ astPath: _astPath, // omit
93
+ source, target, kind, links, id, ...model }) => {
94
+ if (isNonNullish(kind) && this.specs.relationships[kind]) {
95
+ return {
96
+ ...this.specs.relationships[kind],
97
+ ...model,
98
+ ...(links && { links }),
99
+ source,
100
+ target,
101
+ kind,
102
+ id,
103
+ };
104
+ }
105
+ return {
106
+ ...(links && { links }),
107
+ ...model,
108
+ source,
109
+ target,
110
+ id,
111
+ };
112
+ };
113
+ /**
114
+ * Converts a parsed deployment model into a C4 deployment model
115
+ */
116
+ toDeploymentElement = (parsed) => {
117
+ if ('element' in parsed && !('kind' in parsed)) {
118
+ return {
119
+ ...parsed,
120
+ element: FqnRef.flatten(parsed.element),
121
+ };
122
+ }
123
+ if ('element' in parsed) {
124
+ logger.warn `Invalid ParsedAstDeployment ${parsed.id}, has both element and kind properties`;
125
+ return null;
126
+ }
127
+ try {
128
+ const __kind = this.specs.deployments[parsed.kind];
129
+ if (!__kind) {
130
+ logger.warn `No kind ${parsed.kind} found for ${parsed.id}`;
131
+ return null;
132
+ }
133
+ let { id, style, title, ...rest } = parsed;
134
+ title = title === nameFromFqn(parsed.id) && __kind.title ? __kind.title : title;
135
+ return exact({
136
+ ...__kind,
137
+ ...rest,
138
+ title,
139
+ style: exact({
140
+ ...__kind.style,
141
+ ...style,
142
+ }),
143
+ id,
144
+ });
145
+ }
146
+ catch (e) {
147
+ logWarnError(e);
148
+ }
149
+ return null;
150
+ };
151
+ /**
152
+ * Converts a parsed deployment relation into a C4 deployment relation.
153
+ */
154
+ toDeploymentRelation = ({ astPath: _astPath, // omit
155
+ source, target, kind, links, id, ...model }) => {
156
+ if (isNonNullish(kind) && this.specs.relationships[kind]) {
157
+ return {
158
+ ...this.specs.relationships[kind],
159
+ ...model,
160
+ ...(links && { links }),
161
+ source,
162
+ target,
163
+ kind,
164
+ id,
165
+ };
166
+ }
167
+ return {
168
+ ...(links && { links }),
169
+ ...model,
170
+ source,
171
+ target,
172
+ id,
173
+ };
174
+ };
175
+ }
@@ -0,0 +1,176 @@
1
+ import { isDeploymentNode, isGlobalFqn, } from '@likec4/core';
2
+ import { resolveRulesExtendedViews } from '@likec4/core/compute-view';
3
+ import { computeColorValues } from '@likec4/core/styles';
4
+ import { _stage, _type, exact, FqnRef, isExtendsElementView } from '@likec4/core/types';
5
+ import { compareNatural, parentFqn, sortByFqnHierarchically, } from '@likec4/core/utils';
6
+ import { UriUtils } from 'langium';
7
+ import { filter, flatMap, forEach, indexBy, isDefined, isNullish, isTruthy, keys, map, mapValues, omitBy, pipe, prop, reduce, } from 'remeda';
8
+ import { logger } from '../../logger';
9
+ import { MergedExtends } from './MergedExtends';
10
+ import { MergedSpecification } from './MergedSpecification';
11
+ /**
12
+ * Each document was parsed into a ParsedLikeC4LangiumDocument, where elements
13
+ * do not inherit styles from specification.
14
+ *
15
+ * This function builds a model from all documents, merging the specifications
16
+ * and globals, and applying the extends to the elements.
17
+ */
18
+ export function buildModelData(project, docs) {
19
+ const c4Specification = new MergedSpecification(docs);
20
+ const customColors = mapValues(c4Specification.specs.colors, c => computeColorValues(c.color));
21
+ const metadataKeys = new Set();
22
+ const elementExtends = new MergedExtends();
23
+ const deploymentExtends = new MergedExtends();
24
+ const scanMetadataKeys = (obj) => {
25
+ if (obj?.metadata) {
26
+ keys(obj.metadata).forEach(key => metadataKeys.add(key));
27
+ }
28
+ };
29
+ const elements = pipe(docs, flatMap(d => {
30
+ elementExtends.merge(d.c4ExtendElements);
31
+ return map(d.c4Elements, c4Specification.toModelElement);
32
+ }), filter(isTruthy),
33
+ // sort from root elements to nested, so that parent is always present
34
+ // Import to preserve the order from the source
35
+ sortByFqnHierarchically, reduce((acc, el) => {
36
+ const parent = parentFqn(el.id);
37
+ if (parent && isNullish(acc[parent])) {
38
+ logger.debug `No parent found for ${el.id}`;
39
+ return acc;
40
+ }
41
+ acc[el.id] = elementExtends.applyExtended(el);
42
+ scanMetadataKeys(acc[el.id]);
43
+ return acc;
44
+ }, {}));
45
+ const relations = pipe(docs, flatMap(d => map(d.c4Relations, c4Specification.toModelRelation)), filter((rel) => {
46
+ if (!rel)
47
+ return false;
48
+ const source = FqnRef.flatten(rel.source), target = FqnRef.flatten(rel.target);
49
+ if ((isNullish(elements[source]) && !isGlobalFqn(source)) ||
50
+ (isNullish(elements[target]) && !isGlobalFqn(target))) {
51
+ logger.debug `Invalid relation ${rel.id}
52
+ source: ${source} resolved: ${!!elements[source]}
53
+ target: ${target} resolved: ${!!elements[target]}\n`;
54
+ return false;
55
+ }
56
+ return true;
57
+ }), forEach(scanMetadataKeys), indexBy(prop('id')));
58
+ const deploymentElements = pipe(docs, flatMap(d => {
59
+ deploymentExtends.merge(d.c4ExtendDeployments);
60
+ return map(d.c4Deployments, c4Specification.toDeploymentElement);
61
+ }), filter(isTruthy),
62
+ // sort from root elements to nested, so that parent is always present
63
+ // Import to preserve the order from the source
64
+ sortByFqnHierarchically, reduce((acc, el) => {
65
+ const parent = parentFqn(el.id);
66
+ if (parent && isNullish(acc[parent])) {
67
+ logger.debug `No parent found for deployment element ${el.id}`;
68
+ return acc;
69
+ }
70
+ acc[el.id] = isDeploymentNode(el) ? deploymentExtends.applyExtended(el) : el;
71
+ scanMetadataKeys(acc[el.id]);
72
+ return acc;
73
+ }, {}));
74
+ const deploymentRelations = pipe(docs, flatMap(d => map(d.c4DeploymentRelations, c4Specification.toDeploymentRelation)), filter((rel) => {
75
+ if (!rel)
76
+ return false;
77
+ if (isNullish(deploymentElements[rel.source.deployment]) || isNullish(deploymentElements[rel.target.deployment])) {
78
+ logger.debug `Invalid deployment relation ${rel.id}
79
+ source: ${rel.source.deployment} resolved: ${!!deploymentElements[rel.source.deployment]}
80
+ target: ${rel.target.deployment} resolved: ${!!deploymentElements[rel.target.deployment]}\n`;
81
+ return false;
82
+ }
83
+ return true;
84
+ }), reduce((acc, el) => {
85
+ if (isDefined(acc[el.id])) {
86
+ logger.debug `Duplicate deployment relation ${el.id}`;
87
+ return acc;
88
+ }
89
+ scanMetadataKeys(el);
90
+ acc[el.id] = el;
91
+ return acc;
92
+ }, {}));
93
+ function toC4View(doc) {
94
+ const docUri = doc.uri.toString();
95
+ return (parsedAstView) => {
96
+ let { id, title, description,
97
+ // ignore this property
98
+ astPath: _ignore,
99
+ // model should include discriminant __
100
+ ...model } = parsedAstView;
101
+ if (parsedAstView[_type] === 'element' && isNullish(title) && 'viewOf' in parsedAstView) {
102
+ title = elements[parsedAstView.viewOf]?.title ?? null;
103
+ }
104
+ if (isNullish(title) && id === 'index') {
105
+ title = 'Landscape view';
106
+ }
107
+ return {
108
+ ...omitBy(model, v => v === undefined),
109
+ [_stage]: 'parsed',
110
+ sourcePath: UriUtils.relative(project.folderUri, docUri),
111
+ docUri,
112
+ description,
113
+ title,
114
+ id,
115
+ };
116
+ };
117
+ }
118
+ const parsedViews = docs.flatMap(d => map(d.c4Views, toC4View(d)));
119
+ // Add index view if not present
120
+ if (!parsedViews.some(v => v.id === 'index')) {
121
+ parsedViews.unshift({
122
+ [_stage]: 'parsed',
123
+ [_type]: 'element',
124
+ id: 'index',
125
+ title: 'Landscape view',
126
+ description: null,
127
+ rules: [
128
+ {
129
+ include: [
130
+ {
131
+ wildcard: true,
132
+ },
133
+ ],
134
+ },
135
+ ],
136
+ });
137
+ }
138
+ let views = pipe(parsedViews, indexBy(prop('id')));
139
+ if (parsedViews.some(isExtendsElementView)) {
140
+ views = resolveRulesExtendedViews(views);
141
+ }
142
+ return {
143
+ data: {
144
+ [_stage]: 'parsed',
145
+ projectId: project.id,
146
+ project: exact({
147
+ id: project.id,
148
+ title: project.config.title ?? project.config.name,
149
+ styles: project.config.styles,
150
+ manualLayouts: project.config.manualLayouts,
151
+ }),
152
+ specification: {
153
+ tags: c4Specification.tags,
154
+ elements: c4Specification.specs.elements,
155
+ relationships: mapValues(c4Specification.specs.relationships, ({ notation, technology, ...style }) => ({
156
+ ...(notation && { notation }),
157
+ ...(technology && { technology }),
158
+ style,
159
+ })),
160
+ deployments: c4Specification.specs.deployments,
161
+ ...(metadataKeys.size > 0 && { metadataKeys: [...metadataKeys].sort(compareNatural) }),
162
+ customColors,
163
+ },
164
+ elements,
165
+ relations,
166
+ globals: c4Specification.globals,
167
+ views,
168
+ deployments: {
169
+ elements: deploymentElements,
170
+ relations: deploymentRelations,
171
+ },
172
+ imports: {},
173
+ },
174
+ imports: c4Specification.imports,
175
+ };
176
+ }