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