business-as-code 0.2.1 → 2.0.2
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/.turbo/turbo-build.log +5 -0
- package/CHANGELOG.md +17 -0
- package/IMPLEMENTATION.md +226 -0
- package/README.md +1133 -193
- package/dist/business.d.ts +62 -0
- package/dist/business.d.ts.map +1 -0
- package/dist/business.js +109 -0
- package/dist/business.js.map +1 -0
- package/dist/dollar.d.ts +60 -0
- package/dist/dollar.d.ts.map +1 -0
- package/dist/dollar.js +107 -0
- package/dist/dollar.js.map +1 -0
- package/dist/entities/assets.d.ts +21 -0
- package/dist/entities/assets.d.ts.map +1 -0
- package/dist/entities/assets.js +323 -0
- package/dist/entities/assets.js.map +1 -0
- package/dist/entities/business.d.ts +36 -0
- package/dist/entities/business.d.ts.map +1 -0
- package/dist/entities/business.js +370 -0
- package/dist/entities/business.js.map +1 -0
- package/dist/entities/communication.d.ts +21 -0
- package/dist/entities/communication.d.ts.map +1 -0
- package/dist/entities/communication.js +255 -0
- package/dist/entities/communication.js.map +1 -0
- package/dist/entities/customers.d.ts +58 -0
- package/dist/entities/customers.d.ts.map +1 -0
- package/dist/entities/customers.js +989 -0
- package/dist/entities/customers.js.map +1 -0
- package/dist/entities/financials.d.ts +59 -0
- package/dist/entities/financials.d.ts.map +1 -0
- package/dist/entities/financials.js +932 -0
- package/dist/entities/financials.js.map +1 -0
- package/dist/entities/goals.d.ts +58 -0
- package/dist/entities/goals.d.ts.map +1 -0
- package/dist/entities/goals.js +800 -0
- package/dist/entities/goals.js.map +1 -0
- package/dist/entities/index.d.ts +299 -0
- package/dist/entities/index.d.ts.map +1 -0
- package/dist/entities/index.js +198 -0
- package/dist/entities/index.js.map +1 -0
- package/dist/entities/legal.d.ts +21 -0
- package/dist/entities/legal.d.ts.map +1 -0
- package/dist/entities/legal.js +301 -0
- package/dist/entities/legal.js.map +1 -0
- package/dist/entities/market.d.ts +21 -0
- package/dist/entities/market.d.ts.map +1 -0
- package/dist/entities/market.js +301 -0
- package/dist/entities/market.js.map +1 -0
- package/dist/entities/marketing.d.ts +67 -0
- package/dist/entities/marketing.d.ts.map +1 -0
- package/dist/entities/marketing.js +1157 -0
- package/dist/entities/marketing.js.map +1 -0
- package/dist/entities/offerings.d.ts +51 -0
- package/dist/entities/offerings.d.ts.map +1 -0
- package/dist/entities/offerings.js +727 -0
- package/dist/entities/offerings.js.map +1 -0
- package/dist/entities/operations.d.ts +58 -0
- package/dist/entities/operations.d.ts.map +1 -0
- package/dist/entities/operations.js +787 -0
- package/dist/entities/operations.js.map +1 -0
- package/dist/entities/organization.d.ts +57 -0
- package/dist/entities/organization.d.ts.map +1 -0
- package/dist/entities/organization.js +807 -0
- package/dist/entities/organization.js.map +1 -0
- package/dist/entities/partnerships.d.ts +21 -0
- package/dist/entities/partnerships.d.ts.map +1 -0
- package/dist/entities/partnerships.js +300 -0
- package/dist/entities/partnerships.js.map +1 -0
- package/dist/entities/planning.d.ts +87 -0
- package/dist/entities/planning.d.ts.map +1 -0
- package/dist/entities/planning.js +271 -0
- package/dist/entities/planning.js.map +1 -0
- package/dist/entities/projects.d.ts +25 -0
- package/dist/entities/projects.d.ts.map +1 -0
- package/dist/entities/projects.js +349 -0
- package/dist/entities/projects.js.map +1 -0
- package/dist/entities/risk.d.ts +21 -0
- package/dist/entities/risk.d.ts.map +1 -0
- package/dist/entities/risk.js +293 -0
- package/dist/entities/risk.js.map +1 -0
- package/dist/entities/sales.d.ts +72 -0
- package/dist/entities/sales.d.ts.map +1 -0
- package/dist/entities/sales.js +1248 -0
- package/dist/entities/sales.js.map +1 -0
- package/dist/financials.d.ts +130 -0
- package/dist/financials.d.ts.map +1 -0
- package/dist/financials.js +297 -0
- package/dist/financials.js.map +1 -0
- package/dist/goals.d.ts +87 -0
- package/dist/goals.d.ts.map +1 -0
- package/dist/goals.js +215 -0
- package/dist/goals.js.map +1 -0
- package/dist/index.d.ts +97 -4
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +131 -1079
- package/dist/index.js.map +1 -1
- package/dist/kpis.d.ts +118 -0
- package/dist/kpis.d.ts.map +1 -0
- package/dist/kpis.js +232 -0
- package/dist/kpis.js.map +1 -0
- package/dist/metrics.d.ts +448 -0
- package/dist/metrics.d.ts.map +1 -0
- package/dist/metrics.js +325 -0
- package/dist/metrics.js.map +1 -0
- package/dist/okrs.d.ts +123 -0
- package/dist/okrs.d.ts.map +1 -0
- package/dist/okrs.js +269 -0
- package/dist/okrs.js.map +1 -0
- package/dist/organization.d.ts +585 -0
- package/dist/organization.d.ts.map +1 -0
- package/dist/organization.js +173 -0
- package/dist/organization.js.map +1 -0
- package/dist/process.d.ts +112 -0
- package/dist/process.d.ts.map +1 -0
- package/dist/process.js +241 -0
- package/dist/process.js.map +1 -0
- package/dist/product.d.ts +85 -0
- package/dist/product.d.ts.map +1 -0
- package/dist/product.js +145 -0
- package/dist/product.js.map +1 -0
- package/dist/queries.d.ts +304 -0
- package/dist/queries.d.ts.map +1 -0
- package/dist/queries.js +415 -0
- package/dist/queries.js.map +1 -0
- package/dist/roles.d.ts +340 -0
- package/dist/roles.d.ts.map +1 -0
- package/dist/roles.js +255 -0
- package/dist/roles.js.map +1 -0
- package/dist/service.d.ts +61 -0
- package/dist/service.d.ts.map +1 -0
- package/dist/service.js +140 -0
- package/dist/service.js.map +1 -0
- package/dist/types.d.ts +459 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +5 -0
- package/dist/types.js.map +1 -0
- package/dist/vision.d.ts +38 -0
- package/dist/vision.d.ts.map +1 -0
- package/dist/vision.js +68 -0
- package/dist/vision.js.map +1 -0
- package/dist/workflow.d.ts +115 -0
- package/dist/workflow.d.ts.map +1 -0
- package/dist/workflow.js +247 -0
- package/dist/workflow.js.map +1 -0
- package/examples/basic-usage.ts +307 -0
- package/package.json +19 -60
- package/src/business.ts +121 -0
- package/src/dollar.ts +132 -0
- package/src/entities/assets.ts +332 -0
- package/src/entities/business.ts +406 -0
- package/src/entities/communication.ts +264 -0
- package/src/entities/customers.ts +1072 -0
- package/src/entities/financials.ts +1011 -0
- package/src/entities/goals.ts +871 -0
- package/src/entities/index.ts +383 -0
- package/src/entities/legal.ts +310 -0
- package/src/entities/market.ts +310 -0
- package/src/entities/marketing.ts +1249 -0
- package/src/entities/offerings.ts +789 -0
- package/src/entities/operations.ts +861 -0
- package/src/entities/organization.ts +876 -0
- package/src/entities/partnerships.ts +309 -0
- package/src/entities/planning.ts +307 -0
- package/src/entities/projects.ts +360 -0
- package/src/entities/risk.ts +302 -0
- package/src/entities/sales.ts +1352 -0
- package/src/financials.ts +352 -0
- package/src/goals.ts +250 -0
- package/src/index.test.ts +336 -0
- package/src/index.ts +530 -0
- package/src/kpis.ts +275 -0
- package/src/metrics.ts +825 -0
- package/src/okrs.ts +325 -0
- package/src/organization.ts +909 -0
- package/src/process.ts +272 -0
- package/src/product.ts +178 -0
- package/src/queries.ts +767 -0
- package/src/roles.ts +686 -0
- package/src/service.ts +164 -0
- package/src/types.ts +493 -0
- package/src/vision.ts +88 -0
- package/src/workflow.ts +280 -0
- package/tsconfig.json +9 -0
- package/dist/loaders/index.d.ts +0 -174
- package/dist/loaders/index.js +0 -366
- package/dist/loaders/index.js.map +0 -1
- package/dist/schema/index.d.ts +0 -146
- package/dist/schema/index.js +0 -716
- package/dist/schema/index.js.map +0 -1
- package/dist/types-CJ9eGS_C.d.ts +0 -86
package/src/kpis.ts
ADDED
|
@@ -0,0 +1,275 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Key Performance Indicators (KPIs) management
|
|
3
|
+
*/
|
|
4
|
+
|
|
5
|
+
import type { KPIDefinition, TimePeriod } from './types.js'
|
|
6
|
+
|
|
7
|
+
/**
|
|
8
|
+
* Define Key Performance Indicators for tracking business metrics
|
|
9
|
+
*
|
|
10
|
+
* @example
|
|
11
|
+
* ```ts
|
|
12
|
+
* const businessKPIs = kpis([
|
|
13
|
+
* {
|
|
14
|
+
* name: 'Monthly Recurring Revenue',
|
|
15
|
+
* description: 'Total predictable revenue per month',
|
|
16
|
+
* category: 'financial',
|
|
17
|
+
* unit: 'USD',
|
|
18
|
+
* target: 100000,
|
|
19
|
+
* current: 85000,
|
|
20
|
+
* frequency: 'monthly',
|
|
21
|
+
* dataSource: 'Billing System',
|
|
22
|
+
* formula: 'SUM(active_subscriptions.price)',
|
|
23
|
+
* },
|
|
24
|
+
* {
|
|
25
|
+
* name: 'Customer Churn Rate',
|
|
26
|
+
* description: 'Percentage of customers lost per month',
|
|
27
|
+
* category: 'customer',
|
|
28
|
+
* unit: 'percent',
|
|
29
|
+
* target: 5,
|
|
30
|
+
* current: 3.2,
|
|
31
|
+
* frequency: 'monthly',
|
|
32
|
+
* dataSource: 'CRM',
|
|
33
|
+
* formula: '(churned_customers / total_customers) * 100',
|
|
34
|
+
* },
|
|
35
|
+
* {
|
|
36
|
+
* name: 'Net Promoter Score',
|
|
37
|
+
* description: 'Customer satisfaction and loyalty metric',
|
|
38
|
+
* category: 'customer',
|
|
39
|
+
* unit: 'score',
|
|
40
|
+
* target: 50,
|
|
41
|
+
* current: 48,
|
|
42
|
+
* frequency: 'quarterly',
|
|
43
|
+
* dataSource: 'Survey Platform',
|
|
44
|
+
* },
|
|
45
|
+
* ])
|
|
46
|
+
* ```
|
|
47
|
+
*/
|
|
48
|
+
export function kpis(definitions: KPIDefinition[]): KPIDefinition[] {
|
|
49
|
+
return definitions.map(kpi => validateAndNormalizeKPI(kpi))
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
/**
|
|
53
|
+
* Define a single KPI
|
|
54
|
+
*/
|
|
55
|
+
export function kpi(definition: KPIDefinition): KPIDefinition {
|
|
56
|
+
return validateAndNormalizeKPI(definition)
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
/**
|
|
60
|
+
* Validate and normalize a KPI definition
|
|
61
|
+
*/
|
|
62
|
+
function validateAndNormalizeKPI(kpi: KPIDefinition): KPIDefinition {
|
|
63
|
+
if (!kpi.name) {
|
|
64
|
+
throw new Error('KPI name is required')
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
return {
|
|
68
|
+
...kpi,
|
|
69
|
+
category: kpi.category || 'operations',
|
|
70
|
+
frequency: kpi.frequency || 'monthly',
|
|
71
|
+
metadata: kpi.metadata || {},
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
/**
|
|
76
|
+
* Calculate KPI achievement percentage
|
|
77
|
+
*/
|
|
78
|
+
export function calculateAchievement(kpi: KPIDefinition): number {
|
|
79
|
+
if (kpi.target === undefined || kpi.current === undefined) return 0
|
|
80
|
+
if (kpi.target === 0) return 100
|
|
81
|
+
return (kpi.current / kpi.target) * 100
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
/**
|
|
85
|
+
* Check if KPI meets target
|
|
86
|
+
*/
|
|
87
|
+
export function meetsTarget(kpi: KPIDefinition): boolean {
|
|
88
|
+
if (kpi.target === undefined || kpi.current === undefined) return false
|
|
89
|
+
|
|
90
|
+
// For metrics where lower is better (like churn rate)
|
|
91
|
+
const lowerIsBetter = ['churn', 'cost', 'time', 'error', 'downtime'].some(term =>
|
|
92
|
+
kpi.name.toLowerCase().includes(term)
|
|
93
|
+
)
|
|
94
|
+
|
|
95
|
+
if (lowerIsBetter) {
|
|
96
|
+
return kpi.current <= kpi.target
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
return kpi.current >= kpi.target
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
/**
|
|
103
|
+
* Update KPI current value
|
|
104
|
+
*/
|
|
105
|
+
export function updateCurrent(kpi: KPIDefinition, value: number): KPIDefinition {
|
|
106
|
+
return {
|
|
107
|
+
...kpi,
|
|
108
|
+
current: value,
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
/**
|
|
113
|
+
* Update KPI target
|
|
114
|
+
*/
|
|
115
|
+
export function updateTarget(kpi: KPIDefinition, target: number): KPIDefinition {
|
|
116
|
+
return {
|
|
117
|
+
...kpi,
|
|
118
|
+
target,
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
/**
|
|
123
|
+
* Get KPIs by category
|
|
124
|
+
*/
|
|
125
|
+
export function getKPIsByCategory(
|
|
126
|
+
kpis: KPIDefinition[],
|
|
127
|
+
category: KPIDefinition['category']
|
|
128
|
+
): KPIDefinition[] {
|
|
129
|
+
return kpis.filter(k => k.category === category)
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
/**
|
|
133
|
+
* Get KPIs by frequency
|
|
134
|
+
*/
|
|
135
|
+
export function getKPIsByFrequency(kpis: KPIDefinition[], frequency: TimePeriod): KPIDefinition[] {
|
|
136
|
+
return kpis.filter(k => k.frequency === frequency)
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
/**
|
|
140
|
+
* Get KPIs that meet their targets
|
|
141
|
+
*/
|
|
142
|
+
export function getKPIsOnTarget(kpis: KPIDefinition[]): KPIDefinition[] {
|
|
143
|
+
return kpis.filter(meetsTarget)
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
/**
|
|
147
|
+
* Get KPIs that don't meet their targets
|
|
148
|
+
*/
|
|
149
|
+
export function getKPIsOffTarget(kpis: KPIDefinition[]): KPIDefinition[] {
|
|
150
|
+
return kpis.filter(kpi => !meetsTarget(kpi))
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
/**
|
|
154
|
+
* Calculate overall KPI health score (0-100)
|
|
155
|
+
*/
|
|
156
|
+
export function calculateHealthScore(kpis: KPIDefinition[]): number {
|
|
157
|
+
if (kpis.length === 0) return 0
|
|
158
|
+
|
|
159
|
+
const onTarget = getKPIsOnTarget(kpis).length
|
|
160
|
+
return (onTarget / kpis.length) * 100
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
/**
|
|
164
|
+
* Group KPIs by category
|
|
165
|
+
*/
|
|
166
|
+
export function groupByCategory(kpis: KPIDefinition[]): Map<string, KPIDefinition[]> {
|
|
167
|
+
const groups = new Map<string, KPIDefinition[]>()
|
|
168
|
+
|
|
169
|
+
for (const kpi of kpis) {
|
|
170
|
+
const category = kpi.category || 'other'
|
|
171
|
+
const existing = groups.get(category) || []
|
|
172
|
+
groups.set(category, [...existing, kpi])
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
return groups
|
|
176
|
+
}
|
|
177
|
+
|
|
178
|
+
/**
|
|
179
|
+
* Calculate variance from target
|
|
180
|
+
*/
|
|
181
|
+
export function calculateVariance(kpi: KPIDefinition): number {
|
|
182
|
+
if (kpi.target === undefined || kpi.current === undefined) return 0
|
|
183
|
+
return kpi.current - kpi.target
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
/**
|
|
187
|
+
* Calculate variance percentage from target
|
|
188
|
+
*/
|
|
189
|
+
export function calculateVariancePercentage(kpi: KPIDefinition): number {
|
|
190
|
+
if (kpi.target === undefined || kpi.current === undefined) return 0
|
|
191
|
+
if (kpi.target === 0) return 0
|
|
192
|
+
return ((kpi.current - kpi.target) / kpi.target) * 100
|
|
193
|
+
}
|
|
194
|
+
|
|
195
|
+
/**
|
|
196
|
+
* Format KPI value with unit
|
|
197
|
+
*/
|
|
198
|
+
export function formatValue(kpi: KPIDefinition, value?: number): string {
|
|
199
|
+
const val = value ?? kpi.current
|
|
200
|
+
if (val === undefined) return 'N/A'
|
|
201
|
+
|
|
202
|
+
const formatted = val.toLocaleString(undefined, {
|
|
203
|
+
minimumFractionDigits: 0,
|
|
204
|
+
maximumFractionDigits: 2,
|
|
205
|
+
})
|
|
206
|
+
|
|
207
|
+
if (!kpi.unit) return formatted
|
|
208
|
+
|
|
209
|
+
switch (kpi.unit.toLowerCase()) {
|
|
210
|
+
case 'usd':
|
|
211
|
+
case 'eur':
|
|
212
|
+
case 'gbp':
|
|
213
|
+
return `$${formatted}`
|
|
214
|
+
case 'percent':
|
|
215
|
+
case '%':
|
|
216
|
+
return `${formatted}%`
|
|
217
|
+
default:
|
|
218
|
+
return `${formatted} ${kpi.unit}`
|
|
219
|
+
}
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
/**
|
|
223
|
+
* Compare KPI performance over time
|
|
224
|
+
*/
|
|
225
|
+
export function comparePerformance(
|
|
226
|
+
current: KPIDefinition,
|
|
227
|
+
previous: KPIDefinition
|
|
228
|
+
): {
|
|
229
|
+
change: number
|
|
230
|
+
changePercent: number
|
|
231
|
+
improved: boolean
|
|
232
|
+
} {
|
|
233
|
+
if (current.current === undefined || previous.current === undefined) {
|
|
234
|
+
return { change: 0, changePercent: 0, improved: false }
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
const change = current.current - previous.current
|
|
238
|
+
const changePercent =
|
|
239
|
+
previous.current !== 0 ? (change / previous.current) * 100 : 0
|
|
240
|
+
|
|
241
|
+
// Determine if change is an improvement
|
|
242
|
+
const lowerIsBetter = ['churn', 'cost', 'time', 'error', 'downtime'].some(term =>
|
|
243
|
+
current.name.toLowerCase().includes(term)
|
|
244
|
+
)
|
|
245
|
+
|
|
246
|
+
const improved = lowerIsBetter ? change < 0 : change > 0
|
|
247
|
+
|
|
248
|
+
return { change, changePercent, improved }
|
|
249
|
+
}
|
|
250
|
+
|
|
251
|
+
/**
|
|
252
|
+
* Validate KPI definitions
|
|
253
|
+
*/
|
|
254
|
+
export function validateKPIs(kpis: KPIDefinition[]): { valid: boolean; errors: string[] } {
|
|
255
|
+
const errors: string[] = []
|
|
256
|
+
|
|
257
|
+
for (const kpi of kpis) {
|
|
258
|
+
if (!kpi.name) {
|
|
259
|
+
errors.push('KPI name is required')
|
|
260
|
+
}
|
|
261
|
+
|
|
262
|
+
if (kpi.target !== undefined && kpi.target < 0) {
|
|
263
|
+
errors.push(`KPI ${kpi.name} target cannot be negative`)
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
if (kpi.current !== undefined && kpi.current < 0) {
|
|
267
|
+
errors.push(`KPI ${kpi.name} current value cannot be negative`)
|
|
268
|
+
}
|
|
269
|
+
}
|
|
270
|
+
|
|
271
|
+
return {
|
|
272
|
+
valid: errors.length === 0,
|
|
273
|
+
errors,
|
|
274
|
+
}
|
|
275
|
+
}
|