@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.
- package/browser/package.json +1 -1
- package/browser-worker/package.json +1 -1
- package/dist/LikeC4LanguageServices.d.ts +12 -19
- package/dist/LikeC4LanguageServices.js +182 -0
- package/dist/Rpc.js +245 -0
- package/dist/ast.d.ts +10 -6
- package/dist/ast.js +253 -0
- package/dist/browser-worker.js +4 -0
- package/dist/browser.js +35 -0
- package/dist/bundled.js +42 -0
- package/dist/bundled.mjs +3830 -4172
- package/dist/documentation/documentation-provider.js +51 -0
- package/dist/documentation/index.js +1 -0
- package/dist/empty.js +2 -0
- package/dist/filesystem/ChokidarWatcher.js +97 -0
- package/dist/filesystem/FileSystemWatcher.js +14 -0
- package/dist/filesystem/LikeC4FileSystem.d.ts +1 -2
- package/dist/filesystem/LikeC4FileSystem.js +126 -0
- package/dist/filesystem/index.d.ts +26 -0
- package/dist/filesystem/index.js +29 -0
- package/dist/formatting/LikeC4Formatter.js +637 -0
- package/dist/formatting/utils.js +18 -0
- package/dist/generated/ast.d.ts +19 -3
- package/dist/generated/ast.js +2155 -0
- package/dist/generated/grammar.js +7 -0
- package/dist/generated/module.d.ts +6 -1
- package/dist/generated/module.js +27 -0
- package/dist/generated-lib/{icons.mjs → icons.js} +11 -7
- package/dist/index.d.ts +7 -0
- package/dist/index.js +53 -0
- package/dist/{likec4lib.mjs → likec4lib.js} +3 -3
- package/dist/logger.js +81 -0
- package/dist/lsp/CodeActionProvider.d.ts +14 -0
- package/dist/lsp/CodeActionProvider.js +33 -0
- package/dist/lsp/CodeLensProvider.js +44 -0
- package/dist/lsp/CompletionProvider.d.ts +3 -1
- package/dist/lsp/CompletionProvider.js +200 -0
- package/dist/lsp/DocumentHighlightProvider.js +10 -0
- package/dist/lsp/DocumentLinkProvider.js +58 -0
- package/dist/lsp/DocumentSymbolProvider.js +306 -0
- package/dist/lsp/HoverProvider.js +106 -0
- package/dist/lsp/RenameProvider.js +6 -0
- package/dist/lsp/SemanticTokenProvider.d.ts +5 -0
- package/dist/lsp/SemanticTokenProvider.js +257 -0
- package/dist/lsp/index.d.ts +1 -0
- package/dist/lsp/index.js +9 -0
- package/dist/mcp/MCPServerFactory.js +73 -0
- package/dist/mcp/NoopLikeC4MCPServer.js +17 -0
- package/dist/mcp/interfaces.js +5 -0
- package/dist/mcp/server/StdioLikeC4MCPServer.js +47 -0
- package/dist/mcp/server/StreamableLikeC4MCPServer.js +145 -0
- package/dist/mcp/server/WithMCPServer.js +56 -0
- package/dist/mcp/tools/_common.d.ts +8 -7
- package/dist/mcp/tools/_common.js +49 -0
- package/dist/mcp/tools/find-relationships.d.ts +7 -8
- package/dist/mcp/tools/find-relationships.js +150 -0
- package/dist/mcp/tools/list-projects.d.ts +3 -3
- package/dist/mcp/tools/list-projects.js +62 -0
- package/dist/mcp/tools/open-view.d.ts +6 -7
- package/dist/mcp/tools/open-view.js +52 -0
- package/dist/mcp/tools/read-deployment.d.ts +6 -7
- package/dist/mcp/tools/read-deployment.js +132 -0
- package/dist/mcp/tools/read-element.d.ts +6 -7
- package/dist/mcp/tools/read-element.js +194 -0
- package/dist/mcp/tools/read-project-summary.d.ts +5 -6
- package/dist/mcp/tools/read-project-summary.js +176 -0
- package/dist/mcp/tools/read-view.d.ts +6 -7
- package/dist/mcp/tools/read-view.js +203 -0
- package/dist/mcp/tools/search-element.d.ts +3 -3
- package/dist/mcp/tools/search-element.js +177 -0
- package/dist/mcp/utils.d.ts +2 -2
- package/dist/mcp/utils.js +48 -0
- package/dist/model/builder/MergedExtends.d.ts +2 -1
- package/dist/model/builder/MergedExtends.js +74 -0
- package/dist/model/builder/MergedSpecification.js +175 -0
- package/dist/model/builder/buildModel.js +176 -0
- package/dist/model/deployments-index.js +102 -0
- package/dist/model/fqn-index.js +250 -0
- package/dist/model/index.js +6 -0
- package/dist/model/model-builder.d.ts +13 -11
- package/dist/model/model-builder.js +234 -0
- package/dist/model/model-locator.d.ts +6 -5
- package/dist/model/model-locator.js +240 -0
- package/dist/model/model-parser-where.js +81 -0
- package/dist/model/model-parser.d.ts +318 -313
- package/dist/model/model-parser.js +119 -0
- package/dist/model/parser/Base.d.ts +3 -3
- package/dist/model/parser/Base.js +367 -0
- package/dist/model/parser/DeploymentModelParser.d.ts +3 -3
- package/dist/model/parser/DeploymentModelParser.js +176 -0
- package/dist/model/parser/DeploymentViewParser.d.ts +4 -4
- package/dist/model/parser/DeploymentViewParser.js +86 -0
- package/dist/model/parser/FqnRefParser.d.ts +3 -3
- package/dist/model/parser/FqnRefParser.js +382 -0
- package/dist/model/parser/GlobalsParser.d.ts +7 -7
- package/dist/model/parser/GlobalsParser.js +84 -0
- package/dist/model/parser/ImportsParser.d.ts +12 -13
- package/dist/model/parser/ImportsParser.js +24 -0
- package/dist/model/parser/ModelParser.d.ts +3 -3
- package/dist/model/parser/ModelParser.js +165 -0
- package/dist/model/parser/PredicatesParser.d.ts +3 -3
- package/dist/model/parser/PredicatesParser.js +45 -0
- package/dist/model/parser/SpecificationParser.d.ts +3 -3
- package/dist/model/parser/SpecificationParser.js +109 -0
- package/dist/model/parser/ValueConverter.js +12 -0
- package/dist/model/parser/ViewsParser.d.ts +4 -4
- package/dist/model/parser/ViewsParser.js +477 -0
- package/dist/model-change/ModelChanges.d.ts +6 -3
- package/dist/model-change/ModelChanges.js +102 -0
- package/dist/model-change/changeElementStyle.js +134 -0
- package/dist/model-change/changeViewLayout.d.ts +2 -2
- package/dist/model-change/changeViewLayout.js +28 -0
- package/dist/model-change/removeManualLayoutV1.d.ts +7 -0
- package/dist/model-change/removeManualLayoutV1.js +27 -0
- package/dist/module.d.ts +10 -5
- package/dist/module.js +143 -0
- package/dist/protocol.d.ts +1 -17
- package/dist/protocol.js +114 -0
- package/dist/references/index.js +3 -0
- package/dist/references/name-provider.js +37 -0
- package/dist/references/scope-computation.js +288 -0
- package/dist/references/scope-provider.d.ts +3 -3
- package/dist/references/scope-provider.js +242 -0
- package/dist/shared/NodeKindProvider.js +57 -0
- package/dist/shared/{WorkspaceSymbolProvider.mjs → WorkspaceSymbolProvider.js} +1 -1
- package/dist/shared/index.js +2 -0
- package/dist/test/index.js +1 -0
- package/dist/test/testServices.d.ts +16 -16
- package/dist/test/testServices.js +210 -0
- package/dist/utils/disposable.js +26 -0
- package/dist/utils/elementRef.d.ts +1 -1
- package/dist/utils/elementRef.js +27 -0
- package/dist/utils/fqnRef.js +63 -0
- package/dist/utils/index.js +35 -0
- package/dist/utils/printDocs.js +1 -0
- package/dist/utils/projectId.js +16 -0
- package/dist/utils/stringHash.js +5 -0
- package/dist/validation/DocumentValidator.js +17 -0
- package/dist/validation/_shared.js +26 -0
- package/dist/validation/deployment-checks.js +140 -0
- package/dist/validation/dynamic-view.js +67 -0
- package/dist/validation/element-ref.js +12 -0
- package/dist/validation/element.js +49 -0
- package/dist/validation/imports.js +46 -0
- package/dist/validation/index.d.ts +1 -1
- package/dist/validation/index.js +157 -0
- package/dist/validation/property-checks.js +108 -0
- package/dist/validation/relation.js +55 -0
- package/dist/validation/specification.js +190 -0
- package/dist/validation/view-predicates/fqn-expr-with.js +43 -0
- package/dist/validation/view-predicates/fqn-ref-expr.js +51 -0
- package/dist/validation/view-predicates/incoming.js +16 -0
- package/dist/validation/view-predicates/index.js +6 -0
- package/dist/validation/view-predicates/outgoing.js +20 -0
- package/dist/validation/view-predicates/relation-expr.js +46 -0
- package/dist/validation/view-predicates/relation-with.js +16 -0
- package/dist/validation/view.d.ts +1 -1
- package/dist/validation/view.js +42 -0
- package/dist/view-utils/assignNavigateTo.js +27 -0
- package/dist/view-utils/index.d.ts +1 -0
- package/dist/view-utils/index.js +2 -0
- package/dist/view-utils/manual-layout.d.ts +6 -0
- package/dist/view-utils/manual-layout.js +151 -0
- package/dist/views/ConfigurableLayouter.js +51 -0
- package/dist/views/LikeC4ManualLayouts.d.ts +28 -0
- package/dist/views/LikeC4ManualLayouts.js +132 -0
- package/dist/views/{likec4-views.d.ts → LikeC4Views.d.ts} +9 -8
- package/dist/views/LikeC4Views.js +200 -0
- package/dist/views/index.d.ts +4 -1
- package/dist/views/index.js +11 -0
- package/dist/workspace/AstNodeDescriptionProvider.js +15 -0
- package/dist/workspace/IndexManager.js +21 -0
- package/dist/workspace/LangiumDocuments.d.ts +1 -1
- package/dist/workspace/LangiumDocuments.js +58 -0
- package/dist/workspace/ProjectsManager.d.ts +8 -3
- package/dist/workspace/ProjectsManager.js +373 -0
- package/dist/workspace/WorkspaceManager.d.ts +3 -2
- package/dist/workspace/WorkspaceManager.js +93 -0
- package/dist/workspace/index.js +5 -0
- package/likec4lib/package.json +1 -1
- package/package.json +32 -31
- package/protocol/package.json +1 -1
- package/dist/LikeC4LanguageServices.mjs +0 -197
- package/dist/Rpc.mjs +0 -296
- package/dist/ast.mjs +0 -221
- package/dist/browser-worker.mjs +0 -2
- package/dist/browser.mjs +0 -32
- package/dist/documentation/documentation-provider.mjs +0 -48
- package/dist/documentation/index.mjs +0 -1
- package/dist/empty.mjs +0 -1
- package/dist/filesystem/ChokidarWatcher.mjs +0 -68
- package/dist/filesystem/FileSystemWatcher.mjs +0 -11
- package/dist/filesystem/LikeC4FileSystem.mjs +0 -64
- package/dist/filesystem/index.mjs +0 -19
- package/dist/formatting/LikeC4Formatter.mjs +0 -511
- package/dist/formatting/utils.mjs +0 -15
- package/dist/generated/ast.mjs +0 -2118
- package/dist/generated/grammar.mjs +0 -3
- package/dist/generated/module.mjs +0 -23
- package/dist/index.mjs +0 -50
- package/dist/logger.mjs +0 -82
- package/dist/lsp/CodeLensProvider.mjs +0 -42
- package/dist/lsp/CompletionProvider.mjs +0 -208
- package/dist/lsp/DocumentHighlightProvider.mjs +0 -10
- package/dist/lsp/DocumentLinkProvider.mjs +0 -53
- package/dist/lsp/DocumentSymbolProvider.mjs +0 -287
- package/dist/lsp/HoverProvider.mjs +0 -104
- package/dist/lsp/RenameProvider.mjs +0 -6
- package/dist/lsp/SemanticTokenProvider.mjs +0 -350
- package/dist/lsp/index.mjs +0 -7
- package/dist/mcp/MCPServerFactory.mjs +0 -70
- package/dist/mcp/NoopLikeC4MCPServer.mjs +0 -17
- package/dist/mcp/interfaces.mjs +0 -4
- package/dist/mcp/server/StdioLikeC4MCPServer.mjs +0 -46
- package/dist/mcp/server/StreamableLikeC4MCPServer.mjs +0 -153
- package/dist/mcp/server/WithMCPServer.mjs +0 -58
- package/dist/mcp/tools/_common.mjs +0 -42
- package/dist/mcp/tools/find-relationships.mjs +0 -151
- package/dist/mcp/tools/list-projects.mjs +0 -62
- package/dist/mcp/tools/open-view.mjs +0 -52
- package/dist/mcp/tools/read-deployment.mjs +0 -130
- package/dist/mcp/tools/read-element.mjs +0 -198
- package/dist/mcp/tools/read-project-summary.mjs +0 -178
- package/dist/mcp/tools/read-view.mjs +0 -205
- package/dist/mcp/tools/search-element.mjs +0 -171
- package/dist/mcp/utils.mjs +0 -47
- package/dist/model/builder/MergedExtends.mjs +0 -67
- package/dist/model/builder/MergedSpecification.mjs +0 -205
- package/dist/model/builder/assignTagColors.d.ts +0 -7
- package/dist/model/builder/assignTagColors.mjs +0 -51
- package/dist/model/builder/buildModel.mjs +0 -226
- package/dist/model/deployments-index.mjs +0 -100
- package/dist/model/fqn-index.mjs +0 -243
- package/dist/model/index.mjs +0 -6
- package/dist/model/model-builder.mjs +0 -285
- package/dist/model/model-locator.mjs +0 -239
- package/dist/model/model-parser-where.mjs +0 -81
- package/dist/model/model-parser.mjs +0 -127
- package/dist/model/parser/Base.mjs +0 -342
- package/dist/model/parser/DeploymentModelParser.mjs +0 -212
- package/dist/model/parser/DeploymentViewParser.mjs +0 -95
- package/dist/model/parser/FqnRefParser.mjs +0 -398
- package/dist/model/parser/GlobalsParser.mjs +0 -82
- package/dist/model/parser/ImportsParser.mjs +0 -28
- package/dist/model/parser/ModelParser.mjs +0 -190
- package/dist/model/parser/PredicatesParser.mjs +0 -45
- package/dist/model/parser/SpecificationParser.mjs +0 -120
- package/dist/model/parser/ValueConverter.mjs +0 -12
- package/dist/model/parser/ViewsParser.mjs +0 -490
- package/dist/model-change/ModelChanges.mjs +0 -89
- package/dist/model-change/changeElementStyle.mjs +0 -143
- package/dist/model-change/changeViewLayout.mjs +0 -32
- package/dist/model-change/saveManualLayout.d.ts +0 -11
- package/dist/model-change/saveManualLayout.mjs +0 -27
- package/dist/module.mjs +0 -180
- package/dist/protocol.mjs +0 -65
- package/dist/references/index.mjs +0 -3
- package/dist/references/name-provider.mjs +0 -39
- package/dist/references/scope-computation.mjs +0 -312
- package/dist/references/scope-provider.mjs +0 -239
- package/dist/shared/NodeKindProvider.mjs +0 -110
- package/dist/shared/index.mjs +0 -2
- package/dist/test/index.mjs +0 -1
- package/dist/test/testServices.mjs +0 -200
- package/dist/utils/disposable.mjs +0 -25
- package/dist/utils/elementRef.mjs +0 -20
- package/dist/utils/fqnRef.mjs +0 -57
- package/dist/utils/index.mjs +0 -33
- package/dist/utils/printDocs.mjs +0 -1
- package/dist/utils/projectId.mjs +0 -16
- package/dist/utils/stringHash.mjs +0 -5
- package/dist/validation/DocumentValidator.mjs +0 -16
- package/dist/validation/_shared.mjs +0 -25
- package/dist/validation/deployment-checks.mjs +0 -146
- package/dist/validation/dynamic-view.mjs +0 -67
- package/dist/validation/element-ref.mjs +0 -12
- package/dist/validation/element.mjs +0 -50
- package/dist/validation/imports.mjs +0 -25
- package/dist/validation/index.mjs +0 -180
- package/dist/validation/property-checks.mjs +0 -107
- package/dist/validation/relation.mjs +0 -53
- package/dist/validation/specification.mjs +0 -173
- package/dist/validation/view-predicates/fqn-expr-with.mjs +0 -43
- package/dist/validation/view-predicates/fqn-ref-expr.mjs +0 -53
- package/dist/validation/view-predicates/incoming.mjs +0 -16
- package/dist/validation/view-predicates/index.mjs +0 -6
- package/dist/validation/view-predicates/outgoing.mjs +0 -20
- package/dist/validation/view-predicates/relation-expr.mjs +0 -39
- package/dist/validation/view-predicates/relation-with.mjs +0 -16
- package/dist/validation/view.mjs +0 -25
- package/dist/view-utils/assignNavigateTo.mjs +0 -25
- package/dist/view-utils/index.mjs +0 -1
- package/dist/view-utils/manual-layout.mjs +0 -99
- package/dist/views/configurable-layouter.mjs +0 -51
- package/dist/views/index.mjs +0 -1
- package/dist/views/likec4-views.mjs +0 -166
- package/dist/workspace/AstNodeDescriptionProvider.mjs +0 -17
- package/dist/workspace/IndexManager.mjs +0 -17
- package/dist/workspace/LangiumDocuments.mjs +0 -53
- package/dist/workspace/ProjectsManager.mjs +0 -360
- package/dist/workspace/WorkspaceManager.mjs +0 -83
- package/dist/workspace/index.mjs +0 -5
- /package/dist/views/{configurable-layouter.d.ts → ConfigurableLayouter.d.ts} +0 -0
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
import { FqnRef } from '@likec4/core';
|
|
2
|
+
import { AstUtils } from 'langium';
|
|
3
|
+
import { ast } from '../ast';
|
|
4
|
+
import { logWarnError } from '../logger';
|
|
5
|
+
export class LikeC4DocumentationProvider {
|
|
6
|
+
parser;
|
|
7
|
+
locator;
|
|
8
|
+
constructor(services) {
|
|
9
|
+
this.parser = services.likec4.ModelParser;
|
|
10
|
+
this.locator = services.likec4.ModelLocator;
|
|
11
|
+
}
|
|
12
|
+
getDocumentation(node) {
|
|
13
|
+
try {
|
|
14
|
+
if (ast.isDeploymentNode(node)) {
|
|
15
|
+
const doc = AstUtils.getDocument(node);
|
|
16
|
+
const el = this.parser.forDocument(doc).parseDeploymentNode(node);
|
|
17
|
+
const lines = [el.id];
|
|
18
|
+
if (el.title !== node.name) {
|
|
19
|
+
lines.push(' ', `**${el.title}**`);
|
|
20
|
+
}
|
|
21
|
+
return lines.join(' \n');
|
|
22
|
+
}
|
|
23
|
+
if (ast.isDeployedInstance(node)) {
|
|
24
|
+
const doc = AstUtils.getDocument(node);
|
|
25
|
+
const instance = this.parser.forDocument(doc).parseDeployedInstance(node);
|
|
26
|
+
const [projectId, fqn] = FqnRef.isImportRef(instance.element)
|
|
27
|
+
? [instance.element.project, instance.element.model]
|
|
28
|
+
: [doc.likec4ProjectId, instance.element.model];
|
|
29
|
+
const el = projectId ? this.locator.getParsedElement(fqn, projectId) : this.locator.getParsedElement(fqn);
|
|
30
|
+
const lines = [instance.id, `_instance of_ ${fqn}`];
|
|
31
|
+
if (el) {
|
|
32
|
+
lines.push(' ', `**${el.title}**`);
|
|
33
|
+
}
|
|
34
|
+
return lines.join(' \n');
|
|
35
|
+
}
|
|
36
|
+
if (ast.isElement(node)) {
|
|
37
|
+
const doc = AstUtils.getDocument(node);
|
|
38
|
+
const el = this.parser.forDocument(doc).parseElement(node);
|
|
39
|
+
if (!el) {
|
|
40
|
+
return;
|
|
41
|
+
}
|
|
42
|
+
const lines = [el.id, ' ', `**${el.title}**`];
|
|
43
|
+
return lines.join(' \n');
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
catch (e) {
|
|
47
|
+
logWarnError(e);
|
|
48
|
+
}
|
|
49
|
+
return;
|
|
50
|
+
}
|
|
51
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export { LikeC4DocumentationProvider } from './documentation-provider';
|
package/dist/empty.js
ADDED
|
@@ -0,0 +1,97 @@
|
|
|
1
|
+
import { isLikeC4Config } from '@likec4/config/node';
|
|
2
|
+
import { loggable } from '@likec4/log';
|
|
3
|
+
import chokidar from 'chokidar';
|
|
4
|
+
import { URI } from 'langium';
|
|
5
|
+
import { logger as mainLogger } from '../logger';
|
|
6
|
+
import { isManualLayoutFile } from '../views/LikeC4ManualLayouts';
|
|
7
|
+
import { isLikeC4File } from './LikeC4FileSystem';
|
|
8
|
+
const logger = mainLogger.getChild('chokidar');
|
|
9
|
+
export const chokidarFileSystemWatcher = {
|
|
10
|
+
fileSystemWatcher: (services) => new ChokidarFileSystemWatcher(services),
|
|
11
|
+
};
|
|
12
|
+
const isAnyLikeC4File = (path) => isLikeC4File(path) || isLikeC4Config(path) || isManualLayoutFile(path);
|
|
13
|
+
/**
|
|
14
|
+
* A no-op file system watcher.
|
|
15
|
+
*/
|
|
16
|
+
export class ChokidarFileSystemWatcher {
|
|
17
|
+
services;
|
|
18
|
+
watcher;
|
|
19
|
+
constructor(services) {
|
|
20
|
+
this.services = services;
|
|
21
|
+
}
|
|
22
|
+
watch(folder) {
|
|
23
|
+
if (this.watcher) {
|
|
24
|
+
this.watcher.add(folder);
|
|
25
|
+
}
|
|
26
|
+
else {
|
|
27
|
+
this.watcher = this.createWatcher(folder);
|
|
28
|
+
}
|
|
29
|
+
logger.debug `watching folder: ${folder}`;
|
|
30
|
+
}
|
|
31
|
+
async dispose() {
|
|
32
|
+
if (this.watcher) {
|
|
33
|
+
await this.watcher.close();
|
|
34
|
+
this.watcher = undefined;
|
|
35
|
+
}
|
|
36
|
+
return;
|
|
37
|
+
}
|
|
38
|
+
createWatcher(folder) {
|
|
39
|
+
let watcher = chokidar.watch(folder, {
|
|
40
|
+
ignored: [
|
|
41
|
+
path => path.includes('node_modules') || path.includes('.git'),
|
|
42
|
+
(path, stats) => (!!stats && stats.isFile() && !isAnyLikeC4File(path)),
|
|
43
|
+
],
|
|
44
|
+
ignoreInitial: true,
|
|
45
|
+
});
|
|
46
|
+
const onAddOrChange = async (path) => {
|
|
47
|
+
try {
|
|
48
|
+
if (isLikeC4Config(path)) {
|
|
49
|
+
logger.debug `project file changed: ${path}`;
|
|
50
|
+
await this.services.workspace.ProjectsManager.reloadProjects();
|
|
51
|
+
}
|
|
52
|
+
else if (isLikeC4File(path)) {
|
|
53
|
+
logger.debug `file changed: ${path}`;
|
|
54
|
+
await this.services.workspace.DocumentBuilder.update([URI.file(path)], []);
|
|
55
|
+
}
|
|
56
|
+
else if (isManualLayoutFile(path)) {
|
|
57
|
+
logger.debug `manual layout file changed: ${path}`;
|
|
58
|
+
// TODO: optimize to only reload manual layouts instead of all projects
|
|
59
|
+
await this.services.workspace.ProjectsManager.reloadProjects();
|
|
60
|
+
}
|
|
61
|
+
else {
|
|
62
|
+
logger.warn `Unknown file change: ${path}`;
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
catch (error) {
|
|
66
|
+
logger.error(loggable(error));
|
|
67
|
+
}
|
|
68
|
+
};
|
|
69
|
+
const onRemove = async (path) => {
|
|
70
|
+
try {
|
|
71
|
+
if (isLikeC4Config(path)) {
|
|
72
|
+
logger.debug `project file removed: ${path}`;
|
|
73
|
+
await this.services.workspace.ProjectsManager.reloadProjects();
|
|
74
|
+
}
|
|
75
|
+
else if (isLikeC4File(path)) {
|
|
76
|
+
logger.debug `file removed: ${path}`;
|
|
77
|
+
await this.services.workspace.DocumentBuilder.update([], [URI.file(path)]);
|
|
78
|
+
}
|
|
79
|
+
else if (isManualLayoutFile(path)) {
|
|
80
|
+
logger.debug `manual layout file removed: ${path}`;
|
|
81
|
+
// TODO: optimize to only reload manual layouts instead of all projects
|
|
82
|
+
await this.services.workspace.ProjectsManager.reloadProjects();
|
|
83
|
+
}
|
|
84
|
+
else {
|
|
85
|
+
logger.warn `Unknown file removal: ${path}`;
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
catch (error) {
|
|
89
|
+
logger.error(loggable(error));
|
|
90
|
+
}
|
|
91
|
+
};
|
|
92
|
+
watcher.on('add', onAddOrChange)
|
|
93
|
+
.on('change', onAddOrChange)
|
|
94
|
+
.on('unlink', onRemove);
|
|
95
|
+
return watcher;
|
|
96
|
+
}
|
|
97
|
+
}
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
export const noopFileSystemWatcher = {
|
|
2
|
+
fileSystemWatcher: () => new NoopFileSystemWatcher(),
|
|
3
|
+
};
|
|
4
|
+
/**
|
|
5
|
+
* A no-op file system watcher.
|
|
6
|
+
*/
|
|
7
|
+
export class NoopFileSystemWatcher {
|
|
8
|
+
watch() {
|
|
9
|
+
return;
|
|
10
|
+
}
|
|
11
|
+
dispose() {
|
|
12
|
+
return Promise.resolve();
|
|
13
|
+
}
|
|
14
|
+
}
|
|
@@ -1,4 +1,3 @@
|
|
|
1
1
|
import type { FileSystemModuleContext } from './index';
|
|
2
2
|
export declare const LikeC4FileSystem: (ehableWatcher?: boolean) => FileSystemModuleContext;
|
|
3
|
-
export declare const isLikeC4File: (path: string) =>
|
|
4
|
-
export declare const isAnyLikeC4File: (path: string) => any;
|
|
3
|
+
export declare const isLikeC4File: (path: string) => boolean;
|
|
@@ -0,0 +1,126 @@
|
|
|
1
|
+
import { isLikeC4Config, loadConfig } from '@likec4/config/node';
|
|
2
|
+
import { fdir } from 'fdir';
|
|
3
|
+
import { URI } from 'langium';
|
|
4
|
+
import { NodeFileSystemProvider } from 'langium/node';
|
|
5
|
+
import { mkdirSync } from 'node:fs';
|
|
6
|
+
import { stat, unlink, writeFile } from 'node:fs/promises';
|
|
7
|
+
import { dirname } from 'node:path';
|
|
8
|
+
import { LikeC4LanguageMetaData } from '../generated/module';
|
|
9
|
+
import { Content, isLikeC4Builtin } from '../likec4lib';
|
|
10
|
+
import { logger as rootLogger } from '../logger';
|
|
11
|
+
import { chokidarFileSystemWatcher } from './ChokidarWatcher';
|
|
12
|
+
import { noopFileSystemWatcher } from './FileSystemWatcher';
|
|
13
|
+
const logger = rootLogger.getChild('filesystem');
|
|
14
|
+
export const LikeC4FileSystem = (ehableWatcher = true) => ({
|
|
15
|
+
fileSystemProvider: () => new SymLinkTraversingFileSystemProvider(),
|
|
16
|
+
...ehableWatcher ? chokidarFileSystemWatcher : noopFileSystemWatcher,
|
|
17
|
+
});
|
|
18
|
+
export const isLikeC4File = (path) => LikeC4LanguageMetaData.fileExtensions.some((ext) => path.endsWith(ext));
|
|
19
|
+
/**
|
|
20
|
+
* A file system provider that follows symbolic links.
|
|
21
|
+
* @see https://github.com/likec4/likec4/pull/1213
|
|
22
|
+
*/
|
|
23
|
+
class SymLinkTraversingFileSystemProvider extends NodeFileSystemProvider {
|
|
24
|
+
async readFile(uri) {
|
|
25
|
+
if (isLikeC4Builtin(uri)) {
|
|
26
|
+
return Promise.resolve(Content);
|
|
27
|
+
}
|
|
28
|
+
try {
|
|
29
|
+
return await super.readFile(uri);
|
|
30
|
+
}
|
|
31
|
+
catch (error) {
|
|
32
|
+
logger.warn(`Failed to read file ${uri.fsPath}`, { error });
|
|
33
|
+
return '';
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
async readDirectory(folderPath, opts) {
|
|
37
|
+
const recursive = opts?.recursive ?? true;
|
|
38
|
+
const entries = [];
|
|
39
|
+
try {
|
|
40
|
+
let crawler = new fdir()
|
|
41
|
+
.withSymlinks({ resolvePaths: false })
|
|
42
|
+
.withFullPaths()
|
|
43
|
+
.filter(isLikeC4File);
|
|
44
|
+
if (!recursive) {
|
|
45
|
+
crawler = crawler.withMaxDepth(1);
|
|
46
|
+
}
|
|
47
|
+
const crawled = await crawler
|
|
48
|
+
.crawl(folderPath.fsPath)
|
|
49
|
+
.withPromise();
|
|
50
|
+
for (const path of crawled) {
|
|
51
|
+
entries.push({
|
|
52
|
+
isFile: true,
|
|
53
|
+
isDirectory: false,
|
|
54
|
+
uri: URI.file(path),
|
|
55
|
+
});
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
catch (error) {
|
|
59
|
+
logger.warn(`Failed to read directory ${folderPath.fsPath}`, { error });
|
|
60
|
+
}
|
|
61
|
+
return entries;
|
|
62
|
+
}
|
|
63
|
+
async scanProjectFiles(folderUri) {
|
|
64
|
+
return await this.scanDirectory(folderUri, isLikeC4Config);
|
|
65
|
+
}
|
|
66
|
+
async scanDirectory(directory, filter) {
|
|
67
|
+
const entries = [];
|
|
68
|
+
try {
|
|
69
|
+
const crawled = await new fdir()
|
|
70
|
+
.withSymlinks({ resolvePaths: false })
|
|
71
|
+
.withFullPaths()
|
|
72
|
+
.filter(filter)
|
|
73
|
+
.crawl(directory.fsPath)
|
|
74
|
+
.withPromise();
|
|
75
|
+
for (const path of crawled) {
|
|
76
|
+
entries.push({
|
|
77
|
+
isFile: true,
|
|
78
|
+
isDirectory: false,
|
|
79
|
+
uri: URI.file(path),
|
|
80
|
+
});
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
catch (error) {
|
|
84
|
+
logger.warn(`Failed to scan directory ${directory.fsPath}`, { error });
|
|
85
|
+
}
|
|
86
|
+
return entries;
|
|
87
|
+
}
|
|
88
|
+
async loadProjectConfig(filepath) {
|
|
89
|
+
return await loadConfig(filepath);
|
|
90
|
+
}
|
|
91
|
+
async writeFile(uri, content) {
|
|
92
|
+
const dir = dirname(uri.fsPath);
|
|
93
|
+
const exists = await stat(dir).catch(() => null);
|
|
94
|
+
if (exists?.isFile()) {
|
|
95
|
+
throw new Error(`Cannot create directory ${dir} because a file with the same name exists.`);
|
|
96
|
+
}
|
|
97
|
+
if (!exists) {
|
|
98
|
+
logger.debug('creating directory {path}', { path: dir });
|
|
99
|
+
// Create the directory synchronously on purpose
|
|
100
|
+
// to prevent watchers from picking up the change too early
|
|
101
|
+
mkdirSync(dir, { recursive: true });
|
|
102
|
+
}
|
|
103
|
+
logger.debug('writing file {path}', { path: uri.fsPath });
|
|
104
|
+
return await writeFile(uri.fsPath, content, {
|
|
105
|
+
encoding: 'utf-8',
|
|
106
|
+
});
|
|
107
|
+
}
|
|
108
|
+
async deleteFile(uri) {
|
|
109
|
+
try {
|
|
110
|
+
const path = uri.fsPath;
|
|
111
|
+
const exists = await stat(path);
|
|
112
|
+
if (exists.isFile() || exists.isSymbolicLink()) {
|
|
113
|
+
await unlink(path);
|
|
114
|
+
logger.debug('deleted file {path}', { path });
|
|
115
|
+
return true;
|
|
116
|
+
}
|
|
117
|
+
else {
|
|
118
|
+
logger.warn('deleteFile failed: {path} does not exist, or is not a file', { path });
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
catch (error) {
|
|
122
|
+
logger.warn(`Failed to delete file ${uri.fsPath}`, { error });
|
|
123
|
+
}
|
|
124
|
+
return false;
|
|
125
|
+
}
|
|
126
|
+
}
|
|
@@ -14,14 +14,40 @@ export interface FileSystemProvider extends LangiumFileSystemProvider {
|
|
|
14
14
|
* @returns The project config.
|
|
15
15
|
*/
|
|
16
16
|
loadProjectConfig(filepath: URI): Promise<LikeC4ProjectConfig>;
|
|
17
|
+
/**
|
|
18
|
+
* Reads the directory information for the given URI.
|
|
19
|
+
* @param options.recursive If true, recursively reads the directory,
|
|
20
|
+
* @default true
|
|
21
|
+
*/
|
|
22
|
+
readDirectory(uri: URI, options?: {
|
|
23
|
+
recursive?: boolean;
|
|
24
|
+
}): Promise<FileSystemNode[]>;
|
|
25
|
+
/**
|
|
26
|
+
* Finds all files in the given directory, matching the given filter.
|
|
27
|
+
*/
|
|
28
|
+
scanDirectory(directory: URI, filter: (filepath: string) => boolean): Promise<FileSystemNode[]>;
|
|
29
|
+
/**
|
|
30
|
+
* Writes the content to the file system.
|
|
31
|
+
* Used by manual layouts.
|
|
32
|
+
*/
|
|
33
|
+
writeFile(uri: URI, content: string): Promise<void>;
|
|
34
|
+
/**
|
|
35
|
+
* Deletes the file from the file system.
|
|
36
|
+
* Used by manual layouts.
|
|
37
|
+
* @return true if the file was deleted, false if the file did not exist.
|
|
38
|
+
*/
|
|
39
|
+
deleteFile(uri: URI): Promise<boolean>;
|
|
17
40
|
}
|
|
18
41
|
export interface FileSystemModuleContext extends FileSystemWatcherModuleContext {
|
|
19
42
|
fileSystemProvider: (services: LangiumSharedCoreServices) => FileSystemProvider;
|
|
20
43
|
}
|
|
21
44
|
export declare class NoopFileSystemProvider implements FileSystemProvider {
|
|
22
45
|
scanProjectFiles(): Promise<FileSystemNode[]>;
|
|
46
|
+
scanDirectory(): Promise<FileSystemNode[]>;
|
|
23
47
|
readFile(): Promise<string>;
|
|
24
48
|
readDirectory(): Promise<FileSystemNode[]>;
|
|
25
49
|
loadProjectConfig(): Promise<LikeC4ProjectConfig>;
|
|
50
|
+
writeFile(): Promise<void>;
|
|
51
|
+
deleteFile(): Promise<boolean>;
|
|
26
52
|
}
|
|
27
53
|
export declare const NoopFileSystem: FileSystemModuleContext;
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
import { URI } from 'vscode-uri';
|
|
2
|
+
import { noopFileSystemWatcher } from './FileSystemWatcher';
|
|
3
|
+
export class NoopFileSystemProvider {
|
|
4
|
+
scanProjectFiles() {
|
|
5
|
+
return Promise.resolve([]);
|
|
6
|
+
}
|
|
7
|
+
scanDirectory() {
|
|
8
|
+
return Promise.resolve([]);
|
|
9
|
+
}
|
|
10
|
+
readFile() {
|
|
11
|
+
throw new Error('No file system is available.');
|
|
12
|
+
}
|
|
13
|
+
readDirectory() {
|
|
14
|
+
return Promise.resolve([]);
|
|
15
|
+
}
|
|
16
|
+
loadProjectConfig() {
|
|
17
|
+
throw new Error('No file system is available.');
|
|
18
|
+
}
|
|
19
|
+
writeFile() {
|
|
20
|
+
throw new Error('No file system is available.');
|
|
21
|
+
}
|
|
22
|
+
deleteFile() {
|
|
23
|
+
throw new Error('No file system is available.');
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
export const NoopFileSystem = {
|
|
27
|
+
fileSystemProvider: () => new NoopFileSystemProvider(),
|
|
28
|
+
...noopFileSystemWatcher,
|
|
29
|
+
};
|