@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
@@ -1,200 +0,0 @@
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/index.mjs";
8
- import { createLanguageServices } from "../module.mjs";
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) return;
27
- isInitialized = true;
28
- services.shared.workspace.ConfigurationProvider.updateConfiguration({
29
- settings: { likec4: { formatting: { quoteStyle: "single" } } }
30
- });
31
- services.shared.workspace.WorkspaceManager.initialize({
32
- capabilities: {},
33
- processId: null,
34
- rootUri: workspaceFolder.uri,
35
- workspaceFolders: [workspaceFolder]
36
- });
37
- await services.shared.workspace.WorkspaceManager.initializeWorkspace([workspaceFolder]);
38
- if (projectConfig) {
39
- const projectFolderUri = Utils.resolvePath(workspaceUri, "src");
40
- await services.shared.workspace.ProjectsManager.registerProject({
41
- config: {
42
- name: projectConfig?.name || "test-project",
43
- title: projectConfig?.title || "Test Project",
44
- contactPerson: projectConfig?.contactPerson || "Unknown",
45
- imageAliases: projectConfig?.imageAliases || {},
46
- exclude: projectConfig?.exclude || ["node_modules"]
47
- },
48
- folderUri: projectFolderUri
49
- });
50
- }
51
- }
52
- const addDocument = async (input, uri) => {
53
- await initialize();
54
- const docUri = Utils.resolvePath(
55
- workspaceUri,
56
- "./src/",
57
- uri ?? `${documentIndex++}${metaData.fileExtensions[0]}`
58
- );
59
- const document = services.shared.workspace.LangiumDocumentFactory.fromString(
60
- stripIndent(input),
61
- docUri
62
- );
63
- langiumDocuments.addDocument(document);
64
- return document;
65
- };
66
- const removeDocument = async (doc) => {
67
- const uri = doc instanceof URI ? doc : doc.uri;
68
- await documentBuilder.update([], [uri]);
69
- };
70
- const parse = async (input, uri) => {
71
- const document = await addDocument(input, uri);
72
- await documentBuilder.build([document], { validation: false });
73
- return document;
74
- };
75
- const validate = async (input, uri) => {
76
- const document = typeof input === "string" ? await addDocument(input, uri) : input;
77
- await documentBuilder.build([document], { validation: true });
78
- const diagnostics = document.diagnostics ?? [];
79
- const warnings = diagnostics.flatMap((d) => d.severity === DiagnosticSeverity.Warning ? d.message : []);
80
- const errors = diagnostics.flatMap((d) => d.severity === DiagnosticSeverity.Error ? d.message : []);
81
- return {
82
- document,
83
- diagnostics,
84
- warnings,
85
- errors
86
- };
87
- };
88
- const format = async (input, uri) => {
89
- const document = typeof input === "string" ? await parse(stripIndent(input), uri) : input;
90
- await documentBuilder.build([document], { validation: true });
91
- const edits = await formatter?.formatDocument(
92
- document,
93
- {
94
- options: { tabSize: 2, insertSpaces: true },
95
- textDocument: { uri: document.uri.toString() }
96
- }
97
- );
98
- return TextDocument.applyEdits(document.textDocument, edits ?? []);
99
- };
100
- const validateAll = async () => {
101
- const docs = langiumDocuments.all.toArray();
102
- assert.ok(docs.length > 0, "no documents to validate");
103
- await documentBuilder.build(docs, { validation: true });
104
- const diagnostics = docs.flatMap((doc) => doc.diagnostics ?? []);
105
- const warnings = diagnostics.flatMap((d) => d.severity === DiagnosticSeverity.Warning ? d.message : []);
106
- const errors = diagnostics.flatMap((d) => d.severity === DiagnosticSeverity.Error ? d.message : []);
107
- return {
108
- diagnostics,
109
- errors,
110
- warnings
111
- };
112
- };
113
- const buildModel = async () => {
114
- if (langiumDocuments.all.some((doc) => doc.state < DocumentState.Validated)) {
115
- await validateAll();
116
- }
117
- const likec4model = await modelBuilder.buildLikeC4Model();
118
- if (!likec4model) throw new Error("No model found");
119
- return likec4model.$data;
120
- };
121
- const buildLikeC4Model = async () => {
122
- if (langiumDocuments.all.some((doc) => doc.state < DocumentState.Validated)) {
123
- await validateAll();
124
- }
125
- const likec4model = await modelBuilder.buildLikeC4Model();
126
- if (!likec4model) throw new Error("No model found");
127
- return likec4model;
128
- };
129
- const resetState = async () => {
130
- await services.shared.workspace.WorkspaceLock.write(async (cancelToken) => {
131
- const docs = langiumDocuments.allExcludingBuiltin.toArray().map((doc) => doc.uri);
132
- await documentBuilder.update([], docs, cancelToken);
133
- });
134
- };
135
- return {
136
- services,
137
- addDocument,
138
- removeDocument,
139
- parse,
140
- validate,
141
- validateAll,
142
- buildModel,
143
- buildLikeC4Model,
144
- resetState,
145
- format
146
- };
147
- }
148
- export async function createMultiProjectTestServices(data) {
149
- const workspace = "file:///test/workspace";
150
- const {
151
- services,
152
- addDocument,
153
- validateAll
154
- } = createTestServices({ workspace });
155
- const projects = {};
156
- for (const [name, files] of entries(data)) {
157
- const folderUri = UriUtils.joinPath(URI.parse(workspace), "src", name);
158
- await services.shared.workspace.ProjectsManager.registerProject({
159
- config: {
160
- name,
161
- exclude: ["node_modules"]
162
- },
163
- folderUri
164
- });
165
- projects[name] = {};
166
- for (let [docName, content] of entries(files)) {
167
- const fileName = docName.endsWith(".c4") ? docName : `${docName}.c4`;
168
- projects[name][docName] = await addDocument(content, `${name}/${fileName}`);
169
- }
170
- }
171
- async function buildLikeC4Model(projectId) {
172
- if (services.shared.workspace.LangiumDocuments.all.some((doc) => doc.state < DocumentState.Validated)) {
173
- await validateAll();
174
- }
175
- const likec4model = await services.likec4.ModelBuilder.buildLikeC4Model(projectId);
176
- if (!likec4model) throw new Error("No model found");
177
- return likec4model;
178
- }
179
- async function buildModel(projectId) {
180
- const model = await buildLikeC4Model(projectId);
181
- return model.$data;
182
- }
183
- return {
184
- services,
185
- projects,
186
- projectsManager: services.shared.workspace.ProjectsManager,
187
- addDocument: async (uri, input) => {
188
- return await addDocument(input, uri.toString());
189
- },
190
- /**
191
- * Add document outside of projects
192
- */
193
- addDocumentOutside: async (input) => {
194
- return await addDocument(input);
195
- },
196
- validateAll,
197
- buildModel,
198
- buildLikeC4Model
199
- };
200
- }
@@ -1,25 +0,0 @@
1
- import { logWarnError } from "../logger.mjs";
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
- } catch (e) {
16
- logWarnError(e);
17
- }
18
- }
19
- }
20
- throwIfDisposed() {
21
- if (this.isDisposed) {
22
- throw new Error("This has already been disposed");
23
- }
24
- }
25
- }
@@ -1,20 +0,0 @@
1
- import { ast } from "../ast.mjs";
2
- export function elementRef(node) {
3
- let el = ast.isStrictFqnElementRef(node) ? node.el.ref : node.modelElement.value.ref;
4
- if (el?.$type === "Imported") {
5
- el = el.imported.ref;
6
- }
7
- return el?.$type === "Element" ? el : void 0;
8
- }
9
- export function readStrictFqn(node) {
10
- const name = [node.$type === "StrictFqnRef" ? node.value.$refText : node.el.$refText];
11
- let parent = node.parent;
12
- while (parent) {
13
- name.push(parent.$type === "StrictFqnRef" ? parent.value.$refText : parent.el.$refText);
14
- parent = parent.parent;
15
- }
16
- if (name.length === 1) {
17
- return name[0];
18
- }
19
- return name.reverse().join(".");
20
- }
@@ -1,57 +0,0 @@
1
- import { AstUtils } from "langium";
2
- import { isNullish } from "remeda";
3
- import { ast } from "../ast.mjs";
4
- export function referenceableParent(node) {
5
- while (node.parent) {
6
- node = node.parent;
7
- }
8
- return node.value.ref ?? null;
9
- }
10
- export function instanceRef(deploymentRef) {
11
- let referenceable;
12
- while (referenceable = deploymentRef.value?.ref) {
13
- if (ast.isDeploymentNode(referenceable)) {
14
- return null;
15
- }
16
- if (ast.isDeployedInstance(referenceable)) {
17
- return referenceable;
18
- }
19
- if (isNullish(deploymentRef.parent)) {
20
- return null;
21
- }
22
- deploymentRef = deploymentRef.parent;
23
- }
24
- return null;
25
- }
26
- export function deploymentNodeRef(deploymentRef) {
27
- let referenceable = deploymentRef.value.ref ?? null;
28
- if (!referenceable || ast.isDeploymentNode(referenceable)) {
29
- return referenceable;
30
- }
31
- const artifact = instanceRef(deploymentRef);
32
- return artifact ? AstUtils.getContainerOfType(artifact, ast.isDeploymentNode) ?? null : null;
33
- }
34
- export function importsRef(node) {
35
- const referenceable = referenceableParent(node);
36
- return referenceable?.$type === "Imported" ? referenceable : null;
37
- }
38
- export function isImportsRef(node) {
39
- return !!importsRef(node);
40
- }
41
- export function isReferenceToLogicalModel(node) {
42
- const referenceable = referenceableParent(node);
43
- return referenceable?.$type === "Element";
44
- }
45
- export function isReferenceToDeploymentModel(node) {
46
- let referenceable;
47
- while (referenceable = node.value?.ref) {
48
- if (ast.isDeploymentElement(referenceable)) {
49
- return true;
50
- }
51
- if (isNullish(node.parent)) {
52
- return false;
53
- }
54
- node = node.parent;
55
- }
56
- return false;
57
- }
@@ -1,33 +0,0 @@
1
- import prettyMs from "pretty-ms";
2
- import { logger } from "../logger.mjs";
3
- export * from "./disposable.mjs";
4
- export * from "./elementRef.mjs";
5
- export * from "./fqnRef.mjs";
6
- export * from "./projectId.mjs";
7
- export * from "./stringHash.mjs";
8
- export function safeCall(fn) {
9
- try {
10
- return fn();
11
- } catch (e) {
12
- logger.debug(`Safe call failed`, { error: e });
13
- return void 0;
14
- }
15
- }
16
- export function performanceNow() {
17
- try {
18
- return performance.now();
19
- } catch {
20
- return Date.now();
21
- }
22
- }
23
- export function performanceMark() {
24
- const t0 = performanceNow();
25
- return {
26
- get ms() {
27
- return performanceNow() - t0;
28
- },
29
- get pretty() {
30
- return prettyMs(performanceNow() - t0);
31
- }
32
- };
33
- }
@@ -1 +0,0 @@
1
- export const printDocs = (docs) => docs.map((d) => " - " + d.uri.toString(true)).join("\n");
@@ -1,16 +0,0 @@
1
- import { invariant, nonNullable } from "@likec4/core";
2
- import { AstUtils, isAstNode } from "langium";
3
- import { ast } from "../ast.mjs";
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
- }
@@ -1,5 +0,0 @@
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
- }
@@ -1,16 +0,0 @@
1
- import { DefaultDocumentValidator } from "langium";
2
- export class LikeC4DocumentValidator extends DefaultDocumentValidator {
3
- constructor(services) {
4
- super(services);
5
- this.services = services;
6
- }
7
- /**
8
- * If the document is excluded, then we skip validation and return an empty array of diagnostics.
9
- */
10
- async validateDocument(document, options, cancelToken) {
11
- if (this.services.shared.workspace.ProjectsManager.checkIfExcluded(document)) {
12
- return [];
13
- }
14
- return await super.validateDocument(document, options, cancelToken);
15
- }
16
- }
@@ -1,25 +0,0 @@
1
- import { isPromise } from "remeda";
2
- import { logger } from "../logger.mjs";
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
- } catch (e) {
20
- const message = e instanceof Error ? e.message : String(e);
21
- accept("error", `Validation failed: ${message}`, { node });
22
- logger.debug(`Validation failed: ${message}`, { error: e });
23
- }
24
- };
25
- }
@@ -1,146 +0,0 @@
1
- import { FqnRef, isSameHierarchy, nonNullable } from "@likec4/core";
2
- import { loggable } from "@likec4/log";
3
- import { AstUtils } from "langium";
4
- import { ast } from "../ast.mjs";
5
- import { logger } from "../logger.mjs";
6
- import { projectIdFrom } from "../utils/index.mjs";
7
- import { RESERVED_WORDS, tryOrLog } from "./_shared.mjs";
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(
32
- "error",
33
- `Duplicate node name "${fqnName}"`,
34
- {
35
- node: el,
36
- range
37
- }
38
- );
39
- }
40
- });
41
- };
42
- export const deployedInstanceChecks = (services) => {
43
- const DeploymentsIndex = services.likec4.DeploymentsIndex;
44
- const Names = services.references.NameProvider;
45
- return tryOrLog((el, accept) => {
46
- const artifactName = Names.getName(el);
47
- if (!artifactName) {
48
- accept("error", "Deployed instance must be named, unique inside node", {
49
- node: el
50
- });
51
- return;
52
- }
53
- const range = nonNullable(Names.getNameNode(el), "name CstNode not found").range;
54
- if (RESERVED_WORDS.includes(artifactName)) {
55
- accept("error", `Reserved word: ${artifactName}`, {
56
- node: el,
57
- range
58
- });
59
- }
60
- const projectId = projectIdFrom(el);
61
- const fqnName = DeploymentsIndex.getFqn(el);
62
- const withSameName = DeploymentsIndex.byFqn(projectId, fqnName).limit(2).toArray();
63
- if (withSameName.length > 1) {
64
- accept(
65
- "error",
66
- `Duplicate instance name "${fqnName}"`,
67
- {
68
- node: el,
69
- range
70
- }
71
- );
72
- }
73
- });
74
- };
75
- export const deploymentRelationChecks = (services) => {
76
- const ModelParser = services.likec4.ModelParser;
77
- return tryOrLog((el, accept) => {
78
- const target = el.target?.value?.ref;
79
- if (!target) {
80
- let targetCstText = el.target?.$cstNode?.text ?? "";
81
- accept("error", `DeploymentRelation target '${targetCstText}' not resolved`, {
82
- node: el,
83
- property: "target"
84
- });
85
- return;
86
- }
87
- const doc = getDocument(el);
88
- const parser = ModelParser.forDocument(doc);
89
- let sourceFqnRef;
90
- try {
91
- sourceFqnRef = parser._resolveDeploymentRelationSource(el);
92
- } catch (e) {
93
- logger.warn(loggable(e));
94
- accept("error", "DeploymentRelation source not resolved", {
95
- node: el,
96
- property: "source"
97
- });
98
- return;
99
- }
100
- if (FqnRef.isImportRef(sourceFqnRef)) {
101
- accept("error", "DeploymentRelation cannot refer imported model (not implemented yet)", {
102
- node: el,
103
- property: "source"
104
- });
105
- return;
106
- }
107
- if (FqnRef.isModelRef(sourceFqnRef)) {
108
- accept("error", "DeploymentRelation must refer deployment element", {
109
- node: el,
110
- property: "source"
111
- });
112
- return;
113
- }
114
- const targetFqnRef = parser.parseFqnRef(el.target);
115
- if (FqnRef.isImportRef(targetFqnRef)) {
116
- accept("error", "DeploymentRelation cannot refer imported model (not implemented yet)", {
117
- node: el,
118
- property: "target"
119
- });
120
- return;
121
- }
122
- if (FqnRef.isModelRef(targetFqnRef)) {
123
- accept("error", "DeploymentRelation must refer deployment element", {
124
- node: el,
125
- property: "target"
126
- });
127
- return;
128
- }
129
- if (isSameHierarchy(sourceFqnRef.deployment, targetFqnRef.deployment)) {
130
- accept("error", "Invalid parent-child relationship", {
131
- node: el
132
- });
133
- }
134
- });
135
- };
136
- export const extendDeploymentChecks = (_services) => {
137
- return tryOrLog((el, accept) => {
138
- const target = el.deploymentNode.value.ref;
139
- if (!target || !ast.isDeploymentNode(target)) {
140
- accept("error", "ExtendDeployment allows only DeploymentNode", {
141
- node: el,
142
- property: "deploymentNode"
143
- });
144
- }
145
- });
146
- };
@@ -1,67 +0,0 @@
1
- import { isAncestor } from "@likec4/core";
2
- import { AstUtils } from "langium";
3
- import { isEmpty } from "remeda";
4
- import { ast } from "../ast.mjs";
5
- import { elementRef } from "../utils/elementRef.mjs";
6
- import { tryOrLog } from "./_shared.mjs";
7
- export const dynamicViewStepSingle = (services) => {
8
- const fqnIndex = services.likec4.FqnIndex;
9
- return tryOrLog((el, accept) => {
10
- const sourceEl = elementRef(el.source);
11
- const source = sourceEl && fqnIndex.getFqn(sourceEl);
12
- if (!source) {
13
- accept("error", "Source not found (not parsed/indexed yet)", {
14
- node: el,
15
- property: "source"
16
- });
17
- }
18
- const targetEl = elementRef(el.target);
19
- const target = targetEl && fqnIndex.getFqn(targetEl);
20
- if (!target) {
21
- accept("error", "Target not found (not parsed/indexed yet)", {
22
- node: el,
23
- property: "target"
24
- });
25
- }
26
- if (source && target && (isAncestor(source, target) || isAncestor(target, source))) {
27
- accept("error", "Invalid parent-child relationship", {
28
- node: el
29
- });
30
- }
31
- });
32
- };
33
- export const dynamicViewStepChain = (services) => {
34
- const fqnIndex = services.likec4.FqnIndex;
35
- return tryOrLog((el, accept) => {
36
- const source = el.source;
37
- if (ast.isDynamicStepSingle(source) && source.isBackward) {
38
- accept("error", "Invalid chain after backward step", {
39
- node: el
40
- });
41
- }
42
- const targetEl = elementRef(el.target);
43
- const target = targetEl && fqnIndex.getFqn(targetEl);
44
- if (!target) {
45
- accept("error", "Target not found (not parsed/indexed yet)", {
46
- node: el,
47
- property: "target"
48
- });
49
- }
50
- });
51
- };
52
- export const dynamicViewDisplayVariant = (_services) => {
53
- return tryOrLog((prop, accept) => {
54
- if (isEmpty(prop.value) || prop.value !== "diagram" && prop.value !== "sequence") {
55
- accept("error", 'Invalid display variant: "diagram" or "sequence" are allowed', {
56
- node: prop,
57
- property: "value"
58
- });
59
- return;
60
- }
61
- if (!AstUtils.hasContainerOfType(prop, ast.isDynamicViewBody)) {
62
- accept("error", `Display mode can be defined only inside dynamic view`, {
63
- node: prop
64
- });
65
- }
66
- });
67
- };
@@ -1,12 +0,0 @@
1
- import { isReferenceToDeploymentModel } from "../utils/index.mjs";
2
- import { tryOrLog } from "./_shared.mjs";
3
- export const checkElementRef = (_services) => {
4
- return tryOrLog((el, accept) => {
5
- if (isReferenceToDeploymentModel(el.modelElement)) {
6
- accept("error", "Only model elements allowed here", {
7
- node: el,
8
- property: "modelElement"
9
- });
10
- }
11
- });
12
- };
@@ -1,50 +0,0 @@
1
- import { AstUtils } from "langium";
2
- import { projectIdFrom } from "../utils/index.mjs";
3
- import { RESERVED_WORDS, tryOrLog } from "./_shared.mjs";
4
- const { getDocument } = AstUtils;
5
- export const checkElement = (services) => {
6
- const fqnIndex = services.likec4.FqnIndex;
7
- const locator = services.workspace.AstNodeLocator;
8
- return tryOrLog((el, accept) => {
9
- const fqn = fqnIndex.getFqn(el);
10
- if (!fqn) {
11
- accept("error", "Not indexed element", {
12
- node: el,
13
- property: "name"
14
- });
15
- return;
16
- }
17
- if (RESERVED_WORDS.includes(el.name)) {
18
- accept("error", `Reserved word: ${el.name}`, {
19
- node: el,
20
- property: "name"
21
- });
22
- }
23
- const doc = getDocument(el);
24
- const docUri = doc.uri;
25
- const elPath = locator.getAstNodePath(el);
26
- const withSameFqn = fqnIndex.byFqn(projectIdFrom(doc), fqn).filter((v) => v.documentUri !== docUri || v.path !== elPath).head();
27
- if (withSameFqn) {
28
- const isAnotherDoc = withSameFqn.documentUri !== docUri;
29
- accept(
30
- "error",
31
- `Duplicate element name ${el.name !== fqn ? el.name + " (" + fqn + ")" : el.name}`,
32
- {
33
- node: el,
34
- property: "name",
35
- ...isAnotherDoc && {
36
- relatedInformation: [
37
- {
38
- location: {
39
- range: withSameFqn.nameSegment?.range ?? withSameFqn.selectionSegment?.range,
40
- uri: withSameFqn.documentUri.toString()
41
- },
42
- message: `conflicting element`
43
- }
44
- ]
45
- }
46
- }
47
- );
48
- }
49
- });
50
- };