@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,119 @@
1
+ import { DefaultWeakMap, invariant, MultiMap } from '@likec4/core/utils';
2
+ import { loggable } from '@likec4/log';
3
+ import { DocumentState, UriUtils } from 'langium';
4
+ import { pipe } from 'remeda';
5
+ import { DiagnosticSeverity } from 'vscode-languageserver-types';
6
+ import { isLikeC4LangiumDocument } from '../ast';
7
+ import { isLikeC4Builtin } from '../likec4lib';
8
+ import { logger as rootLogger } from '../logger';
9
+ import { BaseParser } from './parser/Base';
10
+ import { DeploymentModelParser } from './parser/DeploymentModelParser';
11
+ import { DeploymentViewParser } from './parser/DeploymentViewParser';
12
+ import { ExpressionV2Parser } from './parser/FqnRefParser';
13
+ import { GlobalsParser } from './parser/GlobalsParser';
14
+ import { ImportsParser } from './parser/ImportsParser';
15
+ import { ModelParser } from './parser/ModelParser';
16
+ import { PredicatesParser } from './parser/PredicatesParser';
17
+ import { SpecificationParser } from './parser/SpecificationParser';
18
+ import { ViewsParser } from './parser/ViewsParser';
19
+ const DocumentParserFromMixins = pipe(BaseParser, ExpressionV2Parser, ImportsParser, ModelParser, DeploymentModelParser, DeploymentViewParser, PredicatesParser, SpecificationParser, ViewsParser, GlobalsParser);
20
+ export class DocumentParser extends DocumentParserFromMixins {
21
+ }
22
+ const logger = rootLogger.getChild('ModelParser');
23
+ export class LikeC4ModelParser {
24
+ services;
25
+ cachedParsers = new DefaultWeakMap((doc) => this.createParser(doc));
26
+ constructor(services) {
27
+ this.services = services;
28
+ services.shared.workspace.DocumentBuilder.onDocumentPhase(DocumentState.Linked, doc => {
29
+ try {
30
+ if (services.shared.workspace.ProjectsManager.isExcluded(doc)) {
31
+ return;
32
+ }
33
+ if (!isLikeC4Builtin(doc.uri)) {
34
+ this.cachedParsers.set(doc, this.createParser(doc));
35
+ }
36
+ }
37
+ catch (e) {
38
+ logger.warn(loggable(e));
39
+ }
40
+ });
41
+ // We need to clean up cached parser when document is validated and has errors
42
+ // Because after that parser takes into account validation results
43
+ services.shared.workspace.DocumentBuilder.onDocumentPhase(DocumentState.Validated, doc => {
44
+ if (doc.diagnostics?.some(d => d.severity === DiagnosticSeverity.Error) && this.cachedParsers.has(doc)) {
45
+ logger.debug('clear cached parser {projectId} document {doc}', {
46
+ projectId: doc.likec4ProjectId,
47
+ doc: UriUtils.basename(doc.uri),
48
+ });
49
+ this.cachedParsers.delete(doc);
50
+ }
51
+ });
52
+ }
53
+ documents(projectId) {
54
+ return this.services.shared.workspace.LangiumDocuments.projectDocuments(projectId).map(d => this.parse(d));
55
+ }
56
+ parse(doc) {
57
+ try {
58
+ const parser = this.forDocument(doc);
59
+ return parser.doc;
60
+ }
61
+ catch (cause) {
62
+ throw new Error(`Error parsing document ${doc.uri.toString()}`, { cause });
63
+ }
64
+ }
65
+ forDocument(doc) {
66
+ if (doc.state < DocumentState.Linked) {
67
+ logger.warn(`Document {doc} is not linked`, { doc: doc.uri.toString() });
68
+ }
69
+ return this.cachedParsers.get(doc);
70
+ }
71
+ createParser(doc) {
72
+ invariant(isLikeC4LangiumDocument(doc), `Document ${doc.uri.toString()} is not a LikeC4 document`);
73
+ if (doc.likec4ProjectId) {
74
+ logger.debug(`create parser {projectId} document {doc}`, {
75
+ projectId: doc.likec4ProjectId,
76
+ doc: UriUtils.basename(doc.uri),
77
+ });
78
+ }
79
+ else {
80
+ logger.warn(`create parser for document without project {doc}`, { doc: doc.uri.fsPath });
81
+ }
82
+ const props = {
83
+ c4Specification: {
84
+ tags: {},
85
+ elements: {},
86
+ relationships: {},
87
+ colors: {},
88
+ deployments: {},
89
+ },
90
+ c4Elements: [],
91
+ c4ExtendElements: [],
92
+ c4ExtendDeployments: [],
93
+ c4Relations: [],
94
+ c4Deployments: [],
95
+ c4DeploymentRelations: [],
96
+ c4Globals: {
97
+ predicates: {},
98
+ dynamicPredicates: {},
99
+ styles: {},
100
+ },
101
+ c4Views: [],
102
+ c4Imports: new MultiMap(Set),
103
+ };
104
+ doc = Object.assign(doc, props);
105
+ const parser = new DocumentParser(this.services, doc);
106
+ try {
107
+ parser.parseSpecification();
108
+ parser.parseImports();
109
+ parser.parseModel();
110
+ parser.parseGlobals();
111
+ parser.parseDeployment();
112
+ parser.parseViews();
113
+ }
114
+ catch (e) {
115
+ logger.error(`Error parsing document {doc}`, { doc: doc.uri.toString(), error: e });
116
+ }
117
+ return parser;
118
+ }
119
+ }
@@ -25,14 +25,14 @@ export declare class BaseParser {
25
25
  constructor(services: LikeC4Services, doc: ParsedLikeC4LangiumDocument);
26
26
  get project(): Project;
27
27
  resolveFqn(node: ast.FqnReferenceable): c4.Fqn;
28
- getAstNodePath(node: AstNode): any;
28
+ getAstNodePath(node: AstNode): string;
29
29
  getMetadata(metadataAstNode: ast.MetadataProperty | undefined): {
30
30
  [key: string]: string | string[];
31
31
  } | undefined;
32
32
  parseMarkdownOrString(markdownOrString: ast.MarkdownOrString | undefined): c4.MarkdownOrString | undefined;
33
33
  convertTags<E extends {
34
34
  tags?: ast.Tags;
35
- }>(withTags?: E): any;
35
+ }>(withTags?: E): c4.NonEmptyArray<c4.Tag<string>> | null;
36
36
  parseTags<E extends {
37
37
  tags?: ast.Tags;
38
38
  }>(withTags?: E): c4.NonEmptyArray<c4.Tag> | null;
@@ -0,0 +1,367 @@
1
+ import { exact, GlobalFqn, isNonEmptyArray, memoizeProp, nonexhaustive, nonNullable, } from '@likec4/core';
2
+ import { filter, flatMap, groupBy, isArray, isBoolean, isEmpty, isNumber, isString, isTruthy, mapValues, pipe, unique, } from 'remeda';
3
+ import { dedent } from 'strip-indent';
4
+ import { hasLeadingSlash, hasProtocol, isRelative, joinRelativeURL, joinURL } from 'ufo';
5
+ import { ast, parseAstOpacityProperty, parseAstPercent, parseAstSizeValue, parseMarkdownAsString, toColor, } from '../../ast';
6
+ import { serverLogger } from '../../logger';
7
+ import { projectIdFrom } from '../../utils';
8
+ import { readStrictFqn } from '../../utils/elementRef';
9
+ import { checksFromDiagnostics } from '../../validation';
10
+ const logger = serverLogger.getChild('BaseParser');
11
+ export function toSingleLine(str) {
12
+ if (str === null || str === undefined) {
13
+ return undefined;
14
+ }
15
+ const without = removeIndent(str);
16
+ if (isString(without)) {
17
+ return without.split('\n').join(' ');
18
+ }
19
+ if ('md' in without) {
20
+ return {
21
+ md: without.md.split('\n').join(' '),
22
+ };
23
+ }
24
+ return {
25
+ txt: without.txt.split('\n').join(' '),
26
+ };
27
+ }
28
+ // export function removeIndent(str: ast.MarkdownOrString | string): MarkdownOrString | string
29
+ export function removeIndent(str) {
30
+ if (str === null || str === undefined) {
31
+ return undefined;
32
+ }
33
+ switch (true) {
34
+ case isString(str):
35
+ return dedent(str).trim();
36
+ case ast.isMarkdownOrString(str) && isTruthy(str.markdown):
37
+ return {
38
+ md: dedent(str.markdown).trim(),
39
+ };
40
+ case ast.isMarkdownOrString(str) && isTruthy(str.text):
41
+ return {
42
+ txt: dedent(str.text).trim(),
43
+ };
44
+ default:
45
+ return undefined;
46
+ }
47
+ }
48
+ export class BaseParser {
49
+ services;
50
+ doc;
51
+ isValid;
52
+ constructor(services, doc) {
53
+ this.services = services;
54
+ this.doc = doc;
55
+ // do nothing
56
+ this.isValid = checksFromDiagnostics(doc).isValid;
57
+ }
58
+ get project() {
59
+ return memoizeProp(this, 'project', () => this.services.shared.workspace.ProjectsManager.getProject(this.doc));
60
+ }
61
+ resolveFqn(node) {
62
+ if (ast.isImported(node)) {
63
+ const project = projectIdFrom(node);
64
+ const fqn = this.resolveFqn(nonNullable(node.imported.ref, `FqnRef is empty of imported: ${node.$cstNode?.text}`));
65
+ this.doc.c4Imports.set(project, fqn);
66
+ return GlobalFqn(project, fqn);
67
+ }
68
+ if (ast.isExtendElement(node)) {
69
+ return readStrictFqn(node.element);
70
+ }
71
+ if (ast.isExtendDeployment(node)) {
72
+ return readStrictFqn(node.deploymentNode);
73
+ }
74
+ if (ast.isDeploymentElement(node)) {
75
+ return this.services.likec4.DeploymentsIndex.getFqn(node);
76
+ }
77
+ return this.services.likec4.FqnIndex.getFqn(node);
78
+ }
79
+ getAstNodePath(node) {
80
+ return this.services.workspace.AstNodeLocator.getAstNodePath(node);
81
+ }
82
+ getMetadata(metadataAstNode) {
83
+ if (!metadataAstNode || !this.isValid(metadataAstNode) || isEmpty(metadataAstNode.props)) {
84
+ return undefined;
85
+ }
86
+ // Helper function to extract string values from MetadataValue
87
+ const extractValues = (value) => {
88
+ if (ast.isMarkdownOrString(value)) {
89
+ const mdOrStr = removeIndent(value);
90
+ if (!mdOrStr)
91
+ return [];
92
+ // Handle both string and MarkdownOrString types
93
+ if (typeof mdOrStr === 'string') {
94
+ return isTruthy(mdOrStr) ? [mdOrStr] : [];
95
+ }
96
+ else {
97
+ // MarkdownOrString object with txt or md property
98
+ const strValue = mdOrStr.md || mdOrStr.txt;
99
+ return isTruthy(strValue) ? [strValue] : [];
100
+ }
101
+ }
102
+ else if (ast.isMetadataArray(value)) {
103
+ return value.values
104
+ .map(v => removeIndent(v))
105
+ .map(v => {
106
+ if (typeof v === 'string') {
107
+ return v;
108
+ }
109
+ else {
110
+ // MarkdownOrString object
111
+ return v.md || v.txt;
112
+ }
113
+ })
114
+ .filter(isTruthy);
115
+ }
116
+ return [];
117
+ };
118
+ // Transform metadata attributes into key-value pairs
119
+ const keyValuePairs = pipe(metadataAstNode.props, flatMap(p => extractValues(p.value).map(v => [p.key, v])), filter(([_, value]) => isTruthy(value)));
120
+ if (isEmpty(keyValuePairs)) {
121
+ return undefined;
122
+ }
123
+ // Group by key to handle duplicate keys
124
+ const groupedData = pipe(keyValuePairs, groupBy(([key]) => key), mapValues(pairs => pairs.map(([_, value]) => value)));
125
+ // Convert to final format: single values as string, multiple values as string[]
126
+ const data = {};
127
+ for (const [key, values] of Object.entries(groupedData)) {
128
+ if (values && values.length > 0) {
129
+ data[key] = values.length === 1 ? values[0] : values;
130
+ }
131
+ }
132
+ return isEmpty(data) ? undefined : data;
133
+ }
134
+ parseMarkdownOrString(markdownOrString) {
135
+ if (ast.isMarkdownOrString(markdownOrString)) {
136
+ return removeIndent(markdownOrString);
137
+ }
138
+ return undefined;
139
+ }
140
+ convertTags(withTags) {
141
+ return this.parseTags(withTags);
142
+ }
143
+ parseTags(withTags) {
144
+ let iter = withTags?.tags;
145
+ if (!iter) {
146
+ return null;
147
+ }
148
+ let tags = [];
149
+ while (iter) {
150
+ try {
151
+ if (this.isValid(iter)) {
152
+ const values = iter.values.map(t => t.tag.ref?.name).filter(isTruthy);
153
+ if (values.length > 0) {
154
+ tags.push(...values);
155
+ }
156
+ }
157
+ }
158
+ catch {
159
+ // ignore
160
+ }
161
+ iter = iter.prev;
162
+ }
163
+ return isNonEmptyArray(tags) ? unique(tags) : null;
164
+ }
165
+ convertLinks(source) {
166
+ return this.parseLinks(source);
167
+ }
168
+ parseLinks(source) {
169
+ if (!source?.props || source.props.length === 0) {
170
+ return undefined;
171
+ }
172
+ return pipe(source.props, filter(ast.isLinkProperty), flatMap(p => {
173
+ if (!this.isValid(p)) {
174
+ return [];
175
+ }
176
+ const url = p.value;
177
+ if (isTruthy(url)) {
178
+ const title = isTruthy(p.title) ? toSingleLine(p.title) : undefined;
179
+ const relative = this.services.lsp.DocumentLinkProvider.relativeLink(this.doc, url);
180
+ return {
181
+ url,
182
+ ...(title && { title }),
183
+ ...(relative && relative !== url && { relative }),
184
+ };
185
+ }
186
+ return [];
187
+ }));
188
+ }
189
+ parseIconProperty(prop) {
190
+ if (!prop || !this.isValid(prop)) {
191
+ return undefined;
192
+ }
193
+ const { libicon, value } = prop;
194
+ switch (true) {
195
+ case !!libicon: {
196
+ return nonNullable(libicon.ref, `Library icon ${libicon.$refText} has empty ref`).name;
197
+ }
198
+ case value && value === 'none': {
199
+ return value;
200
+ }
201
+ case value && hasProtocol(value): {
202
+ if (value.startsWith('file:')) {
203
+ logger.warn(`Icon property '${value}' used the 'file' protocol which is not supported`);
204
+ return undefined;
205
+ }
206
+ return value;
207
+ }
208
+ case value && value.startsWith('@'): {
209
+ return this.parseImageAlias(value);
210
+ }
211
+ case value && isRelative(value): {
212
+ return joinRelativeURL(this.doc.uri.toString(), '../', value);
213
+ }
214
+ case value && hasLeadingSlash(value): {
215
+ return joinURL(this.project.folderUri.toString(), value);
216
+ }
217
+ default: {
218
+ logger.warn(`Icon property '${value}' is not a valid URL, library icon, image alias or 'none'`);
219
+ return undefined;
220
+ }
221
+ }
222
+ }
223
+ parseImageAlias(value) {
224
+ // Extract the alias name (e.g., '@infra' from '@infra/backend.svg')
225
+ const slashIndex = value.indexOf('/');
226
+ const aliasName = slashIndex > 0 ? value.substring(0, slashIndex) : value;
227
+ const remainingPath = slashIndex > 0 ? value.substring(slashIndex + 1) : '';
228
+ // Get imageAliases from project config, or use default '@' -> './images' mapping
229
+ const imageAliases = { '@': './images', ...this.project.config.imageAliases };
230
+ // Look up the alias path
231
+ const aliasPath = imageAliases[aliasName];
232
+ if (!aliasPath) {
233
+ logger.warn(`Image alias "${aliasName}" not found in project configuration`);
234
+ return undefined;
235
+ }
236
+ // Combine the alias path with the remaining path
237
+ const fullPath = remainingPath ? joinURL(aliasPath, remainingPath) : aliasPath;
238
+ // Make it relative to the **project root**, not the current document.
239
+ return joinURL(this.project.folderUri.toString(), fullPath);
240
+ }
241
+ parseColorLiteral(astNode) {
242
+ if (!this.isValid(astNode)) {
243
+ return undefined;
244
+ }
245
+ if (ast.isHexColor(astNode)) {
246
+ return `#${astNode.hex}`;
247
+ }
248
+ if (ast.isRGBAColor(astNode)) {
249
+ let alpha = isNumber(astNode.alpha) ? astNode.alpha : undefined;
250
+ if (isString(astNode.alpha)) {
251
+ alpha = parseAstPercent(astNode.alpha) / 100;
252
+ }
253
+ if (alpha !== undefined) {
254
+ return `rgba(${astNode.red},${astNode.green},${astNode.blue},${alpha})`;
255
+ }
256
+ return `rgb(${astNode.red},${astNode.green},${astNode.blue})`;
257
+ }
258
+ nonexhaustive(astNode);
259
+ }
260
+ parseElementStyle(elementProps) {
261
+ if (!elementProps) {
262
+ return {};
263
+ }
264
+ if (isArray(elementProps)) {
265
+ const style = this.parseStyleProps(elementProps.find(ast.isElementStyleProperty)?.props);
266
+ // Property on element has higher priority than from style
267
+ const iconProp = this.parseIconProperty(elementProps.find(ast.isIconProperty));
268
+ if (iconProp) {
269
+ style.icon = iconProp;
270
+ }
271
+ return style;
272
+ }
273
+ return this.parseStyleProps(elementProps.props);
274
+ }
275
+ parseStyleProps(styleProps) {
276
+ const result = {};
277
+ if (!styleProps || styleProps.length === 0) {
278
+ return result;
279
+ }
280
+ for (const prop of styleProps) {
281
+ if (!this.isValid(prop)) {
282
+ continue;
283
+ }
284
+ switch (true) {
285
+ case ast.isBorderProperty(prop): {
286
+ if (isTruthy(prop.value)) {
287
+ result.border = prop.value;
288
+ }
289
+ break;
290
+ }
291
+ case ast.isColorProperty(prop): {
292
+ const color = toColor(prop);
293
+ if (isTruthy(color)) {
294
+ result.color = color;
295
+ }
296
+ break;
297
+ }
298
+ case ast.isShapeProperty(prop): {
299
+ if (isTruthy(prop.value)) {
300
+ result.shape = prop.value;
301
+ }
302
+ break;
303
+ }
304
+ case ast.isIconProperty(prop): {
305
+ const icon = this.parseIconProperty(prop);
306
+ if (isTruthy(icon)) {
307
+ result.icon = icon;
308
+ }
309
+ break;
310
+ }
311
+ case ast.isOpacityProperty(prop): {
312
+ result.opacity = parseAstOpacityProperty(prop);
313
+ break;
314
+ }
315
+ case ast.isMultipleProperty(prop): {
316
+ result.multiple = isBoolean(prop.value) ? prop.value : false;
317
+ break;
318
+ }
319
+ case ast.isShapeSizeProperty(prop): {
320
+ if (isTruthy(prop.value)) {
321
+ result.size = parseAstSizeValue(prop);
322
+ }
323
+ break;
324
+ }
325
+ case ast.isPaddingSizeProperty(prop): {
326
+ if (isTruthy(prop.value)) {
327
+ result.padding = parseAstSizeValue(prop);
328
+ }
329
+ break;
330
+ }
331
+ case ast.isTextSizeProperty(prop): {
332
+ if (isTruthy(prop.value)) {
333
+ result.textSize = parseAstSizeValue(prop);
334
+ }
335
+ break;
336
+ }
337
+ default:
338
+ nonexhaustive(prop);
339
+ }
340
+ }
341
+ return exact(result);
342
+ }
343
+ /**
344
+ * Parse base properties: title, description and technology
345
+ *
346
+ * @param props - body properties (inside '{...}')
347
+ * @param override - optional, inline properties (right on the node)
348
+ * have higher priority and override body properties
349
+ */
350
+ parseBaseProps(props, override) {
351
+ const title = removeIndent(override?.title ?? parseMarkdownAsString(props.title));
352
+ const description = override?.description
353
+ ? { txt: removeIndent(override.description) }
354
+ : this.parseMarkdownOrString(props.description);
355
+ const summary = override?.summary
356
+ ? { txt: removeIndent(override.summary) }
357
+ : this.parseMarkdownOrString(props.summary);
358
+ const technology = toSingleLine(override?.technology) ??
359
+ removeIndent(parseMarkdownAsString(props.technology));
360
+ return exact({
361
+ title,
362
+ summary,
363
+ description,
364
+ technology,
365
+ });
366
+ }
367
+ }
@@ -33,14 +33,14 @@ export declare function DeploymentModelParser<TBase extends WithExpressionV2>(B:
33
33
  readonly doc: import("../../ast").ParsedLikeC4LangiumDocument;
34
34
  get project(): import("../../workspace").Project;
35
35
  resolveFqn(node: ast.FqnReferenceable): c4.Fqn;
36
- getAstNodePath(node: c4): any;
36
+ getAstNodePath(node: import("langium").AstNode): string;
37
37
  getMetadata(metadataAstNode: ast.MetadataProperty | undefined): {
38
38
  [key: string]: string | string[];
39
39
  } | undefined;
40
40
  parseMarkdownOrString(markdownOrString: ast.MarkdownOrString | undefined): c4.MarkdownOrString | undefined;
41
41
  convertTags<E extends {
42
42
  tags?: ast.Tags;
43
- }>(withTags?: E | undefined): any;
43
+ }>(withTags?: E | undefined): c4.NonEmptyArray<c4.Tag<string>> | null;
44
44
  parseTags<E extends {
45
45
  tags?: ast.Tags;
46
46
  }>(withTags?: E): c4.NonEmptyArray<c4.Tag> | null;