@likec4/language-server 1.47.0 → 1.49.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 +2 -2
- package/browser-worker/package.json +2 -2
- package/bundled/package.json +4 -0
- package/dist/THIRD-PARTY-LICENSES.md +178 -0
- package/dist/_chunks/ConfigurableLayouter.mjs +1 -0
- package/dist/_chunks/LikeC4FileSystem.mjs +3 -0
- package/dist/_chunks/LikeC4Views.mjs +34 -0
- package/dist/_chunks/ProjectsManager.mjs +1 -0
- package/dist/_chunks/WithMCPServer.mjs +481 -0
- package/dist/_chunks/icons.mjs +2 -0
- package/dist/_chunks/index.d.mts +3107 -0
- package/dist/_chunks/libs/@msgpack/msgpack.mjs +1 -0
- package/dist/_chunks/libs/eventemitter3.mjs +1 -0
- package/dist/_chunks/libs/fast-equals.mjs +1 -0
- package/dist/_chunks/libs/p-queue.mjs +1 -0
- package/dist/_chunks/libs/parse-ms.mjs +1 -0
- package/dist/_chunks/libs/picomatch.mjs +1 -0
- package/dist/_chunks/libs/pretty-ms.mjs +1 -0
- package/dist/_chunks/libs/remeda.mjs +1 -0
- package/dist/_chunks/libs/strip-indent.mjs +1 -0
- package/dist/_chunks/libs/ufo.mjs +1 -0
- package/dist/_chunks/logger.mjs +1 -0
- package/dist/_chunks/rolldown-runtime.mjs +1 -0
- package/dist/_chunks/utils.mjs +1 -0
- package/dist/browser/index.d.mts +10 -0
- package/dist/browser/index.mjs +1 -0
- package/dist/browser/worker.d.mts +1 -0
- package/dist/browser/worker.mjs +1 -0
- package/dist/bundled.d.mts +27 -0
- package/dist/bundled.mjs +1 -4853
- package/dist/filesystem/index.d.mts +2 -0
- package/dist/filesystem/index.mjs +1 -0
- package/dist/index.d.mts +39 -0
- package/dist/index.mjs +1 -0
- package/dist/likec4lib.d.mts +16 -0
- package/dist/likec4lib.mjs +1 -0
- package/dist/mcp/index.d.mts +2 -0
- package/dist/mcp/index.mjs +1 -0
- package/dist/module.d.mts +126 -0
- package/dist/module.mjs +1 -0
- package/dist/protocol.d.mts +315 -0
- package/dist/protocol.mjs +1 -0
- package/filesystem/package.json +4 -0
- package/likec4lib/package.json +2 -2
- package/mcp/package.json +4 -0
- package/module/package.json +4 -0
- package/package.json +128 -62
- package/protocol/package.json +2 -2
- package/LICENSE +0 -21
- package/dist/LikeC4LanguageServices.d.ts +0 -115
- package/dist/LikeC4LanguageServices.js +0 -164
- package/dist/Rpc.d.ts +0 -9
- package/dist/Rpc.js +0 -276
- package/dist/ast.d.ts +0 -226
- package/dist/ast.js +0 -264
- package/dist/browser-worker.d.ts +0 -1
- package/dist/browser-worker.js +0 -4
- package/dist/browser.d.ts +0 -10
- package/dist/browser.js +0 -34
- package/dist/bundled.d.ts +0 -8
- package/dist/bundled.js +0 -44
- package/dist/documentation/documentation-provider.d.ts +0 -8
- package/dist/documentation/documentation-provider.js +0 -51
- package/dist/documentation/index.d.ts +0 -1
- package/dist/documentation/index.js +0 -1
- package/dist/empty.d.ts +0 -2
- package/dist/empty.js +0 -2
- package/dist/filesystem/ChokidarWatcher.d.ts +0 -19
- package/dist/filesystem/ChokidarWatcher.js +0 -133
- package/dist/filesystem/FileSystemWatcher.d.ts +0 -19
- package/dist/filesystem/FileSystemWatcher.js +0 -14
- package/dist/filesystem/LikeC4FileSystem.d.ts +0 -3
- package/dist/filesystem/LikeC4FileSystem.js +0 -140
- package/dist/filesystem/index.d.ts +0 -55
- package/dist/filesystem/index.js +0 -29
- package/dist/formatting/LikeC4Formatter.d.ts +0 -59
- package/dist/formatting/LikeC4Formatter.js +0 -637
- package/dist/formatting/utils.d.ts +0 -6
- package/dist/formatting/utils.js +0 -18
- package/dist/generated/ast.d.ts +0 -1411
- package/dist/generated/ast.js +0 -2207
- package/dist/generated/grammar.d.ts +0 -6
- package/dist/generated/grammar.js +0 -7
- package/dist/generated/module.d.ts +0 -14
- package/dist/generated/module.js +0 -27
- package/dist/generated-lib/icons.d.ts +0 -1
- package/dist/generated-lib/icons.js +0 -18
- package/dist/index.d.ts +0 -37
- package/dist/index.js +0 -54
- package/dist/likec4lib.d.ts +0 -6
- package/dist/likec4lib.js +0 -7
- package/dist/logger.d.ts +0 -17
- package/dist/logger.js +0 -81
- package/dist/lsp/CodeActionProvider.d.ts +0 -14
- package/dist/lsp/CodeActionProvider.js +0 -33
- package/dist/lsp/CodeLensProvider.d.ts +0 -9
- package/dist/lsp/CodeLensProvider.js +0 -44
- package/dist/lsp/CompletionProvider.d.ts +0 -13
- package/dist/lsp/CompletionProvider.js +0 -238
- package/dist/lsp/DocumentHighlightProvider.d.ts +0 -9
- package/dist/lsp/DocumentHighlightProvider.js +0 -10
- package/dist/lsp/DocumentLinkProvider.d.ts +0 -11
- package/dist/lsp/DocumentLinkProvider.js +0 -58
- package/dist/lsp/DocumentSymbolProvider.d.ts +0 -33
- package/dist/lsp/DocumentSymbolProvider.js +0 -317
- package/dist/lsp/HoverProvider.d.ts +0 -10
- package/dist/lsp/HoverProvider.js +0 -106
- package/dist/lsp/RenameProvider.d.ts +0 -5
- package/dist/lsp/RenameProvider.js +0 -6
- package/dist/lsp/SemanticTokenProvider.d.ts +0 -19
- package/dist/lsp/SemanticTokenProvider.js +0 -305
- package/dist/lsp/index.d.ts +0 -8
- package/dist/lsp/index.js +0 -9
- package/dist/mcp/MCPServerFactory.d.ts +0 -8
- package/dist/mcp/MCPServerFactory.js +0 -73
- package/dist/mcp/NoopLikeC4MCPServer.d.ts +0 -9
- package/dist/mcp/NoopLikeC4MCPServer.js +0 -17
- package/dist/mcp/interfaces.d.ts +0 -13
- package/dist/mcp/interfaces.js +0 -4
- package/dist/mcp/server/StdioLikeC4MCPServer.d.ts +0 -16
- package/dist/mcp/server/StdioLikeC4MCPServer.js +0 -51
- package/dist/mcp/server/StreamableLikeC4MCPServer.d.ts +0 -16
- package/dist/mcp/server/StreamableLikeC4MCPServer.js +0 -121
- package/dist/mcp/server/WithMCPServer.d.ts +0 -4
- package/dist/mcp/server/WithMCPServer.js +0 -54
- package/dist/mcp/tools/_common.d.ts +0 -88
- package/dist/mcp/tools/_common.js +0 -49
- package/dist/mcp/tools/find-relationships.d.ts +0 -202
- package/dist/mcp/tools/find-relationships.js +0 -150
- package/dist/mcp/tools/list-projects.d.ts +0 -194
- package/dist/mcp/tools/list-projects.js +0 -62
- package/dist/mcp/tools/open-view.d.ts +0 -200
- package/dist/mcp/tools/open-view.js +0 -52
- package/dist/mcp/tools/read-deployment.d.ts +0 -200
- package/dist/mcp/tools/read-deployment.js +0 -150
- package/dist/mcp/tools/read-element.d.ts +0 -200
- package/dist/mcp/tools/read-element.js +0 -218
- package/dist/mcp/tools/read-project-summary.d.ts +0 -198
- package/dist/mcp/tools/read-project-summary.js +0 -176
- package/dist/mcp/tools/read-view.d.ts +0 -200
- package/dist/mcp/tools/read-view.js +0 -203
- package/dist/mcp/tools/search-element.d.ts +0 -198
- package/dist/mcp/tools/search-element.js +0 -177
- package/dist/mcp/utils.d.ts +0 -18
- package/dist/mcp/utils.js +0 -48
- package/dist/model/builder/MergedExtends.d.ts +0 -13
- package/dist/model/builder/MergedExtends.js +0 -74
- package/dist/model/builder/MergedSpecification.d.ts +0 -32
- package/dist/model/builder/MergedSpecification.js +0 -175
- package/dist/model/builder/buildModel.d.ts +0 -16
- package/dist/model/builder/buildModel.js +0 -245
- package/dist/model/deployments-index.d.ts +0 -10
- package/dist/model/deployments-index.js +0 -102
- package/dist/model/fqn-index.d.ts +0 -61
- package/dist/model/fqn-index.js +0 -253
- package/dist/model/index.d.ts +0 -6
- package/dist/model/index.js +0 -6
- package/dist/model/model-builder.d.ts +0 -54
- package/dist/model/model-builder.js +0 -233
- package/dist/model/model-locator.d.ts +0 -39
- package/dist/model/model-locator.js +0 -240
- package/dist/model/model-parser-where.d.ts +0 -4
- package/dist/model/model-parser-where.js +0 -81
- package/dist/model/model-parser.d.ts +0 -645
- package/dist/model/model-parser.js +0 -133
- package/dist/model/parser/Base.d.ts +0 -69
- package/dist/model/parser/Base.js +0 -382
- package/dist/model/parser/DeploymentModelParser.d.ts +0 -71
- package/dist/model/parser/DeploymentModelParser.js +0 -176
- package/dist/model/parser/DeploymentViewParser.d.ts +0 -75
- package/dist/model/parser/DeploymentViewParser.js +0 -86
- package/dist/model/parser/FqnRefParser.d.ts +0 -66
- package/dist/model/parser/FqnRefParser.js +0 -382
- package/dist/model/parser/GlobalsParser.d.ts +0 -109
- package/dist/model/parser/GlobalsParser.js +0 -84
- package/dist/model/parser/ImportsParser.d.ts +0 -46
- package/dist/model/parser/ImportsParser.js +0 -24
- package/dist/model/parser/ModelParser.d.ts +0 -71
- package/dist/model/parser/ModelParser.js +0 -209
- package/dist/model/parser/PredicatesParser.d.ts +0 -75
- package/dist/model/parser/PredicatesParser.js +0 -45
- package/dist/model/parser/SpecificationParser.d.ts +0 -53
- package/dist/model/parser/SpecificationParser.js +0 -113
- package/dist/model/parser/ValueConverter.d.ts +0 -4
- package/dist/model/parser/ValueConverter.js +0 -12
- package/dist/model/parser/ViewsParser.d.ts +0 -112
- package/dist/model/parser/ViewsParser.js +0 -492
- package/dist/model-change/ModelChanges.d.ts +0 -18
- package/dist/model-change/ModelChanges.js +0 -129
- package/dist/model-change/changeElementStyle.d.ts +0 -16
- package/dist/model-change/changeElementStyle.js +0 -134
- package/dist/model-change/changeViewLayout.d.ts +0 -12
- package/dist/model-change/changeViewLayout.js +0 -28
- package/dist/model-change/removeManualLayoutV1.d.ts +0 -7
- package/dist/model-change/removeManualLayoutV1.js +0 -27
- package/dist/module.d.ts +0 -92
- package/dist/module.js +0 -143
- package/dist/protocol.d.ts +0 -289
- package/dist/protocol.js +0 -123
- package/dist/references/index.d.ts +0 -3
- package/dist/references/index.js +0 -3
- package/dist/references/name-provider.d.ts +0 -9
- package/dist/references/name-provider.js +0 -37
- package/dist/references/scope-computation.d.ts +0 -20
- package/dist/references/scope-computation.js +0 -288
- package/dist/references/scope-provider.d.ts +0 -40
- package/dist/references/scope-provider.js +0 -239
- package/dist/shared/NodeKindProvider.d.ts +0 -15
- package/dist/shared/NodeKindProvider.js +0 -57
- package/dist/shared/WorkspaceSymbolProvider.d.ts +0 -3
- package/dist/shared/WorkspaceSymbolProvider.js +0 -3
- package/dist/shared/index.d.ts +0 -2
- package/dist/shared/index.js +0 -2
- package/dist/test/index.d.ts +0 -1
- package/dist/test/index.js +0 -1
- package/dist/test/testServices.d.ts +0 -64
- package/dist/test/testServices.js +0 -210
- package/dist/utils/disposable.d.ts +0 -8
- package/dist/utils/disposable.js +0 -26
- package/dist/utils/elementRef.d.ts +0 -11
- package/dist/utils/elementRef.js +0 -33
- package/dist/utils/fqnRef.d.ts +0 -11
- package/dist/utils/fqnRef.js +0 -63
- package/dist/utils/index.d.ts +0 -11
- package/dist/utils/index.js +0 -35
- package/dist/utils/printDocs.d.ts +0 -2
- package/dist/utils/printDocs.js +0 -1
- package/dist/utils/projectId.d.ts +0 -4
- package/dist/utils/projectId.js +0 -16
- package/dist/utils/stringHash.d.ts +0 -1
- package/dist/utils/stringHash.js +0 -5
- package/dist/validation/DocumentValidator.d.ts +0 -11
- package/dist/validation/DocumentValidator.js +0 -17
- package/dist/validation/_shared.d.ts +0 -3
- package/dist/validation/_shared.js +0 -26
- package/dist/validation/deployment-checks.d.ts +0 -7
- package/dist/validation/deployment-checks.js +0 -140
- package/dist/validation/dynamic-view.d.ts +0 -6
- package/dist/validation/dynamic-view.js +0 -67
- package/dist/validation/element-ref.d.ts +0 -4
- package/dist/validation/element-ref.js +0 -12
- package/dist/validation/element.d.ts +0 -4
- package/dist/validation/element.js +0 -49
- package/dist/validation/imports.d.ts +0 -4
- package/dist/validation/imports.js +0 -46
- package/dist/validation/index.d.ts +0 -15
- package/dist/validation/index.js +0 -167
- package/dist/validation/property-checks.d.ts +0 -7
- package/dist/validation/property-checks.js +0 -108
- package/dist/validation/relation.d.ts +0 -6
- package/dist/validation/relation.js +0 -141
- package/dist/validation/specification.d.ts +0 -12
- package/dist/validation/specification.js +0 -190
- package/dist/validation/view-checks.d.ts +0 -4
- package/dist/validation/view-checks.js +0 -46
- package/dist/validation/view-predicates/fqn-expr-with.d.ts +0 -4
- package/dist/validation/view-predicates/fqn-expr-with.js +0 -43
- package/dist/validation/view-predicates/fqn-ref-expr.d.ts +0 -4
- package/dist/validation/view-predicates/fqn-ref-expr.js +0 -51
- package/dist/validation/view-predicates/incoming.d.ts +0 -4
- package/dist/validation/view-predicates/incoming.js +0 -16
- package/dist/validation/view-predicates/index.d.ts +0 -6
- package/dist/validation/view-predicates/index.js +0 -6
- package/dist/validation/view-predicates/outgoing.d.ts +0 -4
- package/dist/validation/view-predicates/outgoing.js +0 -20
- package/dist/validation/view-predicates/relation-expr.d.ts +0 -4
- package/dist/validation/view-predicates/relation-expr.js +0 -46
- package/dist/validation/view-predicates/relation-with.d.ts +0 -4
- package/dist/validation/view-predicates/relation-with.js +0 -16
- package/dist/validation/view.d.ts +0 -4
- package/dist/validation/view.js +0 -42
- package/dist/view-utils/assignNavigateTo.d.ts +0 -2
- package/dist/view-utils/assignNavigateTo.js +0 -27
- package/dist/view-utils/index.d.ts +0 -2
- package/dist/view-utils/index.js +0 -2
- package/dist/view-utils/manual-layout.d.ts +0 -13
- package/dist/view-utils/manual-layout.js +0 -149
- package/dist/views/ConfigurableLayouter.d.ts +0 -7
- package/dist/views/ConfigurableLayouter.js +0 -51
- package/dist/views/LikeC4ManualLayouts.d.ts +0 -42
- package/dist/views/LikeC4ManualLayouts.js +0 -209
- package/dist/views/LikeC4Views.d.ts +0 -89
- package/dist/views/LikeC4Views.js +0 -216
- package/dist/views/index.d.ts +0 -4
- package/dist/views/index.js +0 -11
- package/dist/workspace/AstNodeDescriptionProvider.d.ts +0 -7
- package/dist/workspace/AstNodeDescriptionProvider.js +0 -18
- package/dist/workspace/IndexManager.d.ts +0 -10
- package/dist/workspace/IndexManager.js +0 -26
- package/dist/workspace/LangiumDocuments.d.ts +0 -29
- package/dist/workspace/LangiumDocuments.js +0 -104
- package/dist/workspace/ProjectsManager.d.ts +0 -134
- package/dist/workspace/ProjectsManager.js +0 -610
- package/dist/workspace/WorkspaceManager.d.ts +0 -31
- package/dist/workspace/WorkspaceManager.js +0 -132
- package/dist/workspace/index.d.ts +0 -5
- package/dist/workspace/index.js +0 -5
|
@@ -1,141 +0,0 @@
|
|
|
1
|
-
// SPDX-License-Identifier: MIT
|
|
2
|
-
//
|
|
3
|
-
// Copyright (c) 2023-2025 Denis Davydkov
|
|
4
|
-
// Copyright (c) 2025 NVIDIA CORPORATION & AFFILIATES. All rights reserved.
|
|
5
|
-
//
|
|
6
|
-
// Portions of this file have been modified by NVIDIA CORPORATION & AFFILIATES.
|
|
7
|
-
import { FqnRef, isSameHierarchy } from '@likec4/core';
|
|
8
|
-
import { AstUtils } from 'langium';
|
|
9
|
-
import { ast, isParsedLikeC4LangiumDocument } from '../ast';
|
|
10
|
-
import { safeCall } from '../utils';
|
|
11
|
-
import { stringHash } from '../utils/stringHash';
|
|
12
|
-
import { tryOrLog } from './_shared';
|
|
13
|
-
// Cache of relation match keys to avoid recomputing for every extend validation
|
|
14
|
-
let cachedRelationKeys = null;
|
|
15
|
-
let cachedDocsFingerprint = null;
|
|
16
|
-
const computeDocsFingerprint = (docs) => {
|
|
17
|
-
return docs
|
|
18
|
-
.map(doc => {
|
|
19
|
-
const relationHashes = (doc.c4Relations ?? [])
|
|
20
|
-
.map(rel => stringHash('extend-relation', FqnRef.flatten(rel.source), FqnRef.flatten(rel.target), rel.kind ?? 'default', rel.title ?? ''))
|
|
21
|
-
.sort()
|
|
22
|
-
.join(',');
|
|
23
|
-
return `${doc.uri.toString()}:${relationHashes}`;
|
|
24
|
-
})
|
|
25
|
-
.join('|');
|
|
26
|
-
};
|
|
27
|
-
export const relationChecks = (services) => {
|
|
28
|
-
const modelParser = services.likec4.ModelParser;
|
|
29
|
-
return tryOrLog((el, accept) => {
|
|
30
|
-
const parser = modelParser.forDocument(AstUtils.getDocument(el));
|
|
31
|
-
const source = safeCall(() => parser._resolveRelationSource(el));
|
|
32
|
-
if (!source) {
|
|
33
|
-
accept('error', 'Source not resolved', {
|
|
34
|
-
node: el,
|
|
35
|
-
property: 'source',
|
|
36
|
-
});
|
|
37
|
-
return;
|
|
38
|
-
}
|
|
39
|
-
const target = safeCall(() => parser.parseFqnRef(el.target));
|
|
40
|
-
if (!target) {
|
|
41
|
-
accept('error', 'Target not resolved', {
|
|
42
|
-
node: el,
|
|
43
|
-
property: 'target',
|
|
44
|
-
});
|
|
45
|
-
return;
|
|
46
|
-
}
|
|
47
|
-
if (FqnRef.isImportRef(source)) {
|
|
48
|
-
if (FqnRef.isImportRef(target)) {
|
|
49
|
-
accept('warning', 'Relationship between imported elements may not be visible in origin projects', {
|
|
50
|
-
node: el,
|
|
51
|
-
});
|
|
52
|
-
}
|
|
53
|
-
else {
|
|
54
|
-
accept('warning', 'Relationship from imported element to local element may not be visible in origin project', {
|
|
55
|
-
node: el,
|
|
56
|
-
property: 'source',
|
|
57
|
-
});
|
|
58
|
-
}
|
|
59
|
-
}
|
|
60
|
-
if (isSameHierarchy(FqnRef.flatten(source), FqnRef.flatten(target))) {
|
|
61
|
-
accept('error', 'Invalid parent-child relationship', {
|
|
62
|
-
node: el,
|
|
63
|
-
});
|
|
64
|
-
}
|
|
65
|
-
});
|
|
66
|
-
};
|
|
67
|
-
export const checkRelationBody = (_services) => {
|
|
68
|
-
return tryOrLog((body, accept) => {
|
|
69
|
-
const relation = body.$container;
|
|
70
|
-
if (relation.tags?.values && body.tags?.values) {
|
|
71
|
-
accept('error', 'Relation cannot have tags in both header and body', {
|
|
72
|
-
node: body.tags,
|
|
73
|
-
});
|
|
74
|
-
}
|
|
75
|
-
});
|
|
76
|
-
};
|
|
77
|
-
export const extendRelationChecks = (services) => {
|
|
78
|
-
const modelParser = services.likec4.ModelParser;
|
|
79
|
-
return tryOrLog((el, accept) => {
|
|
80
|
-
const parser = modelParser.forDocument(AstUtils.getDocument(el));
|
|
81
|
-
const source = safeCall(() => parser.parseFqnRef(el.source));
|
|
82
|
-
if (!source) {
|
|
83
|
-
accept('error', 'Source not resolved', {
|
|
84
|
-
node: el,
|
|
85
|
-
property: 'source',
|
|
86
|
-
});
|
|
87
|
-
return;
|
|
88
|
-
}
|
|
89
|
-
const target = safeCall(() => parser.parseFqnRef(el.target));
|
|
90
|
-
if (!target) {
|
|
91
|
-
accept('error', 'Target not resolved', {
|
|
92
|
-
node: el,
|
|
93
|
-
property: 'target',
|
|
94
|
-
});
|
|
95
|
-
return;
|
|
96
|
-
}
|
|
97
|
-
if (!FqnRef.isModelRef(source) && !FqnRef.isImportRef(source)) {
|
|
98
|
-
accept('error', 'Source must reference a model element', {
|
|
99
|
-
node: el,
|
|
100
|
-
property: 'source',
|
|
101
|
-
});
|
|
102
|
-
return;
|
|
103
|
-
}
|
|
104
|
-
if (!FqnRef.isModelRef(target) && !FqnRef.isImportRef(target)) {
|
|
105
|
-
accept('error', 'Target must reference a model element', {
|
|
106
|
-
node: el,
|
|
107
|
-
property: 'target',
|
|
108
|
-
});
|
|
109
|
-
return;
|
|
110
|
-
}
|
|
111
|
-
// Warn if this extend does not match any relation in the workspace
|
|
112
|
-
// Build a match key identical to buildModel.ts
|
|
113
|
-
const kind = (el.kind ?? el.dotKind?.kind)?.ref?.name ?? 'default';
|
|
114
|
-
// Normalize title using the same parser helper
|
|
115
|
-
const { title = '' } = parser.parseBaseProps({}, { title: el.title });
|
|
116
|
-
const extendKey = stringHash('extend-relation', FqnRef.flatten(source), FqnRef.flatten(target), kind, title);
|
|
117
|
-
// Build (or reuse) a Set of all relation match keys across the workspace.
|
|
118
|
-
// This avoids O(E x D x R) scans on large workspaces.
|
|
119
|
-
const docs = services.shared.workspace.LangiumDocuments.all
|
|
120
|
-
.toArray()
|
|
121
|
-
.filter(isParsedLikeC4LangiumDocument);
|
|
122
|
-
const fingerprint = computeDocsFingerprint(docs);
|
|
123
|
-
if (fingerprint !== cachedDocsFingerprint) {
|
|
124
|
-
const keys = new Set();
|
|
125
|
-
for (const d of docs) {
|
|
126
|
-
for (const rel of d.c4Relations ?? []) {
|
|
127
|
-
const key = stringHash('extend-relation', FqnRef.flatten(rel.source), FqnRef.flatten(rel.target), rel.kind ?? 'default', rel.title ?? '');
|
|
128
|
-
keys.add(key);
|
|
129
|
-
}
|
|
130
|
-
}
|
|
131
|
-
cachedRelationKeys = keys;
|
|
132
|
-
cachedDocsFingerprint = fingerprint;
|
|
133
|
-
}
|
|
134
|
-
const hasMatch = cachedRelationKeys?.has(extendKey) ?? false;
|
|
135
|
-
if (!hasMatch) {
|
|
136
|
-
accept('warning', 'This extend does not match any relation (by source, kind, target, title)', {
|
|
137
|
-
node: el,
|
|
138
|
-
});
|
|
139
|
-
}
|
|
140
|
-
});
|
|
141
|
-
};
|
|
@@ -1,12 +0,0 @@
|
|
|
1
|
-
import { type ValidationCheck } from 'langium';
|
|
2
|
-
import { ast } from '../ast';
|
|
3
|
-
import type { LikeC4Services } from '../module';
|
|
4
|
-
export declare const checkSpecificationRule: (_: LikeC4Services) => ValidationCheck<ast.SpecificationRule>;
|
|
5
|
-
export declare const checkModel: (_: LikeC4Services) => ValidationCheck<ast.Model>;
|
|
6
|
-
export declare const checkGlobals: (_: LikeC4Services) => ValidationCheck<ast.Globals>;
|
|
7
|
-
export declare const checkElementKind: (services: LikeC4Services) => ValidationCheck<ast.ElementKind>;
|
|
8
|
-
export declare const checkDeploymentNodeKind: (services: LikeC4Services) => ValidationCheck<ast.DeploymentNodeKind>;
|
|
9
|
-
export declare const checkTag: (services: LikeC4Services) => ValidationCheck<ast.Tag>;
|
|
10
|
-
export declare const checkRelationshipKind: (services: LikeC4Services) => ValidationCheck<ast.RelationshipKind>;
|
|
11
|
-
export declare const checkGlobalPredicate: (services: LikeC4Services) => ValidationCheck<ast.GlobalPredicateGroup | ast.GlobalDynamicPredicateGroup>;
|
|
12
|
-
export declare const checkGlobalStyleId: (services: LikeC4Services) => ValidationCheck<ast.GlobalStyleId>;
|
|
@@ -1,190 +0,0 @@
|
|
|
1
|
-
import { AstUtils } from 'langium';
|
|
2
|
-
import { ast } from '../ast';
|
|
3
|
-
import { projectIdFrom } from '../utils';
|
|
4
|
-
import { RESERVED_WORDS, tryOrLog } from './_shared';
|
|
5
|
-
export const checkSpecificationRule = (_) => {
|
|
6
|
-
return tryOrLog((node, accept) => {
|
|
7
|
-
if (node.$containerIndex && node.$containerIndex > 0) {
|
|
8
|
-
accept('warning', `Prefer one specification per document`, {
|
|
9
|
-
node: node,
|
|
10
|
-
property: 'name',
|
|
11
|
-
});
|
|
12
|
-
}
|
|
13
|
-
});
|
|
14
|
-
};
|
|
15
|
-
export const checkModel = (_) => {
|
|
16
|
-
return tryOrLog((node, accept) => {
|
|
17
|
-
if (node.$containerIndex && node.$containerIndex > 0) {
|
|
18
|
-
accept('warning', `Prefer one model per document`, {
|
|
19
|
-
node: node,
|
|
20
|
-
property: 'name',
|
|
21
|
-
});
|
|
22
|
-
}
|
|
23
|
-
});
|
|
24
|
-
};
|
|
25
|
-
export const checkGlobals = (_) => {
|
|
26
|
-
return tryOrLog((node, accept) => {
|
|
27
|
-
if (node.$containerIndex && node.$containerIndex > 0) {
|
|
28
|
-
accept('warning', `Prefer one global block per document`, {
|
|
29
|
-
node: node,
|
|
30
|
-
property: 'name',
|
|
31
|
-
});
|
|
32
|
-
}
|
|
33
|
-
});
|
|
34
|
-
};
|
|
35
|
-
export const checkElementKind = (services) => {
|
|
36
|
-
const index = services.shared.workspace.IndexManager;
|
|
37
|
-
return tryOrLog((node, accept) => {
|
|
38
|
-
if (RESERVED_WORDS.includes(node.name)) {
|
|
39
|
-
accept('error', `Reserved word: ${node.name}`, {
|
|
40
|
-
node: node,
|
|
41
|
-
property: 'name',
|
|
42
|
-
});
|
|
43
|
-
}
|
|
44
|
-
const projectId = projectIdFrom(node);
|
|
45
|
-
const sameKind = index
|
|
46
|
-
.projectElements(projectId, ast.ElementKind)
|
|
47
|
-
.filter(n => n.name === node.name && n.node !== node)
|
|
48
|
-
.head();
|
|
49
|
-
if (sameKind) {
|
|
50
|
-
const isAnotherDoc = sameKind.documentUri !== AstUtils.getDocument(node).uri;
|
|
51
|
-
accept('error', `Duplicate element kind '${node.name}'`, {
|
|
52
|
-
node: node,
|
|
53
|
-
property: 'name',
|
|
54
|
-
...isAnotherDoc && {
|
|
55
|
-
relatedInformation: [
|
|
56
|
-
{
|
|
57
|
-
location: {
|
|
58
|
-
range: sameKind.nameSegment.range,
|
|
59
|
-
uri: sameKind.documentUri.toString(),
|
|
60
|
-
},
|
|
61
|
-
message: `conflicting definition`,
|
|
62
|
-
},
|
|
63
|
-
],
|
|
64
|
-
},
|
|
65
|
-
});
|
|
66
|
-
}
|
|
67
|
-
});
|
|
68
|
-
};
|
|
69
|
-
export const checkDeploymentNodeKind = (services) => {
|
|
70
|
-
const index = services.shared.workspace.IndexManager;
|
|
71
|
-
return tryOrLog((node, accept) => {
|
|
72
|
-
if (RESERVED_WORDS.includes(node.name)) {
|
|
73
|
-
accept('error', `Reserved word: ${node.name}`, {
|
|
74
|
-
node: node,
|
|
75
|
-
property: 'name',
|
|
76
|
-
});
|
|
77
|
-
}
|
|
78
|
-
const projectId = projectIdFrom(node);
|
|
79
|
-
const sameKind = index
|
|
80
|
-
.projectElements(projectId, ast.DeploymentNodeKind)
|
|
81
|
-
.filter(n => n.name === node.name && n.node !== node)
|
|
82
|
-
.head();
|
|
83
|
-
if (sameKind) {
|
|
84
|
-
const isAnotherDoc = sameKind.documentUri !== AstUtils.getDocument(node).uri;
|
|
85
|
-
accept('error', `Duplicate deploymentNode kind '${node.name}'`, {
|
|
86
|
-
node: node,
|
|
87
|
-
property: 'name',
|
|
88
|
-
...isAnotherDoc && {
|
|
89
|
-
relatedInformation: [
|
|
90
|
-
{
|
|
91
|
-
location: {
|
|
92
|
-
range: sameKind.nameSegment.range,
|
|
93
|
-
uri: sameKind.documentUri.toString(),
|
|
94
|
-
},
|
|
95
|
-
message: `conflicting definition`,
|
|
96
|
-
},
|
|
97
|
-
],
|
|
98
|
-
},
|
|
99
|
-
});
|
|
100
|
-
}
|
|
101
|
-
});
|
|
102
|
-
};
|
|
103
|
-
export const checkTag = (services) => {
|
|
104
|
-
const index = services.shared.workspace.IndexManager;
|
|
105
|
-
return tryOrLog((node, accept) => {
|
|
106
|
-
const tagname = node.name;
|
|
107
|
-
const projectId = projectIdFrom(node);
|
|
108
|
-
const sameTag = index
|
|
109
|
-
.projectElements(projectId, ast.Tag)
|
|
110
|
-
.filter(n => n.name === tagname && n.node !== node)
|
|
111
|
-
.head();
|
|
112
|
-
if (sameTag) {
|
|
113
|
-
const isAnotherDoc = sameTag.documentUri !== AstUtils.getDocument(node).uri;
|
|
114
|
-
accept('error', `Duplicate tag '${node.name}'`, {
|
|
115
|
-
node,
|
|
116
|
-
property: 'name',
|
|
117
|
-
...isAnotherDoc && {
|
|
118
|
-
relatedInformation: [
|
|
119
|
-
{
|
|
120
|
-
location: {
|
|
121
|
-
range: sameTag.nameSegment.range,
|
|
122
|
-
uri: sameTag.documentUri.toString(),
|
|
123
|
-
},
|
|
124
|
-
message: `conflicting definition`,
|
|
125
|
-
},
|
|
126
|
-
],
|
|
127
|
-
},
|
|
128
|
-
});
|
|
129
|
-
}
|
|
130
|
-
});
|
|
131
|
-
};
|
|
132
|
-
export const checkRelationshipKind = (services) => {
|
|
133
|
-
const index = services.shared.workspace.IndexManager;
|
|
134
|
-
return tryOrLog((node, accept) => {
|
|
135
|
-
if (RESERVED_WORDS.includes(node.name)) {
|
|
136
|
-
accept('error', `Reserved word: ${node.name}`, {
|
|
137
|
-
node: node,
|
|
138
|
-
property: 'name',
|
|
139
|
-
});
|
|
140
|
-
}
|
|
141
|
-
const projectId = projectIdFrom(node);
|
|
142
|
-
const sameKinds = index
|
|
143
|
-
.projectElements(projectId, ast.RelationshipKind)
|
|
144
|
-
.filter(n => n.name === node.name)
|
|
145
|
-
.limit(2)
|
|
146
|
-
.count();
|
|
147
|
-
if (sameKinds > 1) {
|
|
148
|
-
accept('error', `Duplicate RelationshipKind '${node.name}'`, {
|
|
149
|
-
node: node,
|
|
150
|
-
property: 'name',
|
|
151
|
-
});
|
|
152
|
-
}
|
|
153
|
-
});
|
|
154
|
-
};
|
|
155
|
-
export const checkGlobalPredicate = (services) => {
|
|
156
|
-
const index = services.shared.workspace.IndexManager;
|
|
157
|
-
return tryOrLog((node, accept) => {
|
|
158
|
-
const projectId = projectIdFrom(node);
|
|
159
|
-
const predicateGroups = index.projectElements(projectId, ast.GlobalPredicateGroup);
|
|
160
|
-
const dynamicPredicateGroups = index.projectElements(projectId, ast.GlobalDynamicPredicateGroup);
|
|
161
|
-
const sameName = predicateGroups
|
|
162
|
-
.concat(dynamicPredicateGroups)
|
|
163
|
-
.filter(s => s.name === node.name)
|
|
164
|
-
.limit(2)
|
|
165
|
-
.count();
|
|
166
|
-
if (sameName > 1) {
|
|
167
|
-
accept('error', `Duplicate GlobalPredicateGroup or GlobalDynamicPredicateGroup name '${node.name}'`, {
|
|
168
|
-
node: node,
|
|
169
|
-
property: 'name',
|
|
170
|
-
});
|
|
171
|
-
}
|
|
172
|
-
});
|
|
173
|
-
};
|
|
174
|
-
export const checkGlobalStyleId = (services) => {
|
|
175
|
-
const index = services.shared.workspace.IndexManager;
|
|
176
|
-
return tryOrLog((node, accept) => {
|
|
177
|
-
const projectId = projectIdFrom(node);
|
|
178
|
-
const sameName = index
|
|
179
|
-
.projectElements(projectId, ast.GlobalStyleId)
|
|
180
|
-
.filter(s => s.name === node.name)
|
|
181
|
-
.limit(2)
|
|
182
|
-
.count();
|
|
183
|
-
if (sameName > 1) {
|
|
184
|
-
accept('error', `Duplicate GlobalStyleId name '${node.name}'`, {
|
|
185
|
-
node: node,
|
|
186
|
-
property: 'name',
|
|
187
|
-
});
|
|
188
|
-
}
|
|
189
|
-
});
|
|
190
|
-
};
|
|
@@ -1,46 +0,0 @@
|
|
|
1
|
-
import { ast } from '../ast';
|
|
2
|
-
import { tryOrLog } from './_shared';
|
|
3
|
-
// Helper to collect FqnExpr values from FqnExpressions linked list
|
|
4
|
-
function collectFqnExprs(exprs) {
|
|
5
|
-
const result = [];
|
|
6
|
-
let iter = exprs;
|
|
7
|
-
while (iter) {
|
|
8
|
-
if (iter.value) {
|
|
9
|
-
result.push(iter.value);
|
|
10
|
-
}
|
|
11
|
-
iter = iter.prev;
|
|
12
|
-
}
|
|
13
|
-
return result.reverse();
|
|
14
|
-
}
|
|
15
|
-
export const viewRuleRankChecks = (_services) => {
|
|
16
|
-
return tryOrLog((el, accept) => {
|
|
17
|
-
const targetExprs = collectFqnExprs(el.targets);
|
|
18
|
-
if (targetExprs.length < 2 && el.value === 'same') {
|
|
19
|
-
accept('warning', 'Rank rule should have at least 2 targets', {
|
|
20
|
-
node: el,
|
|
21
|
-
property: 'targets',
|
|
22
|
-
});
|
|
23
|
-
}
|
|
24
|
-
// Filter to only FqnRefExpr for parent comparison
|
|
25
|
-
const fqnRefExprs = targetExprs.filter(ast.isFqnRefExpr);
|
|
26
|
-
const firstParent = fqnRefExprs[0]?.ref?.parent;
|
|
27
|
-
for (let i = 1; i < fqnRefExprs.length; i++) {
|
|
28
|
-
const target = fqnRefExprs[i];
|
|
29
|
-
if (el.value === 'same' && !areSame(firstParent, target?.ref?.parent)) {
|
|
30
|
-
accept('error', 'All targets must have the same parent rank same', {
|
|
31
|
-
node: el,
|
|
32
|
-
property: 'targets',
|
|
33
|
-
});
|
|
34
|
-
}
|
|
35
|
-
}
|
|
36
|
-
});
|
|
37
|
-
};
|
|
38
|
-
function areSame(a, b) {
|
|
39
|
-
if (!a && !b)
|
|
40
|
-
return true;
|
|
41
|
-
if (!a || !b)
|
|
42
|
-
return false;
|
|
43
|
-
if (a.value.ref !== b.value.ref)
|
|
44
|
-
return false;
|
|
45
|
-
return areSame(a.parent, b.parent);
|
|
46
|
-
}
|
|
@@ -1,43 +0,0 @@
|
|
|
1
|
-
import { FqnExpr } from '@likec4/core/types';
|
|
2
|
-
import { nonexhaustive } from '@likec4/core/utils';
|
|
3
|
-
import { AstUtils } from 'langium';
|
|
4
|
-
import { ast, getViewRulePredicateContainer } from '../../ast';
|
|
5
|
-
import { tryOrLog } from '../_shared';
|
|
6
|
-
export const checkFqnExprWith = (services) => {
|
|
7
|
-
const modelParser = services.likec4.ModelParser;
|
|
8
|
-
return tryOrLog((el, accept) => {
|
|
9
|
-
const container = getViewRulePredicateContainer(el);
|
|
10
|
-
if (container?.$type !== 'DynamicViewIncludePredicate' && container?.isInclude !== true) {
|
|
11
|
-
accept('error', 'Invalid usage inside "exclude"', {
|
|
12
|
-
node: el,
|
|
13
|
-
});
|
|
14
|
-
return;
|
|
15
|
-
}
|
|
16
|
-
const isInsideDynamicView = container.$type === 'DynamicViewIncludePredicate';
|
|
17
|
-
const parser = modelParser.forDocument(AstUtils.getDocument(container));
|
|
18
|
-
let expr = FqnExpr.unwrap(parser.parseFqnExprWith(el).custom.expr);
|
|
19
|
-
switch (true) {
|
|
20
|
-
case FqnExpr.isWildcard(expr) && isInsideDynamicView:
|
|
21
|
-
case FqnExpr.isElementKindExpr(expr) && isInsideDynamicView:
|
|
22
|
-
case FqnExpr.isElementTagExpr(expr) && isInsideDynamicView: {
|
|
23
|
-
accept('warning', `Predicate is ignored, as not supported in dynamic views`, {
|
|
24
|
-
node: el,
|
|
25
|
-
});
|
|
26
|
-
return;
|
|
27
|
-
}
|
|
28
|
-
case FqnExpr.isWildcard(expr):
|
|
29
|
-
case FqnExpr.isModelRef(expr):
|
|
30
|
-
case FqnExpr.isDeploymentRef(expr):
|
|
31
|
-
return;
|
|
32
|
-
case FqnExpr.isElementKindExpr(expr):
|
|
33
|
-
case FqnExpr.isElementTagExpr(expr):
|
|
34
|
-
accept('error', 'Invalid target (expect reference to specific element)', {
|
|
35
|
-
node: el,
|
|
36
|
-
property: 'subject',
|
|
37
|
-
});
|
|
38
|
-
return;
|
|
39
|
-
default:
|
|
40
|
-
nonexhaustive(expr);
|
|
41
|
-
}
|
|
42
|
-
});
|
|
43
|
-
};
|
|
@@ -1,51 +0,0 @@
|
|
|
1
|
-
import { FqnExpr, FqnRef } from '@likec4/core/types';
|
|
2
|
-
import { AstUtils } from 'langium';
|
|
3
|
-
import { isTruthy } from 'remeda';
|
|
4
|
-
import { ast, getViewRulePredicateContainer, isFqnRefInsideDeployment } from '../../ast';
|
|
5
|
-
import { tryOrLog } from '../_shared';
|
|
6
|
-
export const checkFqnRefExpr = (services) => {
|
|
7
|
-
const modelParser = services.likec4.ModelParser;
|
|
8
|
-
return tryOrLog((node, accept) => {
|
|
9
|
-
const parser = modelParser.forDocument(AstUtils.getDocument(node));
|
|
10
|
-
const expr = parser.parseFqnRefExpr(node);
|
|
11
|
-
const viewRulePredicate = getViewRulePredicateContainer(node);
|
|
12
|
-
const isInsideDeploymentButNotStyle = isFqnRefInsideDeployment(node) && !AstUtils.hasContainerOfType(node, n => ast.isDeploymentViewRuleStyle(n) || ast.isViewRuleStyle(n));
|
|
13
|
-
if (viewRulePredicate?.$type === 'DeploymentViewRulePredicate' || isInsideDeploymentButNotStyle) {
|
|
14
|
-
const isPartOfRelationExpr = AstUtils.hasContainerOfType(node, ast.isRelationExpr);
|
|
15
|
-
// This expression is part of element predicate
|
|
16
|
-
if (!isPartOfRelationExpr) {
|
|
17
|
-
if (FqnExpr.isModelRef(expr)) {
|
|
18
|
-
accept('error', 'Deployment view predicate must reference deployment model', {
|
|
19
|
-
node,
|
|
20
|
-
});
|
|
21
|
-
return;
|
|
22
|
-
}
|
|
23
|
-
if (FqnExpr.isDeploymentRef(expr) && FqnRef.isInsideInstanceRef(expr.ref)) {
|
|
24
|
-
accept('error', 'Must reference deployment nodes or instances, but not internals', {
|
|
25
|
-
node,
|
|
26
|
-
});
|
|
27
|
-
return;
|
|
28
|
-
}
|
|
29
|
-
}
|
|
30
|
-
if (isTruthy(node.selector) && !ast.isDeploymentNode(node.ref.value?.ref)) {
|
|
31
|
-
accept('warning', `Selector '${node.selector}' applies to deployment nodes only, ignored here`, {
|
|
32
|
-
node,
|
|
33
|
-
property: 'selector',
|
|
34
|
-
});
|
|
35
|
-
}
|
|
36
|
-
return;
|
|
37
|
-
}
|
|
38
|
-
if (viewRulePredicate?.$type === 'DynamicViewIncludePredicate') {
|
|
39
|
-
switch (true) {
|
|
40
|
-
case FqnExpr.isElementKindExpr(expr):
|
|
41
|
-
case FqnExpr.isElementTagExpr(expr):
|
|
42
|
-
case FqnExpr.isWildcard(expr): {
|
|
43
|
-
accept('warning', `Predicate is ignored, as not supported in dynamic views`, {
|
|
44
|
-
node,
|
|
45
|
-
});
|
|
46
|
-
return;
|
|
47
|
-
}
|
|
48
|
-
}
|
|
49
|
-
}
|
|
50
|
-
});
|
|
51
|
-
};
|
|
@@ -1,16 +0,0 @@
|
|
|
1
|
-
import { AstUtils } from 'langium';
|
|
2
|
-
import { isNullish } from 'remeda';
|
|
3
|
-
import { ast } from '../../ast';
|
|
4
|
-
import { tryOrLog } from '../_shared';
|
|
5
|
-
export const checkIncomingRelationExpr = (_services) => {
|
|
6
|
-
return tryOrLog((el, accept) => {
|
|
7
|
-
if (el.to.$type === 'WildcardExpression' && !ast.isInOutRelationExpr(el.$container)) {
|
|
8
|
-
const view = AstUtils.getContainerOfType(el, ast.isElementView);
|
|
9
|
-
if (isNullish(view?.viewOf)) {
|
|
10
|
-
accept('warning', 'Predicate is ignored as it concerns all relationships', {
|
|
11
|
-
node: el,
|
|
12
|
-
});
|
|
13
|
-
}
|
|
14
|
-
}
|
|
15
|
-
});
|
|
16
|
-
};
|
|
@@ -1,6 +0,0 @@
|
|
|
1
|
-
export { checkFqnExprWith } from './fqn-expr-with';
|
|
2
|
-
export { checkFqnRefExpr } from './fqn-ref-expr';
|
|
3
|
-
export { checkIncomingRelationExpr } from './incoming';
|
|
4
|
-
export { checkOutgoingRelationExpr } from './outgoing';
|
|
5
|
-
export { checkRelationExpr } from './relation-expr';
|
|
6
|
-
export { checkRelationExprWith } from './relation-with';
|
|
@@ -1,6 +0,0 @@
|
|
|
1
|
-
export { checkFqnExprWith } from './fqn-expr-with';
|
|
2
|
-
export { checkFqnRefExpr } from './fqn-ref-expr';
|
|
3
|
-
export { checkIncomingRelationExpr } from './incoming';
|
|
4
|
-
export { checkOutgoingRelationExpr } from './outgoing';
|
|
5
|
-
export { checkRelationExpr } from './relation-expr';
|
|
6
|
-
export { checkRelationExprWith } from './relation-with';
|
|
@@ -1,20 +0,0 @@
|
|
|
1
|
-
import { AstUtils } from 'langium';
|
|
2
|
-
import { isNullish } from 'remeda';
|
|
3
|
-
import { ast, getViewRulePredicateContainer } from '../../ast';
|
|
4
|
-
import { tryOrLog } from '../_shared';
|
|
5
|
-
export const checkOutgoingRelationExpr = (_services) => {
|
|
6
|
-
return tryOrLog((el, accept) => {
|
|
7
|
-
const viewRulePredicate = getViewRulePredicateContainer(el);
|
|
8
|
-
if (viewRulePredicate?.$type !== 'ViewRulePredicate') {
|
|
9
|
-
return;
|
|
10
|
-
}
|
|
11
|
-
if (el.$container.$type !== 'DirectedRelationExpr' && el.from.$type === 'WildcardExpression') {
|
|
12
|
-
const view = AstUtils.getContainerOfType(el, ast.isElementView);
|
|
13
|
-
if (view && isNullish(view.viewOf)) {
|
|
14
|
-
accept('warning', 'Predicate is ignored as it concerns all relationships', {
|
|
15
|
-
node: el,
|
|
16
|
-
});
|
|
17
|
-
}
|
|
18
|
-
}
|
|
19
|
-
});
|
|
20
|
-
};
|
|
@@ -1,46 +0,0 @@
|
|
|
1
|
-
import { FqnExpr, RelationExpr } from '@likec4/core';
|
|
2
|
-
import { AstUtils } from 'langium';
|
|
3
|
-
import { ast } from '../../ast';
|
|
4
|
-
import { tryOrLog } from '../_shared';
|
|
5
|
-
export const checkRelationExpr = (services) => {
|
|
6
|
-
const ModelParser = services.likec4.ModelParser;
|
|
7
|
-
return tryOrLog((node, accept) => {
|
|
8
|
-
// if (node.$container.$type !== 'DeploymentViewRulePredicateExpression') {
|
|
9
|
-
// // skip validation for this node, validated by container
|
|
10
|
-
// return
|
|
11
|
-
// }
|
|
12
|
-
const predicate = AstUtils.getContainerOfType(node, ast.isDeploymentViewRulePredicate);
|
|
13
|
-
if (!predicate || predicate.isInclude !== true) {
|
|
14
|
-
// no restriction for exclude predicate
|
|
15
|
-
return;
|
|
16
|
-
}
|
|
17
|
-
const doc = AstUtils.getDocument(node);
|
|
18
|
-
const parser = ModelParser.forDocument(doc);
|
|
19
|
-
let relationExpr = RelationExpr.unwrap(parser.parseRelationExpr(node));
|
|
20
|
-
const ModelRefOnlyExclude = 'Model reference is allowed in exclude predicate only';
|
|
21
|
-
if (RelationExpr.isDirect(relationExpr)) {
|
|
22
|
-
if (FqnExpr.isModelRef(relationExpr.source) || FqnExpr.isModelRef(relationExpr.target)) {
|
|
23
|
-
accept('error', ModelRefOnlyExclude, {
|
|
24
|
-
node: node,
|
|
25
|
-
});
|
|
26
|
-
}
|
|
27
|
-
return;
|
|
28
|
-
}
|
|
29
|
-
let expr;
|
|
30
|
-
if (RelationExpr.isIncoming(relationExpr)) {
|
|
31
|
-
expr = relationExpr.incoming;
|
|
32
|
-
}
|
|
33
|
-
else if (RelationExpr.isOutgoing(relationExpr)) {
|
|
34
|
-
expr = relationExpr.outgoing;
|
|
35
|
-
}
|
|
36
|
-
else {
|
|
37
|
-
expr = relationExpr.inout;
|
|
38
|
-
}
|
|
39
|
-
if (FqnExpr.isModelRef(expr)) {
|
|
40
|
-
accept('error', ModelRefOnlyExclude, {
|
|
41
|
-
node,
|
|
42
|
-
});
|
|
43
|
-
return;
|
|
44
|
-
}
|
|
45
|
-
});
|
|
46
|
-
};
|
|
@@ -1,16 +0,0 @@
|
|
|
1
|
-
import { ast, getViewRulePredicateContainer } from '../../ast';
|
|
2
|
-
import { tryOrLog } from '../_shared';
|
|
3
|
-
export const checkRelationExprWith = (_services) => {
|
|
4
|
-
return tryOrLog((el, accept) => {
|
|
5
|
-
const container = getViewRulePredicateContainer(el);
|
|
6
|
-
if (!container || container.$type == 'DynamicViewIncludePredicate') {
|
|
7
|
-
return;
|
|
8
|
-
}
|
|
9
|
-
if (!container.isInclude) {
|
|
10
|
-
accept('error', 'Invalid usage inside "exclude"', {
|
|
11
|
-
node: el,
|
|
12
|
-
});
|
|
13
|
-
return;
|
|
14
|
-
}
|
|
15
|
-
});
|
|
16
|
-
};
|