@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
@@ -8,25 +8,25 @@ export declare function createTestServices(options?: {
8
8
  workspace?: string;
9
9
  projectConfig?: Partial<LikeC4ProjectJsonConfig>;
10
10
  }): {
11
- services: any;
11
+ services: import("langium").LangiumGeneratedCoreServices & import("langium").LangiumDefaultCoreServices & import("langium/lsp").LangiumLSPServices & import("../module").LikeC4AddedServices;
12
12
  addDocument: (input: string, uri?: string) => Promise<LikeC4LangiumDocument>;
13
13
  removeDocument: (doc: LangiumDocument | URI) => Promise<void>;
14
14
  parse: (input: string, uri?: string) => Promise<LikeC4LangiumDocument>;
15
15
  validate: (input: string | LikeC4LangiumDocument, uri?: string) => Promise<{
16
16
  document: LikeC4LangiumDocument;
17
- diagnostics: any[];
18
- warnings: any[];
19
- errors: any[];
17
+ diagnostics: (import("vscode-languageserver-types").Diagnostic & import("langium").DiagnosticInfo<import("../ast").LikeC4AstNode>)[];
18
+ warnings: string[];
19
+ errors: string[];
20
20
  }>;
21
21
  validateAll: () => Promise<{
22
- diagnostics: any;
23
- errors: any;
24
- warnings: any;
22
+ diagnostics: import("vscode-languageserver-types").Diagnostic[];
23
+ errors: string[];
24
+ warnings: string[];
25
25
  }>;
26
26
  buildModel: () => Promise<ComputedLikeC4ModelData>;
27
- buildLikeC4Model: () => Promise<any>;
27
+ buildLikeC4Model: () => Promise<import("@likec4/core/model").LikeC4Model<import("@likec4/core").UnknownComputed>>;
28
28
  resetState: () => Promise<void>;
29
- format: (input: string | LikeC4LangiumDocument, uri?: string) => Promise<any>;
29
+ format: (input: string | LikeC4LangiumDocument, uri?: string) => Promise<string>;
30
30
  };
31
31
  /**
32
32
  * @example
@@ -43,21 +43,21 @@ export declare function createTestServices(options?: {
43
43
  * ```
44
44
  */
45
45
  export declare function createMultiProjectTestServices<const Projects extends Record<string, Record<string, string>>>(data: Projects): Promise<{
46
- services: any;
46
+ services: import("langium").LangiumGeneratedCoreServices & import("langium").LangiumDefaultCoreServices & import("langium/lsp").LangiumLSPServices & import("../module").LikeC4AddedServices;
47
47
  projects: { readonly [K in keyof Projects]: { readonly [L in keyof Projects[K]]: LikeC4LangiumDocument; }; };
48
- projectsManager: any;
48
+ projectsManager: import("..").ProjectsManager;
49
49
  addDocument: (uri: string | URI, input: string) => Promise<LikeC4LangiumDocument>;
50
50
  /**
51
51
  * Add document outside of projects
52
52
  */
53
53
  addDocumentOutside: (input: string) => Promise<LikeC4LangiumDocument>;
54
54
  validateAll: () => Promise<{
55
- diagnostics: any;
56
- errors: any;
57
- warnings: any;
55
+ diagnostics: import("vscode-languageserver-types").Diagnostic[];
56
+ errors: string[];
57
+ warnings: string[];
58
58
  }>;
59
- buildModel: (projectId: LiteralUnion<keyof Projects, string>) => Promise<ComputedLikeC4ModelData>;
60
- buildLikeC4Model: (projectId: LiteralUnion<keyof Projects, string>) => Promise<any>;
59
+ buildModel: (projectId: LiteralUnion<keyof Projects, string>) => Promise<ComputedLikeC4ModelData<import("@likec4/core").UnknownComputed>>;
60
+ buildLikeC4Model: (projectId: LiteralUnion<keyof Projects, string>) => Promise<import("@likec4/core/model").LikeC4Model<import("@likec4/core").UnknownComputed>>;
61
61
  }>;
62
62
  export type TestServices = ReturnType<typeof createTestServices>;
63
63
  export type TestParseFn = TestServices['validate'];
@@ -0,0 +1,210 @@
1
+ import { DocumentState, TextDocument, UriUtils } from 'langium';
2
+ import * as assert from 'node:assert';
3
+ import { entries } from 'remeda';
4
+ import stripIndent from 'strip-indent';
5
+ import { DiagnosticSeverity } from 'vscode-languageserver-types';
6
+ import { URI, Utils } from 'vscode-uri';
7
+ import { NoopFileSystem } from '../filesystem';
8
+ import { createLanguageServices } from '../module';
9
+ export function createTestServices(options) {
10
+ const workspace = options?.workspace ?? 'file:///test/workspace';
11
+ const projectConfig = options?.projectConfig;
12
+ const services = createLanguageServices(NoopFileSystem).likec4;
13
+ const metaData = services.LanguageMetaData;
14
+ const langiumDocuments = services.shared.workspace.LangiumDocuments;
15
+ const documentBuilder = services.shared.workspace.DocumentBuilder;
16
+ const modelBuilder = services.likec4.ModelBuilder;
17
+ const workspaceUri = URI.parse(workspace);
18
+ const formatter = services.lsp.Formatter;
19
+ const workspaceFolder = {
20
+ name: projectConfig?.name || 'test-project',
21
+ uri: workspaceUri.toString(),
22
+ };
23
+ let isInitialized = false;
24
+ let documentIndex = 1;
25
+ async function initialize() {
26
+ if (isInitialized)
27
+ return;
28
+ isInitialized = true;
29
+ services.shared.workspace.ConfigurationProvider.updateConfiguration({
30
+ settings: { likec4: { formatting: { quoteStyle: 'single' } } },
31
+ });
32
+ services.shared.workspace.WorkspaceManager.initialize({
33
+ capabilities: {},
34
+ processId: null,
35
+ rootUri: workspaceFolder.uri,
36
+ workspaceFolders: [workspaceFolder],
37
+ });
38
+ await services.shared.workspace.WorkspaceManager.initializeWorkspace([workspaceFolder]);
39
+ // Register project with config if provided...
40
+ if (projectConfig) {
41
+ const projectFolderUri = Utils.resolvePath(workspaceUri, 'src');
42
+ await services.shared.workspace.ProjectsManager.registerProject({
43
+ config: {
44
+ name: projectConfig?.name || 'test-project',
45
+ title: projectConfig?.title || 'Test Project',
46
+ contactPerson: projectConfig?.contactPerson || 'Unknown',
47
+ imageAliases: projectConfig?.imageAliases || {},
48
+ exclude: projectConfig?.exclude || ['node_modules'],
49
+ },
50
+ folderUri: projectFolderUri,
51
+ });
52
+ }
53
+ }
54
+ const addDocument = async (input, uri) => {
55
+ await initialize();
56
+ const docUri = Utils.resolvePath(workspaceUri, './src/', uri ?? `${documentIndex++}${metaData.fileExtensions[0]}`);
57
+ const document = services.shared.workspace.LangiumDocumentFactory.fromString(stripIndent(input), docUri);
58
+ langiumDocuments.addDocument(document);
59
+ return document;
60
+ };
61
+ const removeDocument = async (doc) => {
62
+ const uri = doc instanceof URI ? doc : doc.uri;
63
+ await documentBuilder.update([], [uri]);
64
+ };
65
+ const parse = async (input, uri) => {
66
+ const document = await addDocument(input, uri);
67
+ await documentBuilder.build([document], { validation: false });
68
+ return document;
69
+ };
70
+ const validate = async (input, uri) => {
71
+ const document = typeof input === 'string' ? await addDocument(input, uri) : input;
72
+ await documentBuilder.build([document], { validation: true });
73
+ const diagnostics = document.diagnostics ?? [];
74
+ const warnings = diagnostics.flatMap(d => d.severity === DiagnosticSeverity.Warning ? d.message : []);
75
+ const errors = diagnostics.flatMap(d => d.severity === DiagnosticSeverity.Error ? d.message : []);
76
+ return {
77
+ document,
78
+ diagnostics,
79
+ warnings,
80
+ errors,
81
+ };
82
+ };
83
+ const format = async (input, uri) => {
84
+ const document = typeof input === 'string' ? await parse(stripIndent(input), uri) : input;
85
+ await documentBuilder.build([document], { validation: true });
86
+ const edits = await formatter?.formatDocument(document, {
87
+ options: { tabSize: 2, insertSpaces: true },
88
+ textDocument: { uri: document.uri.toString() },
89
+ });
90
+ return TextDocument.applyEdits(document.textDocument, edits ?? []);
91
+ };
92
+ const validateAll = async () => {
93
+ const docs = langiumDocuments.all.toArray();
94
+ assert.ok(docs.length > 0, 'no documents to validate');
95
+ await documentBuilder.build(docs, { validation: true });
96
+ const diagnostics = docs.flatMap(doc => doc.diagnostics ?? []);
97
+ const warnings = diagnostics.flatMap(d => d.severity === DiagnosticSeverity.Warning ? d.message : []);
98
+ const errors = diagnostics.flatMap(d => d.severity === DiagnosticSeverity.Error ? d.message : []);
99
+ return {
100
+ diagnostics,
101
+ errors,
102
+ warnings,
103
+ };
104
+ };
105
+ const buildModel = async () => {
106
+ if (langiumDocuments.all.some(doc => doc.state < DocumentState.Validated)) {
107
+ await validateAll();
108
+ }
109
+ const likec4model = await modelBuilder.computeModel();
110
+ if (!likec4model)
111
+ throw new Error('No model found');
112
+ return likec4model.$data;
113
+ };
114
+ const buildLikeC4Model = async () => {
115
+ if (langiumDocuments.all.some(doc => doc.state < DocumentState.Validated)) {
116
+ await validateAll();
117
+ }
118
+ const likec4model = await modelBuilder.computeModel();
119
+ if (!likec4model)
120
+ throw new Error('No model found');
121
+ return likec4model;
122
+ };
123
+ /**
124
+ * This will clear all documents
125
+ */
126
+ const resetState = async () => {
127
+ await services.shared.workspace.WorkspaceLock.write(async (cancelToken) => {
128
+ const docs = langiumDocuments.allExcludingBuiltin.toArray().map(doc => doc.uri);
129
+ await documentBuilder.update([], docs, cancelToken);
130
+ });
131
+ };
132
+ return {
133
+ services,
134
+ addDocument,
135
+ removeDocument,
136
+ parse,
137
+ validate,
138
+ validateAll,
139
+ buildModel,
140
+ buildLikeC4Model,
141
+ resetState,
142
+ format,
143
+ };
144
+ }
145
+ /**
146
+ * @example
147
+ * ```ts
148
+ * const { projects } = await createMultiProjectTestServices({
149
+ * project1: {
150
+ * doc1: `...`,
151
+ * doc2: `...`,
152
+ * },
153
+ * project2: {
154
+ * doc1: `...`,
155
+ * },
156
+ * })
157
+ * ```
158
+ */
159
+ export async function createMultiProjectTestServices(data) {
160
+ const workspace = 'file:///test/workspace';
161
+ const { services, addDocument, validateAll, } = createTestServices({ workspace });
162
+ const projects = {};
163
+ for (const [name, files] of entries(data)) {
164
+ const folderUri = UriUtils.joinPath(URI.parse(workspace), 'src', name);
165
+ await services.shared.workspace.ProjectsManager.registerProject({
166
+ config: {
167
+ name,
168
+ exclude: ['node_modules'],
169
+ },
170
+ folderUri,
171
+ });
172
+ // @ts-ignore
173
+ projects[name] = {};
174
+ for (let [docName, content] of entries(files)) {
175
+ const fileName = docName.endsWith('.c4') ? docName : `${docName}.c4`;
176
+ // @ts-ignore
177
+ projects[name][docName] = await addDocument(content, `${name}/${fileName}`);
178
+ }
179
+ }
180
+ async function buildLikeC4Model(projectId) {
181
+ if (services.shared.workspace.LangiumDocuments.all.some(doc => doc.state < DocumentState.Validated)) {
182
+ await validateAll();
183
+ }
184
+ const likec4model = await services.likec4.ModelBuilder.computeModel(projectId);
185
+ if (!likec4model)
186
+ throw new Error('No model found');
187
+ return likec4model;
188
+ }
189
+ async function buildModel(projectId) {
190
+ const model = await buildLikeC4Model(projectId);
191
+ return model.$data;
192
+ }
193
+ return {
194
+ services,
195
+ projects,
196
+ projectsManager: services.shared.workspace.ProjectsManager,
197
+ addDocument: async (uri, input) => {
198
+ return await addDocument(input, uri.toString());
199
+ },
200
+ /**
201
+ * Add document outside of projects
202
+ */
203
+ addDocumentOutside: async (input) => {
204
+ return await addDocument(input);
205
+ },
206
+ validateAll,
207
+ buildModel,
208
+ buildLikeC4Model,
209
+ };
210
+ }
@@ -0,0 +1,26 @@
1
+ import { logWarnError } from '../logger';
2
+ export class ADisposable {
3
+ toDispose = [];
4
+ isDisposed = false;
5
+ onDispose(...disposable) {
6
+ this.toDispose.push(...disposable);
7
+ }
8
+ dispose() {
9
+ this.throwIfDisposed();
10
+ this.isDisposed = true;
11
+ let item;
12
+ while ((item = this.toDispose.pop())) {
13
+ try {
14
+ item.dispose();
15
+ }
16
+ catch (e) {
17
+ logWarnError(e);
18
+ }
19
+ }
20
+ }
21
+ throwIfDisposed() {
22
+ if (this.isDisposed) {
23
+ throw new Error('This has already been disposed');
24
+ }
25
+ }
26
+ }
@@ -3,7 +3,7 @@ import { ast } from '../ast';
3
3
  /**
4
4
  * Returns referenced AST Element
5
5
  */
6
- export declare function elementRef(node: ast.ElementRef | ast.StrictFqnElementRef): any;
6
+ export declare function elementRef(node: ast.ElementRef | ast.StrictFqnElementRef): ast.Element | undefined;
7
7
  /**
8
8
  * Returns FQN of StrictFqnElementRef
9
9
  * a.b.c.d - for c node returns a.b.c
@@ -0,0 +1,27 @@
1
+ import { ast } from '../ast';
2
+ /**
3
+ * Returns referenced AST Element
4
+ */
5
+ export function elementRef(node) {
6
+ let el = ast.isStrictFqnElementRef(node) ? node.el.ref : node.modelElement.value.ref;
7
+ if (el?.$type === 'Imported') {
8
+ el = el.imported.ref;
9
+ }
10
+ return el?.$type === 'Element' ? el : undefined;
11
+ }
12
+ /**
13
+ * Returns FQN of StrictFqnElementRef
14
+ * a.b.c.d - for c node returns a.b.c
15
+ */
16
+ export function readStrictFqn(node) {
17
+ const name = [node.$type === 'StrictFqnRef' ? node.value.$refText : node.el.$refText];
18
+ let parent = node.parent;
19
+ while (parent) {
20
+ name.push(parent.$type === 'StrictFqnRef' ? parent.value.$refText : parent.el.$refText);
21
+ parent = parent.parent;
22
+ }
23
+ if (name.length === 1) {
24
+ return name[0];
25
+ }
26
+ return name.reverse().join('.');
27
+ }
@@ -0,0 +1,63 @@
1
+ import { AstUtils } from 'langium';
2
+ import { isNullish } from 'remeda';
3
+ import { ast } from '../ast';
4
+ export function referenceableParent(node) {
5
+ // iterate up the root parent
6
+ while (node.parent) {
7
+ node = node.parent;
8
+ }
9
+ return node.value.ref ?? null;
10
+ }
11
+ export function instanceRef(deploymentRef) {
12
+ let referenceable;
13
+ while ((referenceable = deploymentRef.value?.ref)) {
14
+ if (ast.isDeploymentNode(referenceable)) {
15
+ return null;
16
+ }
17
+ if (ast.isDeployedInstance(referenceable)) {
18
+ return referenceable;
19
+ }
20
+ if (isNullish(deploymentRef.parent)) {
21
+ return null;
22
+ }
23
+ deploymentRef = deploymentRef.parent;
24
+ }
25
+ return null;
26
+ }
27
+ export function deploymentNodeRef(deploymentRef) {
28
+ let referenceable = deploymentRef.value.ref ?? null;
29
+ if (!referenceable || ast.isDeploymentNode(referenceable)) {
30
+ return referenceable;
31
+ }
32
+ const artifact = instanceRef(deploymentRef);
33
+ // Because path in deploymentRef may be omitted,
34
+ // we find artifact first and then its container
35
+ return artifact ? AstUtils.getContainerOfType(artifact, ast.isDeploymentNode) ?? null : null;
36
+ }
37
+ export function importsRef(node) {
38
+ const referenceable = referenceableParent(node);
39
+ return referenceable?.$type === 'Imported' ? referenceable : null;
40
+ }
41
+ export function isImportsRef(node) {
42
+ return !!importsRef(node);
43
+ }
44
+ export function isReferenceToLogicalModel(node) {
45
+ const referenceable = referenceableParent(node);
46
+ return referenceable?.$type === 'Element';
47
+ }
48
+ /**
49
+ * Returns true if node references deployment model
50
+ */
51
+ export function isReferenceToDeploymentModel(node) {
52
+ let referenceable;
53
+ while ((referenceable = node.value?.ref)) {
54
+ if (ast.isDeploymentElement(referenceable)) {
55
+ return true;
56
+ }
57
+ if (isNullish(node.parent)) {
58
+ return false;
59
+ }
60
+ node = node.parent;
61
+ }
62
+ return false;
63
+ }
@@ -0,0 +1,35 @@
1
+ import prettyMs from 'pretty-ms';
2
+ import { logger } from '../logger';
3
+ export * from './disposable';
4
+ export * from './elementRef';
5
+ export * from './fqnRef';
6
+ export * from './projectId';
7
+ export * from './stringHash';
8
+ export function safeCall(fn) {
9
+ try {
10
+ return fn();
11
+ }
12
+ catch (e) {
13
+ logger.debug(`Safe call failed`, { error: e });
14
+ return undefined;
15
+ }
16
+ }
17
+ export function performanceNow() {
18
+ try {
19
+ return performance.now();
20
+ }
21
+ catch {
22
+ return Date.now();
23
+ }
24
+ }
25
+ export function performanceMark() {
26
+ const t0 = performanceNow();
27
+ return {
28
+ get ms() {
29
+ return performanceNow() - t0;
30
+ },
31
+ get pretty() {
32
+ return prettyMs(performanceNow() - t0);
33
+ },
34
+ };
35
+ }
@@ -0,0 +1 @@
1
+ export const printDocs = (docs) => docs.map(d => ' - ' + d.uri.toString(true)).join('\n');
@@ -0,0 +1,16 @@
1
+ import { invariant, nonNullable } from '@likec4/core';
2
+ import { AstUtils, isAstNode } from 'langium';
3
+ import { ast } from '../ast';
4
+ export function projectIdFrom(value) {
5
+ if (ast.isImported(value)) {
6
+ while (value.$type === 'Imported' && value.$container) {
7
+ value = value.$container;
8
+ }
9
+ invariant(ast.isImportsFromPoject(value));
10
+ }
11
+ if (ast.isImportsFromPoject(value)) {
12
+ return value.project;
13
+ }
14
+ const doc = isAstNode(value) ? AstUtils.getDocument(value) : value;
15
+ return nonNullable(doc.likec4ProjectId, () => `Invalid state, document ${doc.uri} has no project ID assigned`);
16
+ }
@@ -0,0 +1,5 @@
1
+ import { stringHash as hash } from '@likec4/core/utils';
2
+ export function stringHash(...str) {
3
+ var s = str.length > 1 ? str.join(':::') : str[0];
4
+ return hash(s);
5
+ }
@@ -0,0 +1,17 @@
1
+ import { Cancellation, DefaultDocumentValidator } from 'langium';
2
+ export class LikeC4DocumentValidator extends DefaultDocumentValidator {
3
+ services;
4
+ constructor(services) {
5
+ super(services);
6
+ this.services = services;
7
+ }
8
+ /**
9
+ * If the document is excluded, then we skip validation and return an empty array of diagnostics.
10
+ */
11
+ async validateDocument(document, options, cancelToken) {
12
+ if (this.services.shared.workspace.ProjectsManager.isExcluded(document)) {
13
+ return [];
14
+ }
15
+ return await super.validateDocument(document, options, cancelToken);
16
+ }
17
+ }
@@ -0,0 +1,26 @@
1
+ import { isPromise } from 'remeda';
2
+ import { logger } from '../logger';
3
+ export const RESERVED_WORDS = [
4
+ 'this',
5
+ 'it',
6
+ 'self',
7
+ 'super',
8
+ 'likec4lib',
9
+ 'global',
10
+ ];
11
+ export function tryOrLog(fn) {
12
+ return async function tryOrLogFn(node, accept, cancelToken) {
13
+ try {
14
+ const result = fn(node, accept, cancelToken);
15
+ if (isPromise(result)) {
16
+ await result;
17
+ }
18
+ return;
19
+ }
20
+ catch (e) {
21
+ const message = e instanceof Error ? e.message : String(e);
22
+ accept('error', `Validation failed: ${message}`, { node });
23
+ logger.debug(`Validation failed: ${message}`, { error: e });
24
+ }
25
+ };
26
+ }
@@ -0,0 +1,140 @@
1
+ import { FqnRef, isSameHierarchy, nonNullable } from '@likec4/core';
2
+ import { loggable } from '@likec4/log';
3
+ import { AstUtils } from 'langium';
4
+ import { ast } from '../ast';
5
+ import { logger } from '../logger';
6
+ import { projectIdFrom } from '../utils';
7
+ import { RESERVED_WORDS, tryOrLog } from './_shared';
8
+ const { getDocument } = AstUtils;
9
+ export const deploymentNodeChecks = (services) => {
10
+ const DeploymentsIndex = services.likec4.DeploymentsIndex;
11
+ const Names = services.references.NameProvider;
12
+ return tryOrLog((el, accept) => {
13
+ const nodeName = Names.getName(el);
14
+ if (!nodeName) {
15
+ accept('error', 'DeploymentNode must be named', {
16
+ node: el,
17
+ });
18
+ return;
19
+ }
20
+ const range = nonNullable(Names.getNameNode(el), 'name CstNode not found').range;
21
+ if (RESERVED_WORDS.includes(nodeName)) {
22
+ accept('error', `Reserved word: ${nodeName}`, {
23
+ node: el,
24
+ range,
25
+ });
26
+ }
27
+ const projectId = projectIdFrom(el);
28
+ const fqnName = DeploymentsIndex.getFqn(el);
29
+ const withSameName = DeploymentsIndex.byFqn(projectId, fqnName).limit(2).toArray();
30
+ if (withSameName.length > 1) {
31
+ accept('error', `Duplicate node name "${fqnName}"`, {
32
+ node: el,
33
+ range,
34
+ });
35
+ }
36
+ });
37
+ };
38
+ export const deployedInstanceChecks = (services) => {
39
+ const DeploymentsIndex = services.likec4.DeploymentsIndex;
40
+ const Names = services.references.NameProvider;
41
+ // const Locator = services.workspace.AstNodeLocator
42
+ return tryOrLog((el, accept) => {
43
+ const artifactName = Names.getName(el);
44
+ if (!artifactName) {
45
+ accept('error', 'Deployed instance must be named, unique inside node', {
46
+ node: el,
47
+ });
48
+ return;
49
+ }
50
+ const range = nonNullable(Names.getNameNode(el), 'name CstNode not found').range;
51
+ if (RESERVED_WORDS.includes(artifactName)) {
52
+ accept('error', `Reserved word: ${artifactName}`, {
53
+ node: el,
54
+ range,
55
+ });
56
+ }
57
+ const projectId = projectIdFrom(el);
58
+ const fqnName = DeploymentsIndex.getFqn(el);
59
+ const withSameName = DeploymentsIndex.byFqn(projectId, fqnName).limit(2).toArray();
60
+ if (withSameName.length > 1) {
61
+ accept('error', `Duplicate instance name "${fqnName}"`, {
62
+ node: el,
63
+ range,
64
+ });
65
+ }
66
+ });
67
+ };
68
+ export const deploymentRelationChecks = (services) => {
69
+ const ModelParser = services.likec4.ModelParser;
70
+ return tryOrLog((el, accept) => {
71
+ const target = el.target?.value?.ref;
72
+ if (!target) {
73
+ let targetCstText = el.target?.$cstNode?.text ?? '';
74
+ accept('error', `DeploymentRelation target '${targetCstText}' not resolved`, {
75
+ node: el,
76
+ property: 'target',
77
+ });
78
+ return;
79
+ }
80
+ const doc = getDocument(el);
81
+ const parser = ModelParser.forDocument(doc);
82
+ let sourceFqnRef;
83
+ try {
84
+ sourceFqnRef = parser._resolveDeploymentRelationSource(el);
85
+ }
86
+ catch (e) {
87
+ logger.warn(loggable(e));
88
+ accept('error', 'DeploymentRelation source not resolved', {
89
+ node: el,
90
+ property: 'source',
91
+ });
92
+ return;
93
+ }
94
+ if (FqnRef.isImportRef(sourceFqnRef)) {
95
+ accept('error', 'DeploymentRelation cannot refer imported model (not implemented yet)', {
96
+ node: el,
97
+ property: 'source',
98
+ });
99
+ return;
100
+ }
101
+ if (FqnRef.isModelRef(sourceFqnRef)) {
102
+ accept('error', 'DeploymentRelation must refer deployment element', {
103
+ node: el,
104
+ property: 'source',
105
+ });
106
+ return;
107
+ }
108
+ const targetFqnRef = parser.parseFqnRef(el.target);
109
+ if (FqnRef.isImportRef(targetFqnRef)) {
110
+ accept('error', 'DeploymentRelation cannot refer imported model (not implemented yet)', {
111
+ node: el,
112
+ property: 'target',
113
+ });
114
+ return;
115
+ }
116
+ if (FqnRef.isModelRef(targetFqnRef)) {
117
+ accept('error', 'DeploymentRelation must refer deployment element', {
118
+ node: el,
119
+ property: 'target',
120
+ });
121
+ return;
122
+ }
123
+ if (isSameHierarchy(sourceFqnRef.deployment, targetFqnRef.deployment)) {
124
+ accept('error', 'Invalid parent-child relationship', {
125
+ node: el,
126
+ });
127
+ }
128
+ });
129
+ };
130
+ export const extendDeploymentChecks = (_services) => {
131
+ return tryOrLog((el, accept) => {
132
+ const target = el.deploymentNode.value.ref;
133
+ if (!target || !ast.isDeploymentNode(target)) {
134
+ accept('error', 'ExtendDeployment allows only DeploymentNode', {
135
+ node: el,
136
+ property: 'deploymentNode',
137
+ });
138
+ }
139
+ });
140
+ };