@likec4/language-server 1.42.1 → 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 (303) 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 +10 -6
  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 +3830 -4172
  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.d.ts +19 -3
  24. package/dist/generated/ast.js +2155 -0
  25. package/dist/generated/grammar.js +7 -0
  26. package/dist/generated/module.d.ts +6 -1
  27. package/dist/generated/module.js +27 -0
  28. package/dist/generated-lib/{icons.mjs → icons.js} +11 -7
  29. package/dist/index.d.ts +7 -0
  30. package/dist/index.js +53 -0
  31. package/dist/{likec4lib.mjs → likec4lib.js} +3 -3
  32. package/dist/logger.js +81 -0
  33. package/dist/lsp/CodeActionProvider.d.ts +14 -0
  34. package/dist/lsp/CodeActionProvider.js +33 -0
  35. package/dist/lsp/CodeLensProvider.js +44 -0
  36. package/dist/lsp/CompletionProvider.d.ts +3 -1
  37. package/dist/lsp/CompletionProvider.js +200 -0
  38. package/dist/lsp/DocumentHighlightProvider.js +10 -0
  39. package/dist/lsp/DocumentLinkProvider.js +58 -0
  40. package/dist/lsp/DocumentSymbolProvider.js +306 -0
  41. package/dist/lsp/HoverProvider.js +106 -0
  42. package/dist/lsp/RenameProvider.js +6 -0
  43. package/dist/lsp/SemanticTokenProvider.d.ts +5 -0
  44. package/dist/lsp/SemanticTokenProvider.js +257 -0
  45. package/dist/lsp/index.d.ts +1 -0
  46. package/dist/lsp/index.js +9 -0
  47. package/dist/mcp/MCPServerFactory.js +73 -0
  48. package/dist/mcp/NoopLikeC4MCPServer.js +17 -0
  49. package/dist/mcp/interfaces.js +5 -0
  50. package/dist/mcp/server/StdioLikeC4MCPServer.js +47 -0
  51. package/dist/mcp/server/StreamableLikeC4MCPServer.js +145 -0
  52. package/dist/mcp/server/WithMCPServer.js +56 -0
  53. package/dist/mcp/tools/_common.d.ts +8 -7
  54. package/dist/mcp/tools/_common.js +49 -0
  55. package/dist/mcp/tools/find-relationships.d.ts +7 -8
  56. package/dist/mcp/tools/find-relationships.js +150 -0
  57. package/dist/mcp/tools/list-projects.d.ts +3 -3
  58. package/dist/mcp/tools/list-projects.js +62 -0
  59. package/dist/mcp/tools/open-view.d.ts +6 -7
  60. package/dist/mcp/tools/open-view.js +52 -0
  61. package/dist/mcp/tools/read-deployment.d.ts +6 -7
  62. package/dist/mcp/tools/read-deployment.js +132 -0
  63. package/dist/mcp/tools/read-element.d.ts +6 -7
  64. package/dist/mcp/tools/read-element.js +194 -0
  65. package/dist/mcp/tools/read-project-summary.d.ts +5 -6
  66. package/dist/mcp/tools/read-project-summary.js +176 -0
  67. package/dist/mcp/tools/read-view.d.ts +6 -7
  68. package/dist/mcp/tools/read-view.js +203 -0
  69. package/dist/mcp/tools/search-element.d.ts +3 -3
  70. package/dist/mcp/tools/search-element.js +177 -0
  71. package/dist/mcp/utils.d.ts +2 -2
  72. package/dist/mcp/utils.js +48 -0
  73. package/dist/model/builder/MergedExtends.d.ts +2 -1
  74. package/dist/model/builder/MergedExtends.js +74 -0
  75. package/dist/model/builder/MergedSpecification.js +175 -0
  76. package/dist/model/builder/buildModel.js +176 -0
  77. package/dist/model/deployments-index.js +102 -0
  78. package/dist/model/fqn-index.js +250 -0
  79. package/dist/model/index.js +6 -0
  80. package/dist/model/model-builder.d.ts +13 -11
  81. package/dist/model/model-builder.js +234 -0
  82. package/dist/model/model-locator.d.ts +6 -5
  83. package/dist/model/model-locator.js +240 -0
  84. package/dist/model/model-parser-where.js +81 -0
  85. package/dist/model/model-parser.d.ts +318 -313
  86. package/dist/model/model-parser.js +119 -0
  87. package/dist/model/parser/Base.d.ts +3 -3
  88. package/dist/model/parser/Base.js +367 -0
  89. package/dist/model/parser/DeploymentModelParser.d.ts +3 -3
  90. package/dist/model/parser/DeploymentModelParser.js +176 -0
  91. package/dist/model/parser/DeploymentViewParser.d.ts +4 -4
  92. package/dist/model/parser/DeploymentViewParser.js +86 -0
  93. package/dist/model/parser/FqnRefParser.d.ts +3 -3
  94. package/dist/model/parser/FqnRefParser.js +382 -0
  95. package/dist/model/parser/GlobalsParser.d.ts +7 -7
  96. package/dist/model/parser/GlobalsParser.js +84 -0
  97. package/dist/model/parser/ImportsParser.d.ts +12 -13
  98. package/dist/model/parser/ImportsParser.js +24 -0
  99. package/dist/model/parser/ModelParser.d.ts +3 -3
  100. package/dist/model/parser/ModelParser.js +165 -0
  101. package/dist/model/parser/PredicatesParser.d.ts +3 -3
  102. package/dist/model/parser/PredicatesParser.js +45 -0
  103. package/dist/model/parser/SpecificationParser.d.ts +3 -3
  104. package/dist/model/parser/SpecificationParser.js +109 -0
  105. package/dist/model/parser/ValueConverter.js +12 -0
  106. package/dist/model/parser/ViewsParser.d.ts +4 -4
  107. package/dist/model/parser/ViewsParser.js +477 -0
  108. package/dist/model-change/ModelChanges.d.ts +6 -3
  109. package/dist/model-change/ModelChanges.js +102 -0
  110. package/dist/model-change/changeElementStyle.js +134 -0
  111. package/dist/model-change/changeViewLayout.d.ts +2 -2
  112. package/dist/model-change/changeViewLayout.js +28 -0
  113. package/dist/model-change/removeManualLayoutV1.d.ts +7 -0
  114. package/dist/model-change/removeManualLayoutV1.js +27 -0
  115. package/dist/module.d.ts +10 -5
  116. package/dist/module.js +143 -0
  117. package/dist/protocol.d.ts +1 -17
  118. package/dist/protocol.js +114 -0
  119. package/dist/references/index.js +3 -0
  120. package/dist/references/name-provider.js +37 -0
  121. package/dist/references/scope-computation.js +288 -0
  122. package/dist/references/scope-provider.d.ts +3 -3
  123. package/dist/references/scope-provider.js +242 -0
  124. package/dist/shared/NodeKindProvider.js +57 -0
  125. package/dist/shared/{WorkspaceSymbolProvider.mjs → WorkspaceSymbolProvider.js} +1 -1
  126. package/dist/shared/index.js +2 -0
  127. package/dist/test/index.js +1 -0
  128. package/dist/test/testServices.d.ts +16 -16
  129. package/dist/test/testServices.js +210 -0
  130. package/dist/utils/disposable.js +26 -0
  131. package/dist/utils/elementRef.d.ts +1 -1
  132. package/dist/utils/elementRef.js +27 -0
  133. package/dist/utils/fqnRef.js +63 -0
  134. package/dist/utils/index.js +35 -0
  135. package/dist/utils/printDocs.js +1 -0
  136. package/dist/utils/projectId.js +16 -0
  137. package/dist/utils/stringHash.js +5 -0
  138. package/dist/validation/DocumentValidator.js +17 -0
  139. package/dist/validation/_shared.js +26 -0
  140. package/dist/validation/deployment-checks.js +140 -0
  141. package/dist/validation/dynamic-view.js +67 -0
  142. package/dist/validation/element-ref.js +12 -0
  143. package/dist/validation/element.js +49 -0
  144. package/dist/validation/imports.js +46 -0
  145. package/dist/validation/index.d.ts +1 -1
  146. package/dist/validation/index.js +157 -0
  147. package/dist/validation/property-checks.js +108 -0
  148. package/dist/validation/relation.js +55 -0
  149. package/dist/validation/specification.js +190 -0
  150. package/dist/validation/view-predicates/fqn-expr-with.js +43 -0
  151. package/dist/validation/view-predicates/fqn-ref-expr.js +51 -0
  152. package/dist/validation/view-predicates/incoming.js +16 -0
  153. package/dist/validation/view-predicates/index.js +6 -0
  154. package/dist/validation/view-predicates/outgoing.js +20 -0
  155. package/dist/validation/view-predicates/relation-expr.js +46 -0
  156. package/dist/validation/view-predicates/relation-with.js +16 -0
  157. package/dist/validation/view.d.ts +1 -1
  158. package/dist/validation/view.js +42 -0
  159. package/dist/view-utils/assignNavigateTo.js +27 -0
  160. package/dist/view-utils/index.d.ts +1 -0
  161. package/dist/view-utils/index.js +2 -0
  162. package/dist/view-utils/manual-layout.d.ts +6 -0
  163. package/dist/view-utils/manual-layout.js +151 -0
  164. package/dist/views/ConfigurableLayouter.js +51 -0
  165. package/dist/views/LikeC4ManualLayouts.d.ts +28 -0
  166. package/dist/views/LikeC4ManualLayouts.js +132 -0
  167. package/dist/views/{likec4-views.d.ts → LikeC4Views.d.ts} +9 -8
  168. package/dist/views/LikeC4Views.js +200 -0
  169. package/dist/views/index.d.ts +4 -1
  170. package/dist/views/index.js +11 -0
  171. package/dist/workspace/AstNodeDescriptionProvider.js +15 -0
  172. package/dist/workspace/IndexManager.js +21 -0
  173. package/dist/workspace/LangiumDocuments.d.ts +1 -1
  174. package/dist/workspace/LangiumDocuments.js +58 -0
  175. package/dist/workspace/ProjectsManager.d.ts +8 -3
  176. package/dist/workspace/ProjectsManager.js +373 -0
  177. package/dist/workspace/WorkspaceManager.d.ts +3 -2
  178. package/dist/workspace/WorkspaceManager.js +93 -0
  179. package/dist/workspace/index.js +5 -0
  180. package/likec4lib/package.json +1 -1
  181. package/package.json +32 -31
  182. package/protocol/package.json +1 -1
  183. package/dist/LikeC4LanguageServices.mjs +0 -197
  184. package/dist/Rpc.mjs +0 -296
  185. package/dist/ast.mjs +0 -221
  186. package/dist/browser-worker.mjs +0 -2
  187. package/dist/browser.mjs +0 -32
  188. package/dist/documentation/documentation-provider.mjs +0 -48
  189. package/dist/documentation/index.mjs +0 -1
  190. package/dist/empty.mjs +0 -1
  191. package/dist/filesystem/ChokidarWatcher.mjs +0 -68
  192. package/dist/filesystem/FileSystemWatcher.mjs +0 -11
  193. package/dist/filesystem/LikeC4FileSystem.mjs +0 -64
  194. package/dist/filesystem/index.mjs +0 -19
  195. package/dist/formatting/LikeC4Formatter.mjs +0 -511
  196. package/dist/formatting/utils.mjs +0 -15
  197. package/dist/generated/ast.mjs +0 -2118
  198. package/dist/generated/grammar.mjs +0 -3
  199. package/dist/generated/module.mjs +0 -23
  200. package/dist/index.mjs +0 -50
  201. package/dist/logger.mjs +0 -82
  202. package/dist/lsp/CodeLensProvider.mjs +0 -42
  203. package/dist/lsp/CompletionProvider.mjs +0 -208
  204. package/dist/lsp/DocumentHighlightProvider.mjs +0 -10
  205. package/dist/lsp/DocumentLinkProvider.mjs +0 -53
  206. package/dist/lsp/DocumentSymbolProvider.mjs +0 -287
  207. package/dist/lsp/HoverProvider.mjs +0 -104
  208. package/dist/lsp/RenameProvider.mjs +0 -6
  209. package/dist/lsp/SemanticTokenProvider.mjs +0 -350
  210. package/dist/lsp/index.mjs +0 -7
  211. package/dist/mcp/MCPServerFactory.mjs +0 -70
  212. package/dist/mcp/NoopLikeC4MCPServer.mjs +0 -17
  213. package/dist/mcp/interfaces.mjs +0 -4
  214. package/dist/mcp/server/StdioLikeC4MCPServer.mjs +0 -46
  215. package/dist/mcp/server/StreamableLikeC4MCPServer.mjs +0 -153
  216. package/dist/mcp/server/WithMCPServer.mjs +0 -58
  217. package/dist/mcp/tools/_common.mjs +0 -42
  218. package/dist/mcp/tools/find-relationships.mjs +0 -151
  219. package/dist/mcp/tools/list-projects.mjs +0 -62
  220. package/dist/mcp/tools/open-view.mjs +0 -52
  221. package/dist/mcp/tools/read-deployment.mjs +0 -130
  222. package/dist/mcp/tools/read-element.mjs +0 -198
  223. package/dist/mcp/tools/read-project-summary.mjs +0 -178
  224. package/dist/mcp/tools/read-view.mjs +0 -205
  225. package/dist/mcp/tools/search-element.mjs +0 -171
  226. package/dist/mcp/utils.mjs +0 -47
  227. package/dist/model/builder/MergedExtends.mjs +0 -67
  228. package/dist/model/builder/MergedSpecification.mjs +0 -205
  229. package/dist/model/builder/assignTagColors.d.ts +0 -7
  230. package/dist/model/builder/assignTagColors.mjs +0 -51
  231. package/dist/model/builder/buildModel.mjs +0 -226
  232. package/dist/model/deployments-index.mjs +0 -100
  233. package/dist/model/fqn-index.mjs +0 -243
  234. package/dist/model/index.mjs +0 -6
  235. package/dist/model/model-builder.mjs +0 -285
  236. package/dist/model/model-locator.mjs +0 -239
  237. package/dist/model/model-parser-where.mjs +0 -81
  238. package/dist/model/model-parser.mjs +0 -127
  239. package/dist/model/parser/Base.mjs +0 -342
  240. package/dist/model/parser/DeploymentModelParser.mjs +0 -212
  241. package/dist/model/parser/DeploymentViewParser.mjs +0 -95
  242. package/dist/model/parser/FqnRefParser.mjs +0 -398
  243. package/dist/model/parser/GlobalsParser.mjs +0 -82
  244. package/dist/model/parser/ImportsParser.mjs +0 -28
  245. package/dist/model/parser/ModelParser.mjs +0 -190
  246. package/dist/model/parser/PredicatesParser.mjs +0 -45
  247. package/dist/model/parser/SpecificationParser.mjs +0 -120
  248. package/dist/model/parser/ValueConverter.mjs +0 -12
  249. package/dist/model/parser/ViewsParser.mjs +0 -490
  250. package/dist/model-change/ModelChanges.mjs +0 -89
  251. package/dist/model-change/changeElementStyle.mjs +0 -143
  252. package/dist/model-change/changeViewLayout.mjs +0 -32
  253. package/dist/model-change/saveManualLayout.d.ts +0 -11
  254. package/dist/model-change/saveManualLayout.mjs +0 -27
  255. package/dist/module.mjs +0 -180
  256. package/dist/protocol.mjs +0 -65
  257. package/dist/references/index.mjs +0 -3
  258. package/dist/references/name-provider.mjs +0 -39
  259. package/dist/references/scope-computation.mjs +0 -312
  260. package/dist/references/scope-provider.mjs +0 -239
  261. package/dist/shared/NodeKindProvider.mjs +0 -110
  262. package/dist/shared/index.mjs +0 -2
  263. package/dist/test/index.mjs +0 -1
  264. package/dist/test/testServices.mjs +0 -200
  265. package/dist/utils/disposable.mjs +0 -25
  266. package/dist/utils/elementRef.mjs +0 -20
  267. package/dist/utils/fqnRef.mjs +0 -57
  268. package/dist/utils/index.mjs +0 -33
  269. package/dist/utils/printDocs.mjs +0 -1
  270. package/dist/utils/projectId.mjs +0 -16
  271. package/dist/utils/stringHash.mjs +0 -5
  272. package/dist/validation/DocumentValidator.mjs +0 -16
  273. package/dist/validation/_shared.mjs +0 -25
  274. package/dist/validation/deployment-checks.mjs +0 -146
  275. package/dist/validation/dynamic-view.mjs +0 -67
  276. package/dist/validation/element-ref.mjs +0 -12
  277. package/dist/validation/element.mjs +0 -50
  278. package/dist/validation/imports.mjs +0 -25
  279. package/dist/validation/index.mjs +0 -180
  280. package/dist/validation/property-checks.mjs +0 -107
  281. package/dist/validation/relation.mjs +0 -53
  282. package/dist/validation/specification.mjs +0 -173
  283. package/dist/validation/view-predicates/fqn-expr-with.mjs +0 -43
  284. package/dist/validation/view-predicates/fqn-ref-expr.mjs +0 -53
  285. package/dist/validation/view-predicates/incoming.mjs +0 -16
  286. package/dist/validation/view-predicates/index.mjs +0 -6
  287. package/dist/validation/view-predicates/outgoing.mjs +0 -20
  288. package/dist/validation/view-predicates/relation-expr.mjs +0 -39
  289. package/dist/validation/view-predicates/relation-with.mjs +0 -16
  290. package/dist/validation/view.mjs +0 -25
  291. package/dist/view-utils/assignNavigateTo.mjs +0 -25
  292. package/dist/view-utils/index.mjs +0 -1
  293. package/dist/view-utils/manual-layout.mjs +0 -99
  294. package/dist/views/configurable-layouter.mjs +0 -51
  295. package/dist/views/index.mjs +0 -1
  296. package/dist/views/likec4-views.mjs +0 -166
  297. package/dist/workspace/AstNodeDescriptionProvider.mjs +0 -17
  298. package/dist/workspace/IndexManager.mjs +0 -17
  299. package/dist/workspace/LangiumDocuments.mjs +0 -53
  300. package/dist/workspace/ProjectsManager.mjs +0 -360
  301. package/dist/workspace/WorkspaceManager.mjs +0 -83
  302. package/dist/workspace/index.mjs +0 -5
  303. /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
+ }
@@ -2,11 +2,12 @@ import type * as c4 from '@likec4/core';
2
2
  import type { ParsedAstExtend } from '../../ast';
3
3
  export declare class MergedExtends {
4
4
  private mergedData;
5
+ private mergeMetadata;
5
6
  merge(parsedExtends: ParsedAstExtend[]): void;
6
7
  applyExtended<E extends {
7
8
  id: string;
8
9
  tags?: readonly string[] | null;
9
10
  links?: readonly c4.Link[] | null;
10
- metadata?: Record<string, string>;
11
+ metadata?: Record<string, string | string[]>;
11
12
  }>(el: E): E;
12
13
  }
@@ -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
+ }