@dssp/dkpi 1.0.0-alpha.40 → 1.0.0-alpha.41
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-client/pages/kpi-metric-value/kpi-metric-value-editor-page.d.ts +58 -0
- package/dist-client/pages/kpi-metric-value/kpi-metric-value-editor-page.js +731 -0
- package/dist-client/pages/kpi-metric-value/kpi-metric-value-editor-page.js.map +1 -0
- package/dist-client/pages/kpi-metric-value/kpi-metric-value-importer.d.ts +23 -0
- package/dist-client/pages/kpi-metric-value/kpi-metric-value-importer.js +76 -0
- package/dist-client/pages/kpi-metric-value/kpi-metric-value-importer.js.map +1 -0
- package/dist-client/pages/kpi-metric-value/kpi-metric-value-list-page.d.ts +69 -0
- package/dist-client/pages/kpi-metric-value/kpi-metric-value-list-page.js +385 -0
- package/dist-client/pages/kpi-metric-value/kpi-metric-value-list-page.js.map +1 -0
- package/dist-client/pages/kpi-metric-value/kpi-metric-value-manual-entry-form.d.ts +12 -0
- package/dist-client/pages/kpi-metric-value/kpi-metric-value-manual-entry-form.js +174 -0
- package/dist-client/pages/kpi-metric-value/kpi-metric-value-manual-entry-form.js.map +1 -0
- package/dist-client/pages/kpi-metric-value/kpi-metric-value-manual-entry-page.d.ts +41 -0
- package/dist-client/pages/kpi-metric-value/kpi-metric-value-manual-entry-page.js +191 -0
- package/dist-client/pages/kpi-metric-value/kpi-metric-value-manual-entry-page.js.map +1 -0
- package/dist-client/pages/kpi-value/kpi-value-importer.d.ts +23 -0
- package/dist-client/pages/kpi-value/kpi-value-importer.js +93 -0
- package/dist-client/pages/kpi-value/kpi-value-importer.js.map +1 -0
- package/dist-client/pages/kpi-value/kpi-value-list-page.d.ts +72 -0
- package/dist-client/pages/kpi-value/kpi-value-list-page.js +465 -0
- package/dist-client/pages/kpi-value/kpi-value-list-page.js.map +1 -0
- package/dist-client/route.d.ts +1 -1
- package/dist-client/route.js +9 -0
- package/dist-client/route.js.map +1 -1
- package/dist-client/tsconfig.tsbuildinfo +1 -1
- package/dist-server/index.d.ts +1 -0
- package/dist-server/index.js +1 -0
- package/dist-server/index.js.map +1 -1
- package/dist-server/scripts/propagate-parent-kpi-values.d.ts +10 -0
- package/dist-server/scripts/propagate-parent-kpi-values.js +440 -0
- package/dist-server/scripts/propagate-parent-kpi-values.js.map +1 -0
- package/dist-server/service/index.d.ts +6 -0
- package/dist-server/service/index.js +21 -0
- package/dist-server/service/index.js.map +1 -0
- package/dist-server/service/kpi-metric-value/index.d.ts +3 -0
- package/dist-server/service/kpi-metric-value/index.js +7 -0
- package/dist-server/service/kpi-metric-value/index.js.map +1 -0
- package/dist-server/service/kpi-metric-value/kpi-metric-value-query.d.ts +7 -0
- package/dist-server/service/kpi-metric-value/kpi-metric-value-query.js +47 -0
- package/dist-server/service/kpi-metric-value/kpi-metric-value-query.js.map +1 -0
- package/dist-server/service/kpi-value/index.d.ts +3 -0
- package/dist-server/service/kpi-value/index.js +7 -0
- package/dist-server/service/kpi-value/index.js.map +1 -0
- package/dist-server/service/kpi-value/kpi-value-query.d.ts +7 -0
- package/dist-server/service/kpi-value/kpi-value-query.js +47 -0
- package/dist-server/service/kpi-value/kpi-value-query.js.map +1 -0
- package/dist-server/tsconfig.tsbuildinfo +1 -1
- package/package.json +6 -3
- package/schema.graphql +2063 -18
package/dist-server/index.d.ts
CHANGED
package/dist-server/index.js
CHANGED
package/dist-server/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","sourceRoot":"","sources":["../server/index.ts"],"names":[],"mappings":";;;AAAA,gEAAqC","sourcesContent":["export * from './migrations/index.js'\n"]}
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../server/index.ts"],"names":[],"mappings":";;;AAAA,gEAAqC;AACrC,6DAAkC","sourcesContent":["export * from './migrations/index.js'\nexport * from './service/index.js'\n"]}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
#!/usr/bin/env ts-node
|
|
2
|
+
/**
|
|
3
|
+
* Propagate parent KPI values based on child KPI scores and weights
|
|
4
|
+
* Processes hierarchy from leaf to root, calculating weighted averages
|
|
5
|
+
*/
|
|
6
|
+
export declare function propagateParentKpiValues(specificKpiName?: string, forceRecalculate?: boolean, targetDate?: string, orgScope?: string, version?: number): Promise<void>;
|
|
7
|
+
/**
|
|
8
|
+
* List parent KPIs and their hierarchy
|
|
9
|
+
*/
|
|
10
|
+
export declare function listParentKpiHierarchy(): Promise<void>;
|
|
@@ -0,0 +1,440 @@
|
|
|
1
|
+
#!/usr/bin/env ts-node
|
|
2
|
+
"use strict";
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.propagateParentKpiValues = propagateParentKpiValues;
|
|
5
|
+
exports.listParentKpiHierarchy = listParentKpiHierarchy;
|
|
6
|
+
const typeorm_1 = require("typeorm");
|
|
7
|
+
/**
|
|
8
|
+
* Propagate parent KPI values based on child KPI scores and weights
|
|
9
|
+
* Processes hierarchy from leaf to root, calculating weighted averages
|
|
10
|
+
*/
|
|
11
|
+
async function propagateParentKpiValues(specificKpiName, forceRecalculate = false, targetDate, orgScope, version) {
|
|
12
|
+
// Set NODE_ENV if not set
|
|
13
|
+
if (!process.env.NODE_ENV) {
|
|
14
|
+
process.env.NODE_ENV = 'development';
|
|
15
|
+
}
|
|
16
|
+
// Initialize Things-Factory environment
|
|
17
|
+
const { config } = require('@things-factory/env');
|
|
18
|
+
const ormconfig = require('@things-factory/shell/dist-server/initializers/ormconfig.js');
|
|
19
|
+
const connectionConfig = config.get('ormconfig');
|
|
20
|
+
let connection;
|
|
21
|
+
console.log('🚀 Starting Parent KPI Value Propagation...');
|
|
22
|
+
try {
|
|
23
|
+
// Create database connection using Things-Factory pattern
|
|
24
|
+
console.log('🔌 Connecting to database...');
|
|
25
|
+
connection = await (0, typeorm_1.createConnection)(Object.assign(Object.assign(Object.assign({}, ormconfig), connectionConfig), { logging: false }));
|
|
26
|
+
// Register the connection with Things-Factory shell
|
|
27
|
+
const { addDataSource } = require('@things-factory/shell');
|
|
28
|
+
addDataSource('default', connection);
|
|
29
|
+
console.log('✅ Database connected');
|
|
30
|
+
// Now we can use getRepository with the registered connection
|
|
31
|
+
const { getRepository } = require('@things-factory/shell');
|
|
32
|
+
const { Kpi } = require('@things-factory/kpi');
|
|
33
|
+
const kpiRepository = getRepository(Kpi);
|
|
34
|
+
// Get KPI-Value repository
|
|
35
|
+
let kpiValueRepository;
|
|
36
|
+
try {
|
|
37
|
+
const { KpiValue } = require('@things-factory/kpi');
|
|
38
|
+
kpiValueRepository = getRepository(KpiValue);
|
|
39
|
+
}
|
|
40
|
+
catch (error) {
|
|
41
|
+
console.error('❌ KpiValue entity not found. Please ensure it exists in @things-factory/kpi');
|
|
42
|
+
return;
|
|
43
|
+
}
|
|
44
|
+
// Get Domain repository and find SYSTEM domain
|
|
45
|
+
let systemDomain;
|
|
46
|
+
try {
|
|
47
|
+
const { Domain } = require('@things-factory/shell');
|
|
48
|
+
const domainRepository = getRepository(Domain);
|
|
49
|
+
systemDomain = await domainRepository.findOne({ where: { name: 'SYSTEM' } });
|
|
50
|
+
if (!systemDomain) {
|
|
51
|
+
console.error('❌ SYSTEM domain not found');
|
|
52
|
+
return;
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
catch (error) {
|
|
56
|
+
console.error('❌ Domain entity not found:', error.message);
|
|
57
|
+
return;
|
|
58
|
+
}
|
|
59
|
+
// Get all KPIs with their hierarchy
|
|
60
|
+
let kpisQuery = kpiRepository
|
|
61
|
+
.createQueryBuilder('kpi')
|
|
62
|
+
.leftJoinAndSelect('kpi.domain', 'domain')
|
|
63
|
+
.leftJoinAndSelect('kpi.parent', 'parent')
|
|
64
|
+
.leftJoinAndSelect('kpi.children', 'children')
|
|
65
|
+
.where('domain.name = :domainName', { domainName: 'SYSTEM' });
|
|
66
|
+
if (specificKpiName) {
|
|
67
|
+
kpisQuery = kpisQuery.andWhere('kpi.name = :name', { name: specificKpiName });
|
|
68
|
+
console.log(`🎯 Processing specific KPI: ${specificKpiName}`);
|
|
69
|
+
}
|
|
70
|
+
else {
|
|
71
|
+
console.log('🎯 Processing all KPIs in hierarchy');
|
|
72
|
+
}
|
|
73
|
+
const allKpis = await kpisQuery.getMany();
|
|
74
|
+
if (allKpis.length === 0) {
|
|
75
|
+
console.log('⚠️ No KPIs found');
|
|
76
|
+
return;
|
|
77
|
+
}
|
|
78
|
+
console.log(`📊 Found ${allKpis.length} KPIs total`);
|
|
79
|
+
// Get parent KPIs (non-leaf KPIs) ordered by level (deepest first)
|
|
80
|
+
const parentKpis = allKpis.filter(kpi => !kpi.isLeaf);
|
|
81
|
+
if (parentKpis.length === 0) {
|
|
82
|
+
console.log('⚠️ No parent KPIs found');
|
|
83
|
+
return;
|
|
84
|
+
}
|
|
85
|
+
// Sort parent KPIs by hierarchy level (deepest first for bottom-up processing)
|
|
86
|
+
const sortedParentKpis = await sortKpisByLevel(parentKpis, allKpis);
|
|
87
|
+
console.log(`🔄 Processing ${sortedParentKpis.length} parent KPIs`);
|
|
88
|
+
let totalProcessed = 0;
|
|
89
|
+
let totalCreated = 0;
|
|
90
|
+
let totalUpdated = 0;
|
|
91
|
+
let errorCount = 0;
|
|
92
|
+
// Get all distinct (group, valueDate) combinations from existing KPI values
|
|
93
|
+
let groupDateCombinations = [];
|
|
94
|
+
if (orgScope && targetDate) {
|
|
95
|
+
groupDateCombinations = [{ group: orgScope, valueDate: targetDate }];
|
|
96
|
+
}
|
|
97
|
+
else if (orgScope) {
|
|
98
|
+
// Get all dates for specific group
|
|
99
|
+
const dates = await kpiValueRepository.query(`
|
|
100
|
+
SELECT DISTINCT "value_date"
|
|
101
|
+
FROM kpi_values
|
|
102
|
+
WHERE "group" = $1 AND "group" IS NOT NULL
|
|
103
|
+
ORDER BY "value_date"
|
|
104
|
+
`, [orgScope]);
|
|
105
|
+
groupDateCombinations = dates.map(row => ({ group: orgScope, valueDate: row.value_date }));
|
|
106
|
+
}
|
|
107
|
+
else {
|
|
108
|
+
// Get all distinct (group, valueDate) combinations
|
|
109
|
+
const combinations = await kpiValueRepository.query(`
|
|
110
|
+
SELECT DISTINCT "group", "value_date"
|
|
111
|
+
FROM kpi_values
|
|
112
|
+
WHERE "group" IS NOT NULL AND "group" != ''
|
|
113
|
+
ORDER BY "group", "value_date"
|
|
114
|
+
`);
|
|
115
|
+
groupDateCombinations = combinations.map(row => ({ group: row.group, valueDate: row.value_date }));
|
|
116
|
+
}
|
|
117
|
+
console.log(`📍 Found ${groupDateCombinations.length} distinct (group, date) combinations`);
|
|
118
|
+
// Process each (group, date) combination
|
|
119
|
+
for (const { group: currentGroup, valueDate } of groupDateCombinations) {
|
|
120
|
+
console.log(`\\n🏢 Processing Group: ${currentGroup}, Date: ${valueDate}`);
|
|
121
|
+
// Process each parent KPI from deepest to shallowest for this group and date
|
|
122
|
+
for (const parentKpi of sortedParentKpis) {
|
|
123
|
+
try {
|
|
124
|
+
console.log(`\\n📄 Processing Parent KPI: ${parentKpi.name} (Level ${parentKpi.__level || '?'}) - Group: ${currentGroup}, Date: ${valueDate}`);
|
|
125
|
+
// Get child KPIs
|
|
126
|
+
const childKpis = allKpis.filter(kpi => { var _a; return ((_a = kpi.parent) === null || _a === void 0 ? void 0 : _a.id) === parentKpi.id; });
|
|
127
|
+
if (childKpis.length === 0) {
|
|
128
|
+
console.log(` ⚠️ No child KPIs found, skipping`);
|
|
129
|
+
continue;
|
|
130
|
+
}
|
|
131
|
+
console.log(` 👶 Found ${childKpis.length} child KPIs`);
|
|
132
|
+
// Get child KPI values/scores for the specific group and date
|
|
133
|
+
const childValues = await getChildKpiValues(kpiValueRepository, childKpis, valueDate, version, currentGroup);
|
|
134
|
+
if (childValues.length === 0) {
|
|
135
|
+
console.log(` ⚠️ No child KPI values found for date ${valueDate} and group ${currentGroup}, skipping`);
|
|
136
|
+
continue;
|
|
137
|
+
}
|
|
138
|
+
console.log(` 📊 Found values for ${childValues.length}/${childKpis.length} child KPIs`);
|
|
139
|
+
// Calculate weighted average
|
|
140
|
+
const { weightedValue, weightedScore } = calculateWeightedAverage(childValues, childKpis);
|
|
141
|
+
console.log(` 🧮 Calculated: value=${weightedValue.toFixed(4)}, score=${weightedScore.toFixed(4)}`);
|
|
142
|
+
// Check if parent KPI value already exists for this org scope
|
|
143
|
+
let existingValueQuery = kpiValueRepository
|
|
144
|
+
.createQueryBuilder('kpiValue')
|
|
145
|
+
.leftJoinAndSelect('kpiValue.kpi', 'kpi')
|
|
146
|
+
.leftJoinAndSelect('kpiValue.domain', 'domain')
|
|
147
|
+
.where('kpi.id = :kpiId', { kpiId: parentKpi.id })
|
|
148
|
+
.andWhere('kpiValue.valueDate = :valueDate', { valueDate })
|
|
149
|
+
.andWhere('domain.name = :domainName', { domainName: 'SYSTEM' });
|
|
150
|
+
if (version !== undefined) {
|
|
151
|
+
existingValueQuery = existingValueQuery.andWhere('kpiValue.version = :version', { version });
|
|
152
|
+
}
|
|
153
|
+
// Filter by group scope
|
|
154
|
+
existingValueQuery = existingValueQuery.andWhere('kpiValue.group = :groupScope', { groupScope: currentGroup });
|
|
155
|
+
const existingValue = await existingValueQuery.getOne();
|
|
156
|
+
console.log(` 🔍 Checking existing record: ${existingValue ? 'FOUND' : 'NOT FOUND'}`);
|
|
157
|
+
if (existingValue) {
|
|
158
|
+
// Update existing value
|
|
159
|
+
if (forceRecalculate || existingValue.value !== weightedValue || existingValue.score !== weightedScore) {
|
|
160
|
+
const oldValue = existingValue.value;
|
|
161
|
+
const oldScore = existingValue.score;
|
|
162
|
+
existingValue.value = weightedValue;
|
|
163
|
+
existingValue.score = weightedScore;
|
|
164
|
+
existingValue.kpiScope = currentGroup; // Set kpiScope same as group
|
|
165
|
+
existingValue.updatedAt = new Date();
|
|
166
|
+
try {
|
|
167
|
+
await kpiValueRepository.save(existingValue);
|
|
168
|
+
totalUpdated++;
|
|
169
|
+
console.log(` ✅ Updated: ${oldValue.toFixed(4)} → ${weightedValue.toFixed(4)}, score: ${oldScore.toFixed(4)} → ${weightedScore.toFixed(4)}`);
|
|
170
|
+
}
|
|
171
|
+
catch (saveError) {
|
|
172
|
+
console.log(` ❌ Update failed: ${saveError.message}`);
|
|
173
|
+
errorCount++;
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
else {
|
|
177
|
+
console.log(` ⏭️ No change needed`);
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
else {
|
|
181
|
+
// Create new value with group scope
|
|
182
|
+
const newKpiValue = {
|
|
183
|
+
kpi: parentKpi,
|
|
184
|
+
value: weightedValue,
|
|
185
|
+
score: weightedScore,
|
|
186
|
+
valueDate: valueDate,
|
|
187
|
+
version: version || parentKpi.version || 1,
|
|
188
|
+
group: currentGroup,
|
|
189
|
+
kpiScope: currentGroup, // Set kpiScope same as group
|
|
190
|
+
domain: systemDomain, // Set domain entity
|
|
191
|
+
inputType: 'AUTO', // Set inputType to AUTO
|
|
192
|
+
createdAt: new Date(),
|
|
193
|
+
updatedAt: new Date()
|
|
194
|
+
};
|
|
195
|
+
try {
|
|
196
|
+
const savedValue = await kpiValueRepository.save(newKpiValue);
|
|
197
|
+
totalCreated++;
|
|
198
|
+
console.log(` ✨ Created: value=${weightedValue.toFixed(4)}, score=${weightedScore.toFixed(4)}, id=${savedValue.id}`);
|
|
199
|
+
}
|
|
200
|
+
catch (saveError) {
|
|
201
|
+
console.log(` ❌ Save failed: ${saveError.message}`);
|
|
202
|
+
errorCount++;
|
|
203
|
+
}
|
|
204
|
+
}
|
|
205
|
+
totalProcessed++;
|
|
206
|
+
}
|
|
207
|
+
catch (kpiError) {
|
|
208
|
+
console.log(` ❌ Error processing KPI ${parentKpi.name}: ${kpiError.message}`);
|
|
209
|
+
errorCount++;
|
|
210
|
+
}
|
|
211
|
+
}
|
|
212
|
+
}
|
|
213
|
+
console.log(`\\n🎉 Parent KPI Value Propagation Complete!`);
|
|
214
|
+
console.log(` 📊 Parent KPIs processed: ${totalProcessed}`);
|
|
215
|
+
console.log(` ✨ Values created: ${totalCreated}`);
|
|
216
|
+
console.log(` ✅ Values updated: ${totalUpdated}`);
|
|
217
|
+
console.log(` ❌ Errors: ${errorCount}`);
|
|
218
|
+
if (errorCount > 0) {
|
|
219
|
+
console.log(`\\n⚠️ Some parent KPIs had errors. Check the logs above for details.`);
|
|
220
|
+
}
|
|
221
|
+
}
|
|
222
|
+
catch (error) {
|
|
223
|
+
console.error(`❌ Fatal error during parent KPI value propagation:`, error);
|
|
224
|
+
throw error;
|
|
225
|
+
}
|
|
226
|
+
finally {
|
|
227
|
+
if (connection) {
|
|
228
|
+
await connection.close();
|
|
229
|
+
console.log('🔌 Database connection closed');
|
|
230
|
+
}
|
|
231
|
+
}
|
|
232
|
+
}
|
|
233
|
+
/**
|
|
234
|
+
* Sort KPIs by hierarchy level (deepest first for bottom-up processing)
|
|
235
|
+
*/
|
|
236
|
+
async function sortKpisByLevel(parentKpis, allKpis) {
|
|
237
|
+
// Calculate hierarchy level for each KPI
|
|
238
|
+
const kpiLevels = new Map();
|
|
239
|
+
function calculateLevel(kpi) {
|
|
240
|
+
if (kpiLevels.has(kpi.id)) {
|
|
241
|
+
return kpiLevels.get(kpi.id);
|
|
242
|
+
}
|
|
243
|
+
if (!kpi.parent) {
|
|
244
|
+
kpiLevels.set(kpi.id, 0);
|
|
245
|
+
return 0;
|
|
246
|
+
}
|
|
247
|
+
const parentKpi = allKpis.find(k => { var _a; return k.id === ((_a = kpi.parent) === null || _a === void 0 ? void 0 : _a.id); });
|
|
248
|
+
if (!parentKpi) {
|
|
249
|
+
kpiLevels.set(kpi.id, 1);
|
|
250
|
+
return 1;
|
|
251
|
+
}
|
|
252
|
+
const level = calculateLevel(parentKpi) + 1;
|
|
253
|
+
kpiLevels.set(kpi.id, level);
|
|
254
|
+
return level;
|
|
255
|
+
}
|
|
256
|
+
// Calculate levels for all parent KPIs
|
|
257
|
+
parentKpis.forEach(kpi => {
|
|
258
|
+
const level = calculateLevel(kpi);
|
|
259
|
+
kpi.__level = level;
|
|
260
|
+
});
|
|
261
|
+
// Sort by level (deepest first)
|
|
262
|
+
return parentKpis.sort((a, b) => (b.__level || 0) - (a.__level || 0));
|
|
263
|
+
}
|
|
264
|
+
/**
|
|
265
|
+
* Get child KPI values for a specific date
|
|
266
|
+
*/
|
|
267
|
+
async function getChildKpiValues(kpiValueRepository, childKpis, valueDate, version, orgScope) {
|
|
268
|
+
const childKpiIds = childKpis.map(kpi => kpi.id);
|
|
269
|
+
let valuesQuery = kpiValueRepository
|
|
270
|
+
.createQueryBuilder('kpiValue')
|
|
271
|
+
.leftJoinAndSelect('kpiValue.kpi', 'kpi')
|
|
272
|
+
.where('kpi.id IN (:...kpiIds)', { kpiIds: childKpiIds })
|
|
273
|
+
.andWhere('kpiValue.valueDate = :valueDate', { valueDate })
|
|
274
|
+
.andWhere('kpiValue.score IS NOT NULL');
|
|
275
|
+
if (version !== undefined) {
|
|
276
|
+
valuesQuery = valuesQuery.andWhere('kpiValue.version = :version', { version });
|
|
277
|
+
}
|
|
278
|
+
// Filter by group scope
|
|
279
|
+
if (orgScope) {
|
|
280
|
+
valuesQuery = valuesQuery.andWhere('kpiValue.group = :groupScope', { groupScope: orgScope });
|
|
281
|
+
}
|
|
282
|
+
const values = await valuesQuery.getMany();
|
|
283
|
+
return values;
|
|
284
|
+
}
|
|
285
|
+
/**
|
|
286
|
+
* Calculate weighted average from child KPI values
|
|
287
|
+
*/
|
|
288
|
+
function calculateWeightedAverage(childValues, childKpis) {
|
|
289
|
+
let totalWeightedValue = 0;
|
|
290
|
+
let totalWeightedScore = 0;
|
|
291
|
+
let totalWeight = 0;
|
|
292
|
+
for (const value of childValues) {
|
|
293
|
+
const childKpi = childKpis.find(kpi => kpi.id === value.kpi.id);
|
|
294
|
+
const weight = (childKpi === null || childKpi === void 0 ? void 0 : childKpi.weight) || 1;
|
|
295
|
+
totalWeightedValue += value.score * weight;
|
|
296
|
+
totalWeightedScore += value.score * weight;
|
|
297
|
+
totalWeight += weight;
|
|
298
|
+
}
|
|
299
|
+
if (totalWeight === 0) {
|
|
300
|
+
return { weightedValue: 0, weightedScore: 0 };
|
|
301
|
+
}
|
|
302
|
+
return {
|
|
303
|
+
weightedValue: totalWeightedValue / totalWeight,
|
|
304
|
+
weightedScore: totalWeightedScore / totalWeight
|
|
305
|
+
};
|
|
306
|
+
}
|
|
307
|
+
/**
|
|
308
|
+
* List parent KPIs and their hierarchy
|
|
309
|
+
*/
|
|
310
|
+
async function listParentKpiHierarchy() {
|
|
311
|
+
// Set NODE_ENV if not set
|
|
312
|
+
if (!process.env.NODE_ENV) {
|
|
313
|
+
process.env.NODE_ENV = 'development';
|
|
314
|
+
}
|
|
315
|
+
// Initialize Things-Factory environment
|
|
316
|
+
const { config } = require('@things-factory/env');
|
|
317
|
+
const ormconfig = require('@things-factory/shell/dist-server/initializers/ormconfig.js');
|
|
318
|
+
const connectionConfig = config.get('ormconfig');
|
|
319
|
+
let connection;
|
|
320
|
+
try {
|
|
321
|
+
console.log('🔌 Connecting to database...');
|
|
322
|
+
connection = await (0, typeorm_1.createConnection)(Object.assign(Object.assign(Object.assign({}, ormconfig), connectionConfig), { logging: false }));
|
|
323
|
+
const { addDataSource } = require('@things-factory/shell');
|
|
324
|
+
addDataSource('default', connection);
|
|
325
|
+
const { getRepository } = require('@things-factory/shell');
|
|
326
|
+
const { Kpi } = require('@things-factory/kpi');
|
|
327
|
+
const kpiRepository = getRepository(Kpi);
|
|
328
|
+
// Get all KPIs with hierarchy
|
|
329
|
+
const kpis = await kpiRepository
|
|
330
|
+
.createQueryBuilder('kpi')
|
|
331
|
+
.leftJoinAndSelect('kpi.domain', 'domain')
|
|
332
|
+
.leftJoinAndSelect('kpi.parent', 'parent')
|
|
333
|
+
.leftJoinAndSelect('kpi.children', 'children')
|
|
334
|
+
.where('domain.name = :domainName', { domainName: 'SYSTEM' })
|
|
335
|
+
.getMany();
|
|
336
|
+
const parentKpis = kpis.filter(kpi => !kpi.isLeaf);
|
|
337
|
+
const sortedParentKpis = await sortKpisByLevel(parentKpis, kpis);
|
|
338
|
+
console.log(`📁 Parent KPI Hierarchy (${sortedParentKpis.length} parents):\\n`);
|
|
339
|
+
if (sortedParentKpis.length === 0) {
|
|
340
|
+
console.log(' No parent KPIs found.');
|
|
341
|
+
return;
|
|
342
|
+
}
|
|
343
|
+
sortedParentKpis.forEach((kpi, index) => {
|
|
344
|
+
const level = kpi.__level || 0;
|
|
345
|
+
const indent = ' '.repeat(level);
|
|
346
|
+
const childCount = kpis.filter(k => { var _a; return ((_a = k.parent) === null || _a === void 0 ? void 0 : _a.id) === kpi.id; }).length;
|
|
347
|
+
const weight = kpi.weight || 1;
|
|
348
|
+
console.log(` ${(index + 1).toString().padStart(2)}: ${indent}${kpi.name} (L${level}, ${childCount} children, weight: ${weight})`);
|
|
349
|
+
});
|
|
350
|
+
console.log(`\\n📈 Total: ${sortedParentKpis.length} parent KPIs`);
|
|
351
|
+
}
|
|
352
|
+
catch (error) {
|
|
353
|
+
console.error('❌ Error listing parent KPIs:', error);
|
|
354
|
+
}
|
|
355
|
+
finally {
|
|
356
|
+
if (connection) {
|
|
357
|
+
await connection.close();
|
|
358
|
+
}
|
|
359
|
+
}
|
|
360
|
+
}
|
|
361
|
+
/**
|
|
362
|
+
* CLI execution when called directly
|
|
363
|
+
*/
|
|
364
|
+
async function main() {
|
|
365
|
+
const args = process.argv.slice(2);
|
|
366
|
+
try {
|
|
367
|
+
if (args.includes('--help') || args.includes('-h')) {
|
|
368
|
+
console.log(`
|
|
369
|
+
🎯 Parent KPI Value Propagator
|
|
370
|
+
|
|
371
|
+
Usage:
|
|
372
|
+
ts-node server/scripts/propagate-parent-kpi-values.ts [options] [kpiName]
|
|
373
|
+
|
|
374
|
+
Options:
|
|
375
|
+
--list, -l List parent KPIs and their hierarchy
|
|
376
|
+
--force, -f Force recalculate all values (not just missing ones)
|
|
377
|
+
--date YYYY-MM-DD Target date for value calculation (default: today)
|
|
378
|
+
--org orgScope Organization scope filter (default: all orgs)
|
|
379
|
+
--version N KPI version filter (default: latest version)
|
|
380
|
+
--help, -h Show this help
|
|
381
|
+
|
|
382
|
+
Examples:
|
|
383
|
+
ts-node server/scripts/propagate-parent-kpi-values.ts # Process all parent KPIs for today
|
|
384
|
+
ts-node server/scripts/propagate-parent-kpi-values.ts --force # Recalculate ALL parent values
|
|
385
|
+
ts-node server/scripts/propagate-parent-kpi-values.ts --date 2024-01-15 # Process for specific date
|
|
386
|
+
ts-node server/scripts/propagate-parent-kpi-values.ts --org "PROJECT-123" # Process for specific org
|
|
387
|
+
ts-node server/scripts/propagate-parent-kpi-values.ts --version 2 # Process for specific version
|
|
388
|
+
ts-node server/scripts/propagate-parent-kpi-values.ts "상위 KPI명" # Process specific parent KPI
|
|
389
|
+
ts-node server/scripts/propagate-parent-kpi-values.ts --list # List parent KPI hierarchy
|
|
390
|
+
|
|
391
|
+
Note: Processes parent KPIs from deepest level to root, calculating weighted averages.
|
|
392
|
+
Child KPI scores and weights are used to calculate parent values.
|
|
393
|
+
`);
|
|
394
|
+
return;
|
|
395
|
+
}
|
|
396
|
+
if (args.includes('--list') || args.includes('-l')) {
|
|
397
|
+
await listParentKpiHierarchy();
|
|
398
|
+
return;
|
|
399
|
+
}
|
|
400
|
+
const forceRecalculate = args.includes('--force') || args.includes('-f');
|
|
401
|
+
const dateIndex = args.indexOf('--date');
|
|
402
|
+
const targetDate = dateIndex !== -1 && dateIndex + 1 < args.length ? args[dateIndex + 1] : undefined;
|
|
403
|
+
const orgIndex = args.indexOf('--org');
|
|
404
|
+
const orgScope = orgIndex !== -1 && orgIndex + 1 < args.length ? args[orgIndex + 1] : undefined;
|
|
405
|
+
const versionIndex = args.indexOf('--version');
|
|
406
|
+
const version = versionIndex !== -1 && versionIndex + 1 < args.length ? parseInt(args[versionIndex + 1]) : undefined;
|
|
407
|
+
const specificKpiName = args.find(arg => !arg.startsWith('--') && arg !== targetDate && arg !== orgScope && (version === undefined || arg !== version.toString()));
|
|
408
|
+
if (specificKpiName) {
|
|
409
|
+
console.log(`🎯 Propagating values for specific KPI: ${specificKpiName}`);
|
|
410
|
+
console.log(` Mode: ${forceRecalculate ? 'Force recalculate ALL' : 'Only missing values'}`);
|
|
411
|
+
console.log(` Date: ${targetDate || 'today'}`);
|
|
412
|
+
console.log(` Org: ${orgScope || 'all'}`);
|
|
413
|
+
console.log(` Version: ${version || 'latest'}`);
|
|
414
|
+
await propagateParentKpiValues(specificKpiName, forceRecalculate, targetDate, orgScope, version);
|
|
415
|
+
}
|
|
416
|
+
else {
|
|
417
|
+
console.log(`🎯 Propagating values for all parent KPIs`);
|
|
418
|
+
console.log(` Mode: ${forceRecalculate ? 'Force recalculate ALL' : 'Only missing values'}`);
|
|
419
|
+
console.log(` Date: ${targetDate || 'today'}`);
|
|
420
|
+
console.log(` Org: ${orgScope || 'all'}`);
|
|
421
|
+
console.log(` Version: ${version || 'latest'}`);
|
|
422
|
+
await propagateParentKpiValues(undefined, forceRecalculate, targetDate, orgScope, version);
|
|
423
|
+
}
|
|
424
|
+
console.log('\\n✨ Done!');
|
|
425
|
+
}
|
|
426
|
+
catch (error) {
|
|
427
|
+
console.error('\\n❌ Script failed:', error.message);
|
|
428
|
+
process.exit(1);
|
|
429
|
+
}
|
|
430
|
+
}
|
|
431
|
+
// Handle uncaught errors
|
|
432
|
+
process.on('unhandledRejection', (error) => {
|
|
433
|
+
console.error('❌ Unhandled error:', error.message || error);
|
|
434
|
+
process.exit(1);
|
|
435
|
+
});
|
|
436
|
+
// Run if called directly
|
|
437
|
+
if (require.main === module) {
|
|
438
|
+
main();
|
|
439
|
+
}
|
|
440
|
+
//# sourceMappingURL=propagate-parent-kpi-values.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"propagate-parent-kpi-values.js","sourceRoot":"","sources":["../../server/scripts/propagate-parent-kpi-values.ts"],"names":[],"mappings":"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","sourcesContent":["#!/usr/bin/env ts-node\n\nimport { createConnection } from 'typeorm'\nimport * as fs from 'fs'\nimport * as path from 'path'\n\ninterface KpiRecord {\n id: string\n name: string\n isLeaf: boolean\n weight: number\n version?: number\n parent?: {\n id: string\n name: string\n }\n children?: KpiRecord[]\n __level?: number // Added for hierarchy level tracking\n}\n\ninterface KpiValueRecord {\n id: string\n value: number\n score: number\n kpi: {\n id: string\n name: string\n }\n valueDate?: Date\n org?: any\n}\n\n/**\n * Propagate parent KPI values based on child KPI scores and weights\n * Processes hierarchy from leaf to root, calculating weighted averages\n */\nexport async function propagateParentKpiValues(\n specificKpiName?: string,\n forceRecalculate = false,\n targetDate?: string,\n orgScope?: string,\n version?: number\n): Promise<void> {\n // Set NODE_ENV if not set\n if (!process.env.NODE_ENV) {\n process.env.NODE_ENV = 'development'\n }\n\n // Initialize Things-Factory environment\n const { config } = require('@things-factory/env')\n const ormconfig = require('@things-factory/shell/dist-server/initializers/ormconfig.js')\n const connectionConfig = config.get('ormconfig')\n\n let connection\n\n console.log('🚀 Starting Parent KPI Value Propagation...')\n\n try {\n // Create database connection using Things-Factory pattern\n console.log('🔌 Connecting to database...')\n connection = await createConnection({\n ...ormconfig,\n ...connectionConfig,\n logging: false\n })\n\n // Register the connection with Things-Factory shell\n const { addDataSource } = require('@things-factory/shell')\n addDataSource('default', connection)\n\n console.log('✅ Database connected')\n\n // Now we can use getRepository with the registered connection\n const { getRepository } = require('@things-factory/shell')\n const { Kpi } = require('@things-factory/kpi')\n const kpiRepository = getRepository(Kpi)\n\n // Get KPI-Value repository\n let kpiValueRepository\n try {\n const { KpiValue } = require('@things-factory/kpi')\n kpiValueRepository = getRepository(KpiValue)\n } catch (error) {\n console.error('❌ KpiValue entity not found. Please ensure it exists in @things-factory/kpi')\n return\n }\n\n // Get Domain repository and find SYSTEM domain\n let systemDomain\n try {\n const { Domain } = require('@things-factory/shell')\n const domainRepository = getRepository(Domain)\n systemDomain = await domainRepository.findOne({ where: { name: 'SYSTEM' } })\n if (!systemDomain) {\n console.error('❌ SYSTEM domain not found')\n return\n }\n } catch (error) {\n console.error('❌ Domain entity not found:', error.message)\n return\n }\n\n // Get all KPIs with their hierarchy\n let kpisQuery = kpiRepository\n .createQueryBuilder('kpi')\n .leftJoinAndSelect('kpi.domain', 'domain')\n .leftJoinAndSelect('kpi.parent', 'parent')\n .leftJoinAndSelect('kpi.children', 'children')\n .where('domain.name = :domainName', { domainName: 'SYSTEM' })\n\n if (specificKpiName) {\n kpisQuery = kpisQuery.andWhere('kpi.name = :name', { name: specificKpiName })\n console.log(`🎯 Processing specific KPI: ${specificKpiName}`)\n } else {\n console.log('🎯 Processing all KPIs in hierarchy')\n }\n\n const allKpis = await kpisQuery.getMany()\n\n if (allKpis.length === 0) {\n console.log('⚠️ No KPIs found')\n return\n }\n\n console.log(`📊 Found ${allKpis.length} KPIs total`)\n\n // Get parent KPIs (non-leaf KPIs) ordered by level (deepest first)\n const parentKpis = allKpis.filter(kpi => !kpi.isLeaf)\n\n if (parentKpis.length === 0) {\n console.log('⚠️ No parent KPIs found')\n return\n }\n\n // Sort parent KPIs by hierarchy level (deepest first for bottom-up processing)\n const sortedParentKpis = await sortKpisByLevel(parentKpis, allKpis)\n\n console.log(`🔄 Processing ${sortedParentKpis.length} parent KPIs`)\n\n let totalProcessed = 0\n let totalCreated = 0\n let totalUpdated = 0\n let errorCount = 0\n\n // Get all distinct (group, valueDate) combinations from existing KPI values\n let groupDateCombinations: { group: string; valueDate: string }[] = []\n if (orgScope && targetDate) {\n groupDateCombinations = [{ group: orgScope, valueDate: targetDate }]\n } else if (orgScope) {\n // Get all dates for specific group\n const dates = await kpiValueRepository.query(\n `\n SELECT DISTINCT \"value_date\" \n FROM kpi_values \n WHERE \"group\" = $1 AND \"group\" IS NOT NULL\n ORDER BY \"value_date\"\n `,\n [orgScope]\n )\n groupDateCombinations = dates.map(row => ({ group: orgScope, valueDate: row.value_date }))\n } else {\n // Get all distinct (group, valueDate) combinations\n const combinations = await kpiValueRepository.query(`\n SELECT DISTINCT \"group\", \"value_date\" \n FROM kpi_values \n WHERE \"group\" IS NOT NULL AND \"group\" != ''\n ORDER BY \"group\", \"value_date\"\n `)\n groupDateCombinations = combinations.map(row => ({ group: row.group, valueDate: row.value_date }))\n }\n\n console.log(`📍 Found ${groupDateCombinations.length} distinct (group, date) combinations`)\n\n // Process each (group, date) combination\n for (const { group: currentGroup, valueDate } of groupDateCombinations) {\n console.log(`\\\\n🏢 Processing Group: ${currentGroup}, Date: ${valueDate}`)\n\n // Process each parent KPI from deepest to shallowest for this group and date\n for (const parentKpi of sortedParentKpis) {\n try {\n console.log(\n `\\\\n📄 Processing Parent KPI: ${parentKpi.name} (Level ${parentKpi.__level || '?'}) - Group: ${currentGroup}, Date: ${valueDate}`\n )\n\n // Get child KPIs\n const childKpis = allKpis.filter(kpi => kpi.parent?.id === parentKpi.id)\n\n if (childKpis.length === 0) {\n console.log(` ⚠️ No child KPIs found, skipping`)\n continue\n }\n\n console.log(` 👶 Found ${childKpis.length} child KPIs`)\n\n // Get child KPI values/scores for the specific group and date\n const childValues = await getChildKpiValues(kpiValueRepository, childKpis, valueDate, version, currentGroup)\n\n if (childValues.length === 0) {\n console.log(` ⚠️ No child KPI values found for date ${valueDate} and group ${currentGroup}, skipping`)\n continue\n }\n\n console.log(` 📊 Found values for ${childValues.length}/${childKpis.length} child KPIs`)\n\n // Calculate weighted average\n const { weightedValue, weightedScore } = calculateWeightedAverage(childValues, childKpis)\n\n console.log(` 🧮 Calculated: value=${weightedValue.toFixed(4)}, score=${weightedScore.toFixed(4)}`)\n\n // Check if parent KPI value already exists for this org scope\n let existingValueQuery = kpiValueRepository\n .createQueryBuilder('kpiValue')\n .leftJoinAndSelect('kpiValue.kpi', 'kpi')\n .leftJoinAndSelect('kpiValue.domain', 'domain')\n .where('kpi.id = :kpiId', { kpiId: parentKpi.id })\n .andWhere('kpiValue.valueDate = :valueDate', { valueDate })\n .andWhere('domain.name = :domainName', { domainName: 'SYSTEM' })\n\n if (version !== undefined) {\n existingValueQuery = existingValueQuery.andWhere('kpiValue.version = :version', { version })\n }\n\n // Filter by group scope\n existingValueQuery = existingValueQuery.andWhere('kpiValue.group = :groupScope', { groupScope: currentGroup })\n\n const existingValue = await existingValueQuery.getOne()\n\n console.log(` 🔍 Checking existing record: ${existingValue ? 'FOUND' : 'NOT FOUND'}`)\n\n if (existingValue) {\n // Update existing value\n if (forceRecalculate || existingValue.value !== weightedValue || existingValue.score !== weightedScore) {\n const oldValue = existingValue.value\n const oldScore = existingValue.score\n\n existingValue.value = weightedValue\n existingValue.score = weightedScore\n existingValue.kpiScope = currentGroup // Set kpiScope same as group\n existingValue.updatedAt = new Date()\n\n try {\n await kpiValueRepository.save(existingValue)\n totalUpdated++\n console.log(\n ` ✅ Updated: ${oldValue.toFixed(4)} → ${weightedValue.toFixed(4)}, score: ${oldScore.toFixed(4)} → ${weightedScore.toFixed(4)}`\n )\n } catch (saveError) {\n console.log(` ❌ Update failed: ${saveError.message}`)\n errorCount++\n }\n } else {\n console.log(` ⏭️ No change needed`)\n }\n } else {\n // Create new value with group scope\n const newKpiValue: any = {\n kpi: parentKpi,\n value: weightedValue,\n score: weightedScore,\n valueDate: valueDate,\n version: version || parentKpi.version || 1,\n group: currentGroup,\n kpiScope: currentGroup, // Set kpiScope same as group\n domain: systemDomain, // Set domain entity\n inputType: 'AUTO', // Set inputType to AUTO\n createdAt: new Date(),\n updatedAt: new Date()\n }\n\n try {\n const savedValue = await kpiValueRepository.save(newKpiValue)\n totalCreated++\n console.log(\n ` ✨ Created: value=${weightedValue.toFixed(4)}, score=${weightedScore.toFixed(4)}, id=${savedValue.id}`\n )\n } catch (saveError) {\n console.log(` ❌ Save failed: ${saveError.message}`)\n errorCount++\n }\n }\n\n totalProcessed++\n } catch (kpiError) {\n console.log(` ❌ Error processing KPI ${parentKpi.name}: ${kpiError.message}`)\n errorCount++\n }\n }\n }\n\n console.log(`\\\\n🎉 Parent KPI Value Propagation Complete!`)\n console.log(` 📊 Parent KPIs processed: ${totalProcessed}`)\n console.log(` ✨ Values created: ${totalCreated}`)\n console.log(` ✅ Values updated: ${totalUpdated}`)\n console.log(` ❌ Errors: ${errorCount}`)\n\n if (errorCount > 0) {\n console.log(`\\\\n⚠️ Some parent KPIs had errors. Check the logs above for details.`)\n }\n } catch (error) {\n console.error(`❌ Fatal error during parent KPI value propagation:`, error)\n throw error\n } finally {\n if (connection) {\n await connection.close()\n console.log('🔌 Database connection closed')\n }\n }\n}\n\n/**\n * Sort KPIs by hierarchy level (deepest first for bottom-up processing)\n */\nasync function sortKpisByLevel(parentKpis: KpiRecord[], allKpis: KpiRecord[]): Promise<KpiRecord[]> {\n // Calculate hierarchy level for each KPI\n const kpiLevels = new Map<string, number>()\n\n function calculateLevel(kpi: KpiRecord): number {\n if (kpiLevels.has(kpi.id)) {\n return kpiLevels.get(kpi.id)!\n }\n\n if (!kpi.parent) {\n kpiLevels.set(kpi.id, 0)\n return 0\n }\n\n const parentKpi = allKpis.find(k => k.id === kpi.parent?.id)\n if (!parentKpi) {\n kpiLevels.set(kpi.id, 1)\n return 1\n }\n\n const level = calculateLevel(parentKpi) + 1\n kpiLevels.set(kpi.id, level)\n return level\n }\n\n // Calculate levels for all parent KPIs\n parentKpis.forEach(kpi => {\n const level = calculateLevel(kpi)\n ;(kpi as any).__level = level\n })\n\n // Sort by level (deepest first)\n return parentKpis.sort((a, b) => ((b as any).__level || 0) - ((a as any).__level || 0))\n}\n\n/**\n * Get child KPI values for a specific date\n */\nasync function getChildKpiValues(\n kpiValueRepository: any,\n childKpis: KpiRecord[],\n valueDate: string,\n version?: number,\n orgScope?: string\n): Promise<KpiValueRecord[]> {\n const childKpiIds = childKpis.map(kpi => kpi.id)\n\n let valuesQuery = kpiValueRepository\n .createQueryBuilder('kpiValue')\n .leftJoinAndSelect('kpiValue.kpi', 'kpi')\n .where('kpi.id IN (:...kpiIds)', { kpiIds: childKpiIds })\n .andWhere('kpiValue.valueDate = :valueDate', { valueDate })\n .andWhere('kpiValue.score IS NOT NULL')\n\n if (version !== undefined) {\n valuesQuery = valuesQuery.andWhere('kpiValue.version = :version', { version })\n }\n\n // Filter by group scope\n if (orgScope) {\n valuesQuery = valuesQuery.andWhere('kpiValue.group = :groupScope', { groupScope: orgScope })\n }\n\n const values = await valuesQuery.getMany()\n\n return values\n}\n\n/**\n * Calculate weighted average from child KPI values\n */\nfunction calculateWeightedAverage(\n childValues: KpiValueRecord[],\n childKpis: KpiRecord[]\n): { weightedValue: number; weightedScore: number } {\n let totalWeightedValue = 0\n let totalWeightedScore = 0\n let totalWeight = 0\n\n for (const value of childValues) {\n const childKpi = childKpis.find(kpi => kpi.id === value.kpi.id)\n const weight = childKpi?.weight || 1\n\n totalWeightedValue += value.score * weight\n totalWeightedScore += value.score * weight\n totalWeight += weight\n }\n\n if (totalWeight === 0) {\n return { weightedValue: 0, weightedScore: 0 }\n }\n\n return {\n weightedValue: totalWeightedValue / totalWeight,\n weightedScore: totalWeightedScore / totalWeight\n }\n}\n\n/**\n * List parent KPIs and their hierarchy\n */\nexport async function listParentKpiHierarchy(): Promise<void> {\n // Set NODE_ENV if not set\n if (!process.env.NODE_ENV) {\n process.env.NODE_ENV = 'development'\n }\n\n // Initialize Things-Factory environment\n const { config } = require('@things-factory/env')\n const ormconfig = require('@things-factory/shell/dist-server/initializers/ormconfig.js')\n const connectionConfig = config.get('ormconfig')\n\n let connection\n\n try {\n console.log('🔌 Connecting to database...')\n connection = await createConnection({\n ...ormconfig,\n ...connectionConfig,\n logging: false\n })\n\n const { addDataSource } = require('@things-factory/shell')\n addDataSource('default', connection)\n\n const { getRepository } = require('@things-factory/shell')\n const { Kpi } = require('@things-factory/kpi')\n const kpiRepository = getRepository(Kpi)\n\n // Get all KPIs with hierarchy\n const kpis = await kpiRepository\n .createQueryBuilder('kpi')\n .leftJoinAndSelect('kpi.domain', 'domain')\n .leftJoinAndSelect('kpi.parent', 'parent')\n .leftJoinAndSelect('kpi.children', 'children')\n .where('domain.name = :domainName', { domainName: 'SYSTEM' })\n .getMany()\n\n const parentKpis = kpis.filter(kpi => !kpi.isLeaf)\n const sortedParentKpis = await sortKpisByLevel(parentKpis, kpis)\n\n console.log(`📁 Parent KPI Hierarchy (${sortedParentKpis.length} parents):\\\\n`)\n\n if (sortedParentKpis.length === 0) {\n console.log(' No parent KPIs found.')\n return\n }\n\n sortedParentKpis.forEach((kpi, index) => {\n const level = (kpi as any).__level || 0\n const indent = ' '.repeat(level)\n const childCount = kpis.filter(k => k.parent?.id === kpi.id).length\n const weight = kpi.weight || 1\n\n console.log(\n ` ${(index + 1).toString().padStart(2)}: ${indent}${kpi.name} (L${level}, ${childCount} children, weight: ${weight})`\n )\n })\n\n console.log(`\\\\n📈 Total: ${sortedParentKpis.length} parent KPIs`)\n } catch (error) {\n console.error('❌ Error listing parent KPIs:', error)\n } finally {\n if (connection) {\n await connection.close()\n }\n }\n}\n\n/**\n * CLI execution when called directly\n */\nasync function main() {\n const args = process.argv.slice(2)\n\n try {\n if (args.includes('--help') || args.includes('-h')) {\n console.log(`\n🎯 Parent KPI Value Propagator\n\nUsage:\n ts-node server/scripts/propagate-parent-kpi-values.ts [options] [kpiName]\n\nOptions:\n --list, -l List parent KPIs and their hierarchy\n --force, -f Force recalculate all values (not just missing ones)\n --date YYYY-MM-DD Target date for value calculation (default: today)\n --org orgScope Organization scope filter (default: all orgs)\n --version N KPI version filter (default: latest version)\n --help, -h Show this help\n\nExamples:\n ts-node server/scripts/propagate-parent-kpi-values.ts # Process all parent KPIs for today\n ts-node server/scripts/propagate-parent-kpi-values.ts --force # Recalculate ALL parent values\n ts-node server/scripts/propagate-parent-kpi-values.ts --date 2024-01-15 # Process for specific date\n ts-node server/scripts/propagate-parent-kpi-values.ts --org \"PROJECT-123\" # Process for specific org\n ts-node server/scripts/propagate-parent-kpi-values.ts --version 2 # Process for specific version\n ts-node server/scripts/propagate-parent-kpi-values.ts \"상위 KPI명\" # Process specific parent KPI\n ts-node server/scripts/propagate-parent-kpi-values.ts --list # List parent KPI hierarchy\n\nNote: Processes parent KPIs from deepest level to root, calculating weighted averages.\n Child KPI scores and weights are used to calculate parent values.\n `)\n return\n }\n\n if (args.includes('--list') || args.includes('-l')) {\n await listParentKpiHierarchy()\n return\n }\n\n const forceRecalculate = args.includes('--force') || args.includes('-f')\n const dateIndex = args.indexOf('--date')\n const targetDate = dateIndex !== -1 && dateIndex + 1 < args.length ? args[dateIndex + 1] : undefined\n const orgIndex = args.indexOf('--org')\n const orgScope = orgIndex !== -1 && orgIndex + 1 < args.length ? args[orgIndex + 1] : undefined\n const versionIndex = args.indexOf('--version')\n const version = versionIndex !== -1 && versionIndex + 1 < args.length ? parseInt(args[versionIndex + 1]) : undefined\n const specificKpiName = args.find(\n arg =>\n !arg.startsWith('--') && arg !== targetDate && arg !== orgScope && (version === undefined || arg !== version.toString())\n )\n\n if (specificKpiName) {\n console.log(`🎯 Propagating values for specific KPI: ${specificKpiName}`)\n console.log(` Mode: ${forceRecalculate ? 'Force recalculate ALL' : 'Only missing values'}`)\n console.log(` Date: ${targetDate || 'today'}`)\n console.log(` Org: ${orgScope || 'all'}`)\n console.log(` Version: ${version || 'latest'}`)\n await propagateParentKpiValues(specificKpiName, forceRecalculate, targetDate, orgScope, version)\n } else {\n console.log(`🎯 Propagating values for all parent KPIs`)\n console.log(` Mode: ${forceRecalculate ? 'Force recalculate ALL' : 'Only missing values'}`)\n console.log(` Date: ${targetDate || 'today'}`)\n console.log(` Org: ${orgScope || 'all'}`)\n console.log(` Version: ${version || 'latest'}`)\n await propagateParentKpiValues(undefined, forceRecalculate, targetDate, orgScope, version)\n }\n\n console.log('\\\\n✨ Done!')\n } catch (error) {\n console.error('\\\\n❌ Script failed:', error.message)\n process.exit(1)\n }\n}\n\n// Handle uncaught errors\nprocess.on('unhandledRejection', (error: any) => {\n console.error('❌ Unhandled error:', error.message || error)\n process.exit(1)\n})\n\n// Run if called directly\nif (require.main === module) {\n main()\n}\n"]}
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
export * from './kpi-value/kpi-value-query.js';
|
|
2
|
+
export * from './kpi-metric-value/kpi-metric-value-query.js';
|
|
3
|
+
export declare const entities: any[];
|
|
4
|
+
export declare const schema: {
|
|
5
|
+
resolverClasses: (typeof import("./kpi-value/kpi-value-query.js").KpiValueQueryForProject | typeof import("./kpi-metric-value/kpi-metric-value-query.js").KpiMetricValueQuery)[];
|
|
6
|
+
};
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.schema = exports.entities = void 0;
|
|
4
|
+
const tslib_1 = require("tslib");
|
|
5
|
+
/* EXPORT ENTITY TYPES */
|
|
6
|
+
tslib_1.__exportStar(require("./kpi-value/kpi-value-query.js"), exports);
|
|
7
|
+
tslib_1.__exportStar(require("./kpi-metric-value/kpi-metric-value-query.js"), exports);
|
|
8
|
+
/* IMPORT ENTITIES AND RESOLVERS */
|
|
9
|
+
const kpi_value_1 = require("./kpi-value");
|
|
10
|
+
const kpi_metric_value_1 = require("./kpi-metric-value");
|
|
11
|
+
exports.entities = [
|
|
12
|
+
/* ENTITIES */
|
|
13
|
+
];
|
|
14
|
+
exports.schema = {
|
|
15
|
+
resolverClasses: [
|
|
16
|
+
/* RESOLVER CLASSES */
|
|
17
|
+
...kpi_value_1.resolvers,
|
|
18
|
+
...kpi_metric_value_1.resolvers
|
|
19
|
+
]
|
|
20
|
+
};
|
|
21
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../server/service/index.ts"],"names":[],"mappings":";;;;AAAA,yBAAyB;AACzB,yEAA8C;AAC9C,uFAA4D;AAE5D,mCAAmC;AACnC,2CAA4D;AAC5D,yDAAyE;AAE5D,QAAA,QAAQ,GAAG;AACtB,cAAc;CACf,CAAA;AAEY,QAAA,MAAM,GAAG;IACpB,eAAe,EAAE;QACf,sBAAsB;QACtB,GAAG,qBAAiB;QACpB,GAAG,4BAAuB;KAC3B;CACF,CAAA","sourcesContent":["/* EXPORT ENTITY TYPES */\nexport * from './kpi-value/kpi-value-query.js'\nexport * from './kpi-metric-value/kpi-metric-value-query.js'\n\n/* IMPORT ENTITIES AND RESOLVERS */\nimport { resolvers as KpiValueResolvers } from './kpi-value'\nimport { resolvers as KpiMetricValueResolvers } from './kpi-metric-value'\n\nexport const entities = [\n /* ENTITIES */\n]\n\nexport const schema = {\n resolverClasses: [\n /* RESOLVER CLASSES */\n ...KpiValueResolvers,\n ...KpiMetricValueResolvers\n ]\n}\n"]}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.subscribers = exports.resolvers = void 0;
|
|
4
|
+
const kpi_metric_value_query_1 = require("./kpi-metric-value-query");
|
|
5
|
+
exports.resolvers = [kpi_metric_value_query_1.KpiMetricValueQuery];
|
|
6
|
+
exports.subscribers = [];
|
|
7
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../server/service/kpi-metric-value/index.ts"],"names":[],"mappings":";;;AAAA,qEAA8D;AAEjD,QAAA,SAAS,GAAG,CAAC,4CAAmB,CAAC,CAAA;AACjC,QAAA,WAAW,GAAG,EAAE,CAAA","sourcesContent":["import { KpiMetricValueQuery } from './kpi-metric-value-query'\n\nexport const resolvers = [KpiMetricValueQuery]\nexport const subscribers = []\n"]}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import { ListParam } from '@things-factory/shell';
|
|
2
|
+
import { KpiMetricValue, KpiMetricValueList } from '@things-factory/kpi';
|
|
3
|
+
import { Project } from '@dssp/project';
|
|
4
|
+
export declare class KpiMetricValueQuery {
|
|
5
|
+
kpiMetricValues(params: ListParam, context: ResolverContext): Promise<KpiMetricValueList>;
|
|
6
|
+
project(kpiMetricValue: KpiMetricValue): Promise<Project | null>;
|
|
7
|
+
}
|