@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.
Files changed (147) hide show
  1. package/data/connectwise_api.db +0 -0
  2. package/data/manage.json +298179 -0
  3. package/dist/index.d.ts +10 -0
  4. package/dist/index.d.ts.map +1 -0
  5. package/dist/index.js +116 -0
  6. package/dist/index.js.map +1 -0
  7. package/dist/operations/analytics-extended.d.ts +6 -0
  8. package/dist/operations/analytics-extended.d.ts.map +1 -0
  9. package/dist/operations/analytics-extended.js +825 -0
  10. package/dist/operations/analytics-extended.js.map +1 -0
  11. package/dist/operations/analytics-msp-assets.d.ts +3 -0
  12. package/dist/operations/analytics-msp-assets.d.ts.map +1 -0
  13. package/dist/operations/analytics-msp-assets.js +180 -0
  14. package/dist/operations/analytics-msp-assets.js.map +1 -0
  15. package/dist/operations/analytics-msp-clients.d.ts +3 -0
  16. package/dist/operations/analytics-msp-clients.d.ts.map +1 -0
  17. package/dist/operations/analytics-msp-clients.js +198 -0
  18. package/dist/operations/analytics-msp-clients.js.map +1 -0
  19. package/dist/operations/analytics-msp-comms.d.ts +3 -0
  20. package/dist/operations/analytics-msp-comms.d.ts.map +1 -0
  21. package/dist/operations/analytics-msp-comms.js +127 -0
  22. package/dist/operations/analytics-msp-comms.js.map +1 -0
  23. package/dist/operations/analytics-msp-contracts.d.ts +3 -0
  24. package/dist/operations/analytics-msp-contracts.d.ts.map +1 -0
  25. package/dist/operations/analytics-msp-contracts.js +91 -0
  26. package/dist/operations/analytics-msp-contracts.js.map +1 -0
  27. package/dist/operations/analytics-msp-financial.d.ts +3 -0
  28. package/dist/operations/analytics-msp-financial.d.ts.map +1 -0
  29. package/dist/operations/analytics-msp-financial.js +300 -0
  30. package/dist/operations/analytics-msp-financial.js.map +1 -0
  31. package/dist/operations/analytics-msp-procurement.d.ts +3 -0
  32. package/dist/operations/analytics-msp-procurement.d.ts.map +1 -0
  33. package/dist/operations/analytics-msp-procurement.js +78 -0
  34. package/dist/operations/analytics-msp-procurement.js.map +1 -0
  35. package/dist/operations/analytics-msp-projects.d.ts +3 -0
  36. package/dist/operations/analytics-msp-projects.d.ts.map +1 -0
  37. package/dist/operations/analytics-msp-projects.js +190 -0
  38. package/dist/operations/analytics-msp-projects.js.map +1 -0
  39. package/dist/operations/analytics-msp-sales.d.ts +3 -0
  40. package/dist/operations/analytics-msp-sales.d.ts.map +1 -0
  41. package/dist/operations/analytics-msp-sales.js +99 -0
  42. package/dist/operations/analytics-msp-sales.js.map +1 -0
  43. package/dist/operations/analytics-msp-schedule.d.ts +3 -0
  44. package/dist/operations/analytics-msp-schedule.d.ts.map +1 -0
  45. package/dist/operations/analytics-msp-schedule.js +339 -0
  46. package/dist/operations/analytics-msp-schedule.js.map +1 -0
  47. package/dist/operations/analytics-msp-team.d.ts +3 -0
  48. package/dist/operations/analytics-msp-team.d.ts.map +1 -0
  49. package/dist/operations/analytics-msp-team.js +195 -0
  50. package/dist/operations/analytics-msp-team.js.map +1 -0
  51. package/dist/operations/analytics-msp-tickets.d.ts +3 -0
  52. package/dist/operations/analytics-msp-tickets.d.ts.map +1 -0
  53. package/dist/operations/analytics-msp-tickets.js +578 -0
  54. package/dist/operations/analytics-msp-tickets.js.map +1 -0
  55. package/dist/operations/analytics-msp-time.d.ts +3 -0
  56. package/dist/operations/analytics-msp-time.d.ts.map +1 -0
  57. package/dist/operations/analytics-msp-time.js +485 -0
  58. package/dist/operations/analytics-msp-time.js.map +1 -0
  59. package/dist/operations/analytics-msp-utils.d.ts +49 -0
  60. package/dist/operations/analytics-msp-utils.d.ts.map +1 -0
  61. package/dist/operations/analytics-msp-utils.js +157 -0
  62. package/dist/operations/analytics-msp-utils.js.map +1 -0
  63. package/dist/operations/analytics.d.ts +9 -0
  64. package/dist/operations/analytics.d.ts.map +1 -0
  65. package/dist/operations/analytics.js +742 -0
  66. package/dist/operations/analytics.js.map +1 -0
  67. package/dist/operations/executor.d.ts +10 -0
  68. package/dist/operations/executor.d.ts.map +1 -0
  69. package/dist/operations/executor.js +243 -0
  70. package/dist/operations/executor.js.map +1 -0
  71. package/dist/operations/registry.d.ts +16 -0
  72. package/dist/operations/registry.d.ts.map +1 -0
  73. package/dist/operations/registry.js +847 -0
  74. package/dist/operations/registry.js.map +1 -0
  75. package/dist/services/api-database.d.ts +38 -0
  76. package/dist/services/api-database.d.ts.map +1 -0
  77. package/dist/services/api-database.js +191 -0
  78. package/dist/services/api-database.js.map +1 -0
  79. package/dist/services/cache.d.ts +12 -0
  80. package/dist/services/cache.d.ts.map +1 -0
  81. package/dist/services/cache.js +32 -0
  82. package/dist/services/cache.js.map +1 -0
  83. package/dist/services/connectwise-api.d.ts +43 -0
  84. package/dist/services/connectwise-api.d.ts.map +1 -0
  85. package/dist/services/connectwise-api.js +198 -0
  86. package/dist/services/connectwise-api.js.map +1 -0
  87. package/dist/services/db-builder.d.ts +11 -0
  88. package/dist/services/db-builder.d.ts.map +1 -0
  89. package/dist/services/db-builder.js +237 -0
  90. package/dist/services/db-builder.js.map +1 -0
  91. package/dist/services/fast-memory.d.ts +39 -0
  92. package/dist/services/fast-memory.d.ts.map +1 -0
  93. package/dist/services/fast-memory.js +147 -0
  94. package/dist/services/fast-memory.js.map +1 -0
  95. package/dist/services/load-env.d.ts +15 -0
  96. package/dist/services/load-env.d.ts.map +1 -0
  97. package/dist/services/load-env.js +59 -0
  98. package/dist/services/load-env.js.map +1 -0
  99. package/dist/tools/batch.d.ts +9 -0
  100. package/dist/tools/batch.d.ts.map +1 -0
  101. package/dist/tools/batch.js +159 -0
  102. package/dist/tools/batch.js.map +1 -0
  103. package/dist/tools/composite.d.ts +9 -0
  104. package/dist/tools/composite.d.ts.map +1 -0
  105. package/dist/tools/composite.js +353 -0
  106. package/dist/tools/composite.js.map +1 -0
  107. package/dist/tools/discovery.d.ts +9 -0
  108. package/dist/tools/discovery.d.ts.map +1 -0
  109. package/dist/tools/discovery.js +245 -0
  110. package/dist/tools/discovery.js.map +1 -0
  111. package/dist/tools/execution.d.ts +9 -0
  112. package/dist/tools/execution.d.ts.map +1 -0
  113. package/dist/tools/execution.js +130 -0
  114. package/dist/tools/execution.js.map +1 -0
  115. package/dist/tools/memory.d.ts +9 -0
  116. package/dist/tools/memory.d.ts.map +1 -0
  117. package/dist/tools/memory.js +152 -0
  118. package/dist/tools/memory.js.map +1 -0
  119. package/dist/tools/operations.d.ts +9 -0
  120. package/dist/tools/operations.d.ts.map +1 -0
  121. package/dist/tools/operations.js +214 -0
  122. package/dist/tools/operations.js.map +1 -0
  123. package/dist/tools/pagination.d.ts +9 -0
  124. package/dist/tools/pagination.d.ts.map +1 -0
  125. package/dist/tools/pagination.js +133 -0
  126. package/dist/tools/pagination.js.map +1 -0
  127. package/dist/tools/validation.d.ts +9 -0
  128. package/dist/tools/validation.d.ts.map +1 -0
  129. package/dist/tools/validation.js +705 -0
  130. package/dist/tools/validation.js.map +1 -0
  131. package/dist/types/index.d.ts +145 -0
  132. package/dist/types/index.d.ts.map +1 -0
  133. package/dist/types/index.js +3 -0
  134. package/dist/types/index.js.map +1 -0
  135. package/dist/types/operations.d.ts +30 -0
  136. package/dist/types/operations.d.ts.map +1 -0
  137. package/dist/types/operations.js +3 -0
  138. package/dist/types/operations.js.map +1 -0
  139. package/dist/utils/conditions.d.ts +20 -0
  140. package/dist/utils/conditions.d.ts.map +1 -0
  141. package/dist/utils/conditions.js +78 -0
  142. package/dist/utils/conditions.js.map +1 -0
  143. package/dist/utils/formatters.d.ts +35 -0
  144. package/dist/utils/formatters.d.ts.map +1 -0
  145. package/dist/utils/formatters.js +337 -0
  146. package/dist/utils/formatters.js.map +1 -0
  147. 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,3 @@
1
+ import { type MspHandler } from './analytics-msp-utils.js';
2
+ export declare const mspProjectHandlers: Record<string, MspHandler>;
3
+ //# sourceMappingURL=analytics-msp-projects.d.ts.map
@@ -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":"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"}
@@ -0,0 +1,3 @@
1
+ import { type MspHandler } from './analytics-msp-utils.js';
2
+ export declare const mspSalesHandlers: Record<string, MspHandler>;
3
+ //# sourceMappingURL=analytics-msp-sales.d.ts.map
@@ -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":"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"}
@@ -0,0 +1,3 @@
1
+ import { type MspHandler } from './analytics-msp-utils.js';
2
+ export declare const mspScheduleHandlers: Record<string, MspHandler>;
3
+ //# sourceMappingURL=analytics-msp-schedule.d.ts.map
@@ -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"}