eventmodeler 0.4.7 → 0.6.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/dist/api/client-config.js +10 -0
- package/dist/api/generated/client/client.gen.js +235 -0
- package/dist/api/generated/client/index.js +6 -0
- package/dist/api/generated/client/types.gen.js +2 -0
- package/dist/api/generated/client/utils.gen.js +228 -0
- package/dist/api/generated/client.gen.js +4 -0
- package/dist/api/generated/core/auth.gen.js +14 -0
- package/dist/api/generated/core/bodySerializer.gen.js +57 -0
- package/dist/api/generated/core/params.gen.js +100 -0
- package/dist/api/generated/core/pathSerializer.gen.js +106 -0
- package/dist/api/generated/core/queryKeySerializer.gen.js +92 -0
- package/dist/api/generated/core/serverSentEvents.gen.js +133 -0
- package/dist/api/generated/core/types.gen.js +2 -0
- package/dist/api/generated/core/utils.gen.js +87 -0
- package/dist/api/generated/index.js +2 -0
- package/dist/api/generated/sdk.gen.js +4222 -0
- package/dist/api/generated/types.gen.js +2 -0
- package/dist/api/generated/zod.gen.js +7217 -0
- package/dist/commands/add.js +315 -0
- package/dist/commands/auth.js +14 -0
- package/dist/commands/create.js +192 -0
- package/dist/commands/design.js +108 -0
- package/dist/commands/guide.js +15 -0
- package/dist/commands/init.js +21 -0
- package/dist/commands/list-schemas.js +177 -0
- package/dist/commands/list.js +39 -0
- package/dist/commands/loop.js +101 -0
- package/dist/commands/map.js +40 -0
- package/dist/commands/mark.js +27 -0
- package/dist/commands/move.js +35 -0
- package/dist/commands/remove.js +170 -0
- package/dist/commands/rename.js +53 -0
- package/dist/commands/resize.js +30 -0
- package/dist/commands/search.js +14 -0
- package/dist/commands/set.js +199 -0
- package/dist/commands/show-schemas.js +259 -0
- package/dist/commands/show.js +56 -0
- package/dist/commands/summary.js +13 -0
- package/dist/commands/update.js +240 -0
- package/dist/index.js +46 -2379
- package/dist/lib/auth.js +1 -1
- package/dist/lib/config.js +0 -15
- package/dist/lib/excalidraw-schema.js +66 -0
- package/dist/lib/globals.js +8 -0
- package/dist/lib/model.js +11 -0
- package/dist/lib/project-config.js +20 -0
- package/dist/lib/resolve.js +59 -0
- package/dist/lib/scenario.js +15 -0
- package/dist/slices/add-scenario/index.js +2 -206
- package/dist/slices/guide/guides/codegen.js +1 -1
- package/dist/slices/guide/guides/connect-slices.js +12 -37
- package/dist/slices/guide/guides/create-slices.js +110 -140
- package/dist/slices/guide/guides/explore.js +37 -26
- package/dist/slices/guide/guides/information-flow.js +70 -82
- package/dist/slices/guide/guides/scenarios.js +82 -137
- package/dist/slices/guide/index.js +6 -6
- package/dist/slices/help/index.js +96 -0
- package/dist/slices/help/topics/build-codegen.js +109 -0
- package/dist/slices/help/topics/build-slice.js +147 -0
- package/dist/slices/help/topics/check-completeness.js +57 -0
- package/dist/slices/help/topics/connect-slices.js +99 -0
- package/dist/slices/help/topics/explore-model.js +112 -0
- package/dist/slices/help/topics/json-reference.js +188 -0
- package/dist/slices/help/topics/linked-copies.js +89 -0
- package/dist/slices/help/topics/manipulate-canvas.js +150 -0
- package/dist/slices/help/topics/write-scenarios.js +162 -0
- package/dist/slices/init/index.js +10 -4
- package/dist/slices/init/loop.js +60 -0
- package/dist/slices/login/index.js +2 -2
- package/dist/slices/logout/index.js +2 -2
- package/dist/slices/whoami/index.js +11 -36
- package/package.json +8 -3
- package/dist/api/index.d.ts +0 -285
- package/dist/api/index.js +0 -323
- package/dist/cloud/slices/index.d.ts +0 -276
- package/dist/cloud/slices/index.js +0 -406
- package/dist/eventmodeler.js +0 -5646
- package/dist/formatters.d.ts +0 -17
- package/dist/formatters.js +0 -482
- package/dist/index.d.ts +0 -2
- package/dist/lib/auth.d.ts +0 -24
- package/dist/lib/backend.d.ts +0 -43
- package/dist/lib/backend.js +0 -73
- package/dist/lib/chapter-utils.d.ts +0 -13
- package/dist/lib/chapter-utils.js +0 -71
- package/dist/lib/cloud-client.d.ts +0 -69
- package/dist/lib/cloud-client.js +0 -364
- package/dist/lib/config.d.ts +0 -30
- package/dist/lib/diff/merge-rules.d.ts +0 -45
- package/dist/lib/diff/merge-rules.js +0 -210
- package/dist/lib/diff/model-differ.d.ts +0 -8
- package/dist/lib/diff/model-differ.js +0 -568
- package/dist/lib/diff/three-way-merge.d.ts +0 -7
- package/dist/lib/diff/three-way-merge.js +0 -390
- package/dist/lib/diff/types.d.ts +0 -75
- package/dist/lib/diff/types.js +0 -1
- package/dist/lib/element-lookup.d.ts +0 -58
- package/dist/lib/element-lookup.js +0 -126
- package/dist/lib/file-loader.d.ts +0 -8
- package/dist/lib/file-loader.js +0 -108
- package/dist/lib/flow-utils.d.ts +0 -53
- package/dist/lib/flow-utils.js +0 -348
- package/dist/lib/format.d.ts +0 -10
- package/dist/lib/format.js +0 -23
- package/dist/lib/project-config.d.ts +0 -27
- package/dist/lib/slice-utils.d.ts +0 -59
- package/dist/lib/slice-utils.js +0 -140
- package/dist/local/slices/index.d.ts +0 -11
- package/dist/local/slices/index.js +0 -13
- package/dist/projection.d.ts +0 -3
- package/dist/projection.js +0 -828
- package/dist/slices/add-field/index.d.ts +0 -8
- package/dist/slices/add-field/index.js +0 -211
- package/dist/slices/add-scenario/index.d.ts +0 -27
- package/dist/slices/codegen-chapter-events/index.d.ts +0 -2
- package/dist/slices/codegen-chapter-events/index.js +0 -145
- package/dist/slices/codegen-slice/index.d.ts +0 -2
- package/dist/slices/codegen-slice/index.js +0 -448
- package/dist/slices/create-automation-slice/index.d.ts +0 -2
- package/dist/slices/create-automation-slice/index.js +0 -304
- package/dist/slices/create-flow/index.d.ts +0 -2
- package/dist/slices/create-flow/index.js +0 -183
- package/dist/slices/create-state-change-slice/index.d.ts +0 -2
- package/dist/slices/create-state-change-slice/index.js +0 -263
- package/dist/slices/create-state-view-slice/index.d.ts +0 -2
- package/dist/slices/create-state-view-slice/index.js +0 -128
- package/dist/slices/diff/index.d.ts +0 -11
- package/dist/slices/diff/index.js +0 -293
- package/dist/slices/export-eventmodel-to-json/index.d.ts +0 -2
- package/dist/slices/export-eventmodel-to-json/index.js +0 -355
- package/dist/slices/git/index.d.ts +0 -2
- package/dist/slices/git/index.js +0 -125
- package/dist/slices/guide/guides/codegen.d.ts +0 -5
- package/dist/slices/guide/guides/connect-slices.d.ts +0 -5
- package/dist/slices/guide/guides/create-slices.d.ts +0 -5
- package/dist/slices/guide/guides/explore.d.ts +0 -5
- package/dist/slices/guide/guides/information-flow.d.ts +0 -5
- package/dist/slices/guide/guides/scenarios.d.ts +0 -5
- package/dist/slices/guide/index.d.ts +0 -1
- package/dist/slices/import/index.d.ts +0 -8
- package/dist/slices/import/index.js +0 -63
- package/dist/slices/init/index.d.ts +0 -5
- package/dist/slices/list-chapters/index.d.ts +0 -3
- package/dist/slices/list-chapters/index.js +0 -21
- package/dist/slices/list-commands/index.d.ts +0 -3
- package/dist/slices/list-commands/index.js +0 -20
- package/dist/slices/list-events/index.d.ts +0 -3
- package/dist/slices/list-events/index.js +0 -98
- package/dist/slices/list-processors/index.d.ts +0 -3
- package/dist/slices/list-processors/index.js +0 -20
- package/dist/slices/list-readmodels/index.d.ts +0 -3
- package/dist/slices/list-readmodels/index.js +0 -21
- package/dist/slices/list-scenarios/index.d.ts +0 -3
- package/dist/slices/list-scenarios/index.js +0 -35
- package/dist/slices/list-screens/index.d.ts +0 -3
- package/dist/slices/list-screens/index.js +0 -47
- package/dist/slices/list-slices/index.d.ts +0 -3
- package/dist/slices/list-slices/index.js +0 -35
- package/dist/slices/login/index.d.ts +0 -1
- package/dist/slices/logout/index.d.ts +0 -1
- package/dist/slices/map-fields/index.d.ts +0 -2
- package/dist/slices/map-fields/index.js +0 -269
- package/dist/slices/mark-slice-status/index.d.ts +0 -2
- package/dist/slices/mark-slice-status/index.js +0 -31
- package/dist/slices/merge/index.d.ts +0 -19
- package/dist/slices/merge/index.js +0 -147
- package/dist/slices/open-app/index.d.ts +0 -1
- package/dist/slices/remove-field/index.d.ts +0 -8
- package/dist/slices/remove-field/index.js +0 -167
- package/dist/slices/remove-scenario/index.d.ts +0 -2
- package/dist/slices/remove-scenario/index.js +0 -77
- package/dist/slices/search/index.d.ts +0 -3
- package/dist/slices/search/index.js +0 -302
- package/dist/slices/show-actor/index.d.ts +0 -4
- package/dist/slices/show-actor/index.js +0 -115
- package/dist/slices/show-aggregate/index.d.ts +0 -3
- package/dist/slices/show-aggregate/index.js +0 -108
- package/dist/slices/show-aggregate-completeness/index.d.ts +0 -4
- package/dist/slices/show-aggregate-completeness/index.js +0 -181
- package/dist/slices/show-chapter/index.d.ts +0 -3
- package/dist/slices/show-chapter/index.js +0 -195
- package/dist/slices/show-command/index.d.ts +0 -3
- package/dist/slices/show-command/index.js +0 -133
- package/dist/slices/show-completeness/index.d.ts +0 -4
- package/dist/slices/show-completeness/index.js +0 -731
- package/dist/slices/show-event/index.d.ts +0 -3
- package/dist/slices/show-event/index.js +0 -118
- package/dist/slices/show-model-summary/index.d.ts +0 -3
- package/dist/slices/show-model-summary/index.js +0 -31
- package/dist/slices/show-processor/index.d.ts +0 -3
- package/dist/slices/show-processor/index.js +0 -111
- package/dist/slices/show-readmodel/index.d.ts +0 -3
- package/dist/slices/show-readmodel/index.js +0 -158
- package/dist/slices/show-scenario/index.d.ts +0 -3
- package/dist/slices/show-scenario/index.js +0 -196
- package/dist/slices/show-screen/index.d.ts +0 -3
- package/dist/slices/show-screen/index.js +0 -139
- package/dist/slices/show-slice/index.d.ts +0 -3
- package/dist/slices/show-slice/index.js +0 -696
- package/dist/slices/update-field/index.d.ts +0 -15
- package/dist/slices/update-field/index.js +0 -208
- package/dist/slices/whoami/index.d.ts +0 -2
- package/dist/types.d.ts +0 -195
- package/dist/types.js +0 -1
|
@@ -1,167 +0,0 @@
|
|
|
1
|
-
import { appendEvent } from '../../lib/file-loader.js';
|
|
2
|
-
import { findElementOrExit, excludeLinkedCopies } from '../../lib/element-lookup.js';
|
|
3
|
-
export function removeField(model, filePath, options, fieldName) {
|
|
4
|
-
// Determine which entity type
|
|
5
|
-
const entityCount = [options.command, options.event, options.readModel, options.screen, options.processor].filter(Boolean).length;
|
|
6
|
-
if (entityCount === 0) {
|
|
7
|
-
console.error('Error: Must specify one of --command, --event, --read-model, --screen, or --processor');
|
|
8
|
-
process.exit(1);
|
|
9
|
-
}
|
|
10
|
-
if (entityCount > 1) {
|
|
11
|
-
console.error('Error: Can only specify one of --command, --event, --read-model, --screen, or --processor');
|
|
12
|
-
process.exit(1);
|
|
13
|
-
}
|
|
14
|
-
if (options.command) {
|
|
15
|
-
removeFieldFromCommand(model, filePath, options.command, fieldName);
|
|
16
|
-
}
|
|
17
|
-
else if (options.event) {
|
|
18
|
-
removeFieldFromEvent(model, filePath, options.event, fieldName);
|
|
19
|
-
}
|
|
20
|
-
else if (options.readModel) {
|
|
21
|
-
removeFieldFromReadModel(model, filePath, options.readModel, fieldName);
|
|
22
|
-
}
|
|
23
|
-
else if (options.screen) {
|
|
24
|
-
removeFieldFromScreen(model, filePath, options.screen, fieldName);
|
|
25
|
-
}
|
|
26
|
-
else if (options.processor) {
|
|
27
|
-
removeFieldFromProcessor(model, filePath, options.processor, fieldName);
|
|
28
|
-
}
|
|
29
|
-
}
|
|
30
|
-
function removeFieldFromCommand(model, filePath, commandName, fieldName) {
|
|
31
|
-
const command = findElementOrExit(model.commands, commandName, 'command');
|
|
32
|
-
const fieldNameLower = fieldName.toLowerCase();
|
|
33
|
-
const field = command.fields.find(f => f.name.toLowerCase() === fieldNameLower);
|
|
34
|
-
if (!field) {
|
|
35
|
-
console.error(`Error: Field "${fieldName}" not found on command "${command.name}"`);
|
|
36
|
-
if (command.fields.length > 0) {
|
|
37
|
-
console.error('Available fields:');
|
|
38
|
-
for (const f of command.fields) {
|
|
39
|
-
console.error(` - ${f.name} (${f.fieldType})`);
|
|
40
|
-
}
|
|
41
|
-
}
|
|
42
|
-
else {
|
|
43
|
-
console.error('This command has no fields.');
|
|
44
|
-
}
|
|
45
|
-
process.exit(1);
|
|
46
|
-
}
|
|
47
|
-
appendEvent(filePath, {
|
|
48
|
-
type: 'CommandFieldRemoved',
|
|
49
|
-
data: {
|
|
50
|
-
commandStickyId: command.id,
|
|
51
|
-
fieldId: field.id,
|
|
52
|
-
timestamp: Date.now(),
|
|
53
|
-
},
|
|
54
|
-
});
|
|
55
|
-
console.log(`Removed field "${field.name}" from command "${command.name}"`);
|
|
56
|
-
}
|
|
57
|
-
function removeFieldFromEvent(model, filePath, eventName, fieldName) {
|
|
58
|
-
// Exclude linked copies - they share fields with their canonical original
|
|
59
|
-
const event = findElementOrExit(excludeLinkedCopies(model.events), eventName, 'event');
|
|
60
|
-
const fieldNameLower = fieldName.toLowerCase();
|
|
61
|
-
const field = event.fields.find(f => f.name.toLowerCase() === fieldNameLower);
|
|
62
|
-
if (!field) {
|
|
63
|
-
console.error(`Error: Field "${fieldName}" not found on event "${event.name}"`);
|
|
64
|
-
if (event.fields.length > 0) {
|
|
65
|
-
console.error('Available fields:');
|
|
66
|
-
for (const f of event.fields) {
|
|
67
|
-
console.error(` - ${f.name} (${f.fieldType})`);
|
|
68
|
-
}
|
|
69
|
-
}
|
|
70
|
-
else {
|
|
71
|
-
console.error('This event has no fields.');
|
|
72
|
-
}
|
|
73
|
-
process.exit(1);
|
|
74
|
-
}
|
|
75
|
-
appendEvent(filePath, {
|
|
76
|
-
type: 'EventFieldRemoved',
|
|
77
|
-
data: {
|
|
78
|
-
eventStickyId: event.id,
|
|
79
|
-
fieldId: field.id,
|
|
80
|
-
timestamp: Date.now(),
|
|
81
|
-
},
|
|
82
|
-
});
|
|
83
|
-
console.log(`Removed field "${field.name}" from event "${event.name}"`);
|
|
84
|
-
}
|
|
85
|
-
function removeFieldFromReadModel(model, filePath, readModelName, fieldName) {
|
|
86
|
-
// Exclude linked copies - they share fields with their canonical original
|
|
87
|
-
const readModel = findElementOrExit(excludeLinkedCopies(model.readModels), readModelName, 'read model');
|
|
88
|
-
const fieldNameLower = fieldName.toLowerCase();
|
|
89
|
-
const field = readModel.fields.find(f => f.name.toLowerCase() === fieldNameLower);
|
|
90
|
-
if (!field) {
|
|
91
|
-
console.error(`Error: Field "${fieldName}" not found on read model "${readModel.name}"`);
|
|
92
|
-
if (readModel.fields.length > 0) {
|
|
93
|
-
console.error('Available fields:');
|
|
94
|
-
for (const f of readModel.fields) {
|
|
95
|
-
console.error(` - ${f.name} (${f.fieldType})`);
|
|
96
|
-
}
|
|
97
|
-
}
|
|
98
|
-
else {
|
|
99
|
-
console.error('This read model has no fields.');
|
|
100
|
-
}
|
|
101
|
-
process.exit(1);
|
|
102
|
-
}
|
|
103
|
-
appendEvent(filePath, {
|
|
104
|
-
type: 'ReadModelFieldRemoved',
|
|
105
|
-
data: {
|
|
106
|
-
readModelStickyId: readModel.id,
|
|
107
|
-
fieldId: field.id,
|
|
108
|
-
timestamp: Date.now(),
|
|
109
|
-
},
|
|
110
|
-
});
|
|
111
|
-
console.log(`Removed field "${field.name}" from read model "${readModel.name}"`);
|
|
112
|
-
}
|
|
113
|
-
function removeFieldFromScreen(model, filePath, screenName, fieldName) {
|
|
114
|
-
// Exclude linked copies - they share fields with their canonical original
|
|
115
|
-
const screen = findElementOrExit(excludeLinkedCopies(model.screens), screenName, 'screen');
|
|
116
|
-
const fieldNameLower = fieldName.toLowerCase();
|
|
117
|
-
const field = screen.fields.find(f => f.name.toLowerCase() === fieldNameLower);
|
|
118
|
-
if (!field) {
|
|
119
|
-
console.error(`Error: Field "${fieldName}" not found on screen "${screen.name}"`);
|
|
120
|
-
if (screen.fields.length > 0) {
|
|
121
|
-
console.error('Available fields:');
|
|
122
|
-
for (const f of screen.fields) {
|
|
123
|
-
console.error(` - ${f.name} (${f.fieldType})`);
|
|
124
|
-
}
|
|
125
|
-
}
|
|
126
|
-
else {
|
|
127
|
-
console.error('This screen has no fields.');
|
|
128
|
-
}
|
|
129
|
-
process.exit(1);
|
|
130
|
-
}
|
|
131
|
-
appendEvent(filePath, {
|
|
132
|
-
type: 'ScreenFieldRemoved',
|
|
133
|
-
data: {
|
|
134
|
-
screenId: screen.id,
|
|
135
|
-
fieldId: field.id,
|
|
136
|
-
timestamp: Date.now(),
|
|
137
|
-
},
|
|
138
|
-
});
|
|
139
|
-
console.log(`Removed field "${field.name}" from screen "${screen.name}"`);
|
|
140
|
-
}
|
|
141
|
-
function removeFieldFromProcessor(model, filePath, processorName, fieldName) {
|
|
142
|
-
const processor = findElementOrExit(model.processors, processorName, 'processor');
|
|
143
|
-
const fieldNameLower = fieldName.toLowerCase();
|
|
144
|
-
const field = processor.fields.find(f => f.name.toLowerCase() === fieldNameLower);
|
|
145
|
-
if (!field) {
|
|
146
|
-
console.error(`Error: Field "${fieldName}" not found on processor "${processor.name}"`);
|
|
147
|
-
if (processor.fields.length > 0) {
|
|
148
|
-
console.error('Available fields:');
|
|
149
|
-
for (const f of processor.fields) {
|
|
150
|
-
console.error(` - ${f.name} (${f.fieldType})`);
|
|
151
|
-
}
|
|
152
|
-
}
|
|
153
|
-
else {
|
|
154
|
-
console.error('This processor has no fields.');
|
|
155
|
-
}
|
|
156
|
-
process.exit(1);
|
|
157
|
-
}
|
|
158
|
-
appendEvent(filePath, {
|
|
159
|
-
type: 'ProcessorFieldRemoved',
|
|
160
|
-
data: {
|
|
161
|
-
processorId: processor.id,
|
|
162
|
-
fieldId: field.id,
|
|
163
|
-
timestamp: Date.now(),
|
|
164
|
-
},
|
|
165
|
-
});
|
|
166
|
-
console.log(`Removed field "${field.name}" from processor "${processor.name}"`);
|
|
167
|
-
}
|
|
@@ -1,77 +0,0 @@
|
|
|
1
|
-
import { appendEvent } from '../../lib/file-loader.js';
|
|
2
|
-
import { findElement, formatElementWithId } from '../../lib/element-lookup.js';
|
|
3
|
-
export function removeScenario(model, filePath, scenarioName, sliceName) {
|
|
4
|
-
// Check for UUID lookup
|
|
5
|
-
if (scenarioName.startsWith('id:')) {
|
|
6
|
-
const idSearch = scenarioName.slice(3).toLowerCase();
|
|
7
|
-
const scenario = [...model.scenarios.values()].find(s => s.id.toLowerCase().startsWith(idSearch));
|
|
8
|
-
if (!scenario) {
|
|
9
|
-
console.error(`Error: Scenario not found with ID starting with: "${idSearch}"`);
|
|
10
|
-
console.error('Available scenarios:');
|
|
11
|
-
for (const s of model.scenarios.values()) {
|
|
12
|
-
const slice = model.slices.get(s.sliceId);
|
|
13
|
-
console.error(` - "${s.name}" (id: ${s.id.slice(0, 8)}) (in slice "${slice?.name ?? 'unknown'}")`);
|
|
14
|
-
}
|
|
15
|
-
process.exit(1);
|
|
16
|
-
}
|
|
17
|
-
const slice = model.slices.get(scenario.sliceId);
|
|
18
|
-
appendEvent(filePath, {
|
|
19
|
-
type: 'ScenarioRemoved',
|
|
20
|
-
data: {
|
|
21
|
-
scenarioId: scenario.id,
|
|
22
|
-
timestamp: Date.now(),
|
|
23
|
-
},
|
|
24
|
-
});
|
|
25
|
-
console.log(`Removed scenario "${scenario.name}" from slice "${slice?.name ?? 'unknown'}"`);
|
|
26
|
-
return;
|
|
27
|
-
}
|
|
28
|
-
// Case-insensitive exact name match
|
|
29
|
-
const scenarioNameLower = scenarioName.toLowerCase();
|
|
30
|
-
let matchingScenarios = [...model.scenarios.values()].filter(s => s.name.toLowerCase() === scenarioNameLower);
|
|
31
|
-
// If slice name provided, filter to that slice using exact match
|
|
32
|
-
if (sliceName && matchingScenarios.length > 0) {
|
|
33
|
-
const sliceResult = findElement(model.slices, sliceName);
|
|
34
|
-
if (sliceResult.success) {
|
|
35
|
-
matchingScenarios = matchingScenarios.filter(s => s.sliceId === sliceResult.element.id);
|
|
36
|
-
}
|
|
37
|
-
else if (sliceResult.error === 'ambiguous') {
|
|
38
|
-
console.error(`Error: Multiple slices found with name "${sliceName}"`);
|
|
39
|
-
console.error('Please specify using the slice ID:');
|
|
40
|
-
for (const s of sliceResult.matches) {
|
|
41
|
-
console.error(` - ${formatElementWithId(s, false)}`);
|
|
42
|
-
}
|
|
43
|
-
process.exit(1);
|
|
44
|
-
}
|
|
45
|
-
// If slice not found, just proceed without filtering
|
|
46
|
-
}
|
|
47
|
-
if (matchingScenarios.length === 0) {
|
|
48
|
-
console.error(`Error: Scenario not found: "${scenarioName}"`);
|
|
49
|
-
console.error('Available scenarios:');
|
|
50
|
-
for (const s of model.scenarios.values()) {
|
|
51
|
-
const slice = model.slices.get(s.sliceId);
|
|
52
|
-
console.error(` - "${s.name}" (id: ${s.id.slice(0, 8)}) (in slice "${slice?.name ?? 'unknown'}")`);
|
|
53
|
-
}
|
|
54
|
-
process.exit(1);
|
|
55
|
-
}
|
|
56
|
-
if (matchingScenarios.length > 1) {
|
|
57
|
-
console.error(`Error: Multiple scenarios found with name "${scenarioName}"`);
|
|
58
|
-
console.error('Please specify using the scenario ID or --slice to disambiguate:');
|
|
59
|
-
for (const s of matchingScenarios) {
|
|
60
|
-
const slice = model.slices.get(s.sliceId);
|
|
61
|
-
console.error(` - "${s.name}" (id: ${s.id.slice(0, 8)}) (in slice "${slice?.name ?? 'unknown'}")`);
|
|
62
|
-
}
|
|
63
|
-
console.error('');
|
|
64
|
-
console.error(`Usage: eventmodeler remove scenario "id:${matchingScenarios[0].id.slice(0, 8)}"`);
|
|
65
|
-
process.exit(1);
|
|
66
|
-
}
|
|
67
|
-
const scenario = matchingScenarios[0];
|
|
68
|
-
const slice = model.slices.get(scenario.sliceId);
|
|
69
|
-
appendEvent(filePath, {
|
|
70
|
-
type: 'ScenarioRemoved',
|
|
71
|
-
data: {
|
|
72
|
-
scenarioId: scenario.id,
|
|
73
|
-
timestamp: Date.now(),
|
|
74
|
-
},
|
|
75
|
-
});
|
|
76
|
-
console.log(`Removed scenario "${scenario.name}" from slice "${slice?.name ?? 'unknown'}"`);
|
|
77
|
-
}
|
|
@@ -1,302 +0,0 @@
|
|
|
1
|
-
import { escapeXml, outputJson } from '../../lib/format.js';
|
|
2
|
-
function isInSlice(slice, pos, width, height) {
|
|
3
|
-
const centerX = pos.x + width / 2;
|
|
4
|
-
const centerY = pos.y + height / 2;
|
|
5
|
-
return (centerX >= slice.position.x &&
|
|
6
|
-
centerX <= slice.position.x + slice.size.width &&
|
|
7
|
-
centerY >= slice.position.y &&
|
|
8
|
-
centerY <= slice.position.y + slice.size.height);
|
|
9
|
-
}
|
|
10
|
-
function findSlicesContaining(model, pos, width, height) {
|
|
11
|
-
return [...model.slices.values()].filter(slice => isInSlice(slice, pos, width, height));
|
|
12
|
-
}
|
|
13
|
-
export function search(model, term, format) {
|
|
14
|
-
const termLower = term.toLowerCase();
|
|
15
|
-
const xmlResults = [];
|
|
16
|
-
const jsonResults = [];
|
|
17
|
-
// Search commands
|
|
18
|
-
for (const cmd of model.commands.values()) {
|
|
19
|
-
if (cmd.name.toLowerCase().includes(termLower)) {
|
|
20
|
-
const slices = findSlicesContaining(model, cmd.position, cmd.width, cmd.height);
|
|
21
|
-
const incomingFlows = [...model.flows.values()].filter(f => f.targetId === cmd.id);
|
|
22
|
-
const outgoingFlows = [...model.flows.values()].filter(f => f.sourceId === cmd.id);
|
|
23
|
-
// JSON result
|
|
24
|
-
const jsonResult = { type: 'command', name: cmd.name, fields: cmd.fields.length };
|
|
25
|
-
if (slices.length > 0) {
|
|
26
|
-
jsonResult.inSlices = slices.map(s => ({ name: s.name, status: s.status }));
|
|
27
|
-
}
|
|
28
|
-
const triggeredBy = [];
|
|
29
|
-
for (const flow of incomingFlows) {
|
|
30
|
-
const screen = model.screens.get(flow.sourceId);
|
|
31
|
-
const processor = model.processors.get(flow.sourceId);
|
|
32
|
-
if (screen)
|
|
33
|
-
triggeredBy.push(screen.name);
|
|
34
|
-
if (processor)
|
|
35
|
-
triggeredBy.push(processor.name);
|
|
36
|
-
}
|
|
37
|
-
if (triggeredBy.length > 0)
|
|
38
|
-
jsonResult.triggeredBy = triggeredBy;
|
|
39
|
-
const produces = [];
|
|
40
|
-
for (const flow of outgoingFlows) {
|
|
41
|
-
const evt = model.events.get(flow.targetId);
|
|
42
|
-
if (evt)
|
|
43
|
-
produces.push(evt.name);
|
|
44
|
-
}
|
|
45
|
-
if (produces.length > 0)
|
|
46
|
-
jsonResult.produces = produces;
|
|
47
|
-
jsonResults.push(jsonResult);
|
|
48
|
-
// XML result
|
|
49
|
-
let xml = ` <command name="${escapeXml(cmd.name)}" fields="${cmd.fields.length}">\n`;
|
|
50
|
-
if (slices.length > 0) {
|
|
51
|
-
xml += ' <in-slices>\n';
|
|
52
|
-
for (const slice of slices) {
|
|
53
|
-
xml += ` <slice name="${escapeXml(slice.name)}" status="${slice.status}"/>\n`;
|
|
54
|
-
}
|
|
55
|
-
xml += ' </in-slices>\n';
|
|
56
|
-
}
|
|
57
|
-
if (incomingFlows.length > 0) {
|
|
58
|
-
xml += ' <triggered-by>\n';
|
|
59
|
-
for (const flow of incomingFlows) {
|
|
60
|
-
const screen = model.screens.get(flow.sourceId);
|
|
61
|
-
const processor = model.processors.get(flow.sourceId);
|
|
62
|
-
if (screen)
|
|
63
|
-
xml += ` <screen name="${escapeXml(screen.name)}"/>\n`;
|
|
64
|
-
if (processor)
|
|
65
|
-
xml += ` <processor name="${escapeXml(processor.name)}"/>\n`;
|
|
66
|
-
}
|
|
67
|
-
xml += ' </triggered-by>\n';
|
|
68
|
-
}
|
|
69
|
-
if (outgoingFlows.length > 0) {
|
|
70
|
-
xml += ' <produces>\n';
|
|
71
|
-
for (const flow of outgoingFlows) {
|
|
72
|
-
const evt = model.events.get(flow.targetId);
|
|
73
|
-
if (evt)
|
|
74
|
-
xml += ` <event name="${escapeXml(evt.name)}"/>\n`;
|
|
75
|
-
}
|
|
76
|
-
xml += ' </produces>\n';
|
|
77
|
-
}
|
|
78
|
-
xml += ' </command>';
|
|
79
|
-
xmlResults.push(xml);
|
|
80
|
-
}
|
|
81
|
-
}
|
|
82
|
-
// Search events
|
|
83
|
-
for (const evt of model.events.values()) {
|
|
84
|
-
if (evt.name.toLowerCase().includes(termLower)) {
|
|
85
|
-
const slices = findSlicesContaining(model, evt.position, evt.width, evt.height);
|
|
86
|
-
const incomingFlows = [...model.flows.values()].filter(f => f.targetId === evt.id);
|
|
87
|
-
const outgoingFlows = [...model.flows.values()].filter(f => f.sourceId === evt.id);
|
|
88
|
-
// JSON result
|
|
89
|
-
const jsonResult = { type: 'event', name: evt.name, fields: evt.fields.length };
|
|
90
|
-
if (slices.length > 0) {
|
|
91
|
-
jsonResult.inSlices = slices.map(s => ({ name: s.name, status: s.status }));
|
|
92
|
-
}
|
|
93
|
-
const producedBy = [];
|
|
94
|
-
for (const flow of incomingFlows) {
|
|
95
|
-
const cmd = model.commands.get(flow.sourceId);
|
|
96
|
-
if (cmd)
|
|
97
|
-
producedBy.push(cmd.name);
|
|
98
|
-
}
|
|
99
|
-
if (producedBy.length > 0)
|
|
100
|
-
jsonResult.producedBy = producedBy;
|
|
101
|
-
const consumedBy = [];
|
|
102
|
-
for (const flow of outgoingFlows) {
|
|
103
|
-
const rm = model.readModels.get(flow.targetId);
|
|
104
|
-
if (rm)
|
|
105
|
-
consumedBy.push(rm.name);
|
|
106
|
-
}
|
|
107
|
-
if (consumedBy.length > 0)
|
|
108
|
-
jsonResult.consumedBy = consumedBy;
|
|
109
|
-
jsonResults.push(jsonResult);
|
|
110
|
-
// XML result
|
|
111
|
-
let xml = ` <event name="${escapeXml(evt.name)}" fields="${evt.fields.length}">\n`;
|
|
112
|
-
if (slices.length > 0) {
|
|
113
|
-
xml += ' <in-slices>\n';
|
|
114
|
-
for (const slice of slices) {
|
|
115
|
-
xml += ` <slice name="${escapeXml(slice.name)}" status="${slice.status}"/>\n`;
|
|
116
|
-
}
|
|
117
|
-
xml += ' </in-slices>\n';
|
|
118
|
-
}
|
|
119
|
-
if (incomingFlows.length > 0) {
|
|
120
|
-
xml += ' <produced-by>\n';
|
|
121
|
-
for (const flow of incomingFlows) {
|
|
122
|
-
const cmd = model.commands.get(flow.sourceId);
|
|
123
|
-
if (cmd)
|
|
124
|
-
xml += ` <command name="${escapeXml(cmd.name)}"/>\n`;
|
|
125
|
-
}
|
|
126
|
-
xml += ' </produced-by>\n';
|
|
127
|
-
}
|
|
128
|
-
if (outgoingFlows.length > 0) {
|
|
129
|
-
xml += ' <consumed-by>\n';
|
|
130
|
-
for (const flow of outgoingFlows) {
|
|
131
|
-
const rm = model.readModels.get(flow.targetId);
|
|
132
|
-
if (rm)
|
|
133
|
-
xml += ` <read-model name="${escapeXml(rm.name)}"/>\n`;
|
|
134
|
-
}
|
|
135
|
-
xml += ' </consumed-by>\n';
|
|
136
|
-
}
|
|
137
|
-
xml += ' </event>';
|
|
138
|
-
xmlResults.push(xml);
|
|
139
|
-
}
|
|
140
|
-
}
|
|
141
|
-
// Search read models
|
|
142
|
-
for (const rm of model.readModels.values()) {
|
|
143
|
-
if (rm.name.toLowerCase().includes(termLower)) {
|
|
144
|
-
const slices = findSlicesContaining(model, rm.position, rm.width, rm.height);
|
|
145
|
-
// JSON result
|
|
146
|
-
const jsonResult = { type: 'read-model', name: rm.name, fields: rm.fields.length };
|
|
147
|
-
if (slices.length > 0) {
|
|
148
|
-
jsonResult.inSlices = slices.map(s => ({ name: s.name, status: s.status }));
|
|
149
|
-
}
|
|
150
|
-
jsonResults.push(jsonResult);
|
|
151
|
-
// XML result
|
|
152
|
-
let xml = ` <read-model name="${escapeXml(rm.name)}" fields="${rm.fields.length}">\n`;
|
|
153
|
-
if (slices.length > 0) {
|
|
154
|
-
xml += ' <in-slices>\n';
|
|
155
|
-
for (const slice of slices) {
|
|
156
|
-
xml += ` <slice name="${escapeXml(slice.name)}" status="${slice.status}"/>\n`;
|
|
157
|
-
}
|
|
158
|
-
xml += ' </in-slices>\n';
|
|
159
|
-
}
|
|
160
|
-
xml += ' </read-model>';
|
|
161
|
-
xmlResults.push(xml);
|
|
162
|
-
}
|
|
163
|
-
}
|
|
164
|
-
// Search screens
|
|
165
|
-
for (const scr of model.screens.values()) {
|
|
166
|
-
if (scr.name.toLowerCase().includes(termLower)) {
|
|
167
|
-
const slices = findSlicesContaining(model, scr.position, scr.width, scr.height);
|
|
168
|
-
// JSON result
|
|
169
|
-
const jsonResult = { type: 'screen', name: scr.name, fields: scr.fields.length };
|
|
170
|
-
if (slices.length > 0) {
|
|
171
|
-
jsonResult.inSlices = slices.map(s => ({ name: s.name, status: s.status }));
|
|
172
|
-
}
|
|
173
|
-
jsonResults.push(jsonResult);
|
|
174
|
-
// XML result
|
|
175
|
-
let xml = ` <screen name="${escapeXml(scr.name)}" fields="${scr.fields.length}">\n`;
|
|
176
|
-
if (slices.length > 0) {
|
|
177
|
-
xml += ' <in-slices>\n';
|
|
178
|
-
for (const slice of slices) {
|
|
179
|
-
xml += ` <slice name="${escapeXml(slice.name)}" status="${slice.status}"/>\n`;
|
|
180
|
-
}
|
|
181
|
-
xml += ' </in-slices>\n';
|
|
182
|
-
}
|
|
183
|
-
xml += ' </screen>';
|
|
184
|
-
xmlResults.push(xml);
|
|
185
|
-
}
|
|
186
|
-
}
|
|
187
|
-
// Search processors
|
|
188
|
-
for (const proc of model.processors.values()) {
|
|
189
|
-
if (proc.name.toLowerCase().includes(termLower)) {
|
|
190
|
-
const slices = findSlicesContaining(model, proc.position, proc.width, proc.height);
|
|
191
|
-
// JSON result
|
|
192
|
-
const jsonResult = { type: 'processor', name: proc.name, fields: proc.fields.length };
|
|
193
|
-
if (slices.length > 0) {
|
|
194
|
-
jsonResult.inSlices = slices.map(s => ({ name: s.name, status: s.status }));
|
|
195
|
-
}
|
|
196
|
-
jsonResults.push(jsonResult);
|
|
197
|
-
// XML result
|
|
198
|
-
let xml = ` <processor name="${escapeXml(proc.name)}" fields="${proc.fields.length}">\n`;
|
|
199
|
-
if (slices.length > 0) {
|
|
200
|
-
xml += ' <in-slices>\n';
|
|
201
|
-
for (const slice of slices) {
|
|
202
|
-
xml += ` <slice name="${escapeXml(slice.name)}" status="${slice.status}"/>\n`;
|
|
203
|
-
}
|
|
204
|
-
xml += ' </in-slices>\n';
|
|
205
|
-
}
|
|
206
|
-
xml += ' </processor>';
|
|
207
|
-
xmlResults.push(xml);
|
|
208
|
-
}
|
|
209
|
-
}
|
|
210
|
-
// Search slices
|
|
211
|
-
for (const slice of model.slices.values()) {
|
|
212
|
-
if (slice.name.toLowerCase().includes(termLower)) {
|
|
213
|
-
// JSON result
|
|
214
|
-
jsonResults.push({ type: 'slice', name: slice.name });
|
|
215
|
-
// XML result
|
|
216
|
-
xmlResults.push(` <slice name="${escapeXml(slice.name)}" status="${slice.status}"/>`);
|
|
217
|
-
}
|
|
218
|
-
}
|
|
219
|
-
// Search aggregates
|
|
220
|
-
for (const agg of model.aggregates.values()) {
|
|
221
|
-
if (agg.name.toLowerCase().includes(termLower)) {
|
|
222
|
-
const events = agg.eventIds
|
|
223
|
-
.map(id => model.events.get(id))
|
|
224
|
-
.filter((e) => e !== undefined);
|
|
225
|
-
// JSON result
|
|
226
|
-
const jsonResult = { type: 'aggregate', name: agg.name, events: events.length };
|
|
227
|
-
if (events.length > 0) {
|
|
228
|
-
jsonResult.contains = events.map(e => e.name);
|
|
229
|
-
}
|
|
230
|
-
jsonResults.push(jsonResult);
|
|
231
|
-
// XML result
|
|
232
|
-
let xml = ` <aggregate name="${escapeXml(agg.name)}" events="${events.length}">\n`;
|
|
233
|
-
if (events.length > 0) {
|
|
234
|
-
xml += ' <contains>\n';
|
|
235
|
-
for (const evt of events) {
|
|
236
|
-
xml += ` <event name="${escapeXml(evt.name)}"/>\n`;
|
|
237
|
-
}
|
|
238
|
-
xml += ' </contains>\n';
|
|
239
|
-
}
|
|
240
|
-
xml += ' </aggregate>';
|
|
241
|
-
xmlResults.push(xml);
|
|
242
|
-
}
|
|
243
|
-
}
|
|
244
|
-
// Search actors
|
|
245
|
-
for (const actor of model.actors.values()) {
|
|
246
|
-
if (actor.name.toLowerCase().includes(termLower)) {
|
|
247
|
-
const screens = actor.screenIds
|
|
248
|
-
.map(id => model.screens.get(id))
|
|
249
|
-
.filter((s) => s !== undefined);
|
|
250
|
-
// JSON result
|
|
251
|
-
const jsonResult = { type: 'actor', name: actor.name, screens: screens.length };
|
|
252
|
-
if (screens.length > 0) {
|
|
253
|
-
jsonResult.contains = screens.map(s => s.name);
|
|
254
|
-
}
|
|
255
|
-
jsonResults.push(jsonResult);
|
|
256
|
-
// XML result
|
|
257
|
-
let xml = ` <actor name="${escapeXml(actor.name)}" screens="${screens.length}">\n`;
|
|
258
|
-
if (screens.length > 0) {
|
|
259
|
-
xml += ' <contains>\n';
|
|
260
|
-
for (const scr of screens) {
|
|
261
|
-
xml += ` <screen name="${escapeXml(scr.name)}"/>\n`;
|
|
262
|
-
}
|
|
263
|
-
xml += ' </contains>\n';
|
|
264
|
-
}
|
|
265
|
-
xml += ' </actor>';
|
|
266
|
-
xmlResults.push(xml);
|
|
267
|
-
}
|
|
268
|
-
}
|
|
269
|
-
// Search scenarios
|
|
270
|
-
for (const scenario of model.scenarios.values()) {
|
|
271
|
-
if (scenario.name.toLowerCase().includes(termLower)) {
|
|
272
|
-
const slice = model.slices.get(scenario.sliceId);
|
|
273
|
-
// JSON result
|
|
274
|
-
const jsonResult = { type: 'scenario', name: scenario.name };
|
|
275
|
-
if (slice) {
|
|
276
|
-
jsonResult.inSlice = { name: slice.name, status: slice.status };
|
|
277
|
-
}
|
|
278
|
-
jsonResults.push(jsonResult);
|
|
279
|
-
// XML result
|
|
280
|
-
let xml = ` <scenario name="${escapeXml(scenario.name)}">\n`;
|
|
281
|
-
if (slice) {
|
|
282
|
-
xml += ` <in-slice name="${escapeXml(slice.name)}" status="${slice.status}"/>\n`;
|
|
283
|
-
}
|
|
284
|
-
xml += ' </scenario>';
|
|
285
|
-
xmlResults.push(xml);
|
|
286
|
-
}
|
|
287
|
-
}
|
|
288
|
-
if (format === 'json') {
|
|
289
|
-
outputJson({ query: term, count: jsonResults.length, results: jsonResults });
|
|
290
|
-
return;
|
|
291
|
-
}
|
|
292
|
-
if (xmlResults.length === 0) {
|
|
293
|
-
console.log(`<search-results query="${escapeXml(term)}" count="0"/>`);
|
|
294
|
-
}
|
|
295
|
-
else {
|
|
296
|
-
console.log(`<search-results query="${escapeXml(term)}" count="${xmlResults.length}">`);
|
|
297
|
-
for (const result of xmlResults) {
|
|
298
|
-
console.log(result);
|
|
299
|
-
}
|
|
300
|
-
console.log('</search-results>');
|
|
301
|
-
}
|
|
302
|
-
}
|
|
@@ -1,4 +0,0 @@
|
|
|
1
|
-
import type { EventModel } from '../../types.js';
|
|
2
|
-
import { type OutputFormat } from '../../lib/format.js';
|
|
3
|
-
export declare function showActor(model: EventModel, actorName: string, format: OutputFormat): void;
|
|
4
|
-
export declare function listActors(model: EventModel, format: OutputFormat): void;
|
|
@@ -1,115 +0,0 @@
|
|
|
1
|
-
import { escapeXml, outputJson } from '../../lib/format.js';
|
|
2
|
-
import { findElementOrExit } from '../../lib/element-lookup.js';
|
|
3
|
-
// Get screens whose center point is inside the actor bounds
|
|
4
|
-
function getScreensInActor(model, actor) {
|
|
5
|
-
const bounds = {
|
|
6
|
-
left: actor.position.x,
|
|
7
|
-
right: actor.position.x + actor.size.width,
|
|
8
|
-
top: actor.position.y,
|
|
9
|
-
bottom: actor.position.y + actor.size.height,
|
|
10
|
-
};
|
|
11
|
-
function isInActor(pos, width, height) {
|
|
12
|
-
const centerX = pos.x + width / 2;
|
|
13
|
-
const centerY = pos.y + height / 2;
|
|
14
|
-
return centerX >= bounds.left && centerX <= bounds.right && centerY >= bounds.top && centerY <= bounds.bottom;
|
|
15
|
-
}
|
|
16
|
-
return [...model.screens.values()].filter(s => isInActor(s.position, s.width, s.height));
|
|
17
|
-
}
|
|
18
|
-
// Find which slice contains a screen
|
|
19
|
-
function findSliceForScreen(model, screen) {
|
|
20
|
-
for (const slice of model.slices.values()) {
|
|
21
|
-
const centerX = screen.position.x + screen.width / 2;
|
|
22
|
-
const centerY = screen.position.y + screen.height / 2;
|
|
23
|
-
if (centerX >= slice.position.x &&
|
|
24
|
-
centerX <= slice.position.x + slice.size.width &&
|
|
25
|
-
centerY >= slice.position.y &&
|
|
26
|
-
centerY <= slice.position.y + slice.size.height) {
|
|
27
|
-
return slice.name;
|
|
28
|
-
}
|
|
29
|
-
}
|
|
30
|
-
return null;
|
|
31
|
-
}
|
|
32
|
-
export function showActor(model, actorName, format) {
|
|
33
|
-
// Find the actor
|
|
34
|
-
const actor = findElementOrExit(model.actors, actorName, 'actor');
|
|
35
|
-
// Get screens dynamically based on position (center point inside actor bounds)
|
|
36
|
-
const screensInActor = getScreensInActor(model, actor);
|
|
37
|
-
if (format === 'json') {
|
|
38
|
-
outputJson({
|
|
39
|
-
id: actor.id,
|
|
40
|
-
name: actor.name,
|
|
41
|
-
screensCount: screensInActor.length,
|
|
42
|
-
screens: screensInActor.map(screen => {
|
|
43
|
-
const result = {
|
|
44
|
-
id: screen.id,
|
|
45
|
-
name: screen.name,
|
|
46
|
-
fields: screen.fields.length
|
|
47
|
-
};
|
|
48
|
-
if (screen.originalNodeId) {
|
|
49
|
-
result.linkedCopy = true;
|
|
50
|
-
const original = model.screens.get(screen.originalNodeId);
|
|
51
|
-
const originSlice = original ? findSliceForScreen(model, original) : null;
|
|
52
|
-
if (originSlice)
|
|
53
|
-
result.originSlice = originSlice;
|
|
54
|
-
}
|
|
55
|
-
return result;
|
|
56
|
-
})
|
|
57
|
-
});
|
|
58
|
-
return;
|
|
59
|
-
}
|
|
60
|
-
console.log(`<actor id="${actor.id}" name="${escapeXml(actor.name)}">`);
|
|
61
|
-
console.log(` <screens-count>${screensInActor.length}</screens-count>`);
|
|
62
|
-
if (screensInActor.length > 0) {
|
|
63
|
-
console.log(' <screens>');
|
|
64
|
-
for (const screen of screensInActor) {
|
|
65
|
-
const fieldCount = screen.fields.length;
|
|
66
|
-
if (screen.originalNodeId) {
|
|
67
|
-
// This is a linked copy - show origin info
|
|
68
|
-
const original = model.screens.get(screen.originalNodeId);
|
|
69
|
-
const originSlice = original ? findSliceForScreen(model, original) : null;
|
|
70
|
-
const originAttr = originSlice ? ` origin-slice="${escapeXml(originSlice)}"` : '';
|
|
71
|
-
console.log(` <screen id="${screen.id}" name="${escapeXml(screen.name)}" fields="${fieldCount}" linked-copy="true"${originAttr}/>`);
|
|
72
|
-
}
|
|
73
|
-
else {
|
|
74
|
-
console.log(` <screen id="${screen.id}" name="${escapeXml(screen.name)}" fields="${fieldCount}"/>`);
|
|
75
|
-
}
|
|
76
|
-
}
|
|
77
|
-
console.log(' </screens>');
|
|
78
|
-
}
|
|
79
|
-
console.log('</actor>');
|
|
80
|
-
}
|
|
81
|
-
export function listActors(model, format) {
|
|
82
|
-
const actors = [...model.actors.values()];
|
|
83
|
-
if (format === 'json') {
|
|
84
|
-
outputJson({
|
|
85
|
-
actors: actors.map(actor => {
|
|
86
|
-
const screensInActor = getScreensInActor(model, actor);
|
|
87
|
-
const originalCount = screensInActor.filter(s => !s.originalNodeId).length;
|
|
88
|
-
const copyCount = screensInActor.filter(s => s.originalNodeId).length;
|
|
89
|
-
const result = {
|
|
90
|
-
id: actor.id,
|
|
91
|
-
name: actor.name,
|
|
92
|
-
screens: originalCount
|
|
93
|
-
};
|
|
94
|
-
if (copyCount > 0)
|
|
95
|
-
result.copies = copyCount;
|
|
96
|
-
return result;
|
|
97
|
-
})
|
|
98
|
-
});
|
|
99
|
-
return;
|
|
100
|
-
}
|
|
101
|
-
if (actors.length === 0) {
|
|
102
|
-
console.log('<actors/>');
|
|
103
|
-
return;
|
|
104
|
-
}
|
|
105
|
-
console.log('<actors>');
|
|
106
|
-
for (const actor of actors) {
|
|
107
|
-
// Calculate screens dynamically based on position
|
|
108
|
-
const screensInActor = getScreensInActor(model, actor);
|
|
109
|
-
const originalCount = screensInActor.filter(s => !s.originalNodeId).length;
|
|
110
|
-
const copyCount = screensInActor.filter(s => s.originalNodeId).length;
|
|
111
|
-
const copiesAttr = copyCount > 0 ? ` copies="${copyCount}"` : '';
|
|
112
|
-
console.log(` <actor id="${actor.id}" name="${escapeXml(actor.name)}" screens="${originalCount}"${copiesAttr}/>`);
|
|
113
|
-
}
|
|
114
|
-
console.log('</actors>');
|
|
115
|
-
}
|