@goxtechnologies/connectwise-psa-mcp 1.1.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/data/connectwise_api.db +0 -0
- package/data/manage.json +298179 -0
- package/dist/index.d.ts +10 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +116 -0
- package/dist/index.js.map +1 -0
- package/dist/operations/analytics-extended.d.ts +6 -0
- package/dist/operations/analytics-extended.d.ts.map +1 -0
- package/dist/operations/analytics-extended.js +825 -0
- package/dist/operations/analytics-extended.js.map +1 -0
- package/dist/operations/analytics-msp-assets.d.ts +3 -0
- package/dist/operations/analytics-msp-assets.d.ts.map +1 -0
- package/dist/operations/analytics-msp-assets.js +180 -0
- package/dist/operations/analytics-msp-assets.js.map +1 -0
- package/dist/operations/analytics-msp-clients.d.ts +3 -0
- package/dist/operations/analytics-msp-clients.d.ts.map +1 -0
- package/dist/operations/analytics-msp-clients.js +198 -0
- package/dist/operations/analytics-msp-clients.js.map +1 -0
- package/dist/operations/analytics-msp-comms.d.ts +3 -0
- package/dist/operations/analytics-msp-comms.d.ts.map +1 -0
- package/dist/operations/analytics-msp-comms.js +127 -0
- package/dist/operations/analytics-msp-comms.js.map +1 -0
- package/dist/operations/analytics-msp-contracts.d.ts +3 -0
- package/dist/operations/analytics-msp-contracts.d.ts.map +1 -0
- package/dist/operations/analytics-msp-contracts.js +91 -0
- package/dist/operations/analytics-msp-contracts.js.map +1 -0
- package/dist/operations/analytics-msp-financial.d.ts +3 -0
- package/dist/operations/analytics-msp-financial.d.ts.map +1 -0
- package/dist/operations/analytics-msp-financial.js +300 -0
- package/dist/operations/analytics-msp-financial.js.map +1 -0
- package/dist/operations/analytics-msp-procurement.d.ts +3 -0
- package/dist/operations/analytics-msp-procurement.d.ts.map +1 -0
- package/dist/operations/analytics-msp-procurement.js +78 -0
- package/dist/operations/analytics-msp-procurement.js.map +1 -0
- package/dist/operations/analytics-msp-projects.d.ts +3 -0
- package/dist/operations/analytics-msp-projects.d.ts.map +1 -0
- package/dist/operations/analytics-msp-projects.js +190 -0
- package/dist/operations/analytics-msp-projects.js.map +1 -0
- package/dist/operations/analytics-msp-sales.d.ts +3 -0
- package/dist/operations/analytics-msp-sales.d.ts.map +1 -0
- package/dist/operations/analytics-msp-sales.js +99 -0
- package/dist/operations/analytics-msp-sales.js.map +1 -0
- package/dist/operations/analytics-msp-schedule.d.ts +3 -0
- package/dist/operations/analytics-msp-schedule.d.ts.map +1 -0
- package/dist/operations/analytics-msp-schedule.js +339 -0
- package/dist/operations/analytics-msp-schedule.js.map +1 -0
- package/dist/operations/analytics-msp-team.d.ts +3 -0
- package/dist/operations/analytics-msp-team.d.ts.map +1 -0
- package/dist/operations/analytics-msp-team.js +195 -0
- package/dist/operations/analytics-msp-team.js.map +1 -0
- package/dist/operations/analytics-msp-tickets.d.ts +3 -0
- package/dist/operations/analytics-msp-tickets.d.ts.map +1 -0
- package/dist/operations/analytics-msp-tickets.js +578 -0
- package/dist/operations/analytics-msp-tickets.js.map +1 -0
- package/dist/operations/analytics-msp-time.d.ts +3 -0
- package/dist/operations/analytics-msp-time.d.ts.map +1 -0
- package/dist/operations/analytics-msp-time.js +485 -0
- package/dist/operations/analytics-msp-time.js.map +1 -0
- package/dist/operations/analytics-msp-utils.d.ts +49 -0
- package/dist/operations/analytics-msp-utils.d.ts.map +1 -0
- package/dist/operations/analytics-msp-utils.js +157 -0
- package/dist/operations/analytics-msp-utils.js.map +1 -0
- package/dist/operations/analytics.d.ts +9 -0
- package/dist/operations/analytics.d.ts.map +1 -0
- package/dist/operations/analytics.js +742 -0
- package/dist/operations/analytics.js.map +1 -0
- package/dist/operations/executor.d.ts +10 -0
- package/dist/operations/executor.d.ts.map +1 -0
- package/dist/operations/executor.js +243 -0
- package/dist/operations/executor.js.map +1 -0
- package/dist/operations/registry.d.ts +16 -0
- package/dist/operations/registry.d.ts.map +1 -0
- package/dist/operations/registry.js +847 -0
- package/dist/operations/registry.js.map +1 -0
- package/dist/services/api-database.d.ts +38 -0
- package/dist/services/api-database.d.ts.map +1 -0
- package/dist/services/api-database.js +191 -0
- package/dist/services/api-database.js.map +1 -0
- package/dist/services/cache.d.ts +12 -0
- package/dist/services/cache.d.ts.map +1 -0
- package/dist/services/cache.js +32 -0
- package/dist/services/cache.js.map +1 -0
- package/dist/services/connectwise-api.d.ts +43 -0
- package/dist/services/connectwise-api.d.ts.map +1 -0
- package/dist/services/connectwise-api.js +198 -0
- package/dist/services/connectwise-api.js.map +1 -0
- package/dist/services/db-builder.d.ts +11 -0
- package/dist/services/db-builder.d.ts.map +1 -0
- package/dist/services/db-builder.js +237 -0
- package/dist/services/db-builder.js.map +1 -0
- package/dist/services/fast-memory.d.ts +39 -0
- package/dist/services/fast-memory.d.ts.map +1 -0
- package/dist/services/fast-memory.js +147 -0
- package/dist/services/fast-memory.js.map +1 -0
- package/dist/services/load-env.d.ts +15 -0
- package/dist/services/load-env.d.ts.map +1 -0
- package/dist/services/load-env.js +59 -0
- package/dist/services/load-env.js.map +1 -0
- package/dist/tools/batch.d.ts +9 -0
- package/dist/tools/batch.d.ts.map +1 -0
- package/dist/tools/batch.js +159 -0
- package/dist/tools/batch.js.map +1 -0
- package/dist/tools/composite.d.ts +9 -0
- package/dist/tools/composite.d.ts.map +1 -0
- package/dist/tools/composite.js +353 -0
- package/dist/tools/composite.js.map +1 -0
- package/dist/tools/discovery.d.ts +9 -0
- package/dist/tools/discovery.d.ts.map +1 -0
- package/dist/tools/discovery.js +245 -0
- package/dist/tools/discovery.js.map +1 -0
- package/dist/tools/execution.d.ts +9 -0
- package/dist/tools/execution.d.ts.map +1 -0
- package/dist/tools/execution.js +130 -0
- package/dist/tools/execution.js.map +1 -0
- package/dist/tools/memory.d.ts +9 -0
- package/dist/tools/memory.d.ts.map +1 -0
- package/dist/tools/memory.js +152 -0
- package/dist/tools/memory.js.map +1 -0
- package/dist/tools/operations.d.ts +9 -0
- package/dist/tools/operations.d.ts.map +1 -0
- package/dist/tools/operations.js +214 -0
- package/dist/tools/operations.js.map +1 -0
- package/dist/tools/pagination.d.ts +9 -0
- package/dist/tools/pagination.d.ts.map +1 -0
- package/dist/tools/pagination.js +133 -0
- package/dist/tools/pagination.js.map +1 -0
- package/dist/tools/validation.d.ts +9 -0
- package/dist/tools/validation.d.ts.map +1 -0
- package/dist/tools/validation.js +705 -0
- package/dist/tools/validation.js.map +1 -0
- package/dist/types/index.d.ts +145 -0
- package/dist/types/index.d.ts.map +1 -0
- package/dist/types/index.js +3 -0
- package/dist/types/index.js.map +1 -0
- package/dist/types/operations.d.ts +30 -0
- package/dist/types/operations.d.ts.map +1 -0
- package/dist/types/operations.js +3 -0
- package/dist/types/operations.js.map +1 -0
- package/dist/utils/conditions.d.ts +20 -0
- package/dist/utils/conditions.d.ts.map +1 -0
- package/dist/utils/conditions.js +78 -0
- package/dist/utils/conditions.js.map +1 -0
- package/dist/utils/formatters.d.ts +35 -0
- package/dist/utils/formatters.d.ts.map +1 -0
- package/dist/utils/formatters.js +337 -0
- package/dist/utils/formatters.js.map +1 -0
- package/package.json +46 -0
|
@@ -0,0 +1,78 @@
|
|
|
1
|
+
// ConnectWise PSA MCP Server — MSP Procurement & Expense Analytics Handlers
|
|
2
|
+
import { getAPI } from '../services/connectwise-api.js';
|
|
3
|
+
import { num, nested, daysSince, daysAgo, groupByField, sum, pct, currencyDisplay, mdTable } from './analytics-msp-utils.js';
|
|
4
|
+
export const mspProcurementHandlers = {};
|
|
5
|
+
function reg(name, handler) { mspProcurementHandlers[name] = handler; }
|
|
6
|
+
// ---------------------------------------------------------------------------
|
|
7
|
+
// procurement_spend_summary
|
|
8
|
+
// Fetch purchase orders and group by vendor. Show PO count and total spend.
|
|
9
|
+
// ---------------------------------------------------------------------------
|
|
10
|
+
reg('procurement_spend_summary', async (params) => {
|
|
11
|
+
const api = getAPI();
|
|
12
|
+
const days = num(params['days'], 30);
|
|
13
|
+
const since = daysAgo(days);
|
|
14
|
+
const r = await api.paginatedFetch('/procurement/purchaseorders', `dateEntered>=[${since}]`, 'id,vendorCompany/name,total,status/name', 500);
|
|
15
|
+
if (r.items.length === 0)
|
|
16
|
+
return `No purchase orders found in the last ${days} days.`;
|
|
17
|
+
const byVendor = groupByField(r.items, 'vendorCompany/name');
|
|
18
|
+
const rows = [];
|
|
19
|
+
for (const [vendor, pos] of [...byVendor.entries()].sort((a, b) => sum(b[1], 'total') - sum(a[1], 'total'))) {
|
|
20
|
+
const total = sum(pos, 'total');
|
|
21
|
+
const statuses = [...new Set(pos.map(p => nested(p, 'status', 'name')).filter(s => s !== 'N/A'))].join(', ') || 'N/A';
|
|
22
|
+
rows.push([vendor, String(pos.length), currencyDisplay(total), statuses]);
|
|
23
|
+
}
|
|
24
|
+
const grandTotal = sum(r.items, 'total');
|
|
25
|
+
return [
|
|
26
|
+
`## Procurement Spend Summary (last ${days} days)\n`,
|
|
27
|
+
`**Total POs:** ${r.items.length} | **Total Spend:** ${currencyDisplay(grandTotal)}\n`,
|
|
28
|
+
mdTable(['Vendor', 'PO Count', 'Total $', 'Status'], rows),
|
|
29
|
+
].join('\n');
|
|
30
|
+
});
|
|
31
|
+
// ---------------------------------------------------------------------------
|
|
32
|
+
// expense_pending_approvals
|
|
33
|
+
// Fetch pending expense entries grouped by member.
|
|
34
|
+
// ---------------------------------------------------------------------------
|
|
35
|
+
reg('expense_pending_approvals', async (_params) => {
|
|
36
|
+
const api = getAPI();
|
|
37
|
+
const r = await api.paginatedFetch('/expense/entries', "status/name like '%Pending%'", 'id,amount,expenseDate,member/identifier,status/name', 500);
|
|
38
|
+
if (r.items.length === 0)
|
|
39
|
+
return 'No pending expense entries found.';
|
|
40
|
+
const byMember = groupByField(r.items, 'member/identifier');
|
|
41
|
+
const rows = [];
|
|
42
|
+
for (const [member, entries] of [...byMember.entries()].sort((a, b) => sum(b[1], 'amount') - sum(a[1], 'amount'))) {
|
|
43
|
+
const total = sum(entries, 'amount');
|
|
44
|
+
const oldestAge = Math.max(...entries.map(e => daysSince(String(e['expenseDate'] ?? ''))));
|
|
45
|
+
rows.push([member, String(entries.length), currencyDisplay(total), `${oldestAge}d ago`]);
|
|
46
|
+
}
|
|
47
|
+
const grandTotal = sum(r.items, 'amount');
|
|
48
|
+
return [
|
|
49
|
+
`## Expense Pending Approvals (${r.items.length} entries)\n`,
|
|
50
|
+
`**Total Pending:** ${currencyDisplay(grandTotal)}\n`,
|
|
51
|
+
mdTable(['Member', 'Entries', 'Total $', 'Oldest'], rows),
|
|
52
|
+
].join('\n');
|
|
53
|
+
});
|
|
54
|
+
// ---------------------------------------------------------------------------
|
|
55
|
+
// expense_by_category
|
|
56
|
+
// Fetch expense entries in period and group by classification.
|
|
57
|
+
// ---------------------------------------------------------------------------
|
|
58
|
+
reg('expense_by_category', async (params) => {
|
|
59
|
+
const api = getAPI();
|
|
60
|
+
const days = num(params['days'], 30);
|
|
61
|
+
const since = daysAgo(days);
|
|
62
|
+
const r = await api.paginatedFetch('/expense/entries', `expenseDate>=[${since}]`, 'id,amount,expenseDate,classification/name', 500);
|
|
63
|
+
if (r.items.length === 0)
|
|
64
|
+
return `No expense entries found in the last ${days} days.`;
|
|
65
|
+
const grandTotal = sum(r.items, 'amount');
|
|
66
|
+
const byCategory = groupByField(r.items, 'classification/name');
|
|
67
|
+
const rows = [];
|
|
68
|
+
for (const [category, entries] of [...byCategory.entries()].sort((a, b) => sum(b[1], 'amount') - sum(a[1], 'amount'))) {
|
|
69
|
+
const total = sum(entries, 'amount');
|
|
70
|
+
rows.push([category, String(entries.length), currencyDisplay(total), pct(total, grandTotal)]);
|
|
71
|
+
}
|
|
72
|
+
return [
|
|
73
|
+
`## Expense by Category (last ${days} days)\n`,
|
|
74
|
+
`**Total Entries:** ${r.items.length} | **Total Spend:** ${currencyDisplay(grandTotal)}\n`,
|
|
75
|
+
mdTable(['Category', 'Count', 'Total $', '% of Total'], rows),
|
|
76
|
+
].join('\n');
|
|
77
|
+
});
|
|
78
|
+
//# sourceMappingURL=analytics-msp-procurement.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"analytics-msp-procurement.js","sourceRoot":"","sources":["../../src/operations/analytics-msp-procurement.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"analytics-msp-projects.d.ts","sourceRoot":"","sources":["../../src/operations/analytics-msp-projects.ts"],"names":[],"mappings":"AAGA,OAAO,EACoB,KAAK,UAAU,EAGzC,MAAM,0BAA0B,CAAC;AAElC,eAAO,MAAM,kBAAkB,EAAE,MAAM,CAAC,MAAM,EAAE,UAAU,CAAM,CAAC"}
|
|
@@ -0,0 +1,190 @@
|
|
|
1
|
+
// ConnectWise PSA MCP Server — MSP Project Analytics Handlers
|
|
2
|
+
import { getAPI } from '../services/connectwise-api.js';
|
|
3
|
+
import { num, nested, round2, hoursDisplay, mdTable, settledWarning, } from './analytics-msp-utils.js';
|
|
4
|
+
export const mspProjectHandlers = {};
|
|
5
|
+
function reg(name, handler) { mspProjectHandlers[name] = handler; }
|
|
6
|
+
// ===========================================================================
|
|
7
|
+
// PROJECT ANALYTICS
|
|
8
|
+
// ===========================================================================
|
|
9
|
+
reg('project_budget_burn', async (params) => {
|
|
10
|
+
const api = getAPI();
|
|
11
|
+
const projectId = params['project_id'] !== undefined ? num(params['project_id']) : null;
|
|
12
|
+
const projects = projectId !== null
|
|
13
|
+
? await api.request({ path: `/project/projects/${projectId}`, method: 'GET' }).then(r => ({ items: [r.data] }))
|
|
14
|
+
: await api.paginatedFetch('/project/projects', "status/name='Open'", 'id,name,company/name,budgetHours,estimatedEnd', 200);
|
|
15
|
+
if (projects.items.length === 0)
|
|
16
|
+
return 'No projects found.';
|
|
17
|
+
const budgetResults = await Promise.allSettled(projects.items.map(async (p) => {
|
|
18
|
+
const pid = num(p['id']);
|
|
19
|
+
const time = await api.paginatedFetch('/time/entries', `chargeToType='Project' AND chargeToId=${pid}`, 'id,actualHours', 2000);
|
|
20
|
+
const actual = time.items.reduce((s, e) => s + num(e['actualHours']), 0);
|
|
21
|
+
return { p, actual };
|
|
22
|
+
}));
|
|
23
|
+
const rows = [];
|
|
24
|
+
for (const r of budgetResults) {
|
|
25
|
+
if (r.status !== 'fulfilled')
|
|
26
|
+
continue;
|
|
27
|
+
const { p, actual } = r.value;
|
|
28
|
+
const budget = num(p['budgetHours']);
|
|
29
|
+
const burnPct = budget > 0 ? round2((actual / budget) * 100) : 0;
|
|
30
|
+
const projected = budget > 0 && burnPct > 0 ? round2((actual / (burnPct / 100)) - budget) : 0;
|
|
31
|
+
rows.push([
|
|
32
|
+
String(p['name'] ?? `Project #${p['id']}`),
|
|
33
|
+
hoursDisplay(budget),
|
|
34
|
+
hoursDisplay(actual),
|
|
35
|
+
`${burnPct}%`,
|
|
36
|
+
projected > 0 ? `+${hoursDisplay(projected)}` : 'On track',
|
|
37
|
+
]);
|
|
38
|
+
}
|
|
39
|
+
if (rows.length === 0)
|
|
40
|
+
return 'No project budget data available.';
|
|
41
|
+
const title = projectId !== null
|
|
42
|
+
? `## Project Budget Burn: Project #${projectId}`
|
|
43
|
+
: `## Project Budget Burn — All Open Projects (${rows.length})`;
|
|
44
|
+
return [title + '\n', mdTable(['Project', 'Budget', 'Actual', 'Burn Rate', 'Projected Overrun'], rows)].join('\n') + settledWarning(budgetResults, 'sub-requests');
|
|
45
|
+
});
|
|
46
|
+
reg('project_phase_completion', async (params) => {
|
|
47
|
+
const api = getAPI();
|
|
48
|
+
const projectId = num(params['project_id']);
|
|
49
|
+
if (!projectId)
|
|
50
|
+
return 'Error: project_id required.';
|
|
51
|
+
const [project, phases] = await Promise.all([
|
|
52
|
+
api.request({ path: `/project/projects/${projectId}`, method: 'GET' }),
|
|
53
|
+
api.paginatedFetch(`/project/projects/${projectId}/phases`, undefined, 'id,description,wbsCode', 100),
|
|
54
|
+
]);
|
|
55
|
+
if (phases.items.length === 0) {
|
|
56
|
+
return `No phases found for project #${projectId}.`;
|
|
57
|
+
}
|
|
58
|
+
const projectName = String(project.data['name'] ?? `Project #${projectId}`);
|
|
59
|
+
const phaseResults = await Promise.allSettled(phases.items.map(async (ph) => {
|
|
60
|
+
const tickets = await api.paginatedFetch('/service/tickets', `project/id=${projectId} AND phase/id=${ph['id']}`, 'id,closedFlag', 500);
|
|
61
|
+
const total = tickets.items.length;
|
|
62
|
+
const closed = tickets.items.filter(t => t['closedFlag'] === true).length;
|
|
63
|
+
return { ph, total, closed };
|
|
64
|
+
}));
|
|
65
|
+
const rows = [];
|
|
66
|
+
for (const r of phaseResults) {
|
|
67
|
+
if (r.status !== 'fulfilled')
|
|
68
|
+
continue;
|
|
69
|
+
const { ph, total, closed } = r.value;
|
|
70
|
+
const completePct = total > 0 ? round2((closed / total) * 100) : 0;
|
|
71
|
+
const phaseLabel = ph['wbsCode']
|
|
72
|
+
? `${ph['wbsCode']} — ${ph['description'] ?? ''}`
|
|
73
|
+
: String(ph['description'] ?? `Phase #${ph['id']}`);
|
|
74
|
+
rows.push([phaseLabel, String(total), String(closed), `${completePct}%`]);
|
|
75
|
+
}
|
|
76
|
+
if (rows.length === 0)
|
|
77
|
+
return `No phase ticket data found for project #${projectId}.`;
|
|
78
|
+
return [
|
|
79
|
+
`## Project Phase Completion: ${projectName} (#${projectId})\n`,
|
|
80
|
+
mdTable(['Phase', 'Total Tickets', 'Closed', '% Complete'], rows),
|
|
81
|
+
].join('\n') + settledWarning(phaseResults, 'sub-requests');
|
|
82
|
+
});
|
|
83
|
+
reg('project_health_dashboard', async (_params) => {
|
|
84
|
+
const api = getAPI();
|
|
85
|
+
const projects = await api.paginatedFetch('/project/projects', "status/name='Open'", 'id,name,company/name,budgetHours,estimatedEnd,scheduledStart', 200);
|
|
86
|
+
if (projects.items.length === 0)
|
|
87
|
+
return 'No active (Open) projects found.';
|
|
88
|
+
const today = Date.now();
|
|
89
|
+
const healthResults = await Promise.allSettled(projects.items.map(async (p) => {
|
|
90
|
+
const pid = num(p['id']);
|
|
91
|
+
const time = await api.paginatedFetch('/time/entries', `chargeToType='Project' AND chargeToId=${pid}`, 'id,actualHours', 2000);
|
|
92
|
+
const actual = time.items.reduce((s, e) => s + num(e['actualHours']), 0);
|
|
93
|
+
return { p, actual };
|
|
94
|
+
}));
|
|
95
|
+
const rows = [];
|
|
96
|
+
for (const r of healthResults) {
|
|
97
|
+
if (r.status !== 'fulfilled')
|
|
98
|
+
continue;
|
|
99
|
+
const { p, actual } = r.value;
|
|
100
|
+
const budget = num(p['budgetHours']);
|
|
101
|
+
const burnPct = budget > 0 ? round2((actual / budget) * 100) : 0;
|
|
102
|
+
// Schedule health: compare today vs estimatedEnd
|
|
103
|
+
const estimatedEnd = p['estimatedEnd'] ? new Date(p['estimatedEnd']) : null;
|
|
104
|
+
const scheduledStart = p['scheduledStart'] ? new Date(p['scheduledStart']) : null;
|
|
105
|
+
let scheduleLabel = 'N/A';
|
|
106
|
+
if (estimatedEnd && scheduledStart) {
|
|
107
|
+
const totalSpan = estimatedEnd.getTime() - scheduledStart.getTime();
|
|
108
|
+
const elapsed = today - scheduledStart.getTime();
|
|
109
|
+
const timeProgressPct = totalSpan > 0 ? round2((elapsed / totalSpan) * 100) : 0;
|
|
110
|
+
const diff = timeProgressPct - burnPct;
|
|
111
|
+
scheduleLabel = diff > 20 ? 'Behind' : diff < -20 ? 'Ahead' : 'On Track';
|
|
112
|
+
}
|
|
113
|
+
else if (estimatedEnd && estimatedEnd.getTime() < today) {
|
|
114
|
+
scheduleLabel = 'Overdue';
|
|
115
|
+
}
|
|
116
|
+
// Composite health score
|
|
117
|
+
const budgetHealth = burnPct <= 80 ? 100 : burnPct <= 100 ? 60 : 20;
|
|
118
|
+
const schedHealth = scheduleLabel === 'On Track' || scheduleLabel === 'Ahead' ? 100 : scheduleLabel === 'Behind' ? 50 : 0;
|
|
119
|
+
const composite = round2(budgetHealth * 0.6 + schedHealth * 0.4);
|
|
120
|
+
const healthLabel = composite >= 80 ? 'Healthy' : composite >= 50 ? 'At Risk' : 'Critical';
|
|
121
|
+
rows.push([
|
|
122
|
+
String(p['name'] ?? `#${p['id']}`),
|
|
123
|
+
nested(p, 'company', 'name'),
|
|
124
|
+
`${burnPct}%`,
|
|
125
|
+
scheduleLabel,
|
|
126
|
+
`${composite} — ${healthLabel}`,
|
|
127
|
+
]);
|
|
128
|
+
}
|
|
129
|
+
if (rows.length === 0)
|
|
130
|
+
return 'No project health data available.';
|
|
131
|
+
rows.sort((a, b) => {
|
|
132
|
+
const scoreA = num(a[4]?.split(' — ')[0]);
|
|
133
|
+
const scoreB = num(b[4]?.split(' — ')[0]);
|
|
134
|
+
return scoreA - scoreB;
|
|
135
|
+
});
|
|
136
|
+
return [
|
|
137
|
+
`## Project Health Dashboard (${rows.length} active projects)\n`,
|
|
138
|
+
mdTable(['Project', 'Company', 'Budget %', 'Schedule', 'Health'], rows),
|
|
139
|
+
].join('\n') + settledWarning(healthResults, 'sub-requests');
|
|
140
|
+
});
|
|
141
|
+
reg('project_resource_allocation', async (_params) => {
|
|
142
|
+
const api = getAPI();
|
|
143
|
+
const projects = await api.paginatedFetch('/project/projects', "status/name='Open'", 'id,name', 100);
|
|
144
|
+
if (projects.items.length === 0)
|
|
145
|
+
return 'No active (Open) projects found.';
|
|
146
|
+
// Cap to 20 projects to avoid too many API calls
|
|
147
|
+
const activeProjects = projects.items.slice(0, 20);
|
|
148
|
+
const projectNames = {};
|
|
149
|
+
for (const p of activeProjects)
|
|
150
|
+
projectNames[num(p['id'])] = String(p['name'] ?? `#${p['id']}`);
|
|
151
|
+
// Fetch time entries for all active projects in parallel
|
|
152
|
+
const timeResults = await Promise.allSettled(activeProjects.map(async (p) => {
|
|
153
|
+
const pid = num(p['id']);
|
|
154
|
+
const time = await api.paginatedFetch('/time/entries', `chargeToType='Project' AND chargeToId=${pid}`, 'id,actualHours,member/identifier', 1000);
|
|
155
|
+
return { pid, entries: time.items };
|
|
156
|
+
}));
|
|
157
|
+
// Build: member -> project -> hours
|
|
158
|
+
const allProjects = [];
|
|
159
|
+
const memberHours = {};
|
|
160
|
+
for (const r of timeResults) {
|
|
161
|
+
if (r.status !== 'fulfilled')
|
|
162
|
+
continue;
|
|
163
|
+
const { pid, entries } = r.value;
|
|
164
|
+
allProjects.push(pid);
|
|
165
|
+
for (const e of entries) {
|
|
166
|
+
const id = nested(e, 'member', 'identifier');
|
|
167
|
+
if (!memberHours[id])
|
|
168
|
+
memberHours[id] = {};
|
|
169
|
+
memberHours[id][pid] = (memberHours[id][pid] ?? 0) + num(e['actualHours']);
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
if (Object.keys(memberHours).length === 0)
|
|
173
|
+
return 'No time entries found across active projects.';
|
|
174
|
+
// Build table: columns are projects, rows are members
|
|
175
|
+
const sortedMembers = Object.keys(memberHours).sort();
|
|
176
|
+
const headers = ['Member', ...allProjects.map(pid => projectNames[pid] ?? `#${pid}`).map(n => n.length > 15 ? n.substring(0, 13) + '..' : n), 'Total'];
|
|
177
|
+
const rows = sortedMembers.map(id => {
|
|
178
|
+
const total = allProjects.reduce((s, pid) => s + (memberHours[id][pid] ?? 0), 0);
|
|
179
|
+
return [
|
|
180
|
+
id,
|
|
181
|
+
...allProjects.map(pid => memberHours[id][pid] ? hoursDisplay(memberHours[id][pid]) : '—'),
|
|
182
|
+
hoursDisplay(total),
|
|
183
|
+
];
|
|
184
|
+
});
|
|
185
|
+
return [
|
|
186
|
+
`## Project Resource Allocation — ${allProjects.length} Active Projects\n`,
|
|
187
|
+
mdTable(headers, rows),
|
|
188
|
+
].join('\n') + settledWarning(timeResults, 'sub-requests');
|
|
189
|
+
});
|
|
190
|
+
//# sourceMappingURL=analytics-msp-projects.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"analytics-msp-projects.js","sourceRoot":"","sources":["../../src/operations/analytics-msp-projects.ts"],"names":[],"mappings":"AAAA,8DAA8D;AAE9D,OAAO,EAAE,MAAM,EAAE,MAAM,gCAAgC,CAAC;AACxD,OAAO,EAEL,GAAG,EAAE,MAAM,EAAsB,MAAM,EACpB,YAAY,EAAE,OAAO,EAAE,cAAc,GACzD,MAAM,0BAA0B,CAAC;AAElC,MAAM,CAAC,MAAM,kBAAkB,GAA+B,EAAE,CAAC;AACjE,SAAS,GAAG,CAAC,IAAY,EAAE,OAAmB,IAAU,kBAAkB,CAAC,IAAI,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC;AAE7F,8EAA8E;AAC9E,oBAAoB;AACpB,8EAA8E;AAE9E,GAAG,CAAC,qBAAqB,EAAE,KAAK,EAAE,MAAc,EAAE,EAAE;IAClD,MAAM,GAAG,GAAG,MAAM,EAAE,CAAC;IACrB,MAAM,SAAS,GAAG,MAAM,CAAC,YAAY,CAAC,KAAK,SAAS,CAAC,CAAC,CAAC,GAAG,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;IAExF,MAAM,QAAQ,GAAG,SAAS,KAAK,IAAI;QACjC,CAAC,CAAC,MAAM,GAAG,CAAC,OAAO,CAAQ,EAAE,IAAI,EAAE,qBAAqB,SAAS,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACtH,CAAC,CAAC,MAAM,GAAG,CAAC,cAAc,CACtB,mBAAmB,EACnB,oBAAoB,EACpB,+CAA+C,EAC/C,GAAG,CACJ,CAAC;IAEN,IAAI,QAAQ,CAAC,KAAK,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO,oBAAoB,CAAC;IAE7D,MAAM,aAAa,GAAG,MAAM,OAAO,CAAC,UAAU,CAC5C,QAAQ,CAAC,KAAK,CAAC,GAAG,CAAC,KAAK,EAAE,CAAC,EAAE,EAAE;QAC7B,MAAM,GAAG,GAAG,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;QACzB,MAAM,IAAI,GAAG,MAAM,GAAG,CAAC,cAAc,CACnC,eAAe,EACf,yCAAyC,GAAG,EAAE,EAC9C,gBAAgB,EAChB,IAAI,CACL,CAAC;QACF,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;QACzE,OAAO,EAAE,CAAC,EAAE,MAAM,EAAE,CAAC;IACvB,CAAC,CAAC,CACH,CAAC;IAEF,MAAM,IAAI,GAAe,EAAE,CAAC;IAC5B,KAAK,MAAM,CAAC,IAAI,aAAa,EAAE,CAAC;QAC9B,IAAI,CAAC,CAAC,MAAM,KAAK,WAAW;YAAE,SAAS;QACvC,MAAM,EAAE,CAAC,EAAE,MAAM,EAAE,GAAG,CAAC,CAAC,KAAK,CAAC;QAC9B,MAAM,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC;QACrC,MAAM,OAAO,GAAG,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,MAAM,GAAG,MAAM,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACjE,MAAM,SAAS,GAAG,MAAM,GAAG,CAAC,IAAI,OAAO,GAAG,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,MAAM,GAAG,CAAC,OAAO,GAAG,GAAG,CAAC,CAAC,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAC9F,IAAI,CAAC,IAAI,CAAC;YACR,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,YAAY,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC;YAC1C,YAAY,CAAC,MAAM,CAAC;YACpB,YAAY,CAAC,MAAM,CAAC;YACpB,GAAG,OAAO,GAAG;YACb,SAAS,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,YAAY,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC,UAAU;SAC3D,CAAC,CAAC;IACL,CAAC;IAED,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO,mCAAmC,CAAC;IAElE,MAAM,KAAK,GAAG,SAAS,KAAK,IAAI;QAC9B,CAAC,CAAC,oCAAoC,SAAS,EAAE;QACjD,CAAC,CAAC,+CAA+C,IAAI,CAAC,MAAM,GAAG,CAAC;IAElE,OAAO,CAAC,KAAK,GAAG,IAAI,EAAE,OAAO,CAAC,CAAC,SAAS,EAAE,QAAQ,EAAE,QAAQ,EAAE,WAAW,EAAE,mBAAmB,CAAC,EAAE,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,cAAc,CAAC,aAAa,EAAE,cAAc,CAAC,CAAC;AACrK,CAAC,CAAC,CAAC;AAEH,GAAG,CAAC,0BAA0B,EAAE,KAAK,EAAE,MAAc,EAAE,EAAE;IACvD,MAAM,GAAG,GAAG,MAAM,EAAE,CAAC;IACrB,MAAM,SAAS,GAAG,GAAG,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC,CAAC;IAC5C,IAAI,CAAC,SAAS;QAAE,OAAO,6BAA6B,CAAC;IAErD,MAAM,CAAC,OAAO,EAAE,MAAM,CAAC,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC;QAC1C,GAAG,CAAC,OAAO,CAAQ,EAAE,IAAI,EAAE,qBAAqB,SAAS,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,CAAC;QAC7E,GAAG,CAAC,cAAc,CAChB,qBAAqB,SAAS,SAAS,EACvC,SAAS,EACT,wBAAwB,EACxB,GAAG,CACJ;KACF,CAAC,CAAC;IAEH,IAAI,MAAM,CAAC,KAAK,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QAC9B,OAAO,gCAAgC,SAAS,GAAG,CAAC;IACtD,CAAC;IAED,MAAM,WAAW,GAAG,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,YAAY,SAAS,EAAE,CAAC,CAAC;IAE5E,MAAM,YAAY,GAAG,MAAM,OAAO,CAAC,UAAU,CAC3C,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,KAAK,EAAE,EAAE,EAAE,EAAE;QAC5B,MAAM,OAAO,GAAG,MAAM,GAAG,CAAC,cAAc,CACtC,kBAAkB,EAClB,cAAc,SAAS,iBAAiB,EAAE,CAAC,IAAI,CAAC,EAAE,EAClD,eAAe,EACf,GAAG,CACJ,CAAC;QACF,MAAM,KAAK,GAAG,OAAO,CAAC,KAAK,CAAC,MAAM,CAAC;QACnC,MAAM,MAAM,GAAG,OAAO,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,YAAY,CAAC,KAAK,IAAI,CAAC,CAAC,MAAM,CAAC;QAC1E,OAAO,EAAE,EAAE,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC;IAC/B,CAAC,CAAC,CACH,CAAC;IAEF,MAAM,IAAI,GAAe,EAAE,CAAC;IAC5B,KAAK,MAAM,CAAC,IAAI,YAAY,EAAE,CAAC;QAC7B,IAAI,CAAC,CAAC,MAAM,KAAK,WAAW;YAAE,SAAS;QACvC,MAAM,EAAE,EAAE,EAAE,KAAK,EAAE,MAAM,EAAE,GAAG,CAAC,CAAC,KAAK,CAAC;QACtC,MAAM,WAAW,GAAG,KAAK,GAAG,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,MAAM,GAAG,KAAK,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACnE,MAAM,UAAU,GAAG,EAAE,CAAC,SAAS,CAAC;YAC9B,CAAC,CAAC,GAAG,EAAE,CAAC,SAAS,CAAC,MAAM,EAAE,CAAC,aAAa,CAAC,IAAI,EAAE,EAAE;YACjD,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC,aAAa,CAAC,IAAI,UAAU,EAAE,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACtD,IAAI,CAAC,IAAI,CAAC,CAAC,UAAU,EAAE,MAAM,CAAC,KAAK,CAAC,EAAE,MAAM,CAAC,MAAM,CAAC,EAAE,GAAG,WAAW,GAAG,CAAC,CAAC,CAAC;IAC5E,CAAC;IAED,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO,2CAA2C,SAAS,GAAG,CAAC;IAEtF,OAAO;QACL,gCAAgC,WAAW,MAAM,SAAS,KAAK;QAC/D,OAAO,CAAC,CAAC,OAAO,EAAE,eAAe,EAAE,QAAQ,EAAE,YAAY,CAAC,EAAE,IAAI,CAAC;KAClE,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,cAAc,CAAC,YAAY,EAAE,cAAc,CAAC,CAAC;AAC9D,CAAC,CAAC,CAAC;AAEH,GAAG,CAAC,0BAA0B,EAAE,KAAK,EAAE,OAAe,EAAE,EAAE;IACxD,MAAM,GAAG,GAAG,MAAM,EAAE,CAAC;IAErB,MAAM,QAAQ,GAAG,MAAM,GAAG,CAAC,cAAc,CACvC,mBAAmB,EACnB,oBAAoB,EACpB,8DAA8D,EAC9D,GAAG,CACJ,CAAC;IAEF,IAAI,QAAQ,CAAC,KAAK,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO,kCAAkC,CAAC;IAE3E,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;IAEzB,MAAM,aAAa,GAAG,MAAM,OAAO,CAAC,UAAU,CAC5C,QAAQ,CAAC,KAAK,CAAC,GAAG,CAAC,KAAK,EAAE,CAAC,EAAE,EAAE;QAC7B,MAAM,GAAG,GAAG,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;QACzB,MAAM,IAAI,GAAG,MAAM,GAAG,CAAC,cAAc,CACnC,eAAe,EACf,yCAAyC,GAAG,EAAE,EAC9C,gBAAgB,EAChB,IAAI,CACL,CAAC;QACF,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;QACzE,OAAO,EAAE,CAAC,EAAE,MAAM,EAAE,CAAC;IACvB,CAAC,CAAC,CACH,CAAC;IAEF,MAAM,IAAI,GAAe,EAAE,CAAC;IAE5B,KAAK,MAAM,CAAC,IAAI,aAAa,EAAE,CAAC;QAC9B,IAAI,CAAC,CAAC,MAAM,KAAK,WAAW;YAAE,SAAS;QACvC,MAAM,EAAE,CAAC,EAAE,MAAM,EAAE,GAAG,CAAC,CAAC,KAAK,CAAC;QAC9B,MAAM,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC;QACrC,MAAM,OAAO,GAAG,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,MAAM,GAAG,MAAM,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAEjE,iDAAiD;QACjD,MAAM,YAAY,GAAG,CAAC,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC,CAAC,CAAC,cAAc,CAAW,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;QACtF,MAAM,cAAc,GAAG,CAAC,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC,CAAC,CAAC,gBAAgB,CAAW,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;QAE5F,IAAI,aAAa,GAAG,KAAK,CAAC;QAC1B,IAAI,YAAY,IAAI,cAAc,EAAE,CAAC;YACnC,MAAM,SAAS,GAAG,YAAY,CAAC,OAAO,EAAE,GAAG,cAAc,CAAC,OAAO,EAAE,CAAC;YACpE,MAAM,OAAO,GAAG,KAAK,GAAG,cAAc,CAAC,OAAO,EAAE,CAAC;YACjD,MAAM,eAAe,GAAG,SAAS,GAAG,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,OAAO,GAAG,SAAS,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;YAChF,MAAM,IAAI,GAAG,eAAe,GAAG,OAAO,CAAC;YACvC,aAAa,GAAG,IAAI,GAAG,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,IAAI,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,UAAU,CAAC;QAC3E,CAAC;aAAM,IAAI,YAAY,IAAI,YAAY,CAAC,OAAO,EAAE,GAAG,KAAK,EAAE,CAAC;YAC1D,aAAa,GAAG,SAAS,CAAC;QAC5B,CAAC;QAED,yBAAyB;QACzB,MAAM,YAAY,GAAG,OAAO,IAAI,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,OAAO,IAAI,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;QACpE,MAAM,WAAW,GAAG,aAAa,KAAK,UAAU,IAAI,aAAa,KAAK,OAAO,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,aAAa,KAAK,QAAQ,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;QAC1H,MAAM,SAAS,GAAG,MAAM,CAAC,YAAY,GAAG,GAAG,GAAG,WAAW,GAAG,GAAG,CAAC,CAAC;QACjE,MAAM,WAAW,GAAG,SAAS,IAAI,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS,IAAI,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,UAAU,CAAC;QAE3F,IAAI,CAAC,IAAI,CAAC;YACR,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,IAAI,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC;YAClC,MAAM,CAAC,CAAC,EAAE,SAAS,EAAE,MAAM,CAAC;YAC5B,GAAG,OAAO,GAAG;YACb,aAAa;YACb,GAAG,SAAS,MAAM,WAAW,EAAE;SAChC,CAAC,CAAC;IACL,CAAC;IAED,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO,mCAAmC,CAAC;IAElE,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE;QACjB,MAAM,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAC1C,MAAM,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAC1C,OAAO,MAAM,GAAG,MAAM,CAAC;IACzB,CAAC,CAAC,CAAC;IAEH,OAAO;QACL,gCAAgC,IAAI,CAAC,MAAM,qBAAqB;QAChE,OAAO,CAAC,CAAC,SAAS,EAAE,SAAS,EAAE,UAAU,EAAE,UAAU,EAAE,QAAQ,CAAC,EAAE,IAAI,CAAC;KACxE,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,cAAc,CAAC,aAAa,EAAE,cAAc,CAAC,CAAC;AAC/D,CAAC,CAAC,CAAC;AAEH,GAAG,CAAC,6BAA6B,EAAE,KAAK,EAAE,OAAe,EAAE,EAAE;IAC3D,MAAM,GAAG,GAAG,MAAM,EAAE,CAAC;IAErB,MAAM,QAAQ,GAAG,MAAM,GAAG,CAAC,cAAc,CACvC,mBAAmB,EACnB,oBAAoB,EACpB,SAAS,EACT,GAAG,CACJ,CAAC;IAEF,IAAI,QAAQ,CAAC,KAAK,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO,kCAAkC,CAAC;IAE3E,iDAAiD;IACjD,MAAM,cAAc,GAAG,QAAQ,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;IACnD,MAAM,YAAY,GAA2B,EAAE,CAAC;IAChD,KAAK,MAAM,CAAC,IAAI,cAAc;QAAE,YAAY,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,GAAG,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,IAAI,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IAEhG,yDAAyD;IACzD,MAAM,WAAW,GAAG,MAAM,OAAO,CAAC,UAAU,CAC1C,cAAc,CAAC,GAAG,CAAC,KAAK,EAAE,CAAC,EAAE,EAAE;QAC7B,MAAM,GAAG,GAAG,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;QACzB,MAAM,IAAI,GAAG,MAAM,GAAG,CAAC,cAAc,CACnC,eAAe,EACf,yCAAyC,GAAG,EAAE,EAC9C,kCAAkC,EAClC,IAAI,CACL,CAAC;QACF,OAAO,EAAE,GAAG,EAAE,OAAO,EAAE,IAAI,CAAC,KAAK,EAAE,CAAC;IACtC,CAAC,CAAC,CACH,CAAC;IAEF,oCAAoC;IACpC,MAAM,WAAW,GAAa,EAAE,CAAC;IACjC,MAAM,WAAW,GAA2C,EAAE,CAAC;IAE/D,KAAK,MAAM,CAAC,IAAI,WAAW,EAAE,CAAC;QAC5B,IAAI,CAAC,CAAC,MAAM,KAAK,WAAW;YAAE,SAAS;QACvC,MAAM,EAAE,GAAG,EAAE,OAAO,EAAE,GAAG,CAAC,CAAC,KAAK,CAAC;QACjC,WAAW,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QACtB,KAAK,MAAM,CAAC,IAAI,OAAO,EAAE,CAAC;YACxB,MAAM,EAAE,GAAG,MAAM,CAAC,CAAC,EAAE,QAAQ,EAAE,YAAY,CAAC,CAAC;YAC7C,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC;gBAAE,WAAW,CAAC,EAAE,CAAC,GAAG,EAAE,CAAC;YAC3C,WAAW,CAAC,EAAE,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC;QAC7E,CAAC;IACH,CAAC;IAED,IAAI,MAAM,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO,+CAA+C,CAAC;IAElG,sDAAsD;IACtD,MAAM,aAAa,GAAG,MAAM,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,IAAI,EAAE,CAAC;IACtD,MAAM,OAAO,GAAG,CAAC,QAAQ,EAAE,GAAG,WAAW,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,YAAY,CAAC,GAAG,CAAC,IAAI,IAAI,GAAG,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,GAAG,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC;IAEvJ,MAAM,IAAI,GAAe,aAAa,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE;QAC9C,MAAM,KAAK,GAAG,WAAW,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,GAAG,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;QACjF,OAAO;YACL,EAAE;YACF,GAAG,WAAW,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC;YAC1F,YAAY,CAAC,KAAK,CAAC;SACpB,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,OAAO;QACL,oCAAoC,WAAW,CAAC,MAAM,oBAAoB;QAC1E,OAAO,CAAC,OAAO,EAAE,IAAI,CAAC;KACvB,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,cAAc,CAAC,WAAW,EAAE,cAAc,CAAC,CAAC;AAC7D,CAAC,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"analytics-msp-sales.d.ts","sourceRoot":"","sources":["../../src/operations/analytics-msp-sales.ts"],"names":[],"mappings":"AAGA,OAAO,EAA2B,KAAK,UAAU,EAAgF,MAAM,0BAA0B,CAAC;AAElK,eAAO,MAAM,gBAAgB,EAAE,MAAM,CAAC,MAAM,EAAE,UAAU,CAAM,CAAC"}
|
|
@@ -0,0 +1,99 @@
|
|
|
1
|
+
// ConnectWise PSA MCP Server — MSP Sales Analytics Handlers
|
|
2
|
+
import { getAPI } from '../services/connectwise-api.js';
|
|
3
|
+
import { num, daysSince, daysAgo, round2, groupByField, sum, currencyDisplay, mdTable } from './analytics-msp-utils.js';
|
|
4
|
+
export const mspSalesHandlers = {};
|
|
5
|
+
function reg(name, handler) { mspSalesHandlers[name] = handler; }
|
|
6
|
+
// ---------------------------------------------------------------------------
|
|
7
|
+
// pipeline_value_summary
|
|
8
|
+
// Fetch open opportunities grouped by stage. Compute weighted value.
|
|
9
|
+
// ---------------------------------------------------------------------------
|
|
10
|
+
reg('pipeline_value_summary', async (_params) => {
|
|
11
|
+
const api = getAPI();
|
|
12
|
+
const r = await api.paginatedFetch('/sales/opportunities', 'closedFlag=false', 'id,name,stage/name,amount,probability', 500);
|
|
13
|
+
if (r.items.length === 0)
|
|
14
|
+
return 'No open opportunities found.';
|
|
15
|
+
const byStage = groupByField(r.items, 'stage/name');
|
|
16
|
+
const rows = [];
|
|
17
|
+
for (const [stage, opps] of [...byStage.entries()].sort((a, b) => a[0].localeCompare(b[0]))) {
|
|
18
|
+
const totalValue = sum(opps, 'amount');
|
|
19
|
+
const weightedValue = opps.reduce((acc, o) => acc + num(o['amount']) * (num(o['probability']) / 100), 0);
|
|
20
|
+
rows.push([stage, String(opps.length), currencyDisplay(totalValue), currencyDisplay(round2(weightedValue))]);
|
|
21
|
+
}
|
|
22
|
+
const totalOpps = r.items.length;
|
|
23
|
+
const totalValue = sum(r.items, 'amount');
|
|
24
|
+
const totalWeighted = r.items.reduce((acc, o) => acc + num(o['amount']) * (num(o['probability']) / 100), 0);
|
|
25
|
+
return [
|
|
26
|
+
`## Pipeline Value Summary (${totalOpps} open opportunities)\n`,
|
|
27
|
+
`**Total Pipeline:** ${currencyDisplay(totalValue)} | **Weighted Value:** ${currencyDisplay(round2(totalWeighted))}\n`,
|
|
28
|
+
mdTable(['Stage', 'Count', 'Total Value', 'Weighted Value'], rows),
|
|
29
|
+
].join('\n');
|
|
30
|
+
});
|
|
31
|
+
// ---------------------------------------------------------------------------
|
|
32
|
+
// opportunity_win_rate
|
|
33
|
+
// Fetch won and lost opportunities in the last N days. Compute win rate.
|
|
34
|
+
// ---------------------------------------------------------------------------
|
|
35
|
+
reg('opportunity_win_rate', async (params) => {
|
|
36
|
+
const api = getAPI();
|
|
37
|
+
const days = num(params['days'], 90);
|
|
38
|
+
const since = daysAgo(days);
|
|
39
|
+
const [won, lost] = await Promise.all([
|
|
40
|
+
api.paginatedFetch('/sales/opportunities', `closedFlag=true AND status/name='Won' AND closedDate>=[${since}]`, 'id,amount,closedDate', 500),
|
|
41
|
+
api.paginatedFetch('/sales/opportunities', `closedFlag=true AND status/name='Lost' AND closedDate>=[${since}]`, 'id,amount,closedDate', 500),
|
|
42
|
+
]);
|
|
43
|
+
const wonCount = won.items.length;
|
|
44
|
+
const lostCount = lost.items.length;
|
|
45
|
+
const total = wonCount + lostCount;
|
|
46
|
+
if (total === 0)
|
|
47
|
+
return `No closed opportunities in the last ${days} days.`;
|
|
48
|
+
const winRate = total > 0 ? round2((wonCount / total) * 100) : 0;
|
|
49
|
+
const avgWon = wonCount > 0 ? round2(sum(won.items, 'amount') / wonCount) : 0;
|
|
50
|
+
const avgLost = lostCount > 0 ? round2(sum(lost.items, 'amount') / lostCount) : 0;
|
|
51
|
+
const rows = [
|
|
52
|
+
[String(wonCount), String(lostCount), `${winRate}%`, currencyDisplay(avgWon), currencyDisplay(avgLost)],
|
|
53
|
+
];
|
|
54
|
+
return [
|
|
55
|
+
`## Opportunity Win Rate (last ${days} days)\n`,
|
|
56
|
+
mdTable(['Won', 'Lost', 'Win Rate', 'Avg Won Size', 'Avg Lost Size'], rows),
|
|
57
|
+
].join('\n');
|
|
58
|
+
});
|
|
59
|
+
// ---------------------------------------------------------------------------
|
|
60
|
+
// opportunity_aging
|
|
61
|
+
// Fetch open opportunities and bucket by age.
|
|
62
|
+
// ---------------------------------------------------------------------------
|
|
63
|
+
reg('opportunity_aging', async (_params) => {
|
|
64
|
+
const api = getAPI();
|
|
65
|
+
const r = await api.paginatedFetch('/sales/opportunities', 'closedFlag=false', 'id,name,amount,dateEntered', 500);
|
|
66
|
+
if (r.items.length === 0)
|
|
67
|
+
return 'No open opportunities found.';
|
|
68
|
+
const buckets = {
|
|
69
|
+
'0-30d': { count: 0, totalValue: 0 },
|
|
70
|
+
'30-60d': { count: 0, totalValue: 0 },
|
|
71
|
+
'60-90d': { count: 0, totalValue: 0 },
|
|
72
|
+
'90+d': { count: 0, totalValue: 0 },
|
|
73
|
+
};
|
|
74
|
+
for (const o of r.items) {
|
|
75
|
+
const age = daysSince(String(o['dateEntered'] ?? ''));
|
|
76
|
+
const value = num(o['amount']);
|
|
77
|
+
let key;
|
|
78
|
+
if (age <= 30)
|
|
79
|
+
key = '0-30d';
|
|
80
|
+
else if (age <= 60)
|
|
81
|
+
key = '30-60d';
|
|
82
|
+
else if (age <= 90)
|
|
83
|
+
key = '60-90d';
|
|
84
|
+
else
|
|
85
|
+
key = '90+d';
|
|
86
|
+
buckets[key].count++;
|
|
87
|
+
buckets[key].totalValue += value;
|
|
88
|
+
}
|
|
89
|
+
const rows = Object.entries(buckets).map(([bucket, d]) => [
|
|
90
|
+
bucket,
|
|
91
|
+
String(d.count),
|
|
92
|
+
currencyDisplay(d.totalValue),
|
|
93
|
+
]);
|
|
94
|
+
return [
|
|
95
|
+
`## Opportunity Aging (${r.items.length} open)\n`,
|
|
96
|
+
mdTable(['Age Bucket', 'Count', 'Total Value'], rows),
|
|
97
|
+
].join('\n');
|
|
98
|
+
});
|
|
99
|
+
//# sourceMappingURL=analytics-msp-sales.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"analytics-msp-sales.js","sourceRoot":"","sources":["../../src/operations/analytics-msp-sales.ts"],"names":[],"mappings":"AAAA,4DAA4D;AAE5D,OAAO,EAAE,MAAM,EAAE,MAAM,gCAAgC,CAAC;AACxD,OAAO,EAA4C,GAAG,EAAE,SAAS,EAAE,OAAO,EAAE,MAAM,EAAE,YAAY,EAAE,GAAG,EAAE,eAAe,EAAE,OAAO,EAAE,MAAM,0BAA0B,CAAC;AAElK,MAAM,CAAC,MAAM,gBAAgB,GAA+B,EAAE,CAAC;AAC/D,SAAS,GAAG,CAAC,IAAY,EAAE,OAAmB,IAAU,gBAAgB,CAAC,IAAI,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC;AAE3F,8EAA8E;AAC9E,yBAAyB;AACzB,qEAAqE;AACrE,8EAA8E;AAE9E,GAAG,CAAC,wBAAwB,EAAE,KAAK,EAAE,OAAe,EAAE,EAAE;IACtD,MAAM,GAAG,GAAG,MAAM,EAAE,CAAC;IACrB,MAAM,CAAC,GAAG,MAAM,GAAG,CAAC,cAAc,CAChC,sBAAsB,EACtB,kBAAkB,EAClB,uCAAuC,EACvC,GAAG,CACJ,CAAC;IAEF,IAAI,CAAC,CAAC,KAAK,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO,8BAA8B,CAAC;IAEhE,MAAM,OAAO,GAAG,YAAY,CAAC,CAAC,CAAC,KAAK,EAAE,YAAY,CAAC,CAAC;IAEpD,MAAM,IAAI,GAAe,EAAE,CAAC;IAC5B,KAAK,MAAM,CAAC,KAAK,EAAE,IAAI,CAAC,IAAI,CAAC,GAAG,OAAO,CAAC,OAAO,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;QAC5F,MAAM,UAAU,GAAG,GAAG,CAAC,IAAI,EAAE,QAAQ,CAAC,CAAC;QACvC,MAAM,aAAa,GAAG,IAAI,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,GAAG,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,GAAG,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;QACzG,IAAI,CAAC,IAAI,CAAC,CAAC,KAAK,EAAE,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,eAAe,CAAC,UAAU,CAAC,EAAE,eAAe,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC;IAC/G,CAAC;IAED,MAAM,SAAS,GAAG,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC;IACjC,MAAM,UAAU,GAAG,GAAG,CAAC,CAAC,CAAC,KAAK,EAAE,QAAQ,CAAC,CAAC;IAC1C,MAAM,aAAa,GAAG,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,GAAG,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,GAAG,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;IAE5G,OAAO;QACL,8BAA8B,SAAS,wBAAwB;QAC/D,uBAAuB,eAAe,CAAC,UAAU,CAAC,0BAA0B,eAAe,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC,IAAI;QACtH,OAAO,CAAC,CAAC,OAAO,EAAE,OAAO,EAAE,aAAa,EAAE,gBAAgB,CAAC,EAAE,IAAI,CAAC;KACnE,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;AACf,CAAC,CAAC,CAAC;AAEH,8EAA8E;AAC9E,uBAAuB;AACvB,yEAAyE;AACzE,8EAA8E;AAE9E,GAAG,CAAC,sBAAsB,EAAE,KAAK,EAAE,MAAc,EAAE,EAAE;IACnD,MAAM,GAAG,GAAG,MAAM,EAAE,CAAC;IACrB,MAAM,IAAI,GAAG,GAAG,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,EAAE,CAAC,CAAC;IACrC,MAAM,KAAK,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;IAE5B,MAAM,CAAC,GAAG,EAAE,IAAI,CAAC,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC;QACpC,GAAG,CAAC,cAAc,CAChB,sBAAsB,EACtB,0DAA0D,KAAK,GAAG,EAClE,sBAAsB,EACtB,GAAG,CACJ;QACD,GAAG,CAAC,cAAc,CAChB,sBAAsB,EACtB,2DAA2D,KAAK,GAAG,EACnE,sBAAsB,EACtB,GAAG,CACJ;KACF,CAAC,CAAC;IAEH,MAAM,QAAQ,GAAG,GAAG,CAAC,KAAK,CAAC,MAAM,CAAC;IAClC,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC;IACpC,MAAM,KAAK,GAAG,QAAQ,GAAG,SAAS,CAAC;IAEnC,IAAI,KAAK,KAAK,CAAC;QAAE,OAAO,uCAAuC,IAAI,QAAQ,CAAC;IAE5E,MAAM,OAAO,GAAG,KAAK,GAAG,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,QAAQ,GAAG,KAAK,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IACjE,MAAM,MAAM,GAAG,QAAQ,GAAG,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,CAAC,KAAK,EAAE,QAAQ,CAAC,GAAG,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IAC9E,MAAM,OAAO,GAAG,SAAS,GAAG,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,KAAK,EAAE,QAAQ,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IAElF,MAAM,IAAI,GAAe;QACvB,CAAC,MAAM,CAAC,QAAQ,CAAC,EAAE,MAAM,CAAC,SAAS,CAAC,EAAE,GAAG,OAAO,GAAG,EAAE,eAAe,CAAC,MAAM,CAAC,EAAE,eAAe,CAAC,OAAO,CAAC,CAAC;KACxG,CAAC;IAEF,OAAO;QACL,iCAAiC,IAAI,UAAU;QAC/C,OAAO,CAAC,CAAC,KAAK,EAAE,MAAM,EAAE,UAAU,EAAE,cAAc,EAAE,eAAe,CAAC,EAAE,IAAI,CAAC;KAC5E,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;AACf,CAAC,CAAC,CAAC;AAEH,8EAA8E;AAC9E,oBAAoB;AACpB,8CAA8C;AAC9C,8EAA8E;AAE9E,GAAG,CAAC,mBAAmB,EAAE,KAAK,EAAE,OAAe,EAAE,EAAE;IACjD,MAAM,GAAG,GAAG,MAAM,EAAE,CAAC;IACrB,MAAM,CAAC,GAAG,MAAM,GAAG,CAAC,cAAc,CAChC,sBAAsB,EACtB,kBAAkB,EAClB,4BAA4B,EAC5B,GAAG,CACJ,CAAC;IAEF,IAAI,CAAC,CAAC,KAAK,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO,8BAA8B,CAAC;IAEhE,MAAM,OAAO,GAA0D;QACrE,OAAO,EAAE,EAAE,KAAK,EAAE,CAAC,EAAE,UAAU,EAAE,CAAC,EAAE;QACpC,QAAQ,EAAE,EAAE,KAAK,EAAE,CAAC,EAAE,UAAU,EAAE,CAAC,EAAE;QACrC,QAAQ,EAAE,EAAE,KAAK,EAAE,CAAC,EAAE,UAAU,EAAE,CAAC,EAAE;QACrC,MAAM,EAAE,EAAE,KAAK,EAAE,CAAC,EAAE,UAAU,EAAE,CAAC,EAAE;KACpC,CAAC;IAEF,KAAK,MAAM,CAAC,IAAI,CAAC,CAAC,KAAK,EAAE,CAAC;QACxB,MAAM,GAAG,GAAG,SAAS,CAAC,MAAM,CAAC,CAAC,CAAC,aAAa,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC;QACtD,MAAM,KAAK,GAAG,GAAG,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC;QAC/B,IAAI,GAAW,CAAC;QAChB,IAAI,GAAG,IAAI,EAAE;YAAE,GAAG,GAAG,OAAO,CAAC;aACxB,IAAI,GAAG,IAAI,EAAE;YAAE,GAAG,GAAG,QAAQ,CAAC;aAC9B,IAAI,GAAG,IAAI,EAAE;YAAE,GAAG,GAAG,QAAQ,CAAC;;YAC9B,GAAG,GAAG,MAAM,CAAC;QAClB,OAAO,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,CAAC;QACrB,OAAO,CAAC,GAAG,CAAC,CAAC,UAAU,IAAI,KAAK,CAAC;IACnC,CAAC;IAED,MAAM,IAAI,GAAe,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC;QACpE,MAAM;QACN,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC;QACf,eAAe,CAAC,CAAC,CAAC,UAAU,CAAC;KAC9B,CAAC,CAAC;IAEH,OAAO;QACL,yBAAyB,CAAC,CAAC,KAAK,CAAC,MAAM,UAAU;QACjD,OAAO,CAAC,CAAC,YAAY,EAAE,OAAO,EAAE,aAAa,CAAC,EAAE,IAAI,CAAC;KACtD,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;AACf,CAAC,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"analytics-msp-schedule.d.ts","sourceRoot":"","sources":["../../src/operations/analytics-msp-schedule.ts"],"names":[],"mappings":"AAGA,OAAO,EAGL,KAAK,UAAU,EAWhB,MAAM,0BAA0B,CAAC;AAMlC,eAAO,MAAM,mBAAmB,EAAE,MAAM,CAAC,MAAM,EAAE,UAAU,CAAM,CAAC"}
|