@mgsoftwarebv/mcp-server-bridge 2.10.1 β†’ 2.12.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/README.md CHANGED
@@ -78,6 +78,7 @@ mg-tickets-mcp
78
78
 
79
79
  De bridge geeft toegang tot alle tools van je MCP server:
80
80
 
81
+ ### Tickets & Projects
81
82
  - **get-tickets** - Haal tickets op met filters
82
83
  - **get-ticket-by-id** - Haal een specifieke ticket op
83
84
  - **create-ticket** - Maak een nieuwe ticket aan
@@ -88,6 +89,78 @@ De bridge geeft toegang tot alle tools van je MCP server:
88
89
  - **get-project-by-id** - Haal een specifiek project op
89
90
  - **create-project** - Maak een nieuw project aan
90
91
 
92
+ ### AI Development Tracking
93
+ - **start-ai-session-smart** - Start AI development sessie met time tracking
94
+ - **track-manual-follow-up** - Track manual follow-ups
95
+ - **sync-session-todos** - Synchroniseer todos met sessie
96
+ - **complete-ai-session** - Voltooi AI sessie met samenvatting
97
+ - **get-completion-context** - Haal completion context op
98
+ - **save-customer-response** - Sla customer response op
99
+
100
+ ### Time Tracking
101
+ - **log-hours** πŸ†• - Analyseer chat conversatie en log uren automatisch (zie Cursor Command `/hours`)
102
+
103
+ ## ⏱️ Quick Hour Logging met `/hours` Command
104
+
105
+ De nieuwe `/hours` Cursor command maakt het super eenvoudig om uren te loggen:
106
+
107
+ 1. **Voer je werk uit in Cursor** - gewoon normaal chatten en coderen
108
+ 2. **Type `/hours`** - wanneer je klaar bent
109
+ 3. **AI analyseert de chat** en schat in hoeveel tijd een senior developer (zonder AI) zou besteden
110
+ 4. **Automatische projectkoppeling** - gebruikt je workspace folder naam
111
+ 5. **Draft entry** - wordt aangemaakt in de tracker (niet direct gefactureerd)
112
+
113
+ ### Hoe werkt het?
114
+
115
+ De AI:
116
+ 1. **Haalt alle beschikbare projecten op** via `get-projects`
117
+ 2. **Matched workspace intelligent** (bijv. "tickets-v2" β†’ "Internal Tickets V2")
118
+ - Geen match? β†’ Logt zonder project
119
+ - Meerdere matches? β†’ Vraagt gebruiker
120
+ 3. **Analyseert de chat** - wat is er gebouwd/opgelost
121
+ 4. **Schat realistische tijd** (investigation + implementation + testing)
122
+ 5. **Maakt tracker entry** als draft (met of zonder project)
123
+
124
+ ### Voorbeeld:
125
+
126
+ ```
127
+ Workspace: "tickets-v2"
128
+ Type: /hours
129
+
130
+ AI doet:
131
+ 1. Haalt projecten op: ["Internal Tickets V2", "Client Portal", ...]
132
+ 2. Match: "tickets-v2" β†’ "Internal Tickets V2" βœ“
133
+ 3. Analyseert chat: Auth improvements
134
+ 4. Schatting: 2 uur senior dev tijd
135
+ 5. CreΓ«ert entry:
136
+ - Project: Internal Tickets V2
137
+ - Description: "Authentication improvements: Fixed login bug and added password validation"
138
+ - Hours: 2h
139
+ - Status: DRAFT
140
+
141
+ Of zonder project match:
142
+ Workspace: "random-scripts"
143
+ Type: /hours
144
+
145
+ AI doet:
146
+ 1. Haalt projecten op: ["Internal Tickets V2", "Client Portal", ...]
147
+ 2. Geen match gevonden voor "random-scripts"
148
+ 3. Logt zonder project:
149
+ - Project: (No project assigned)
150
+ - Description: "Script improvements: Added data migration script"
151
+ - Hours: 0.5h
152
+ - Status: DRAFT
153
+ ```
154
+
155
+ ### Voordelen:
156
+
157
+ - βœ… Geen handmatig uren bijhouden tijdens het werk
158
+ - βœ… Realistische schatting (senior dev zonder AI)
159
+ - βœ… **Slimme projectkoppeling** - AI matched workspace naam semantisch
160
+ - βœ… Vraagt bevestiging bij twijfel over project
161
+ - βœ… Chat context als documentatie
162
+ - βœ… Draft entries voor review
163
+
91
164
  ## πŸ“š Beschikbare Resources
92
165
 
93
166
  - **tickets://recent** - Recente tickets across alle teams
package/dist/index.js CHANGED
@@ -179,7 +179,7 @@ var TOOLS = [
179
179
  // === NEW AI SESSION TOOLS ===
180
180
  {
181
181
  name: "start-ai-session-smart",
182
- description: "Start a new AI development session with automatic tracking (time estimate provided by Cursor AI)",
182
+ description: "Start a new AI development session with automatic tracking (time breakdown provided by Cursor AI)",
183
183
  inputSchema: {
184
184
  type: "object",
185
185
  properties: {
@@ -191,9 +191,16 @@ var TOOLS = [
191
191
  items: { type: "string" },
192
192
  description: "Relevant files for complexity analysis"
193
193
  },
194
- aiTimeEstimateMinutes: {
195
- type: "number",
196
- description: "AI estimate in minutes without AI assistance"
194
+ timeBreakdown: {
195
+ type: "object",
196
+ description: "Time estimate breakdown by development phase (REQUIRED)",
197
+ properties: {
198
+ analysisMinutes: { type: "number", description: "Ticket/context analysis time" },
199
+ investigationMinutes: { type: "number", description: "Bug investigation/reproduction (0 if not a bug)" },
200
+ developmentMinutes: { type: "number", description: "Actual coding/fixing time" },
201
+ communicationMinutes: { type: "number", description: "Customer response writing time" }
202
+ },
203
+ required: ["analysisMinutes", "investigationMinutes", "developmentMinutes", "communicationMinutes"]
197
204
  },
198
205
  complexityScore: {
199
206
  type: "number",
@@ -202,7 +209,7 @@ var TOOLS = [
202
209
  description: "Estimated complexity from 1-10"
203
210
  }
204
211
  },
205
- required: ["ticketId", "aiTimeEstimateMinutes"]
212
+ required: ["ticketId", "timeBreakdown"]
206
213
  }
207
214
  },
208
215
  {
@@ -372,6 +379,32 @@ var TOOLS = [
372
379
  },
373
380
  required: ["aiSessionId", "workCompleted"]
374
381
  }
382
+ },
383
+ {
384
+ name: "log-hours",
385
+ description: "Analyze current chat conversation and log hours as draft tracker entry. AI analyzes chat context to estimate hours as a senior developer would (without AI assistance). Cursor AI matches workspace name to correct project from list (optional).",
386
+ inputSchema: {
387
+ type: "object",
388
+ properties: {
389
+ projectId: {
390
+ type: "string",
391
+ description: "Project ID (UUID) - Optional. Cursor AI should call get-projects first to try matching workspace name. If no clear match, omit this field."
392
+ },
393
+ workDescription: {
394
+ type: "string",
395
+ description: "Short description of the work done (for the tracker entry)"
396
+ },
397
+ estimatedHours: {
398
+ type: "number",
399
+ description: "AI-estimated hours as senior developer would spend WITHOUT AI (e.g., 2.5 for 2.5 hours)"
400
+ },
401
+ chatContextSummary: {
402
+ type: "string",
403
+ description: "Brief summary of chat context for internal logging (optional)"
404
+ }
405
+ },
406
+ required: ["workDescription", "estimatedHours"]
407
+ }
375
408
  }
376
409
  ];
377
410
  var RESOURCES = [
@@ -705,28 +738,63 @@ ${description ? `Description: ${description}
705
738
  }
706
739
  // === AI SESSION TOOLS ===
707
740
  case "start-ai-session-smart": {
708
- const { ticketId, ticketUrl, cursorSessionId, codebaseContext, aiTimeEstimateMinutes, complexityScore } = args2;
741
+ const { ticketId, ticketUrl, cursorSessionId, codebaseContext, timeBreakdown, complexityScore } = args2;
742
+ if (!timeBreakdown || !timeBreakdown.analysisMinutes || timeBreakdown.developmentMinutes === void 0) {
743
+ throw new Error("timeBreakdown is required with all phases: analysisMinutes, investigationMinutes, developmentMinutes, communicationMinutes");
744
+ }
745
+ const totalEstimateMinutes = (timeBreakdown.analysisMinutes || 0) + (timeBreakdown.investigationMinutes || 0) + (timeBreakdown.developmentMinutes || 0) + (timeBreakdown.communicationMinutes || 0);
709
746
  const sessionStartTime = /* @__PURE__ */ new Date();
710
747
  const { data: sessionData, error } = await supabase.from("ai_sessions").insert({
711
748
  ticket_id: ticketId,
712
749
  provider_user_id: authContext.userId,
713
750
  team_id: authContext.teamId,
714
751
  cursor_session_id: cursorSessionId || null,
715
- ai_time_estimate_minutes: aiTimeEstimateMinutes,
716
- // Use Cursor's smart estimate
752
+ ai_time_estimate_minutes: totalEstimateMinutes,
717
753
  complexity_score: complexityScore || null,
718
754
  status: "in_progress"
719
- // Start actively tracking time
720
755
  }).select("id, ticket_id, cursor_session_id, created_at").single();
721
756
  if (error) throw error;
722
- await supabase.from("ai_time_logs").insert({
723
- ai_session_id: sessionData.id,
724
- activity_type: "thinking",
725
- description: "AI session started - analysis and planning phase",
726
- duration_seconds: 0,
727
- productivity_score: 8,
728
- logged_at: sessionStartTime
729
- });
757
+ const phaseActivities = [
758
+ {
759
+ ai_session_id: sessionData.id,
760
+ activity_type: "analysis",
761
+ description: "Ticket analysis and context understanding",
762
+ duration_seconds: 0,
763
+ estimated_duration_seconds: timeBreakdown.analysisMinutes * 60,
764
+ status: "in_progress",
765
+ // Analysis starts immediately
766
+ productivity_score: 8,
767
+ started_at: sessionStartTime.toISOString()
768
+ },
769
+ {
770
+ ai_session_id: sessionData.id,
771
+ activity_type: "bug_investigation",
772
+ description: "Bug investigation and root cause analysis",
773
+ duration_seconds: 0,
774
+ estimated_duration_seconds: timeBreakdown.investigationMinutes * 60,
775
+ status: "pending",
776
+ productivity_score: null
777
+ },
778
+ {
779
+ ai_session_id: sessionData.id,
780
+ activity_type: "development",
781
+ description: "Implementation and coding",
782
+ duration_seconds: 0,
783
+ estimated_duration_seconds: timeBreakdown.developmentMinutes * 60,
784
+ status: "pending",
785
+ productivity_score: null
786
+ },
787
+ {
788
+ ai_session_id: sessionData.id,
789
+ activity_type: "communication",
790
+ description: "Customer response and documentation",
791
+ duration_seconds: 0,
792
+ estimated_duration_seconds: timeBreakdown.communicationMinutes * 60,
793
+ status: "pending",
794
+ productivity_score: null
795
+ }
796
+ ];
797
+ await supabase.from("ai_time_logs").insert(phaseActivities);
730
798
  const sessionId = `ai-sess-${sessionData.id.substring(0, 8)}`;
731
799
  return {
732
800
  content: [{
@@ -735,13 +803,20 @@ ${description ? `Description: ${description}
735
803
 
736
804
  \u{1F194} Session ID: **${sessionId}**
737
805
  \u{1F3AB} Ticket: ${ticketId}
738
- \u23F1\uFE0F Time Estimate: ${aiTimeEstimateMinutes} minutes
806
+
807
+ \u{1F4CA} **Time Breakdown:**
808
+ \u2022 Analysis: ${timeBreakdown.analysisMinutes} min
809
+ \u2022 Investigation: ${timeBreakdown.investigationMinutes} min
810
+ \u2022 Development: ${timeBreakdown.developmentMinutes} min
811
+ \u2022 Communication: ${timeBreakdown.communicationMinutes} min
812
+ \u2022 **Total: ${totalEstimateMinutes} min**
813
+
739
814
  ${complexityScore ? `\u{1F3AF} Complexity: ${complexityScore}/10
740
- ` : ""}\u23F1\uFE0F **Automatic Time Tracking Started**
815
+ ` : ""}\u23F1\uFE0F **Phase Tracking Started** (Analysis in progress)
741
816
  ${cursorSessionId ? `\u{1F517} Cursor Session: ${cursorSessionId}
742
817
  ` : ""}\u{1F4C5} Started: ${sessionStartTime.toLocaleString()}
743
818
 
744
- \u2705 Session initialized - ready for development!
819
+ \u2705 Session initialized with phase breakdown!
745
820
  \u{1F4DD} Timetrack entry will be created when you complete the session.`
746
821
  }]
747
822
  };
@@ -784,7 +859,7 @@ ${cursorSessionId ? `\u{1F517} Cursor Session: ${cursorSessionId}
784
859
  description: `Follow-up: ${followUpReason.replace("_", " ")} - ${outcome}`,
785
860
  duration_seconds: (timeSpentMinutes || 0) * 60,
786
861
  productivity_score: outcome === "success" ? 9 : outcome === "partial_success" ? 6 : 4,
787
- logged_at: followUpTime
862
+ started_at: followUpTime.toISOString()
788
863
  });
789
864
  const sessionStartTime = new Date(session.created_at);
790
865
  const totalMinutesElapsed = Math.round((followUpTime.getTime() - sessionStartTime.getTime()) / 6e4);
@@ -1240,6 +1315,70 @@ ${efficiencyNotes}
1240
1315
  }]
1241
1316
  };
1242
1317
  }
1318
+ case "log-hours": {
1319
+ const { projectId, workDescription, estimatedHours, chatContextSummary } = args2;
1320
+ let project = null;
1321
+ if (projectId) {
1322
+ const teamIds = await getAccessibleTeamIds(authContext.teamId);
1323
+ const { data: projectData, error: projectError } = await supabase.from("projects").select("id, name, team_id").eq("id", projectId).in("team_id", teamIds).single();
1324
+ if (projectError || !projectData) {
1325
+ throw new Error(`Project not found or no access: ${projectId}. Please call get-projects first to find the correct project.`);
1326
+ }
1327
+ project = projectData;
1328
+ }
1329
+ const durationSeconds = Math.round(estimatedHours * 3600);
1330
+ const now = /* @__PURE__ */ new Date();
1331
+ const startTime = new Date(now.getTime() - durationSeconds * 1e3);
1332
+ const { data: trackerEntry, error: trackerError } = await supabase.from("tracker_entries").insert({
1333
+ team_id: authContext.teamId,
1334
+ assigned_id: authContext.userId,
1335
+ project_id: project?.id || null,
1336
+ description: workDescription,
1337
+ duration: durationSeconds,
1338
+ start: startTime.toISOString(),
1339
+ stop: now.toISOString(),
1340
+ date: now.toISOString().split("T")[0],
1341
+ billed: false
1342
+ }).select("id, duration, project_id").single();
1343
+ if (trackerError || !trackerEntry) {
1344
+ throw new Error(`Failed to create tracker entry: ${trackerError?.message || "Unknown error"}`);
1345
+ }
1346
+ let responseText = `\u23F1\uFE0F **Hours Logged Successfully!**
1347
+
1348
+ `;
1349
+ responseText += `\u{1F4CB} **Entry Details:**
1350
+ `;
1351
+ if (project) {
1352
+ responseText += ` \u2022 Project: ${project.name}
1353
+ `;
1354
+ } else {
1355
+ responseText += ` \u2022 Project: (No project assigned)
1356
+ `;
1357
+ }
1358
+ responseText += ` \u2022 Description: ${workDescription}
1359
+ `;
1360
+ responseText += ` \u2022 Estimated Hours: ${estimatedHours}h (${Math.floor(estimatedHours)}h ${Math.round(estimatedHours % 1 * 60)}m)
1361
+ `;
1362
+ responseText += ` \u2022 Status: DRAFT (not billed yet)
1363
+ `;
1364
+ responseText += ` \u2022 Entry ID: ${trackerEntry.id}
1365
+
1366
+ `;
1367
+ if (chatContextSummary) {
1368
+ responseText += `\u{1F4CA} **Work Context:**
1369
+ `;
1370
+ responseText += `${chatContextSummary.substring(0, 200)}${chatContextSummary.length > 200 ? "..." : ""}
1371
+
1372
+ `;
1373
+ }
1374
+ responseText += `\u2705 Time entry created and ready for review in the tracker!`;
1375
+ return {
1376
+ content: [{
1377
+ type: "text",
1378
+ text: responseText
1379
+ }]
1380
+ };
1381
+ }
1243
1382
  default:
1244
1383
  throw new Error(`Unknown tool: ${name}`);
1245
1384
  }
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts"],"names":["args","name"],"mappings":"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","file":"index.js","sourcesContent":["#!/usr/bin/env node\r\n\r\nimport { Server } from '@modelcontextprotocol/sdk/server/index.js';\r\nimport { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js';\r\nimport { CallToolRequestSchema, ListToolsRequestSchema, ListResourcesRequestSchema, ReadResourceRequestSchema } from '@modelcontextprotocol/sdk/types.js';\r\nimport { createClient } from '@supabase/supabase-js';\r\nimport { createHash } from 'crypto';\r\n\r\n// Parse command line arguments and environment\r\nconst args = process.argv.slice(2);\r\nconst apiKey = args.find(arg => arg.startsWith('--api-key='))?.split('=')[1] || process.env.MG_TICKETS_API_KEY;\r\n// Default to MG Software's Supabase for SaaS customers\r\nconst supabaseUrl = args.find(arg => arg.startsWith('--supabase-url='))?.split('=')[1] || \r\n process.env.SUPABASE_URL || \r\n 'https://cvjdbczxyczjnatuolsk.supabase.co';\r\n\r\nconst supabaseKey = args.find(arg => arg.startsWith('--supabase-key='))?.split('=')[1] || \r\n process.env.SUPABASE_SERVICE_ROLE_KEY || \r\n 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJzdXBhYmFzZSIsInJlZiI6ImN2amRiY3p4eWN6am5hdHVvbHNrIiwicm9sZSI6InNlcnZpY2Vfcm9sZSIsImlhdCI6MTc1NjE0NzcyNCwiZXhwIjoyMDcxNzIzNzI0fQ.LljuNdCZXDcSIVTeIVOSNsvGNBfWsIM1QIswBJmGXKE';\r\n\r\nif (!apiKey) {\r\n console.error('❌ API key is required. Use --api-key=your_key or set MG_TICKETS_API_KEY environment variable');\r\n process.exit(1);\r\n}\r\n\r\n// Supabase credentials validation (optional with defaults)\r\nif (!supabaseUrl || !supabaseKey) {\r\n console.error('❌ Supabase credentials missing. Using defaults for MG Software SaaS.');\r\n}\r\n\r\n// Initialize Supabase client with service role\r\nconst supabase = createClient(supabaseUrl, supabaseKey);\r\n\r\ninterface AuthContext {\r\n userId: string;\r\n teamId: string;\r\n scopes: string[];\r\n}\r\n\r\n// Smart time estimation and codebase analysis is now handled by Cursor AI\r\n// MCP bridge focuses on data storage and session management only\r\n\r\n// Helper function to get accessible team IDs (own team + child teams)\r\nasync function getAccessibleTeamIds(teamId: string): Promise<string[]> {\r\n const { data: accessibleTeams } = await supabase\r\n .from('teams')\r\n .select('id')\r\n .or(`id.eq.${teamId},parent_team_id.eq.${teamId}`);\r\n \r\n return accessibleTeams?.map(t => t.id) || [teamId];\r\n}\r\n\r\n// API Key validation - direct database access\r\nasync function validateApiKey(key: string): Promise<AuthContext | null> {\r\n if (!key.startsWith('mid_') || key.length !== 68) {\r\n console.error('πŸ”‘ Invalid API key format');\r\n return null;\r\n }\r\n\r\n try {\r\n // Hash the API key\r\n const keyHash = createHash('sha256').update(key).digest('hex');\r\n console.error(`πŸ” Validating API key hash: ${keyHash.substring(0, 16)}...`);\r\n \r\n // Query database for API key\r\n const { data: apiKeyData, error } = await supabase\r\n .from('api_keys')\r\n .select('id, user_id, team_id, scopes, last_used_at')\r\n .eq('key_hash', keyHash)\r\n .single();\r\n\r\n if (error || !apiKeyData) {\r\n console.error('❌ API key not found or invalid:', error?.message);\r\n return null;\r\n }\r\n\r\n // Update last used timestamp\r\n await supabase\r\n .from('api_keys')\r\n .update({ last_used_at: new Date().toISOString() })\r\n .eq('id', apiKeyData.id);\r\n\r\n console.error(`βœ… API key validated for user ${apiKeyData.user_id} in team ${apiKeyData.team_id}`);\r\n \r\n return {\r\n userId: apiKeyData.user_id,\r\n teamId: apiKeyData.team_id,\r\n scopes: apiKeyData.scopes || []\r\n };\r\n } catch (error) {\r\n console.error('πŸ’₯ API key validation error:', error);\r\n return null;\r\n }\r\n}\r\n\r\n// Validate auth context once at startup\r\nlet authContext: AuthContext | null = null;\r\n\r\n// Helper function to check if a file is an image\r\nfunction isImageFile(mimeType: string): boolean {\r\n return mimeType.startsWith('image/') && \r\n ['image/jpeg', 'image/jpg', 'image/png', 'image/gif', 'image/webp'].includes(mimeType);\r\n}\r\n\r\n// Helper function to download and convert image to base64\r\nasync function downloadImageAsBase64(storageKey: string): Promise<string | null> {\r\n try {\r\n // Get signed URL for the file (1 hour expiry)\r\n const { data: urlData, error: urlError } = await supabase.storage\r\n .from('vault')\r\n .createSignedUrl(storageKey, 3600);\r\n \r\n if (urlError || !urlData?.signedUrl) {\r\n console.error(`Failed to create signed URL for ${storageKey}:`, urlError);\r\n return null;\r\n }\r\n \r\n // Download the file\r\n const response = await fetch(urlData.signedUrl);\r\n if (!response.ok) {\r\n console.error(`Failed to download file ${storageKey}: ${response.status}`);\r\n return null;\r\n }\r\n \r\n // Convert to base64\r\n const arrayBuffer = await response.arrayBuffer();\r\n const buffer = Buffer.from(arrayBuffer);\r\n return buffer.toString('base64');\r\n } catch (error) {\r\n console.error(`Error downloading image ${storageKey}:`, error);\r\n return null;\r\n }\r\n}\r\n\r\n/**\r\n * Convert TipTap JSON format to plain text for timetrack/agenda display\r\n * @param description - The description in TipTap JSON format or plain text\r\n * @returns Plain text version of the description\r\n */\r\nfunction convertTipTapToPlainText(description?: string | null): string {\r\n if (!description) {\r\n return \"\";\r\n }\r\n\r\n // If it's already plain text, return as is\r\n if (!description.startsWith('{\"type\":\"doc\"')) {\r\n return description;\r\n }\r\n\r\n try {\r\n const parsed = JSON.parse(description);\r\n \r\n // Recursively extract text from TipTap JSON structure\r\n const extractText = (node: any): string => {\r\n if (!node) return \"\";\r\n\r\n // Text nodes contain the actual text\r\n if (node.type === \"text\") {\r\n return node.text || \"\";\r\n }\r\n\r\n // Handle hard breaks\r\n if (node.type === \"hardBreak\") {\r\n return \"\\n\";\r\n }\r\n\r\n // Handle mentions\r\n if (node.type === \"mention\") {\r\n return `@${node.attrs?.label || node.attrs?.id || \"\"}`;\r\n }\r\n\r\n // Handle ticket mentions\r\n if (node.type === \"ticketMention\") {\r\n return `#${node.attrs?.label || node.attrs?.id || \"\"}`;\r\n }\r\n\r\n // Handle images\r\n if (node.type === \"image\") {\r\n return \"[afbeelding]\";\r\n }\r\n\r\n // Process children recursively\r\n if (node.content && Array.isArray(node.content)) {\r\n const childrenText = node.content.map(extractText).join(\"\");\r\n \r\n // Add appropriate spacing based on node type\r\n switch (node.type) {\r\n case \"paragraph\":\r\n return childrenText + \"\\n\";\r\n case \"heading\":\r\n return childrenText + \"\\n\\n\";\r\n case \"listItem\":\r\n return \"β€’ \" + childrenText + \"\\n\";\r\n case \"blockquote\":\r\n return \"> \" + childrenText + \"\\n\";\r\n default:\r\n return childrenText;\r\n }\r\n }\r\n \r\n // For nodes without content (like empty paragraphs without content array)\r\n if (node.type === \"paragraph\") {\r\n return \"\\n\";\r\n }\r\n\r\n return \"\";\r\n };\r\n\r\n const plainText = extractText(parsed);\r\n return plainText.trim();\r\n } catch (error) {\r\n // If parsing fails, return original string\r\n return description;\r\n }\r\n}\r\n\r\n// MCP Server setup\r\nconst server = new Server(\r\n {\r\n name: 'mg-tickets-mcp-bridge',\r\n version: '2.0.0',\r\n },\r\n {\r\n capabilities: {\r\n tools: {},\r\n resources: {},\r\n },\r\n }\r\n);\r\n\r\n// Available tools definition\r\nconst TOOLS = [\r\n {\r\n name: 'get-tickets',\r\n description: 'Get tickets with optional filtering by status, priority, project, customer, or search query',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n status: { type: 'string', enum: ['open', 'in_progress', 'review', 'resolved', 'closed', 'backlog'] },\r\n priority: { type: 'string', enum: ['low', 'medium', 'high', 'critical'] },\r\n projectId: { type: 'string' },\r\n customerId: { type: 'string' },\r\n q: { type: 'string', description: 'Search query for title or description' },\r\n pageSize: { type: 'number', default: 20, maximum: 100 }\r\n },\r\n required: []\r\n }\r\n },\r\n {\r\n name: 'get-ticket-by-id',\r\n description: 'Get a specific ticket by its ID, including all attachments, comments, and images. Images from ticket attachments and comment attachments are automatically downloaded and returned as base64-encoded content that can be analyzed by AI.',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n id: { type: 'string', description: 'Ticket ID' }\r\n },\r\n required: ['id']\r\n }\r\n },\r\n {\r\n name: 'create-ticket',\r\n description: 'Create a new ticket',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n title: { type: 'string', description: 'Ticket title' },\r\n description: { type: 'string' },\r\n status: { type: 'string', enum: ['open', 'in_progress', 'review', 'resolved', 'closed', 'backlog'], default: 'open' },\r\n priority: { type: 'string', enum: ['low', 'medium', 'high', 'critical'], default: 'medium' },\r\n type: { type: 'string', enum: ['task', 'bug', 'feature', 'support', 'question', 'improvement'], default: 'task' },\r\n projectId: { type: 'string' },\r\n customerId: { type: 'string' }\r\n },\r\n required: ['title']\r\n }\r\n },\r\n {\r\n name: 'get-customers',\r\n description: 'Get customers with optional search',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n q: { type: 'string', description: 'Search query for customer name or email' },\r\n pageSize: { type: 'number', default: 20, maximum: 100 }\r\n },\r\n required: []\r\n }\r\n },\r\n {\r\n name: 'create-customer',\r\n description: 'Create a new customer',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n name: { type: 'string', description: 'Customer name' },\r\n email: { type: 'string' },\r\n website: { type: 'string' }\r\n },\r\n required: ['name']\r\n }\r\n },\r\n {\r\n name: 'get-projects',\r\n description: 'Get projects with optional filtering',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n customerId: { type: 'string', description: 'Filter by customer ID' },\r\n q: { type: 'string', description: 'Search query for project name' },\r\n pageSize: { type: 'number', default: 20, maximum: 100 }\r\n },\r\n required: []\r\n }\r\n },\r\n {\r\n name: 'create-project',\r\n description: 'Create a new project',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n name: { type: 'string', description: 'Project name' },\r\n description: { type: 'string' },\r\n customerId: { type: 'string' },\r\n status: { type: 'string', enum: ['active', 'on_hold', 'completed', 'cancelled'], default: 'active' }\r\n },\r\n required: ['name']\r\n }\r\n },\r\n // === NEW AI SESSION TOOLS ===\r\n {\r\n name: 'start-ai-session-smart',\r\n description: 'Start a new AI development session with automatic tracking (time estimate provided by Cursor AI)',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n ticketId: { type: 'string' },\r\n ticketUrl: { type: 'string', description: 'URL to the ticket' },\r\n cursorSessionId: { type: 'string', description: 'Cursor session identifier' },\r\n codebaseContext: { \r\n type: 'array', \r\n items: { type: 'string' },\r\n description: 'Relevant files for complexity analysis'\r\n },\r\n aiTimeEstimateMinutes: { \r\n type: 'number', \r\n description: 'AI estimate in minutes without AI assistance' \r\n },\r\n complexityScore: {\r\n type: 'number',\r\n minimum: 1,\r\n maximum: 10,\r\n description: 'Estimated complexity from 1-10'\r\n }\r\n },\r\n required: ['ticketId', 'aiTimeEstimateMinutes']\r\n }\r\n },\r\n {\r\n name: 'track-manual-follow-up',\r\n description: 'Track manual follow-up prompt by developer',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n aiSessionId: { type: 'string' },\r\n originalPrompt: { type: 'string' },\r\n aiResponse: { type: 'string' },\r\n developerFollowUp: { type: 'string' },\r\n followUpReason: { \r\n type: 'string', \r\n enum: ['incomplete_result', 'wrong_approach', 'needs_clarification', 'error_in_code']\r\n },\r\n outcome: {\r\n type: 'string',\r\n enum: ['success', 'partial_success', 'still_failed'],\r\n default: 'success'\r\n },\r\n timeSpentMinutes: { type: 'number' }\r\n },\r\n required: ['aiSessionId', 'originalPrompt', 'aiResponse', 'developerFollowUp', 'followUpReason']\r\n }\r\n },\r\n {\r\n name: 'get-session-context',\r\n description: 'Get current session context for follow-up continuity',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n aiSessionId: { type: 'string' },\r\n includeTicketData: { type: 'boolean', default: true },\r\n includeTodoProgress: { type: 'boolean', default: true },\r\n includeFollowUpHistory: { type: 'boolean', default: false }\r\n },\r\n required: ['aiSessionId']\r\n }\r\n },\r\n {\r\n name: 'sync-session-todos',\r\n description: 'Synchronize todo list with AI session (replace existing) or add new todos',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n aiSessionId: { type: 'string' },\r\n todos: {\r\n type: 'array',\r\n items: {\r\n type: 'object',\r\n properties: {\r\n todoId: { type: 'string', description: 'Optional external todo ID for tracking' },\r\n content: { type: 'string' },\r\n status: { type: 'string', enum: ['pending', 'in_progress', 'completed', 'cancelled'] },\r\n estimatedMinutes: { type: 'number' }\r\n },\r\n required: ['content', 'status']\r\n }\r\n },\r\n replaceAll: { \r\n type: 'boolean', \r\n default: true, \r\n description: 'If true, replace all existing todos. If false, add new todos to existing ones' \r\n }\r\n },\r\n required: ['aiSessionId', 'todos']\r\n }\r\n },\r\n {\r\n name: 'add-follow-up-todos',\r\n description: 'Add new todos from follow-up (without replacing existing ones)',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n aiSessionId: { type: 'string' },\r\n newTodos: {\r\n type: 'array',\r\n items: {\r\n type: 'object',\r\n properties: {\r\n content: { type: 'string' },\r\n status: { type: 'string', enum: ['pending', 'in_progress'], default: 'pending' },\r\n estimatedMinutes: { type: 'number' },\r\n addedInFollowUp: { type: 'boolean', default: true }\r\n },\r\n required: ['content']\r\n }\r\n },\r\n followUpReason: { \r\n type: 'string', \r\n description: 'Why were these todos added in follow-up' \r\n }\r\n },\r\n required: ['aiSessionId', 'newTodos']\r\n }\r\n },\r\n {\r\n name: 'update-session-status',\r\n description: 'Update AI session status and completion info',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n aiSessionId: { type: 'string' },\r\n status: {\r\n type: 'string',\r\n enum: ['started', 'in_progress', 'paused', 'completed', 'failed']\r\n },\r\n actualTimeMinutes: { type: 'number' },\r\n completionNotes: { type: 'string' }\r\n },\r\n required: ['aiSessionId', 'status']\r\n }\r\n },\r\n {\r\n name: 'get-completion-context',\r\n description: 'Get all context needed for Cursor AI to generate customer response',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n aiSessionId: { type: 'string' },\r\n includeFollowUps: { type: 'boolean', default: true },\r\n includeTimeMetrics: { type: 'boolean', default: true },\r\n includeTodos: { type: 'boolean', default: true }\r\n },\r\n required: ['aiSessionId']\r\n }\r\n },\r\n {\r\n name: 'save-customer-response',\r\n description: 'Save customer response generated by Cursor AI',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n aiSessionId: { type: 'string' },\r\n customerResponse: { type: 'string', description: 'Customer response generated by Cursor AI' },\r\n responseType: { \r\n type: 'string', \r\n enum: ['completion', 'progress_update', 'needs_clarification'],\r\n default: 'completion'\r\n }\r\n },\r\n required: ['aiSessionId', 'customerResponse']\r\n }\r\n },\r\n {\r\n name: 'complete-ai-session',\r\n description: 'Complete AI session with work summary - time calculated automatically',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n aiSessionId: { type: 'string' },\r\n workCompleted: {\r\n type: 'array',\r\n items: { type: 'string' },\r\n description: 'List of completed tasks/todos in English'\r\n },\r\n technicalSummary: {\r\n type: 'string',\r\n description: 'Technical summary of work done in English'\r\n },\r\n invoiceDescription: {\r\n type: 'string',\r\n description: 'Short invoice-friendly description in the language of the ticket (2-3 sentences max, suitable for billing)'\r\n },\r\n efficiencyNotes: { type: 'string' }\r\n },\r\n required: ['aiSessionId', 'workCompleted']\r\n }\r\n }\r\n];\r\n\r\n// Available resources\r\nconst RESOURCES = [\r\n {\r\n uri: 'tickets://recent',\r\n name: 'Recent Tickets',\r\n description: 'Most recently created tickets',\r\n mimeType: 'application/json'\r\n },\r\n {\r\n uri: 'customers://all',\r\n name: 'All Customers', \r\n description: 'Complete customer directory',\r\n mimeType: 'application/json'\r\n },\r\n {\r\n uri: 'projects://active',\r\n name: 'Active Projects',\r\n description: 'Currently active projects',\r\n mimeType: 'application/json'\r\n }\r\n];\r\n\r\n// List tools handler\r\nserver.setRequestHandler(ListToolsRequestSchema, async () => {\r\n return { tools: TOOLS };\r\n});\r\n\r\n// List resources handler\r\nserver.setRequestHandler(ListResourcesRequestSchema, async () => {\r\n return { resources: RESOURCES };\r\n});\r\n\r\n// Tool execution handler\r\nserver.setRequestHandler(CallToolRequestSchema, async (request) => {\r\n if (!authContext) {\r\n return {\r\n content: [{ type: 'text', text: 'Error: Not authenticated. API key validation failed.' }],\r\n };\r\n }\r\n\r\n const { name, arguments: args } = request.params;\r\n console.error(`πŸ› οΈ Executing tool: ${name} for team ${authContext.teamId}`);\r\n \r\n try {\r\n switch (name) {\r\n case 'get-tickets': {\r\n const { status, priority, projectId, customerId, q, pageSize = 20 } = args as any;\r\n \r\n // Get all accessible team IDs (own team + child teams)\r\n const teamIds = await getAccessibleTeamIds(authContext.teamId);\r\n \r\n let query = supabase\r\n .from('tickets')\r\n .select(`\r\n id,\r\n ticket_number,\r\n title,\r\n description,\r\n status,\r\n priority,\r\n type,\r\n created_at,\r\n project_id,\r\n customer_id,\r\n projects:project_id(id, name),\r\n customers:customer_id(id, name)\r\n `)\r\n .in('team_id', teamIds)\r\n .limit(Math.min(pageSize, 100));\r\n \r\n if (status) query = query.eq('status', status);\r\n if (priority) query = query.eq('priority', priority);\r\n if (projectId) query = query.eq('project_id', projectId);\r\n if (customerId) query = query.eq('customer_id', customerId);\r\n if (q) query = query.or(`title.ilike.%${q}%,description.ilike.%${q}%`);\r\n \r\n const { data, error } = await query.order('created_at', { ascending: false });\r\n \r\n if (error) throw error;\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `Found ${data?.length || 0} tickets:\\n\\n${data?.map(ticket => \r\n `**${ticket.ticket_number}**: ${ticket.title}\\n` +\r\n `Status: ${ticket.status} | Priority: ${ticket.priority}\\n` +\r\n `${(ticket.projects as any)?.name ? `Project: ${(ticket.projects as any).name}\\n` : ''}` +\r\n `${(ticket.customers as any)?.name ? `Customer: ${(ticket.customers as any).name}\\n` : ''}` +\r\n `Created: ${new Date(ticket.created_at).toLocaleDateString()}\\n`\r\n ).join('\\n') || 'No tickets found.'}`\r\n }]\r\n };\r\n }\r\n \r\n case 'get-ticket-by-id': {\r\n const { id } = args as any;\r\n \r\n // Get ticket data - include tickets from provider's team AND child customer teams\r\n const { data: ticketData, error } = await supabase\r\n .from('tickets')\r\n .select(`\r\n *,\r\n projects:project_id(id, name),\r\n customers:customer_id(id, name),\r\n assignee:assignee_id(id, full_name, email),\r\n requester:requester_id(id, full_name, email)\r\n `)\r\n .eq('id', id)\r\n .single();\r\n \r\n if (error) throw error;\r\n \r\n // Check if user has access to this ticket (own team or child team)\r\n const { data: teamAccess } = await supabase\r\n .from('teams')\r\n .select('id')\r\n .or(`id.eq.${ticketData.team_id},parent_team_id.eq.${authContext.teamId}`)\r\n .eq('id', ticketData.team_id)\r\n .single();\r\n \r\n if (!teamAccess && ticketData.team_id !== authContext.teamId) {\r\n throw new Error('Access denied: You do not have permission to view this ticket');\r\n }\r\n \r\n const data = ticketData;\r\n \r\n // Get ticket attachments (no team filter needed - already validated ticket access)\r\n const { data: attachments, error: attachmentsError } = await supabase\r\n .from('ticket_attachments')\r\n .select(`\r\n id,\r\n file_name,\r\n file_size,\r\n mime_type,\r\n storage_key,\r\n created_at,\r\n users:user_id(id, full_name)\r\n `)\r\n .eq('ticket_id', id)\r\n .order('created_at', { ascending: true });\r\n \r\n if (attachmentsError) {\r\n console.error('Error fetching attachments:', attachmentsError);\r\n }\r\n \r\n // Get ticket comments\r\n const { data: comments, error: commentsError } = await supabase\r\n .from('ticket_comments')\r\n .select(`\r\n id,\r\n content,\r\n created_at,\r\n users:user_id(id, full_name)\r\n `)\r\n .eq('ticket_id', id)\r\n .order('created_at', { ascending: true });\r\n \r\n if (commentsError) {\r\n console.error('Error fetching comments:', commentsError);\r\n }\r\n \r\n // Get comment attachments (no team filter needed - already validated ticket access)\r\n const commentIds = comments?.map(c => c.id) || [];\r\n let commentAttachments: any[] = [];\r\n \r\n if (commentIds.length > 0) {\r\n const { data: commAttachments, error: commAttachmentsError } = await supabase\r\n .from('ticket_comment_attachments')\r\n .select(`\r\n id,\r\n comment_id,\r\n file_name,\r\n file_size,\r\n mime_type,\r\n storage_key,\r\n created_at\r\n `)\r\n .in('comment_id', commentIds);\r\n \r\n if (commAttachmentsError) {\r\n console.error('Error fetching comment attachments:', commAttachmentsError);\r\n } else {\r\n commentAttachments = commAttachments || [];\r\n }\r\n }\r\n \r\n // Build content array starting with text\r\n const content: any[] = [{\r\n type: 'text',\r\n text: `**Ticket Details:**\\n\\n` +\r\n `**${data.ticket_number}**: ${data.title}\\n` +\r\n `Status: ${data.status}\\n` +\r\n `Priority: ${data.priority}\\n` +\r\n `Type: ${data.type}\\n` +\r\n `${data.description ? `Description: ${data.description}\\n` : ''}` +\r\n `${(data.projects as any)?.name ? `Project: ${(data.projects as any).name}\\n` : ''}` +\r\n `${(data.customers as any)?.name ? `Customer: ${(data.customers as any).name}\\n` : ''}` +\r\n `${data.assignee?.full_name ? `Assignee: ${data.assignee.full_name}\\n` : ''}` +\r\n `Requester: ${data.requester?.full_name || 'Unknown'}\\n` +\r\n `Created: ${new Date(data.created_at).toLocaleDateString()}\\n` +\r\n `${attachments && attachments.length > 0 ? `\\nπŸ“Ž Attachments: ${attachments.length}\\n` : ''}` +\r\n `${comments && comments.length > 0 ? `πŸ’¬ Comments: ${comments.length}\\n` : ''}`\r\n }];\r\n \r\n // Process ticket attachments - download images\r\n if (attachments && attachments.length > 0) {\r\n console.error(`πŸ“Ž Processing ${attachments.length} ticket attachments...`);\r\n \r\n for (const attachment of attachments) {\r\n if (isImageFile(attachment.mime_type)) {\r\n console.error(`πŸ–ΌοΈ Downloading image: ${attachment.file_name}`);\r\n const base64Data = await downloadImageAsBase64(attachment.storage_key);\r\n \r\n if (base64Data) {\r\n content.push({\r\n type: 'image',\r\n data: base64Data,\r\n mimeType: attachment.mime_type\r\n });\r\n \r\n // Add text context about the image\r\n content.push({\r\n type: 'text',\r\n text: `\\nπŸ“Έ **Image from ticket**: ${attachment.file_name} (${Math.round(attachment.file_size / 1024)}KB, uploaded by ${(attachment.users as any)?.full_name || 'Unknown'} on ${new Date(attachment.created_at).toLocaleDateString()})\\n`\r\n });\r\n }\r\n }\r\n }\r\n }\r\n \r\n // Process comment attachments - download images\r\n if (commentAttachments.length > 0) {\r\n console.error(`πŸ“Ž Processing ${commentAttachments.length} comment attachments...`);\r\n \r\n for (const attachment of commentAttachments) {\r\n if (isImageFile(attachment.mime_type)) {\r\n console.error(`πŸ–ΌοΈ Downloading comment image: ${attachment.file_name}`);\r\n const base64Data = await downloadImageAsBase64(attachment.storage_key);\r\n \r\n if (base64Data) {\r\n // Find the comment this attachment belongs to\r\n const comment = comments?.find(c => c.id === attachment.comment_id);\r\n \r\n content.push({\r\n type: 'image',\r\n data: base64Data,\r\n mimeType: attachment.mime_type\r\n });\r\n \r\n // Add text context about the image\r\n content.push({\r\n type: 'text',\r\n text: `\\nπŸ“Έ **Image from comment** by ${(comment?.users as any)?.full_name || 'Unknown'} on ${new Date(attachment.created_at).toLocaleDateString()}: ${attachment.file_name} (${Math.round(attachment.file_size / 1024)}KB)\\n` +\r\n (comment?.content ? `Comment text: \"${comment.content.substring(0, 100)}${comment.content.length > 100 ? '...' : ''}\"\\n` : '')\r\n });\r\n }\r\n }\r\n }\r\n }\r\n \r\n console.error(`βœ… Returning ticket with ${content.filter(c => c.type === 'image').length} images`);\r\n \r\n return { content };\r\n }\r\n \r\n case 'create-ticket': {\r\n const { title, description, status = 'open', priority = 'medium', type = 'task', projectId, customerId } = args as any;\r\n \r\n // Generate ticket number\r\n const year = new Date().getFullYear();\r\n const { count } = await supabase\r\n .from('tickets')\r\n .select('*', { count: 'exact', head: true })\r\n .eq('team_id', authContext.teamId);\r\n \r\n const ticketNumber = `${year}-${String((count || 0) + 1).padStart(3, '0')}`;\r\n \r\n const { data, error } = await supabase\r\n .from('tickets')\r\n .insert({\r\n team_id: authContext.teamId,\r\n ticket_number: ticketNumber,\r\n title,\r\n description,\r\n status,\r\n priority,\r\n type,\r\n project_id: projectId || null,\r\n customer_id: customerId || null,\r\n requester_id: authContext.userId\r\n })\r\n .select()\r\n .single();\r\n \r\n if (error) throw error;\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `βœ… **Ticket Created Successfully!**\\n\\n` +\r\n `Ticket Number: **${ticketNumber}**\\n` +\r\n `Title: ${title}\\n` +\r\n `Status: ${status}\\n` +\r\n `Priority: ${priority}\\n` +\r\n `Type: ${type}\\n`\r\n }]\r\n };\r\n }\r\n \r\n case 'get-customers': {\r\n const { q, pageSize = 20 } = args as any;\r\n \r\n // Get all accessible team IDs (own team + child teams)\r\n const teamIds = await getAccessibleTeamIds(authContext.teamId);\r\n \r\n let query = supabase\r\n .from('customers')\r\n .select('id, name, email, website, created_at')\r\n .in('team_id', teamIds)\r\n .limit(Math.min(pageSize, 100));\r\n \r\n if (q) query = query.or(`name.ilike.%${q}%,email.ilike.%${q}%`);\r\n \r\n const { data, error } = await query.order('name');\r\n \r\n if (error) throw error;\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `Found ${data?.length || 0} customers:\\n\\n${data?.map(customer => \r\n `**${customer.name}**\\n` +\r\n `${customer.email ? `Email: ${customer.email}\\n` : ''}` +\r\n `${customer.website ? `Website: ${customer.website}\\n` : ''}` +\r\n `Created: ${new Date(customer.created_at).toLocaleDateString()}\\n`\r\n ).join('\\n') || 'No customers found.'}`\r\n }]\r\n };\r\n }\r\n \r\n case 'create-customer': {\r\n const { name, email, website } = args as any;\r\n \r\n const { data, error } = await supabase\r\n .from('customers')\r\n .insert({\r\n team_id: authContext.teamId,\r\n name,\r\n email: email || null,\r\n website: website || null,\r\n user_id: authContext.userId\r\n })\r\n .select()\r\n .single();\r\n \r\n if (error) throw error;\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `βœ… **Customer Created Successfully!**\\n\\n` +\r\n `Name: ${name}\\n` +\r\n `${email ? `Email: ${email}\\n` : ''}` +\r\n `${website ? `Website: ${website}\\n` : ''}`\r\n }]\r\n };\r\n }\r\n \r\n case 'get-projects': {\r\n const { customerId, q, pageSize = 20 } = args as any;\r\n \r\n // Get all accessible team IDs (own team + child teams)\r\n const teamIds = await getAccessibleTeamIds(authContext.teamId);\r\n \r\n let query = supabase\r\n .from('projects')\r\n .select(`\r\n id,\r\n name,\r\n description,\r\n customer_id,\r\n status,\r\n created_at,\r\n customers:customer_id(id, name)\r\n `)\r\n .in('team_id', teamIds)\r\n .limit(Math.min(pageSize, 100));\r\n \r\n if (customerId) query = query.eq('customer_id', customerId);\r\n if (q) query = query.ilike('name', `%${q}%`);\r\n \r\n const { data, error } = await query.order('name');\r\n \r\n if (error) throw error;\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `Found ${data?.length || 0} projects:\\n\\n${data?.map(project => \r\n `**${project.name}**\\n` +\r\n `Status: ${project.status}\\n` +\r\n `${project.description ? `Description: ${project.description}\\n` : ''}` +\r\n `${(project.customers as any)?.name ? `Customer: ${(project.customers as any).name}\\n` : ''}` +\r\n `Created: ${new Date(project.created_at).toLocaleDateString()}\\n`\r\n ).join('\\n') || 'No projects found.'}`\r\n }]\r\n };\r\n }\r\n \r\n case 'create-project': {\r\n const { name, description, customerId, status = 'active' } = args as any;\r\n \r\n const { data, error } = await supabase\r\n .from('projects')\r\n .insert({\r\n team_id: authContext.teamId,\r\n name,\r\n description: description || null,\r\n customer_id: customerId || null,\r\n status,\r\n user_id: authContext.userId\r\n })\r\n .select()\r\n .single();\r\n \r\n if (error) throw error;\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `βœ… **Project Created Successfully!**\\n\\n` +\r\n `Name: ${name}\\n` +\r\n `Status: ${status}\\n` +\r\n `${description ? `Description: ${description}\\n` : ''}`\r\n }]\r\n };\r\n }\r\n \r\n // === AI SESSION TOOLS ===\r\n case 'start-ai-session-smart': {\r\n const { ticketId, ticketUrl, cursorSessionId, codebaseContext, aiTimeEstimateMinutes, complexityScore } = args as any;\r\n \r\n // Store the time estimate as provided by Cursor AI (already smart-adjusted)\r\n const sessionStartTime = new Date();\r\n const { data: sessionData, error } = await supabase\r\n .from('ai_sessions')\r\n .insert({\r\n ticket_id: ticketId,\r\n provider_user_id: authContext.userId,\r\n team_id: authContext.teamId,\r\n cursor_session_id: cursorSessionId || null,\r\n ai_time_estimate_minutes: aiTimeEstimateMinutes, // Use Cursor's smart estimate\r\n complexity_score: complexityScore || null,\r\n status: 'in_progress' // Start actively tracking time\r\n })\r\n .select('id, ticket_id, cursor_session_id, created_at')\r\n .single();\r\n \r\n if (error) throw error;\r\n \r\n // START TIME TRACKING: Log session start\r\n await supabase\r\n .from('ai_time_logs')\r\n .insert({\r\n ai_session_id: sessionData.id,\r\n activity_type: 'thinking',\r\n description: 'AI session started - analysis and planning phase',\r\n duration_seconds: 0,\r\n productivity_score: 8,\r\n logged_at: sessionStartTime\r\n });\r\n \r\n // Generate a readable session ID for UI\r\n const sessionId = `ai-sess-${sessionData.id.substring(0, 8)}`;\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `πŸš€ **AI Session Started Successfully!**\\n\\n` +\r\n `πŸ†” Session ID: **${sessionId}**\\n` +\r\n `🎫 Ticket: ${ticketId}\\n` +\r\n `⏱️ Time Estimate: ${aiTimeEstimateMinutes} minutes\\n` +\r\n `${complexityScore ? `🎯 Complexity: ${complexityScore}/10\\n` : ''}` +\r\n `⏱️ **Automatic Time Tracking Started**\\n` +\r\n `${cursorSessionId ? `πŸ”— Cursor Session: ${cursorSessionId}\\n` : ''}` +\r\n `πŸ“… Started: ${sessionStartTime.toLocaleString()}\\n\\n` +\r\n `βœ… Session initialized - ready for development!\\n` +\r\n `πŸ“ Timetrack entry will be created when you complete the session.`\r\n }]\r\n };\r\n }\r\n \r\n case 'track-manual-follow-up': {\r\n const { aiSessionId, originalPrompt, aiResponse, developerFollowUp, followUpReason, outcome = 'success', timeSpentMinutes } = args as any;\r\n \r\n // Extract actual session UUID from readable ID\r\n const sessionUuid = aiSessionId.replace('ai-sess-', '');\r\n \r\n // Find the full session ID and current session data\r\n // Find the session - get all sessions for accessible teams and filter in JS\r\n const teamIds = await getAccessibleTeamIds(authContext.teamId);\r\n const { data: allSessions, error: sessionError } = await supabase\r\n .from('ai_sessions')\r\n .select('id, status, created_at, ai_time_estimate_minutes')\r\n .in('team_id', teamIds);\r\n \r\n if (sessionError) {\r\n throw new Error(`Database error: ${sessionError.message}`);\r\n }\r\n \r\n // Find session with matching prefix\r\n const session = allSessions?.find(s => s.id.toString().startsWith(sessionUuid));\r\n if (!session) {\r\n throw new Error(`Session not found: ${aiSessionId} (searched ${allSessions?.length || 0} sessions)`);\r\n }\r\n \r\n const followUpTime = new Date();\r\n \r\n // RESTART SESSION: Set status back to in_progress for continued tracking\r\n await supabase\r\n .from('ai_sessions')\r\n .update({\r\n status: 'in_progress', // Restart active tracking\r\n // Don't update completed_at - session continues\r\n })\r\n .eq('id', session.id);\r\n \r\n // LOG FOLLOW-UP: Track the manual follow-up interaction\r\n const { data, error } = await supabase\r\n .from('manual_follow_ups')\r\n .insert({\r\n ai_session_id: session.id,\r\n developer_id: authContext.userId,\r\n team_id: authContext.teamId,\r\n original_prompt: originalPrompt,\r\n ai_response: aiResponse,\r\n follow_up_prompt: developerFollowUp,\r\n follow_up_reason: followUpReason,\r\n outcome: outcome,\r\n time_spent_minutes: timeSpentMinutes || null,\r\n resolved_at: outcome === 'success' ? new Date().toISOString() : null\r\n })\r\n .select()\r\n .single();\r\n \r\n if (error) throw error;\r\n \r\n // RESTART TIME TRACKING: Log follow-up activity start\r\n await supabase\r\n .from('ai_time_logs')\r\n .insert({\r\n ai_session_id: session.id,\r\n activity_type: 'debugging', // Follow-ups are typically debugging/fixing\r\n description: `Follow-up: ${followUpReason.replace('_', ' ')} - ${outcome}`,\r\n duration_seconds: (timeSpentMinutes || 0) * 60,\r\n productivity_score: outcome === 'success' ? 9 : outcome === 'partial_success' ? 6 : 4,\r\n logged_at: followUpTime\r\n });\r\n \r\n // CALCULATE CURRENT EFFICIENCY: Compare time spent vs estimated\r\n const sessionStartTime = new Date(session.created_at);\r\n const totalMinutesElapsed = Math.round((followUpTime.getTime() - sessionStartTime.getTime()) / 60000);\r\n const estimatedMinutes = session.ai_time_estimate_minutes || 60;\r\n const currentEfficiency = totalMinutesElapsed > 0 ? (totalMinutesElapsed / estimatedMinutes) : 1;\r\n \r\n // UPDATE EFFICIENCY TRACKING: Update running efficiency score\r\n await supabase\r\n .from('ai_sessions')\r\n .update({\r\n efficiency_score: currentEfficiency.toFixed(2),\r\n actual_time_minutes: totalMinutesElapsed\r\n })\r\n .eq('id', session.id);\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `πŸ”„ **Follow-up Tracked & Session Restarted!**\\n\\n` +\r\n `πŸ†” Session: ${aiSessionId} (back to active)\\n` +\r\n `πŸ” Reason: ${followUpReason.replace('_', ' ')}\\n` +\r\n `βœ… Outcome: ${outcome}\\n` +\r\n `${timeSpentMinutes ? `⏱️ Time spent: ${timeSpentMinutes} minutes\\n` : ''}` +\r\n `πŸ“Š **Current Progress:**\\n` +\r\n ` β€’ Total time: ${totalMinutesElapsed} minutes\\n` +\r\n ` β€’ Estimated: ${estimatedMinutes} minutes\\n` +\r\n ` β€’ Efficiency: ${currentEfficiency < 1 ? 'πŸš€ ' : currentEfficiency > 1.5 ? '⚠️ ' : '⏱️ '}${(currentEfficiency * 100).toFixed(0)}%\\n` +\r\n `\\n⚑ **Time tracking resumed** - continue with confidence!`\r\n }]\r\n };\r\n }\r\n \r\n case 'get-session-context': {\r\n const { aiSessionId, includeTicketData = true, includeTodoProgress = true, includeFollowUpHistory = false } = args as any;\r\n \r\n // Extract actual session UUID from readable ID\r\n const sessionUuid = aiSessionId.replace('ai-sess-', '');\r\n \r\n // Get session data with related info - find by JavaScript filtering\r\n const teamIds = await getAccessibleTeamIds(authContext.teamId);\r\n const { data: allSessions, error: sessionError } = await supabase\r\n .from('ai_sessions')\r\n .select(`\r\n id,\r\n ticket_id,\r\n status,\r\n ai_time_estimate_minutes,\r\n actual_time_minutes,\r\n complexity_score,\r\n created_at,\r\n cursor_session_id\r\n `)\r\n .in('team_id', teamIds);\r\n \r\n if (sessionError) {\r\n throw new Error(`Database error: ${sessionError.message}`);\r\n }\r\n \r\n // Find session with matching prefix\r\n const session = allSessions?.find(s => s.id.toString().startsWith(sessionUuid));\r\n \r\n if (sessionError || !session) {\r\n throw new Error(`Session not found: ${aiSessionId}`);\r\n }\r\n \r\n let context: any = {\r\n sessionId: aiSessionId,\r\n status: session.status,\r\n timeEstimate: session.ai_time_estimate_minutes,\r\n actualTime: session.actual_time_minutes,\r\n complexity: session.complexity_score,\r\n createdAt: session.created_at\r\n };\r\n \r\n // Include ticket data if requested\r\n if (includeTicketData) {\r\n const { data: ticket } = await supabase\r\n .from('tickets')\r\n .select('id, ticket_number, title, description, status, priority, type')\r\n .eq('id', session.ticket_id)\r\n .single();\r\n \r\n context.ticketData = ticket;\r\n }\r\n \r\n // Include todo progress if requested\r\n if (includeTodoProgress) {\r\n const { data: todos } = await supabase\r\n .from('ai_todos')\r\n .select('id, content, status, estimated_minutes, actual_minutes')\r\n .eq('ai_session_id', session.id)\r\n .order('sequence_order');\r\n \r\n context.todos = todos || [];\r\n context.todoProgress = {\r\n total: todos?.length || 0,\r\n completed: todos?.filter(t => t.status === 'completed').length || 0,\r\n inProgress: todos?.filter(t => t.status === 'in_progress').length || 0\r\n };\r\n }\r\n \r\n // Include follow-up history if requested\r\n if (includeFollowUpHistory) {\r\n const { data: followUps } = await supabase\r\n .from('manual_follow_ups')\r\n .select('follow_up_reason, outcome, time_spent_minutes, created_at')\r\n .eq('ai_session_id', session.id)\r\n .order('created_at');\r\n \r\n context.followUpHistory = followUps || [];\r\n }\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `🎯 **Session Context Retrieved**\\n\\n` +\r\n `Session: ${aiSessionId}\\n` +\r\n `Status: ${session.status}\\n` +\r\n `${context.ticketData ? `Ticket: ${context.ticketData.ticket_number} - ${context.ticketData.title}\\n` : ''}` +\r\n `${context.todoProgress ? `Todo Progress: ${context.todoProgress.completed}/${context.todoProgress.total} completed\\n` : ''}` +\r\n `${context.followUpHistory ? `Follow-ups: ${context.followUpHistory.length}\\n` : ''}` +\r\n `\\nπŸ“‹ Full context preserved for seamless continuation!`\r\n }]\r\n };\r\n }\r\n \r\n case 'sync-session-todos': {\r\n const { aiSessionId, todos, replaceAll = true } = args as any;\r\n \r\n // Extract actual session UUID from readable ID\r\n const sessionUuid = aiSessionId.replace('ai-sess-', '');\r\n \r\n // Find the session - get all sessions for accessible teams and filter in JS\r\n const teamIds = await getAccessibleTeamIds(authContext.teamId);\r\n const { data: allSessions, error: sessionError } = await supabase\r\n .from('ai_sessions')\r\n .select('id')\r\n .in('team_id', teamIds);\r\n \r\n if (sessionError) {\r\n throw new Error(`Database error: ${sessionError.message}`);\r\n }\r\n \r\n // Find session with matching prefix\r\n const session = allSessions?.find(s => s.id.toString().startsWith(sessionUuid));\r\n if (!session) {\r\n throw new Error(`Session not found: ${aiSessionId} (searched ${allSessions?.length || 0} sessions)`);\r\n }\r\n \r\n // If replacing all, clear existing todos first\r\n if (replaceAll) {\r\n await supabase\r\n .from('ai_todos')\r\n .delete()\r\n .eq('ai_session_id', session.id);\r\n }\r\n \r\n if (todos && todos.length > 0) {\r\n // Get current max sequence order if adding to existing\r\n let startSequence = 0;\r\n if (!replaceAll) {\r\n const { data: maxTodo } = await supabase\r\n .from('ai_todos')\r\n .select('sequence_order')\r\n .eq('ai_session_id', session.id)\r\n .order('sequence_order', { ascending: false })\r\n .limit(1)\r\n .single();\r\n startSequence = (maxTodo?.sequence_order || 0) + 1;\r\n }\r\n \r\n const todoInserts = todos.map((todo: any, index: number) => ({\r\n ai_session_id: session.id,\r\n content: todo.content,\r\n status: todo.status,\r\n cursor_todo_id: todo.todoId || null,\r\n estimated_minutes: todo.estimatedMinutes || null,\r\n sequence_order: startSequence + index\r\n }));\r\n \r\n const { error: insertError } = await supabase\r\n .from('ai_todos')\r\n .insert(todoInserts);\r\n \r\n if (insertError) throw insertError;\r\n }\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `βœ… **Todos ${replaceAll ? 'Synced' : 'Added'} Successfully!**\\n\\n` +\r\n `Session: ${aiSessionId}\\n` +\r\n `${replaceAll ? 'Synced' : 'Added'} ${todos?.length || 0} todos\\n` +\r\n `${replaceAll ? '' : 'βž• Added to existing todo list\\n'}` +\r\n `\\nπŸ“ Todo list updated and tracked for progress monitoring!`\r\n }]\r\n };\r\n }\r\n \r\n case 'add-follow-up-todos': {\r\n const { aiSessionId, newTodos, followUpReason } = args as any;\r\n \r\n // Extract actual session UUID from readable ID\r\n const sessionUuid = aiSessionId.replace('ai-sess-', '');\r\n \r\n // Find the session - get all sessions for accessible teams and filter in JS\r\n const teamIds = await getAccessibleTeamIds(authContext.teamId);\r\n const { data: allSessions, error: sessionError } = await supabase\r\n .from('ai_sessions')\r\n .select('id')\r\n .in('team_id', teamIds);\r\n \r\n if (sessionError) {\r\n throw new Error(`Database error: ${sessionError.message}`);\r\n }\r\n \r\n // Find session with matching prefix\r\n const session = allSessions?.find(s => s.id.toString().startsWith(sessionUuid));\r\n if (!session) {\r\n throw new Error(`Session not found: ${aiSessionId} (searched ${allSessions?.length || 0} sessions)`);\r\n }\r\n \r\n if (newTodos && newTodos.length > 0) {\r\n // Get current max sequence order\r\n const { data: maxTodo } = await supabase\r\n .from('ai_todos')\r\n .select('sequence_order')\r\n .eq('ai_session_id', session.id)\r\n .order('sequence_order', { ascending: false })\r\n .limit(1)\r\n .single();\r\n const startSequence = (maxTodo?.sequence_order || 0) + 1;\r\n \r\n const todoInserts = newTodos.map((todo: any, index: number) => ({\r\n ai_session_id: session.id,\r\n content: `[Follow-up] ${todo.content}`,\r\n status: todo.status || 'pending',\r\n estimated_minutes: todo.estimatedMinutes || null,\r\n sequence_order: startSequence + index\r\n }));\r\n \r\n const { error: insertError } = await supabase\r\n .from('ai_todos')\r\n .insert(todoInserts);\r\n \r\n if (insertError) throw insertError;\r\n }\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `βœ… **Follow-up Todos Added Successfully!**\\n\\n` +\r\n `Session: ${aiSessionId}\\n` +\r\n `Added ${newTodos?.length || 0} new todos from follow-up\\n` +\r\n `${followUpReason ? `Reason: ${followUpReason}\\n` : ''}` +\r\n `\\nπŸ“ New tasks identified and added to existing workflow!`\r\n }]\r\n };\r\n }\r\n \r\n case 'update-session-status': {\r\n const { aiSessionId, status, actualTimeMinutes, completionNotes } = args as any;\r\n \r\n // Extract actual session UUID from readable ID\r\n const sessionUuid = aiSessionId.replace('ai-sess-', '');\r\n \r\n // Find and update the session\r\n // First find the session\r\n const teamIds = await getAccessibleTeamIds(authContext.teamId);\r\n const { data: allSessions, error: findError } = await supabase\r\n .from('ai_sessions')\r\n .select('id')\r\n .in('team_id', teamIds);\r\n \r\n if (findError) {\r\n throw new Error(`Database error: ${findError.message}`);\r\n }\r\n \r\n const foundSession = allSessions?.find(s => s.id.toString().startsWith(sessionUuid));\r\n if (!foundSession) {\r\n throw new Error(`Session not found: ${aiSessionId} (searched ${allSessions?.length || 0} sessions)`);\r\n }\r\n \r\n // Now update the found session\r\n const { data, error } = await supabase\r\n .from('ai_sessions')\r\n .update({\r\n status: status,\r\n actual_time_minutes: actualTimeMinutes || null,\r\n completed_at: status === 'completed' ? new Date().toISOString() : null\r\n })\r\n .eq('id', foundSession.id)\r\n .select()\r\n .single();\r\n \r\n if (error) throw error;\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `🎯 **Session Status Updated!**\\n\\n` +\r\n `Session: ${aiSessionId}\\n` +\r\n `Status: ${status}\\n` +\r\n `${actualTimeMinutes ? `Actual Time: ${actualTimeMinutes} minutes\\n` : ''}` +\r\n `${status === 'completed' ? `βœ… Session completed successfully!\\n` : ''}` +\r\n `${completionNotes ? `Notes: ${completionNotes}\\n` : ''}`\r\n }]\r\n };\r\n }\r\n \r\n case 'get-completion-context': {\r\n const { aiSessionId, includeFollowUps = true, includeTimeMetrics = true, includeTodos = true } = args as any;\r\n \r\n // Extract actual session UUID from readable ID\r\n const sessionUuid = aiSessionId.replace('ai-sess-', '');\r\n \r\n // Get session data with ticket info\r\n // Get comprehensive session context - find by JavaScript filtering\r\n const teamIds = await getAccessibleTeamIds(authContext.teamId);\r\n const { data: allSessions, error: sessionError } = await supabase\r\n .from('ai_sessions')\r\n .select(`\r\n id,\r\n ticket_id,\r\n ai_time_estimate_minutes,\r\n actual_time_minutes,\r\n efficiency_score,\r\n created_at,\r\n completed_at,\r\n status,\r\n complexity_score\r\n `)\r\n .in('team_id', teamIds);\r\n \r\n if (sessionError) {\r\n throw new Error(`Database error: ${sessionError.message}`);\r\n }\r\n \r\n // Find session with matching prefix\r\n const session = allSessions?.find(s => s.id.toString().startsWith(sessionUuid));\r\n if (!session) {\r\n throw new Error(`Session not found: ${aiSessionId} (searched ${allSessions?.length || 0} sessions)`);\r\n }\r\n \r\n // Get ticket data\r\n const { data: ticket, error: ticketError } = await supabase\r\n .from('tickets')\r\n .select('ticket_number, title, description, type, priority')\r\n .eq('id', session.ticket_id)\r\n .single();\r\n \r\n if (ticketError || !ticket) {\r\n throw new Error('Ticket not found for session');\r\n }\r\n \r\n let contextData: any = {\r\n session: {\r\n id: aiSessionId,\r\n status: session.status,\r\n complexity: session.complexity_score,\r\n createdAt: session.created_at,\r\n completedAt: session.completed_at\r\n },\r\n ticket: {\r\n number: ticket.ticket_number,\r\n title: ticket.title,\r\n description: ticket.description,\r\n type: ticket.type,\r\n priority: ticket.priority\r\n }\r\n };\r\n \r\n // Add time metrics if requested\r\n if (includeTimeMetrics) {\r\n const timeSaved = session.ai_time_estimate_minutes && session.actual_time_minutes \r\n ? Math.max(0, session.ai_time_estimate_minutes - session.actual_time_minutes)\r\n : null;\r\n \r\n contextData.timeMetrics = {\r\n estimatedMinutes: session.ai_time_estimate_minutes,\r\n actualMinutes: session.actual_time_minutes,\r\n timeSaved: timeSaved,\r\n efficiency: session.efficiency_score,\r\n sessionDuration: session.completed_at && session.created_at \r\n ? Math.round((new Date(session.completed_at).getTime() - new Date(session.created_at).getTime()) / 60000)\r\n : null\r\n };\r\n }\r\n \r\n // Add todos if requested\r\n if (includeTodos) {\r\n const { data: todos } = await supabase\r\n .from('ai_todos')\r\n .select('content, status, estimated_minutes, actual_minutes, completed_at')\r\n .eq('ai_session_id', session.id)\r\n .order('created_at', { ascending: true });\r\n \r\n contextData.todos = todos || [];\r\n }\r\n \r\n // Add follow-ups if requested\r\n if (includeFollowUps) {\r\n const { data: followUps } = await supabase\r\n .from('manual_follow_ups')\r\n .select('follow_up_reason, outcome, time_spent_minutes, created_at')\r\n .eq('ai_session_id', session.id)\r\n .order('created_at', { ascending: true });\r\n \r\n contextData.followUps = followUps || [];\r\n }\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `πŸ“‹ **Completion Context Retrieved!**\\n\\n` +\r\n `🎫 **Ticket:** ${ticket.ticket_number} - ${ticket.title}\\n` +\r\n `πŸ†” **Session:** ${aiSessionId} (${session.status})\\n` +\r\n `⏱️ **Time:** ${session.actual_time_minutes || 'N/A'}/${session.ai_time_estimate_minutes || 'N/A'} minutes\\n` +\r\n `πŸ“‹ **Todos:** ${contextData.todos?.filter((t: any) => t.status === 'completed').length || 0}/${contextData.todos?.length || 0} completed\\n` +\r\n `πŸ”„ **Follow-ups:** ${contextData.followUps?.length || 0}\\n\\n` +\r\n `βœ… **Full context ready for Cursor AI to generate customer response!**\\n\\n` +\r\n `**Context Data:**\\n\\`\\`\\`json\\n${JSON.stringify(contextData, null, 2)}\\`\\`\\``\r\n }]\r\n };\r\n }\r\n \r\n case 'save-customer-response': {\r\n const { aiSessionId, customerResponse, responseType = 'completion' } = args as any;\r\n \r\n // Extract actual session UUID from readable ID\r\n const sessionUuid = aiSessionId.replace('ai-sess-', '');\r\n \r\n // Get session to validate it exists - find by JavaScript filtering\r\n const teamIds = await getAccessibleTeamIds(authContext.teamId);\r\n const { data: allSessions, error: sessionError } = await supabase\r\n .from('ai_sessions')\r\n .select('id')\r\n .in('team_id', teamIds);\r\n \r\n if (sessionError) {\r\n throw new Error(`Database error: ${sessionError.message}`);\r\n }\r\n \r\n // Find session with matching prefix\r\n const session = allSessions?.find(s => s.id.toString().startsWith(sessionUuid));\r\n if (!session) {\r\n throw new Error(`Session not found: ${aiSessionId} (searched ${allSessions?.length || 0} sessions)`);\r\n }\r\n \r\n // Save the response in database\r\n const { data: responseData, error: responseError } = await supabase\r\n .from('ai_responses')\r\n .insert({\r\n ai_session_id: session.id,\r\n response_type: responseType,\r\n content: customerResponse,\r\n is_ready_for_customer: true,\r\n provider_approved: false // Needs manual approval\r\n })\r\n .select()\r\n .single();\r\n \r\n if (responseError) throw responseError;\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `πŸ’Ύ **Customer Response Saved!**\\n\\n` +\r\n `πŸ†” Session: ${aiSessionId}\\n` +\r\n `πŸ“ Response Type: ${responseType}\\n` +\r\n `πŸ“„ Length: ${customerResponse.length} characters\\n\\n` +\r\n `βœ… **Response ready for provider approval**\\n` +\r\n `πŸ” Provider can review in AI tab before sending to customer\\n\\n` +\r\n `**Preview:**\\n\\`\\`\\`\\n${customerResponse.substring(0, 200)}${customerResponse.length > 200 ? '...' : ''}\\`\\`\\``\r\n }]\r\n };\r\n }\r\n \r\n case 'complete-ai-session': {\r\n const { aiSessionId, workCompleted, technicalSummary, invoiceDescription, efficiencyNotes } = args as any;\r\n \r\n // Extract actual session UUID from readable ID\r\n const sessionUuid = aiSessionId.replace('ai-sess-', '');\r\n \r\n // Get session to calculate actual time spent - find by JavaScript filtering\r\n const teamIds = await getAccessibleTeamIds(authContext.teamId);\r\n const { data: allSessions, error: getSessionError } = await supabase\r\n .from('ai_sessions')\r\n .select('id, ticket_id, ai_time_estimate_minutes, created_at')\r\n .in('team_id', teamIds);\r\n \r\n if (getSessionError) {\r\n throw new Error(`Database error: ${getSessionError.message}`);\r\n }\r\n \r\n // Find session with matching prefix\r\n const existingSession = allSessions?.find(s => s.id.toString().startsWith(sessionUuid));\r\n if (!existingSession) {\r\n throw new Error(`Session not found: ${aiSessionId} (searched ${allSessions?.length || 0} sessions)`);\r\n }\r\n\r\n // Calculate actual time spent automatically\r\n const completionTime = new Date();\r\n const sessionStartTime = new Date(existingSession.created_at);\r\n const timeSpentMinutes = Math.round((completionTime.getTime() - sessionStartTime.getTime()) / 60000);\r\n \r\n // Update session to completed with calculated time\r\n const { data: session, error: sessionError } = await supabase\r\n .from('ai_sessions')\r\n .update({\r\n status: 'completed',\r\n actual_time_minutes: timeSpentMinutes,\r\n completed_at: completionTime.toISOString(),\r\n efficiency_score: null // Will be calculated\r\n })\r\n .eq('id', existingSession.id)\r\n .select('id, ticket_id, ai_time_estimate_minutes, created_at')\r\n .single();\r\n \r\n if (sessionError || !session) {\r\n throw new Error(`Failed to update session: ${aiSessionId}`);\r\n }\r\n \r\n // Calculate efficiency score\r\n const efficiencyScore = session.ai_time_estimate_minutes \r\n ? (timeSpentMinutes / session.ai_time_estimate_minutes)\r\n : 1.0;\r\n \r\n // Update efficiency score\r\n await supabase\r\n .from('ai_sessions')\r\n .update({ efficiency_score: efficiencyScore.toFixed(2) })\r\n .eq('id', session.id);\r\n \r\n // Get session duration\r\n const sessionDuration = Math.round(\r\n (completionTime.getTime() - new Date(session.created_at).getTime()) / 60000\r\n );\r\n \r\n // Create work summary\r\n const workSummary = `Completed ${workCompleted.length} tasks including: ${workCompleted.slice(0, 3).join(', ')}${workCompleted.length > 3 ? ' and more' : ''}.`;\r\n \r\n // Get ticket details for agenda event creation\r\n const { data: ticketInfo } = await supabase\r\n .from('tickets')\r\n .select('ticket_number, title, project_id')\r\n .eq('id', session.ticket_id)\r\n .single();\r\n \r\n // CREATE AGENDA EVENT: Now that work is complete, create the timetrack entry\r\n // Use invoice description if provided, otherwise create a technical one\r\n let completionDescription: string;\r\n \r\n if (invoiceDescription) {\r\n // Use the invoice-friendly description provided by Cursor (in ticket language)\r\n completionDescription = `${ticketInfo?.ticket_number || 'Ticket'}: ${invoiceDescription}`;\r\n } else {\r\n // Fallback: Create a technical work description\r\n const workDescription = workCompleted.map((task: string, index: number) => `${index + 1}. ${task}`).join('\\n');\r\n completionDescription = `${ticketInfo?.ticket_number || 'Ticket'}: ${technicalSummary || workSummary}\\n\\nCompleted work:\\n${workDescription}`;\r\n }\r\n \r\n // Create the agenda event with AI estimated time\r\n // Use the original AI estimate for billing, not the actual AI completion time\r\n const estimatedMinutes = session.ai_time_estimate_minutes || timeSpentMinutes;\r\n const sessionStart = new Date(session.created_at);\r\n const estimatedEnd = new Date(sessionStart.getTime() + estimatedMinutes * 60000);\r\n \r\n const { data: agendaEvent, error: agendaError } = await supabase\r\n .from('agenda_events')\r\n .insert({\r\n team_id: authContext.teamId,\r\n user_id: authContext.userId,\r\n title: ticketInfo?.title || 'Development Work',\r\n description: completionDescription,\r\n start_time: sessionStart.toISOString(),\r\n end_time: estimatedEnd.toISOString(),\r\n project_id: ticketInfo?.project_id || null,\r\n ticket_id: session.ticket_id,\r\n ai_session_id: session.id, // Use the actual UUID, not the readable ID\r\n type: 'work',\r\n status: 'draft', // Mark as draft for manual approval\r\n all_day: false,\r\n is_tracked: true,\r\n tracked_duration: estimatedMinutes * 60 // Use AI estimate in seconds for billing\r\n })\r\n .select('id')\r\n .single();\r\n \r\n if (agendaError) {\r\n console.error('⚠️ Failed to create agenda event:', agendaError);\r\n // Don't fail the completion if agenda event creation fails\r\n }\r\n \r\n let responseText = `πŸŽ‰ **AI Session Completed Successfully!**\\n\\n`;\r\n responseText += `πŸ†” Session: ${aiSessionId}\\n`;\r\n responseText += `πŸ“Š **Performance Summary:**\\n`;\r\n responseText += ` β€’ Tasks Completed: ${workCompleted.length}\\n`;\r\n responseText += ` β€’ Time Spent: ${timeSpentMinutes} minutes\\n`;\r\n responseText += ` β€’ Estimated Time: ${session.ai_time_estimate_minutes || 'N/A'} minutes\\n`;\r\n responseText += ` β€’ Efficiency: ${efficiencyScore < 1 ? 'πŸš€' : efficiencyScore > 1.5 ? '⚠️' : '⏱️'} ${(efficiencyScore * 100).toFixed(0)}%\\n`;\r\n responseText += ` β€’ Session Duration: ${sessionDuration} minutes\\n\\n`;\r\n \r\n responseText += `βœ… **Work Completed:**\\n`;\r\n workCompleted.forEach((task: string, index: number) => {\r\n responseText += `${index + 1}. ${task}\\n`;\r\n });\r\n responseText += `\\n`;\r\n \r\n if (technicalSummary) {\r\n responseText += `πŸ”§ **Technical Summary:**\\n${technicalSummary}\\n\\n`;\r\n }\r\n \r\n if (efficiencyNotes) {\r\n responseText += `πŸ“ˆ **Efficiency Notes:**\\n${efficiencyNotes}\\n\\n`;\r\n }\r\n \r\n if (agendaEvent) {\r\n responseText += `πŸ“… **Timetrack Entry Created:**\\n`;\r\n responseText += ` β€’ Agenda event created with work summary\\n`;\r\n responseText += ` β€’ Status: DRAFT (requires approval in agenda)\\n`;\r\n responseText += ` β€’ Duration: ${timeSpentMinutes} minutes\\n`;\r\n responseText += ` β€’ Period: ${sessionStart.toLocaleString()} - ${completionTime.toLocaleString()}\\n\\n`;\r\n }\r\n \r\n // Provide context for Cursor AI to generate customer response\r\n responseText += `πŸ“‹ **Context for Customer Response:**\\n`;\r\n responseText += ` β€’ Use \"get-completion-context\" to retrieve full context\\n`;\r\n responseText += ` β€’ Generate customer-friendly response based on completed work\\n`;\r\n responseText += ` β€’ Focus on business value and customer benefits\\n\\n`;\r\n \r\n responseText += `🎯 **Session archived successfully!**`;\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: responseText\r\n }]\r\n };\r\n }\r\n \r\n default:\r\n throw new Error(`Unknown tool: ${name}`);\r\n }\r\n \r\n } catch (error) {\r\n console.error(`❌ Tool execution error:`, error);\r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `Error executing ${name}: ${error instanceof Error ? error.message : 'Unknown error'}`\r\n }]\r\n };\r\n }\r\n});\r\n\r\n// Resource read handler\r\nserver.setRequestHandler(ReadResourceRequestSchema, async (request) => {\r\n if (!authContext) {\r\n return {\r\n contents: [{\r\n uri: request.params.uri,\r\n mimeType: 'text/plain',\r\n text: 'Error: Not authenticated. API key validation failed.'\r\n }]\r\n };\r\n }\r\n\r\n const { uri } = request.params;\r\n console.error(`πŸ“š Reading resource: ${uri}`);\r\n \r\n try {\r\n switch (uri) {\r\n case 'tickets://recent': {\r\n const teamIds = await getAccessibleTeamIds(authContext!.teamId);\r\n const { data, error } = await supabase\r\n .from('tickets')\r\n .select(`\r\n id,\r\n ticket_number,\r\n title,\r\n status,\r\n priority,\r\n created_at\r\n `)\r\n .in('team_id', teamIds)\r\n .order('created_at', { ascending: false })\r\n .limit(20);\r\n \r\n if (error) throw error;\r\n \r\n return {\r\n contents: [{\r\n uri,\r\n mimeType: 'application/json',\r\n text: JSON.stringify(data, null, 2)\r\n }]\r\n };\r\n }\r\n \r\n case 'customers://all': {\r\n const teamIds = await getAccessibleTeamIds(authContext!.teamId);\r\n const { data, error } = await supabase\r\n .from('customers')\r\n .select('id, name, email, website, created_at')\r\n .in('team_id', teamIds)\r\n .order('name')\r\n .limit(50);\r\n \r\n if (error) throw error;\r\n \r\n return {\r\n contents: [{\r\n uri,\r\n mimeType: 'application/json',\r\n text: JSON.stringify(data, null, 2)\r\n }]\r\n };\r\n }\r\n \r\n case 'projects://active': {\r\n const teamIds = await getAccessibleTeamIds(authContext!.teamId);\r\n const { data, error } = await supabase\r\n .from('projects')\r\n .select(`\r\n id,\r\n name,\r\n description,\r\n status,\r\n created_at,\r\n customers:customer_id(id, name)\r\n `)\r\n .in('team_id', teamIds)\r\n .eq('status', 'active')\r\n .order('name')\r\n .limit(50);\r\n \r\n if (error) throw error;\r\n \r\n return {\r\n contents: [{\r\n uri,\r\n mimeType: 'application/json',\r\n text: JSON.stringify(data, null, 2)\r\n }]\r\n };\r\n }\r\n \r\n default:\r\n throw new Error(`Unknown resource: ${uri}`);\r\n }\r\n \r\n } catch (error) {\r\n console.error(`❌ Resource read error:`, error);\r\n return {\r\n contents: [{\r\n uri,\r\n mimeType: 'text/plain',\r\n text: `Error reading ${uri}: ${error instanceof Error ? error.message : 'Unknown error'}`\r\n }]\r\n };\r\n }\r\n});\r\n\r\n// Main function\r\nasync function main() {\r\n console.error('πŸš€ Starting MG Tickets MCP Bridge Server...');\r\n console.error(`πŸ”‘ API Key: ${apiKey?.substring(0, 10)}...`);\r\n \r\n // Validate API key\r\n authContext = await validateApiKey(apiKey!);\r\n if (!authContext) {\r\n console.error('❌ API key validation failed. Please check your key and try again.');\r\n process.exit(1);\r\n }\r\n \r\n console.error(`βœ… Authenticated as user ${authContext.userId} in team ${authContext.teamId}`);\r\n console.error(`πŸ“‹ Available scopes: ${authContext.scopes.join(', ')}`);\r\n console.error('πŸ“‘ MCP Bridge Server ready for connections');\r\n \r\n const transport = new StdioServerTransport();\r\n await server.connect(transport);\r\n}\r\n\r\n// Handle graceful shutdown\r\nprocess.on('SIGINT', async () => {\r\n console.error('πŸ‘‹ Shutting down MCP Bridge Server...');\r\n process.exit(0);\r\n});\r\n\r\nprocess.on('SIGTERM', async () => {\r\n console.error('πŸ‘‹ Shutting down MCP Bridge Server...');\r\n process.exit(0);\r\n});\r\n\r\n// Start the server\r\nmain().catch((error) => {\r\n console.error('πŸ’₯ Fatal error:', error);\r\n process.exit(1);\r\n});"]}
1
+ {"version":3,"sources":["../src/index.ts"],"names":["args","name"],"mappings":"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","file":"index.js","sourcesContent":["#!/usr/bin/env node\r\n\r\nimport { Server } from '@modelcontextprotocol/sdk/server/index.js';\r\nimport { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js';\r\nimport { CallToolRequestSchema, ListToolsRequestSchema, ListResourcesRequestSchema, ReadResourceRequestSchema } from '@modelcontextprotocol/sdk/types.js';\r\nimport { createClient } from '@supabase/supabase-js';\r\nimport { createHash } from 'crypto';\r\n\r\n// Parse command line arguments and environment\r\nconst args = process.argv.slice(2);\r\nconst apiKey = args.find(arg => arg.startsWith('--api-key='))?.split('=')[1] || process.env.MG_TICKETS_API_KEY;\r\n// Default to MG Software's Supabase for SaaS customers\r\nconst supabaseUrl = args.find(arg => arg.startsWith('--supabase-url='))?.split('=')[1] || \r\n process.env.SUPABASE_URL || \r\n 'https://cvjdbczxyczjnatuolsk.supabase.co';\r\n\r\nconst supabaseKey = args.find(arg => arg.startsWith('--supabase-key='))?.split('=')[1] || \r\n process.env.SUPABASE_SERVICE_ROLE_KEY || \r\n 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJzdXBhYmFzZSIsInJlZiI6ImN2amRiY3p4eWN6am5hdHVvbHNrIiwicm9sZSI6InNlcnZpY2Vfcm9sZSIsImlhdCI6MTc1NjE0NzcyNCwiZXhwIjoyMDcxNzIzNzI0fQ.LljuNdCZXDcSIVTeIVOSNsvGNBfWsIM1QIswBJmGXKE';\r\n\r\nif (!apiKey) {\r\n console.error('❌ API key is required. Use --api-key=your_key or set MG_TICKETS_API_KEY environment variable');\r\n process.exit(1);\r\n}\r\n\r\n// Supabase credentials validation (optional with defaults)\r\nif (!supabaseUrl || !supabaseKey) {\r\n console.error('❌ Supabase credentials missing. Using defaults for MG Software SaaS.');\r\n}\r\n\r\n// Initialize Supabase client with service role\r\nconst supabase = createClient(supabaseUrl, supabaseKey);\r\n\r\ninterface AuthContext {\r\n userId: string;\r\n teamId: string;\r\n scopes: string[];\r\n}\r\n\r\n// Smart time estimation and codebase analysis is now handled by Cursor AI\r\n// MCP bridge focuses on data storage and session management only\r\n\r\n// Helper function to get accessible team IDs (own team + child teams)\r\nasync function getAccessibleTeamIds(teamId: string): Promise<string[]> {\r\n const { data: accessibleTeams } = await supabase\r\n .from('teams')\r\n .select('id')\r\n .or(`id.eq.${teamId},parent_team_id.eq.${teamId}`);\r\n \r\n return accessibleTeams?.map(t => t.id) || [teamId];\r\n}\r\n\r\n// API Key validation - direct database access\r\nasync function validateApiKey(key: string): Promise<AuthContext | null> {\r\n if (!key.startsWith('mid_') || key.length !== 68) {\r\n console.error('πŸ”‘ Invalid API key format');\r\n return null;\r\n }\r\n\r\n try {\r\n // Hash the API key\r\n const keyHash = createHash('sha256').update(key).digest('hex');\r\n console.error(`πŸ” Validating API key hash: ${keyHash.substring(0, 16)}...`);\r\n \r\n // Query database for API key\r\n const { data: apiKeyData, error } = await supabase\r\n .from('api_keys')\r\n .select('id, user_id, team_id, scopes, last_used_at')\r\n .eq('key_hash', keyHash)\r\n .single();\r\n\r\n if (error || !apiKeyData) {\r\n console.error('❌ API key not found or invalid:', error?.message);\r\n return null;\r\n }\r\n\r\n // Update last used timestamp\r\n await supabase\r\n .from('api_keys')\r\n .update({ last_used_at: new Date().toISOString() })\r\n .eq('id', apiKeyData.id);\r\n\r\n console.error(`βœ… API key validated for user ${apiKeyData.user_id} in team ${apiKeyData.team_id}`);\r\n \r\n return {\r\n userId: apiKeyData.user_id,\r\n teamId: apiKeyData.team_id,\r\n scopes: apiKeyData.scopes || []\r\n };\r\n } catch (error) {\r\n console.error('πŸ’₯ API key validation error:', error);\r\n return null;\r\n }\r\n}\r\n\r\n// Validate auth context once at startup\r\nlet authContext: AuthContext | null = null;\r\n\r\n// Helper function to check if a file is an image\r\nfunction isImageFile(mimeType: string): boolean {\r\n return mimeType.startsWith('image/') && \r\n ['image/jpeg', 'image/jpg', 'image/png', 'image/gif', 'image/webp'].includes(mimeType);\r\n}\r\n\r\n// Helper function to download and convert image to base64\r\nasync function downloadImageAsBase64(storageKey: string): Promise<string | null> {\r\n try {\r\n // Get signed URL for the file (1 hour expiry)\r\n const { data: urlData, error: urlError } = await supabase.storage\r\n .from('vault')\r\n .createSignedUrl(storageKey, 3600);\r\n \r\n if (urlError || !urlData?.signedUrl) {\r\n console.error(`Failed to create signed URL for ${storageKey}:`, urlError);\r\n return null;\r\n }\r\n \r\n // Download the file\r\n const response = await fetch(urlData.signedUrl);\r\n if (!response.ok) {\r\n console.error(`Failed to download file ${storageKey}: ${response.status}`);\r\n return null;\r\n }\r\n \r\n // Convert to base64\r\n const arrayBuffer = await response.arrayBuffer();\r\n const buffer = Buffer.from(arrayBuffer);\r\n return buffer.toString('base64');\r\n } catch (error) {\r\n console.error(`Error downloading image ${storageKey}:`, error);\r\n return null;\r\n }\r\n}\r\n\r\n/**\r\n * Convert TipTap JSON format to plain text for timetrack/agenda display\r\n * @param description - The description in TipTap JSON format or plain text\r\n * @returns Plain text version of the description\r\n */\r\nfunction convertTipTapToPlainText(description?: string | null): string {\r\n if (!description) {\r\n return \"\";\r\n }\r\n\r\n // If it's already plain text, return as is\r\n if (!description.startsWith('{\"type\":\"doc\"')) {\r\n return description;\r\n }\r\n\r\n try {\r\n const parsed = JSON.parse(description);\r\n \r\n // Recursively extract text from TipTap JSON structure\r\n const extractText = (node: any): string => {\r\n if (!node) return \"\";\r\n\r\n // Text nodes contain the actual text\r\n if (node.type === \"text\") {\r\n return node.text || \"\";\r\n }\r\n\r\n // Handle hard breaks\r\n if (node.type === \"hardBreak\") {\r\n return \"\\n\";\r\n }\r\n\r\n // Handle mentions\r\n if (node.type === \"mention\") {\r\n return `@${node.attrs?.label || node.attrs?.id || \"\"}`;\r\n }\r\n\r\n // Handle ticket mentions\r\n if (node.type === \"ticketMention\") {\r\n return `#${node.attrs?.label || node.attrs?.id || \"\"}`;\r\n }\r\n\r\n // Handle images\r\n if (node.type === \"image\") {\r\n return \"[afbeelding]\";\r\n }\r\n\r\n // Process children recursively\r\n if (node.content && Array.isArray(node.content)) {\r\n const childrenText = node.content.map(extractText).join(\"\");\r\n \r\n // Add appropriate spacing based on node type\r\n switch (node.type) {\r\n case \"paragraph\":\r\n return childrenText + \"\\n\";\r\n case \"heading\":\r\n return childrenText + \"\\n\\n\";\r\n case \"listItem\":\r\n return \"β€’ \" + childrenText + \"\\n\";\r\n case \"blockquote\":\r\n return \"> \" + childrenText + \"\\n\";\r\n default:\r\n return childrenText;\r\n }\r\n }\r\n \r\n // For nodes without content (like empty paragraphs without content array)\r\n if (node.type === \"paragraph\") {\r\n return \"\\n\";\r\n }\r\n\r\n return \"\";\r\n };\r\n\r\n const plainText = extractText(parsed);\r\n return plainText.trim();\r\n } catch (error) {\r\n // If parsing fails, return original string\r\n return description;\r\n }\r\n}\r\n\r\n// MCP Server setup\r\nconst server = new Server(\r\n {\r\n name: 'mg-tickets-mcp-bridge',\r\n version: '2.0.0',\r\n },\r\n {\r\n capabilities: {\r\n tools: {},\r\n resources: {},\r\n },\r\n }\r\n);\r\n\r\n// Available tools definition\r\nconst TOOLS = [\r\n {\r\n name: 'get-tickets',\r\n description: 'Get tickets with optional filtering by status, priority, project, customer, or search query',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n status: { type: 'string', enum: ['open', 'in_progress', 'review', 'resolved', 'closed', 'backlog'] },\r\n priority: { type: 'string', enum: ['low', 'medium', 'high', 'critical'] },\r\n projectId: { type: 'string' },\r\n customerId: { type: 'string' },\r\n q: { type: 'string', description: 'Search query for title or description' },\r\n pageSize: { type: 'number', default: 20, maximum: 100 }\r\n },\r\n required: []\r\n }\r\n },\r\n {\r\n name: 'get-ticket-by-id',\r\n description: 'Get a specific ticket by its ID, including all attachments, comments, and images. Images from ticket attachments and comment attachments are automatically downloaded and returned as base64-encoded content that can be analyzed by AI.',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n id: { type: 'string', description: 'Ticket ID' }\r\n },\r\n required: ['id']\r\n }\r\n },\r\n {\r\n name: 'create-ticket',\r\n description: 'Create a new ticket',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n title: { type: 'string', description: 'Ticket title' },\r\n description: { type: 'string' },\r\n status: { type: 'string', enum: ['open', 'in_progress', 'review', 'resolved', 'closed', 'backlog'], default: 'open' },\r\n priority: { type: 'string', enum: ['low', 'medium', 'high', 'critical'], default: 'medium' },\r\n type: { type: 'string', enum: ['task', 'bug', 'feature', 'support', 'question', 'improvement'], default: 'task' },\r\n projectId: { type: 'string' },\r\n customerId: { type: 'string' }\r\n },\r\n required: ['title']\r\n }\r\n },\r\n {\r\n name: 'get-customers',\r\n description: 'Get customers with optional search',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n q: { type: 'string', description: 'Search query for customer name or email' },\r\n pageSize: { type: 'number', default: 20, maximum: 100 }\r\n },\r\n required: []\r\n }\r\n },\r\n {\r\n name: 'create-customer',\r\n description: 'Create a new customer',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n name: { type: 'string', description: 'Customer name' },\r\n email: { type: 'string' },\r\n website: { type: 'string' }\r\n },\r\n required: ['name']\r\n }\r\n },\r\n {\r\n name: 'get-projects',\r\n description: 'Get projects with optional filtering',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n customerId: { type: 'string', description: 'Filter by customer ID' },\r\n q: { type: 'string', description: 'Search query for project name' },\r\n pageSize: { type: 'number', default: 20, maximum: 100 }\r\n },\r\n required: []\r\n }\r\n },\r\n {\r\n name: 'create-project',\r\n description: 'Create a new project',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n name: { type: 'string', description: 'Project name' },\r\n description: { type: 'string' },\r\n customerId: { type: 'string' },\r\n status: { type: 'string', enum: ['active', 'on_hold', 'completed', 'cancelled'], default: 'active' }\r\n },\r\n required: ['name']\r\n }\r\n },\r\n // === NEW AI SESSION TOOLS ===\r\n {\r\n name: 'start-ai-session-smart',\r\n description: 'Start a new AI development session with automatic tracking (time breakdown provided by Cursor AI)',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n ticketId: { type: 'string' },\r\n ticketUrl: { type: 'string', description: 'URL to the ticket' },\r\n cursorSessionId: { type: 'string', description: 'Cursor session identifier' },\r\n codebaseContext: { \r\n type: 'array', \r\n items: { type: 'string' },\r\n description: 'Relevant files for complexity analysis'\r\n },\r\n timeBreakdown: {\r\n type: 'object',\r\n description: 'Time estimate breakdown by development phase (REQUIRED)',\r\n properties: {\r\n analysisMinutes: { type: 'number', description: 'Ticket/context analysis time' },\r\n investigationMinutes: { type: 'number', description: 'Bug investigation/reproduction (0 if not a bug)' },\r\n developmentMinutes: { type: 'number', description: 'Actual coding/fixing time' },\r\n communicationMinutes: { type: 'number', description: 'Customer response writing time' }\r\n },\r\n required: ['analysisMinutes', 'investigationMinutes', 'developmentMinutes', 'communicationMinutes']\r\n },\r\n complexityScore: {\r\n type: 'number',\r\n minimum: 1,\r\n maximum: 10,\r\n description: 'Estimated complexity from 1-10'\r\n }\r\n },\r\n required: ['ticketId', 'timeBreakdown']\r\n }\r\n },\r\n {\r\n name: 'track-manual-follow-up',\r\n description: 'Track manual follow-up prompt by developer',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n aiSessionId: { type: 'string' },\r\n originalPrompt: { type: 'string' },\r\n aiResponse: { type: 'string' },\r\n developerFollowUp: { type: 'string' },\r\n followUpReason: { \r\n type: 'string', \r\n enum: ['incomplete_result', 'wrong_approach', 'needs_clarification', 'error_in_code']\r\n },\r\n outcome: {\r\n type: 'string',\r\n enum: ['success', 'partial_success', 'still_failed'],\r\n default: 'success'\r\n },\r\n timeSpentMinutes: { type: 'number' }\r\n },\r\n required: ['aiSessionId', 'originalPrompt', 'aiResponse', 'developerFollowUp', 'followUpReason']\r\n }\r\n },\r\n {\r\n name: 'get-session-context',\r\n description: 'Get current session context for follow-up continuity',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n aiSessionId: { type: 'string' },\r\n includeTicketData: { type: 'boolean', default: true },\r\n includeTodoProgress: { type: 'boolean', default: true },\r\n includeFollowUpHistory: { type: 'boolean', default: false }\r\n },\r\n required: ['aiSessionId']\r\n }\r\n },\r\n {\r\n name: 'sync-session-todos',\r\n description: 'Synchronize todo list with AI session (replace existing) or add new todos',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n aiSessionId: { type: 'string' },\r\n todos: {\r\n type: 'array',\r\n items: {\r\n type: 'object',\r\n properties: {\r\n todoId: { type: 'string', description: 'Optional external todo ID for tracking' },\r\n content: { type: 'string' },\r\n status: { type: 'string', enum: ['pending', 'in_progress', 'completed', 'cancelled'] },\r\n estimatedMinutes: { type: 'number' }\r\n },\r\n required: ['content', 'status']\r\n }\r\n },\r\n replaceAll: { \r\n type: 'boolean', \r\n default: true, \r\n description: 'If true, replace all existing todos. If false, add new todos to existing ones' \r\n }\r\n },\r\n required: ['aiSessionId', 'todos']\r\n }\r\n },\r\n {\r\n name: 'add-follow-up-todos',\r\n description: 'Add new todos from follow-up (without replacing existing ones)',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n aiSessionId: { type: 'string' },\r\n newTodos: {\r\n type: 'array',\r\n items: {\r\n type: 'object',\r\n properties: {\r\n content: { type: 'string' },\r\n status: { type: 'string', enum: ['pending', 'in_progress'], default: 'pending' },\r\n estimatedMinutes: { type: 'number' },\r\n addedInFollowUp: { type: 'boolean', default: true }\r\n },\r\n required: ['content']\r\n }\r\n },\r\n followUpReason: { \r\n type: 'string', \r\n description: 'Why were these todos added in follow-up' \r\n }\r\n },\r\n required: ['aiSessionId', 'newTodos']\r\n }\r\n },\r\n {\r\n name: 'update-session-status',\r\n description: 'Update AI session status and completion info',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n aiSessionId: { type: 'string' },\r\n status: {\r\n type: 'string',\r\n enum: ['started', 'in_progress', 'paused', 'completed', 'failed']\r\n },\r\n actualTimeMinutes: { type: 'number' },\r\n completionNotes: { type: 'string' }\r\n },\r\n required: ['aiSessionId', 'status']\r\n }\r\n },\r\n {\r\n name: 'get-completion-context',\r\n description: 'Get all context needed for Cursor AI to generate customer response',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n aiSessionId: { type: 'string' },\r\n includeFollowUps: { type: 'boolean', default: true },\r\n includeTimeMetrics: { type: 'boolean', default: true },\r\n includeTodos: { type: 'boolean', default: true }\r\n },\r\n required: ['aiSessionId']\r\n }\r\n },\r\n {\r\n name: 'save-customer-response',\r\n description: 'Save customer response generated by Cursor AI',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n aiSessionId: { type: 'string' },\r\n customerResponse: { type: 'string', description: 'Customer response generated by Cursor AI' },\r\n responseType: { \r\n type: 'string', \r\n enum: ['completion', 'progress_update', 'needs_clarification'],\r\n default: 'completion'\r\n }\r\n },\r\n required: ['aiSessionId', 'customerResponse']\r\n }\r\n },\r\n {\r\n name: 'complete-ai-session',\r\n description: 'Complete AI session with work summary - time calculated automatically',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n aiSessionId: { type: 'string' },\r\n workCompleted: {\r\n type: 'array',\r\n items: { type: 'string' },\r\n description: 'List of completed tasks/todos in English'\r\n },\r\n technicalSummary: {\r\n type: 'string',\r\n description: 'Technical summary of work done in English'\r\n },\r\n invoiceDescription: {\r\n type: 'string',\r\n description: 'Short invoice-friendly description in the language of the ticket (2-3 sentences max, suitable for billing)'\r\n },\r\n efficiencyNotes: { type: 'string' }\r\n },\r\n required: ['aiSessionId', 'workCompleted']\r\n }\r\n },\r\n {\r\n name: 'log-hours',\r\n description: 'Analyze current chat conversation and log hours as draft tracker entry. AI analyzes chat context to estimate hours as a senior developer would (without AI assistance). Cursor AI matches workspace name to correct project from list (optional).',\r\n inputSchema: {\r\n type: 'object',\r\n properties: {\r\n projectId: {\r\n type: 'string',\r\n description: 'Project ID (UUID) - Optional. Cursor AI should call get-projects first to try matching workspace name. If no clear match, omit this field.'\r\n },\r\n workDescription: {\r\n type: 'string',\r\n description: 'Short description of the work done (for the tracker entry)'\r\n },\r\n estimatedHours: {\r\n type: 'number',\r\n description: 'AI-estimated hours as senior developer would spend WITHOUT AI (e.g., 2.5 for 2.5 hours)'\r\n },\r\n chatContextSummary: {\r\n type: 'string',\r\n description: 'Brief summary of chat context for internal logging (optional)'\r\n }\r\n },\r\n required: ['workDescription', 'estimatedHours']\r\n }\r\n }\r\n];\r\n\r\n// Available resources\r\nconst RESOURCES = [\r\n {\r\n uri: 'tickets://recent',\r\n name: 'Recent Tickets',\r\n description: 'Most recently created tickets',\r\n mimeType: 'application/json'\r\n },\r\n {\r\n uri: 'customers://all',\r\n name: 'All Customers', \r\n description: 'Complete customer directory',\r\n mimeType: 'application/json'\r\n },\r\n {\r\n uri: 'projects://active',\r\n name: 'Active Projects',\r\n description: 'Currently active projects',\r\n mimeType: 'application/json'\r\n }\r\n];\r\n\r\n// List tools handler\r\nserver.setRequestHandler(ListToolsRequestSchema, async () => {\r\n return { tools: TOOLS };\r\n});\r\n\r\n// List resources handler\r\nserver.setRequestHandler(ListResourcesRequestSchema, async () => {\r\n return { resources: RESOURCES };\r\n});\r\n\r\n// Tool execution handler\r\nserver.setRequestHandler(CallToolRequestSchema, async (request) => {\r\n if (!authContext) {\r\n return {\r\n content: [{ type: 'text', text: 'Error: Not authenticated. API key validation failed.' }],\r\n };\r\n }\r\n\r\n const { name, arguments: args } = request.params;\r\n console.error(`πŸ› οΈ Executing tool: ${name} for team ${authContext.teamId}`);\r\n \r\n try {\r\n switch (name) {\r\n case 'get-tickets': {\r\n const { status, priority, projectId, customerId, q, pageSize = 20 } = args as any;\r\n \r\n // Get all accessible team IDs (own team + child teams)\r\n const teamIds = await getAccessibleTeamIds(authContext.teamId);\r\n \r\n let query = supabase\r\n .from('tickets')\r\n .select(`\r\n id,\r\n ticket_number,\r\n title,\r\n description,\r\n status,\r\n priority,\r\n type,\r\n created_at,\r\n project_id,\r\n customer_id,\r\n projects:project_id(id, name),\r\n customers:customer_id(id, name)\r\n `)\r\n .in('team_id', teamIds)\r\n .limit(Math.min(pageSize, 100));\r\n \r\n if (status) query = query.eq('status', status);\r\n if (priority) query = query.eq('priority', priority);\r\n if (projectId) query = query.eq('project_id', projectId);\r\n if (customerId) query = query.eq('customer_id', customerId);\r\n if (q) query = query.or(`title.ilike.%${q}%,description.ilike.%${q}%`);\r\n \r\n const { data, error } = await query.order('created_at', { ascending: false });\r\n \r\n if (error) throw error;\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `Found ${data?.length || 0} tickets:\\n\\n${data?.map(ticket => \r\n `**${ticket.ticket_number}**: ${ticket.title}\\n` +\r\n `Status: ${ticket.status} | Priority: ${ticket.priority}\\n` +\r\n `${(ticket.projects as any)?.name ? `Project: ${(ticket.projects as any).name}\\n` : ''}` +\r\n `${(ticket.customers as any)?.name ? `Customer: ${(ticket.customers as any).name}\\n` : ''}` +\r\n `Created: ${new Date(ticket.created_at).toLocaleDateString()}\\n`\r\n ).join('\\n') || 'No tickets found.'}`\r\n }]\r\n };\r\n }\r\n \r\n case 'get-ticket-by-id': {\r\n const { id } = args as any;\r\n \r\n // Get ticket data - include tickets from provider's team AND child customer teams\r\n const { data: ticketData, error } = await supabase\r\n .from('tickets')\r\n .select(`\r\n *,\r\n projects:project_id(id, name),\r\n customers:customer_id(id, name),\r\n assignee:assignee_id(id, full_name, email),\r\n requester:requester_id(id, full_name, email)\r\n `)\r\n .eq('id', id)\r\n .single();\r\n \r\n if (error) throw error;\r\n \r\n // Check if user has access to this ticket (own team or child team)\r\n const { data: teamAccess } = await supabase\r\n .from('teams')\r\n .select('id')\r\n .or(`id.eq.${ticketData.team_id},parent_team_id.eq.${authContext.teamId}`)\r\n .eq('id', ticketData.team_id)\r\n .single();\r\n \r\n if (!teamAccess && ticketData.team_id !== authContext.teamId) {\r\n throw new Error('Access denied: You do not have permission to view this ticket');\r\n }\r\n \r\n const data = ticketData;\r\n \r\n // Get ticket attachments (no team filter needed - already validated ticket access)\r\n const { data: attachments, error: attachmentsError } = await supabase\r\n .from('ticket_attachments')\r\n .select(`\r\n id,\r\n file_name,\r\n file_size,\r\n mime_type,\r\n storage_key,\r\n created_at,\r\n users:user_id(id, full_name)\r\n `)\r\n .eq('ticket_id', id)\r\n .order('created_at', { ascending: true });\r\n \r\n if (attachmentsError) {\r\n console.error('Error fetching attachments:', attachmentsError);\r\n }\r\n \r\n // Get ticket comments\r\n const { data: comments, error: commentsError } = await supabase\r\n .from('ticket_comments')\r\n .select(`\r\n id,\r\n content,\r\n created_at,\r\n users:user_id(id, full_name)\r\n `)\r\n .eq('ticket_id', id)\r\n .order('created_at', { ascending: true });\r\n \r\n if (commentsError) {\r\n console.error('Error fetching comments:', commentsError);\r\n }\r\n \r\n // Get comment attachments (no team filter needed - already validated ticket access)\r\n const commentIds = comments?.map(c => c.id) || [];\r\n let commentAttachments: any[] = [];\r\n \r\n if (commentIds.length > 0) {\r\n const { data: commAttachments, error: commAttachmentsError } = await supabase\r\n .from('ticket_comment_attachments')\r\n .select(`\r\n id,\r\n comment_id,\r\n file_name,\r\n file_size,\r\n mime_type,\r\n storage_key,\r\n created_at\r\n `)\r\n .in('comment_id', commentIds);\r\n \r\n if (commAttachmentsError) {\r\n console.error('Error fetching comment attachments:', commAttachmentsError);\r\n } else {\r\n commentAttachments = commAttachments || [];\r\n }\r\n }\r\n \r\n // Build content array starting with text\r\n const content: any[] = [{\r\n type: 'text',\r\n text: `**Ticket Details:**\\n\\n` +\r\n `**${data.ticket_number}**: ${data.title}\\n` +\r\n `Status: ${data.status}\\n` +\r\n `Priority: ${data.priority}\\n` +\r\n `Type: ${data.type}\\n` +\r\n `${data.description ? `Description: ${data.description}\\n` : ''}` +\r\n `${(data.projects as any)?.name ? `Project: ${(data.projects as any).name}\\n` : ''}` +\r\n `${(data.customers as any)?.name ? `Customer: ${(data.customers as any).name}\\n` : ''}` +\r\n `${data.assignee?.full_name ? `Assignee: ${data.assignee.full_name}\\n` : ''}` +\r\n `Requester: ${data.requester?.full_name || 'Unknown'}\\n` +\r\n `Created: ${new Date(data.created_at).toLocaleDateString()}\\n` +\r\n `${attachments && attachments.length > 0 ? `\\nπŸ“Ž Attachments: ${attachments.length}\\n` : ''}` +\r\n `${comments && comments.length > 0 ? `πŸ’¬ Comments: ${comments.length}\\n` : ''}`\r\n }];\r\n \r\n // Process ticket attachments - download images\r\n if (attachments && attachments.length > 0) {\r\n console.error(`πŸ“Ž Processing ${attachments.length} ticket attachments...`);\r\n \r\n for (const attachment of attachments) {\r\n if (isImageFile(attachment.mime_type)) {\r\n console.error(`πŸ–ΌοΈ Downloading image: ${attachment.file_name}`);\r\n const base64Data = await downloadImageAsBase64(attachment.storage_key);\r\n \r\n if (base64Data) {\r\n content.push({\r\n type: 'image',\r\n data: base64Data,\r\n mimeType: attachment.mime_type\r\n });\r\n \r\n // Add text context about the image\r\n content.push({\r\n type: 'text',\r\n text: `\\nπŸ“Έ **Image from ticket**: ${attachment.file_name} (${Math.round(attachment.file_size / 1024)}KB, uploaded by ${(attachment.users as any)?.full_name || 'Unknown'} on ${new Date(attachment.created_at).toLocaleDateString()})\\n`\r\n });\r\n }\r\n }\r\n }\r\n }\r\n \r\n // Process comment attachments - download images\r\n if (commentAttachments.length > 0) {\r\n console.error(`πŸ“Ž Processing ${commentAttachments.length} comment attachments...`);\r\n \r\n for (const attachment of commentAttachments) {\r\n if (isImageFile(attachment.mime_type)) {\r\n console.error(`πŸ–ΌοΈ Downloading comment image: ${attachment.file_name}`);\r\n const base64Data = await downloadImageAsBase64(attachment.storage_key);\r\n \r\n if (base64Data) {\r\n // Find the comment this attachment belongs to\r\n const comment = comments?.find(c => c.id === attachment.comment_id);\r\n \r\n content.push({\r\n type: 'image',\r\n data: base64Data,\r\n mimeType: attachment.mime_type\r\n });\r\n \r\n // Add text context about the image\r\n content.push({\r\n type: 'text',\r\n text: `\\nπŸ“Έ **Image from comment** by ${(comment?.users as any)?.full_name || 'Unknown'} on ${new Date(attachment.created_at).toLocaleDateString()}: ${attachment.file_name} (${Math.round(attachment.file_size / 1024)}KB)\\n` +\r\n (comment?.content ? `Comment text: \"${comment.content.substring(0, 100)}${comment.content.length > 100 ? '...' : ''}\"\\n` : '')\r\n });\r\n }\r\n }\r\n }\r\n }\r\n \r\n console.error(`βœ… Returning ticket with ${content.filter(c => c.type === 'image').length} images`);\r\n \r\n return { content };\r\n }\r\n \r\n case 'create-ticket': {\r\n const { title, description, status = 'open', priority = 'medium', type = 'task', projectId, customerId } = args as any;\r\n \r\n // Generate ticket number\r\n const year = new Date().getFullYear();\r\n const { count } = await supabase\r\n .from('tickets')\r\n .select('*', { count: 'exact', head: true })\r\n .eq('team_id', authContext.teamId);\r\n \r\n const ticketNumber = `${year}-${String((count || 0) + 1).padStart(3, '0')}`;\r\n \r\n const { data, error } = await supabase\r\n .from('tickets')\r\n .insert({\r\n team_id: authContext.teamId,\r\n ticket_number: ticketNumber,\r\n title,\r\n description,\r\n status,\r\n priority,\r\n type,\r\n project_id: projectId || null,\r\n customer_id: customerId || null,\r\n requester_id: authContext.userId\r\n })\r\n .select()\r\n .single();\r\n \r\n if (error) throw error;\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `βœ… **Ticket Created Successfully!**\\n\\n` +\r\n `Ticket Number: **${ticketNumber}**\\n` +\r\n `Title: ${title}\\n` +\r\n `Status: ${status}\\n` +\r\n `Priority: ${priority}\\n` +\r\n `Type: ${type}\\n`\r\n }]\r\n };\r\n }\r\n \r\n case 'get-customers': {\r\n const { q, pageSize = 20 } = args as any;\r\n \r\n // Get all accessible team IDs (own team + child teams)\r\n const teamIds = await getAccessibleTeamIds(authContext.teamId);\r\n \r\n let query = supabase\r\n .from('customers')\r\n .select('id, name, email, website, created_at')\r\n .in('team_id', teamIds)\r\n .limit(Math.min(pageSize, 100));\r\n \r\n if (q) query = query.or(`name.ilike.%${q}%,email.ilike.%${q}%`);\r\n \r\n const { data, error } = await query.order('name');\r\n \r\n if (error) throw error;\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `Found ${data?.length || 0} customers:\\n\\n${data?.map(customer => \r\n `**${customer.name}**\\n` +\r\n `${customer.email ? `Email: ${customer.email}\\n` : ''}` +\r\n `${customer.website ? `Website: ${customer.website}\\n` : ''}` +\r\n `Created: ${new Date(customer.created_at).toLocaleDateString()}\\n`\r\n ).join('\\n') || 'No customers found.'}`\r\n }]\r\n };\r\n }\r\n \r\n case 'create-customer': {\r\n const { name, email, website } = args as any;\r\n \r\n const { data, error } = await supabase\r\n .from('customers')\r\n .insert({\r\n team_id: authContext.teamId,\r\n name,\r\n email: email || null,\r\n website: website || null,\r\n user_id: authContext.userId\r\n })\r\n .select()\r\n .single();\r\n \r\n if (error) throw error;\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `βœ… **Customer Created Successfully!**\\n\\n` +\r\n `Name: ${name}\\n` +\r\n `${email ? `Email: ${email}\\n` : ''}` +\r\n `${website ? `Website: ${website}\\n` : ''}`\r\n }]\r\n };\r\n }\r\n \r\n case 'get-projects': {\r\n const { customerId, q, pageSize = 20 } = args as any;\r\n \r\n // Get all accessible team IDs (own team + child teams)\r\n const teamIds = await getAccessibleTeamIds(authContext.teamId);\r\n \r\n let query = supabase\r\n .from('projects')\r\n .select(`\r\n id,\r\n name,\r\n description,\r\n customer_id,\r\n status,\r\n created_at,\r\n customers:customer_id(id, name)\r\n `)\r\n .in('team_id', teamIds)\r\n .limit(Math.min(pageSize, 100));\r\n \r\n if (customerId) query = query.eq('customer_id', customerId);\r\n if (q) query = query.ilike('name', `%${q}%`);\r\n \r\n const { data, error } = await query.order('name');\r\n \r\n if (error) throw error;\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `Found ${data?.length || 0} projects:\\n\\n${data?.map(project => \r\n `**${project.name}**\\n` +\r\n `Status: ${project.status}\\n` +\r\n `${project.description ? `Description: ${project.description}\\n` : ''}` +\r\n `${(project.customers as any)?.name ? `Customer: ${(project.customers as any).name}\\n` : ''}` +\r\n `Created: ${new Date(project.created_at).toLocaleDateString()}\\n`\r\n ).join('\\n') || 'No projects found.'}`\r\n }]\r\n };\r\n }\r\n \r\n case 'create-project': {\r\n const { name, description, customerId, status = 'active' } = args as any;\r\n \r\n const { data, error } = await supabase\r\n .from('projects')\r\n .insert({\r\n team_id: authContext.teamId,\r\n name,\r\n description: description || null,\r\n customer_id: customerId || null,\r\n status,\r\n user_id: authContext.userId\r\n })\r\n .select()\r\n .single();\r\n \r\n if (error) throw error;\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `βœ… **Project Created Successfully!**\\n\\n` +\r\n `Name: ${name}\\n` +\r\n `Status: ${status}\\n` +\r\n `${description ? `Description: ${description}\\n` : ''}`\r\n }]\r\n };\r\n }\r\n \r\n // === AI SESSION TOOLS ===\r\n case 'start-ai-session-smart': {\r\n const { ticketId, ticketUrl, cursorSessionId, codebaseContext, timeBreakdown, complexityScore } = args as any;\r\n \r\n // Validate timeBreakdown is provided\r\n if (!timeBreakdown || !timeBreakdown.analysisMinutes || timeBreakdown.developmentMinutes === undefined) {\r\n throw new Error('timeBreakdown is required with all phases: analysisMinutes, investigationMinutes, developmentMinutes, communicationMinutes');\r\n }\r\n \r\n // Calculate total time from breakdown\r\n const totalEstimateMinutes = \r\n (timeBreakdown.analysisMinutes || 0) +\r\n (timeBreakdown.investigationMinutes || 0) +\r\n (timeBreakdown.developmentMinutes || 0) +\r\n (timeBreakdown.communicationMinutes || 0);\r\n \r\n const sessionStartTime = new Date();\r\n \r\n // Create AI session with calculated total\r\n const { data: sessionData, error } = await supabase\r\n .from('ai_sessions')\r\n .insert({\r\n ticket_id: ticketId,\r\n provider_user_id: authContext.userId,\r\n team_id: authContext.teamId,\r\n cursor_session_id: cursorSessionId || null,\r\n ai_time_estimate_minutes: totalEstimateMinutes,\r\n complexity_score: complexityScore || null,\r\n status: 'in_progress'\r\n })\r\n .select('id, ticket_id, cursor_session_id, created_at')\r\n .single();\r\n \r\n if (error) throw error;\r\n \r\n // CREATE PHASE TIME LOGS: Store breakdown as separate phase activities\r\n const phaseActivities = [\r\n {\r\n ai_session_id: sessionData.id,\r\n activity_type: 'analysis',\r\n description: 'Ticket analysis and context understanding',\r\n duration_seconds: 0,\r\n estimated_duration_seconds: timeBreakdown.analysisMinutes * 60,\r\n status: 'in_progress', // Analysis starts immediately\r\n productivity_score: 8,\r\n started_at: sessionStartTime.toISOString()\r\n },\r\n {\r\n ai_session_id: sessionData.id,\r\n activity_type: 'bug_investigation',\r\n description: 'Bug investigation and root cause analysis',\r\n duration_seconds: 0,\r\n estimated_duration_seconds: timeBreakdown.investigationMinutes * 60,\r\n status: 'pending',\r\n productivity_score: null\r\n },\r\n {\r\n ai_session_id: sessionData.id,\r\n activity_type: 'development',\r\n description: 'Implementation and coding',\r\n duration_seconds: 0,\r\n estimated_duration_seconds: timeBreakdown.developmentMinutes * 60,\r\n status: 'pending',\r\n productivity_score: null\r\n },\r\n {\r\n ai_session_id: sessionData.id,\r\n activity_type: 'communication',\r\n description: 'Customer response and documentation',\r\n duration_seconds: 0,\r\n estimated_duration_seconds: timeBreakdown.communicationMinutes * 60,\r\n status: 'pending',\r\n productivity_score: null\r\n }\r\n ];\r\n \r\n // Insert all phase activities\r\n await supabase.from('ai_time_logs').insert(phaseActivities);\r\n \r\n // Generate a readable session ID for UI\r\n const sessionId = `ai-sess-${sessionData.id.substring(0, 8)}`;\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `πŸš€ **AI Session Started Successfully!**\\n\\n` +\r\n `πŸ†” Session ID: **${sessionId}**\\n` +\r\n `🎫 Ticket: ${ticketId}\\n` +\r\n `\\nπŸ“Š **Time Breakdown:**\\n` +\r\n ` β€’ Analysis: ${timeBreakdown.analysisMinutes} min\\n` +\r\n ` β€’ Investigation: ${timeBreakdown.investigationMinutes} min\\n` +\r\n ` β€’ Development: ${timeBreakdown.developmentMinutes} min\\n` +\r\n ` β€’ Communication: ${timeBreakdown.communicationMinutes} min\\n` +\r\n ` β€’ **Total: ${totalEstimateMinutes} min**\\n\\n` +\r\n `${complexityScore ? `🎯 Complexity: ${complexityScore}/10\\n` : ''}` +\r\n `⏱️ **Phase Tracking Started** (Analysis in progress)\\n` +\r\n `${cursorSessionId ? `πŸ”— Cursor Session: ${cursorSessionId}\\n` : ''}` +\r\n `πŸ“… Started: ${sessionStartTime.toLocaleString()}\\n\\n` +\r\n `βœ… Session initialized with phase breakdown!\\n` +\r\n `πŸ“ Timetrack entry will be created when you complete the session.`\r\n }]\r\n };\r\n }\r\n \r\n case 'track-manual-follow-up': {\r\n const { aiSessionId, originalPrompt, aiResponse, developerFollowUp, followUpReason, outcome = 'success', timeSpentMinutes } = args as any;\r\n \r\n // Extract actual session UUID from readable ID\r\n const sessionUuid = aiSessionId.replace('ai-sess-', '');\r\n \r\n // Find the full session ID and current session data\r\n // Find the session - get all sessions for accessible teams and filter in JS\r\n const teamIds = await getAccessibleTeamIds(authContext.teamId);\r\n const { data: allSessions, error: sessionError } = await supabase\r\n .from('ai_sessions')\r\n .select('id, status, created_at, ai_time_estimate_minutes')\r\n .in('team_id', teamIds);\r\n \r\n if (sessionError) {\r\n throw new Error(`Database error: ${sessionError.message}`);\r\n }\r\n \r\n // Find session with matching prefix\r\n const session = allSessions?.find(s => s.id.toString().startsWith(sessionUuid));\r\n if (!session) {\r\n throw new Error(`Session not found: ${aiSessionId} (searched ${allSessions?.length || 0} sessions)`);\r\n }\r\n \r\n const followUpTime = new Date();\r\n \r\n // RESTART SESSION: Set status back to in_progress for continued tracking\r\n await supabase\r\n .from('ai_sessions')\r\n .update({\r\n status: 'in_progress', // Restart active tracking\r\n // Don't update completed_at - session continues\r\n })\r\n .eq('id', session.id);\r\n \r\n // LOG FOLLOW-UP: Track the manual follow-up interaction\r\n const { data, error } = await supabase\r\n .from('manual_follow_ups')\r\n .insert({\r\n ai_session_id: session.id,\r\n developer_id: authContext.userId,\r\n team_id: authContext.teamId,\r\n original_prompt: originalPrompt,\r\n ai_response: aiResponse,\r\n follow_up_prompt: developerFollowUp,\r\n follow_up_reason: followUpReason,\r\n outcome: outcome,\r\n time_spent_minutes: timeSpentMinutes || null,\r\n resolved_at: outcome === 'success' ? new Date().toISOString() : null\r\n })\r\n .select()\r\n .single();\r\n \r\n if (error) throw error;\r\n \r\n // RESTART TIME TRACKING: Log follow-up activity start\r\n await supabase\r\n .from('ai_time_logs')\r\n .insert({\r\n ai_session_id: session.id,\r\n activity_type: 'debugging', // Follow-ups are typically debugging/fixing\r\n description: `Follow-up: ${followUpReason.replace('_', ' ')} - ${outcome}`,\r\n duration_seconds: (timeSpentMinutes || 0) * 60,\r\n productivity_score: outcome === 'success' ? 9 : outcome === 'partial_success' ? 6 : 4,\r\n started_at: followUpTime.toISOString()\r\n });\r\n \r\n // CALCULATE CURRENT EFFICIENCY: Compare time spent vs estimated\r\n const sessionStartTime = new Date(session.created_at);\r\n const totalMinutesElapsed = Math.round((followUpTime.getTime() - sessionStartTime.getTime()) / 60000);\r\n const estimatedMinutes = session.ai_time_estimate_minutes || 60;\r\n const currentEfficiency = totalMinutesElapsed > 0 ? (totalMinutesElapsed / estimatedMinutes) : 1;\r\n \r\n // UPDATE EFFICIENCY TRACKING: Update running efficiency score\r\n await supabase\r\n .from('ai_sessions')\r\n .update({\r\n efficiency_score: currentEfficiency.toFixed(2),\r\n actual_time_minutes: totalMinutesElapsed\r\n })\r\n .eq('id', session.id);\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `πŸ”„ **Follow-up Tracked & Session Restarted!**\\n\\n` +\r\n `πŸ†” Session: ${aiSessionId} (back to active)\\n` +\r\n `πŸ” Reason: ${followUpReason.replace('_', ' ')}\\n` +\r\n `βœ… Outcome: ${outcome}\\n` +\r\n `${timeSpentMinutes ? `⏱️ Time spent: ${timeSpentMinutes} minutes\\n` : ''}` +\r\n `πŸ“Š **Current Progress:**\\n` +\r\n ` β€’ Total time: ${totalMinutesElapsed} minutes\\n` +\r\n ` β€’ Estimated: ${estimatedMinutes} minutes\\n` +\r\n ` β€’ Efficiency: ${currentEfficiency < 1 ? 'πŸš€ ' : currentEfficiency > 1.5 ? '⚠️ ' : '⏱️ '}${(currentEfficiency * 100).toFixed(0)}%\\n` +\r\n `\\n⚑ **Time tracking resumed** - continue with confidence!`\r\n }]\r\n };\r\n }\r\n \r\n case 'get-session-context': {\r\n const { aiSessionId, includeTicketData = true, includeTodoProgress = true, includeFollowUpHistory = false } = args as any;\r\n \r\n // Extract actual session UUID from readable ID\r\n const sessionUuid = aiSessionId.replace('ai-sess-', '');\r\n \r\n // Get session data with related info - find by JavaScript filtering\r\n const teamIds = await getAccessibleTeamIds(authContext.teamId);\r\n const { data: allSessions, error: sessionError } = await supabase\r\n .from('ai_sessions')\r\n .select(`\r\n id,\r\n ticket_id,\r\n status,\r\n ai_time_estimate_minutes,\r\n actual_time_minutes,\r\n complexity_score,\r\n created_at,\r\n cursor_session_id\r\n `)\r\n .in('team_id', teamIds);\r\n \r\n if (sessionError) {\r\n throw new Error(`Database error: ${sessionError.message}`);\r\n }\r\n \r\n // Find session with matching prefix\r\n const session = allSessions?.find(s => s.id.toString().startsWith(sessionUuid));\r\n \r\n if (sessionError || !session) {\r\n throw new Error(`Session not found: ${aiSessionId}`);\r\n }\r\n \r\n let context: any = {\r\n sessionId: aiSessionId,\r\n status: session.status,\r\n timeEstimate: session.ai_time_estimate_minutes,\r\n actualTime: session.actual_time_minutes,\r\n complexity: session.complexity_score,\r\n createdAt: session.created_at\r\n };\r\n \r\n // Include ticket data if requested\r\n if (includeTicketData) {\r\n const { data: ticket } = await supabase\r\n .from('tickets')\r\n .select('id, ticket_number, title, description, status, priority, type')\r\n .eq('id', session.ticket_id)\r\n .single();\r\n \r\n context.ticketData = ticket;\r\n }\r\n \r\n // Include todo progress if requested\r\n if (includeTodoProgress) {\r\n const { data: todos } = await supabase\r\n .from('ai_todos')\r\n .select('id, content, status, estimated_minutes, actual_minutes')\r\n .eq('ai_session_id', session.id)\r\n .order('sequence_order');\r\n \r\n context.todos = todos || [];\r\n context.todoProgress = {\r\n total: todos?.length || 0,\r\n completed: todos?.filter(t => t.status === 'completed').length || 0,\r\n inProgress: todos?.filter(t => t.status === 'in_progress').length || 0\r\n };\r\n }\r\n \r\n // Include follow-up history if requested\r\n if (includeFollowUpHistory) {\r\n const { data: followUps } = await supabase\r\n .from('manual_follow_ups')\r\n .select('follow_up_reason, outcome, time_spent_minutes, created_at')\r\n .eq('ai_session_id', session.id)\r\n .order('created_at');\r\n \r\n context.followUpHistory = followUps || [];\r\n }\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `🎯 **Session Context Retrieved**\\n\\n` +\r\n `Session: ${aiSessionId}\\n` +\r\n `Status: ${session.status}\\n` +\r\n `${context.ticketData ? `Ticket: ${context.ticketData.ticket_number} - ${context.ticketData.title}\\n` : ''}` +\r\n `${context.todoProgress ? `Todo Progress: ${context.todoProgress.completed}/${context.todoProgress.total} completed\\n` : ''}` +\r\n `${context.followUpHistory ? `Follow-ups: ${context.followUpHistory.length}\\n` : ''}` +\r\n `\\nπŸ“‹ Full context preserved for seamless continuation!`\r\n }]\r\n };\r\n }\r\n \r\n case 'sync-session-todos': {\r\n const { aiSessionId, todos, replaceAll = true } = args as any;\r\n \r\n // Extract actual session UUID from readable ID\r\n const sessionUuid = aiSessionId.replace('ai-sess-', '');\r\n \r\n // Find the session - get all sessions for accessible teams and filter in JS\r\n const teamIds = await getAccessibleTeamIds(authContext.teamId);\r\n const { data: allSessions, error: sessionError } = await supabase\r\n .from('ai_sessions')\r\n .select('id')\r\n .in('team_id', teamIds);\r\n \r\n if (sessionError) {\r\n throw new Error(`Database error: ${sessionError.message}`);\r\n }\r\n \r\n // Find session with matching prefix\r\n const session = allSessions?.find(s => s.id.toString().startsWith(sessionUuid));\r\n if (!session) {\r\n throw new Error(`Session not found: ${aiSessionId} (searched ${allSessions?.length || 0} sessions)`);\r\n }\r\n \r\n // If replacing all, clear existing todos first\r\n if (replaceAll) {\r\n await supabase\r\n .from('ai_todos')\r\n .delete()\r\n .eq('ai_session_id', session.id);\r\n }\r\n \r\n if (todos && todos.length > 0) {\r\n // Get current max sequence order if adding to existing\r\n let startSequence = 0;\r\n if (!replaceAll) {\r\n const { data: maxTodo } = await supabase\r\n .from('ai_todos')\r\n .select('sequence_order')\r\n .eq('ai_session_id', session.id)\r\n .order('sequence_order', { ascending: false })\r\n .limit(1)\r\n .single();\r\n startSequence = (maxTodo?.sequence_order || 0) + 1;\r\n }\r\n \r\n const todoInserts = todos.map((todo: any, index: number) => ({\r\n ai_session_id: session.id,\r\n content: todo.content,\r\n status: todo.status,\r\n cursor_todo_id: todo.todoId || null,\r\n estimated_minutes: todo.estimatedMinutes || null,\r\n sequence_order: startSequence + index\r\n }));\r\n \r\n const { error: insertError } = await supabase\r\n .from('ai_todos')\r\n .insert(todoInserts);\r\n \r\n if (insertError) throw insertError;\r\n }\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `βœ… **Todos ${replaceAll ? 'Synced' : 'Added'} Successfully!**\\n\\n` +\r\n `Session: ${aiSessionId}\\n` +\r\n `${replaceAll ? 'Synced' : 'Added'} ${todos?.length || 0} todos\\n` +\r\n `${replaceAll ? '' : 'βž• Added to existing todo list\\n'}` +\r\n `\\nπŸ“ Todo list updated and tracked for progress monitoring!`\r\n }]\r\n };\r\n }\r\n \r\n case 'add-follow-up-todos': {\r\n const { aiSessionId, newTodos, followUpReason } = args as any;\r\n \r\n // Extract actual session UUID from readable ID\r\n const sessionUuid = aiSessionId.replace('ai-sess-', '');\r\n \r\n // Find the session - get all sessions for accessible teams and filter in JS\r\n const teamIds = await getAccessibleTeamIds(authContext.teamId);\r\n const { data: allSessions, error: sessionError } = await supabase\r\n .from('ai_sessions')\r\n .select('id')\r\n .in('team_id', teamIds);\r\n \r\n if (sessionError) {\r\n throw new Error(`Database error: ${sessionError.message}`);\r\n }\r\n \r\n // Find session with matching prefix\r\n const session = allSessions?.find(s => s.id.toString().startsWith(sessionUuid));\r\n if (!session) {\r\n throw new Error(`Session not found: ${aiSessionId} (searched ${allSessions?.length || 0} sessions)`);\r\n }\r\n \r\n if (newTodos && newTodos.length > 0) {\r\n // Get current max sequence order\r\n const { data: maxTodo } = await supabase\r\n .from('ai_todos')\r\n .select('sequence_order')\r\n .eq('ai_session_id', session.id)\r\n .order('sequence_order', { ascending: false })\r\n .limit(1)\r\n .single();\r\n const startSequence = (maxTodo?.sequence_order || 0) + 1;\r\n \r\n const todoInserts = newTodos.map((todo: any, index: number) => ({\r\n ai_session_id: session.id,\r\n content: `[Follow-up] ${todo.content}`,\r\n status: todo.status || 'pending',\r\n estimated_minutes: todo.estimatedMinutes || null,\r\n sequence_order: startSequence + index\r\n }));\r\n \r\n const { error: insertError } = await supabase\r\n .from('ai_todos')\r\n .insert(todoInserts);\r\n \r\n if (insertError) throw insertError;\r\n }\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `βœ… **Follow-up Todos Added Successfully!**\\n\\n` +\r\n `Session: ${aiSessionId}\\n` +\r\n `Added ${newTodos?.length || 0} new todos from follow-up\\n` +\r\n `${followUpReason ? `Reason: ${followUpReason}\\n` : ''}` +\r\n `\\nπŸ“ New tasks identified and added to existing workflow!`\r\n }]\r\n };\r\n }\r\n \r\n case 'update-session-status': {\r\n const { aiSessionId, status, actualTimeMinutes, completionNotes } = args as any;\r\n \r\n // Extract actual session UUID from readable ID\r\n const sessionUuid = aiSessionId.replace('ai-sess-', '');\r\n \r\n // Find and update the session\r\n // First find the session\r\n const teamIds = await getAccessibleTeamIds(authContext.teamId);\r\n const { data: allSessions, error: findError } = await supabase\r\n .from('ai_sessions')\r\n .select('id')\r\n .in('team_id', teamIds);\r\n \r\n if (findError) {\r\n throw new Error(`Database error: ${findError.message}`);\r\n }\r\n \r\n const foundSession = allSessions?.find(s => s.id.toString().startsWith(sessionUuid));\r\n if (!foundSession) {\r\n throw new Error(`Session not found: ${aiSessionId} (searched ${allSessions?.length || 0} sessions)`);\r\n }\r\n \r\n // Now update the found session\r\n const { data, error } = await supabase\r\n .from('ai_sessions')\r\n .update({\r\n status: status,\r\n actual_time_minutes: actualTimeMinutes || null,\r\n completed_at: status === 'completed' ? new Date().toISOString() : null\r\n })\r\n .eq('id', foundSession.id)\r\n .select()\r\n .single();\r\n \r\n if (error) throw error;\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `🎯 **Session Status Updated!**\\n\\n` +\r\n `Session: ${aiSessionId}\\n` +\r\n `Status: ${status}\\n` +\r\n `${actualTimeMinutes ? `Actual Time: ${actualTimeMinutes} minutes\\n` : ''}` +\r\n `${status === 'completed' ? `βœ… Session completed successfully!\\n` : ''}` +\r\n `${completionNotes ? `Notes: ${completionNotes}\\n` : ''}`\r\n }]\r\n };\r\n }\r\n \r\n case 'get-completion-context': {\r\n const { aiSessionId, includeFollowUps = true, includeTimeMetrics = true, includeTodos = true } = args as any;\r\n \r\n // Extract actual session UUID from readable ID\r\n const sessionUuid = aiSessionId.replace('ai-sess-', '');\r\n \r\n // Get session data with ticket info\r\n // Get comprehensive session context - find by JavaScript filtering\r\n const teamIds = await getAccessibleTeamIds(authContext.teamId);\r\n const { data: allSessions, error: sessionError } = await supabase\r\n .from('ai_sessions')\r\n .select(`\r\n id,\r\n ticket_id,\r\n ai_time_estimate_minutes,\r\n actual_time_minutes,\r\n efficiency_score,\r\n created_at,\r\n completed_at,\r\n status,\r\n complexity_score\r\n `)\r\n .in('team_id', teamIds);\r\n \r\n if (sessionError) {\r\n throw new Error(`Database error: ${sessionError.message}`);\r\n }\r\n \r\n // Find session with matching prefix\r\n const session = allSessions?.find(s => s.id.toString().startsWith(sessionUuid));\r\n if (!session) {\r\n throw new Error(`Session not found: ${aiSessionId} (searched ${allSessions?.length || 0} sessions)`);\r\n }\r\n \r\n // Get ticket data\r\n const { data: ticket, error: ticketError } = await supabase\r\n .from('tickets')\r\n .select('ticket_number, title, description, type, priority')\r\n .eq('id', session.ticket_id)\r\n .single();\r\n \r\n if (ticketError || !ticket) {\r\n throw new Error('Ticket not found for session');\r\n }\r\n \r\n let contextData: any = {\r\n session: {\r\n id: aiSessionId,\r\n status: session.status,\r\n complexity: session.complexity_score,\r\n createdAt: session.created_at,\r\n completedAt: session.completed_at\r\n },\r\n ticket: {\r\n number: ticket.ticket_number,\r\n title: ticket.title,\r\n description: ticket.description,\r\n type: ticket.type,\r\n priority: ticket.priority\r\n }\r\n };\r\n \r\n // Add time metrics if requested\r\n if (includeTimeMetrics) {\r\n const timeSaved = session.ai_time_estimate_minutes && session.actual_time_minutes \r\n ? Math.max(0, session.ai_time_estimate_minutes - session.actual_time_minutes)\r\n : null;\r\n \r\n contextData.timeMetrics = {\r\n estimatedMinutes: session.ai_time_estimate_minutes,\r\n actualMinutes: session.actual_time_minutes,\r\n timeSaved: timeSaved,\r\n efficiency: session.efficiency_score,\r\n sessionDuration: session.completed_at && session.created_at \r\n ? Math.round((new Date(session.completed_at).getTime() - new Date(session.created_at).getTime()) / 60000)\r\n : null\r\n };\r\n }\r\n \r\n // Add todos if requested\r\n if (includeTodos) {\r\n const { data: todos } = await supabase\r\n .from('ai_todos')\r\n .select('content, status, estimated_minutes, actual_minutes, completed_at')\r\n .eq('ai_session_id', session.id)\r\n .order('created_at', { ascending: true });\r\n \r\n contextData.todos = todos || [];\r\n }\r\n \r\n // Add follow-ups if requested\r\n if (includeFollowUps) {\r\n const { data: followUps } = await supabase\r\n .from('manual_follow_ups')\r\n .select('follow_up_reason, outcome, time_spent_minutes, created_at')\r\n .eq('ai_session_id', session.id)\r\n .order('created_at', { ascending: true });\r\n \r\n contextData.followUps = followUps || [];\r\n }\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `πŸ“‹ **Completion Context Retrieved!**\\n\\n` +\r\n `🎫 **Ticket:** ${ticket.ticket_number} - ${ticket.title}\\n` +\r\n `πŸ†” **Session:** ${aiSessionId} (${session.status})\\n` +\r\n `⏱️ **Time:** ${session.actual_time_minutes || 'N/A'}/${session.ai_time_estimate_minutes || 'N/A'} minutes\\n` +\r\n `πŸ“‹ **Todos:** ${contextData.todos?.filter((t: any) => t.status === 'completed').length || 0}/${contextData.todos?.length || 0} completed\\n` +\r\n `πŸ”„ **Follow-ups:** ${contextData.followUps?.length || 0}\\n\\n` +\r\n `βœ… **Full context ready for Cursor AI to generate customer response!**\\n\\n` +\r\n `**Context Data:**\\n\\`\\`\\`json\\n${JSON.stringify(contextData, null, 2)}\\`\\`\\``\r\n }]\r\n };\r\n }\r\n \r\n case 'save-customer-response': {\r\n const { aiSessionId, customerResponse, responseType = 'completion' } = args as any;\r\n \r\n // Extract actual session UUID from readable ID\r\n const sessionUuid = aiSessionId.replace('ai-sess-', '');\r\n \r\n // Get session to validate it exists - find by JavaScript filtering\r\n const teamIds = await getAccessibleTeamIds(authContext.teamId);\r\n const { data: allSessions, error: sessionError } = await supabase\r\n .from('ai_sessions')\r\n .select('id')\r\n .in('team_id', teamIds);\r\n \r\n if (sessionError) {\r\n throw new Error(`Database error: ${sessionError.message}`);\r\n }\r\n \r\n // Find session with matching prefix\r\n const session = allSessions?.find(s => s.id.toString().startsWith(sessionUuid));\r\n if (!session) {\r\n throw new Error(`Session not found: ${aiSessionId} (searched ${allSessions?.length || 0} sessions)`);\r\n }\r\n \r\n // Save the response in database\r\n const { data: responseData, error: responseError } = await supabase\r\n .from('ai_responses')\r\n .insert({\r\n ai_session_id: session.id,\r\n response_type: responseType,\r\n content: customerResponse,\r\n is_ready_for_customer: true,\r\n provider_approved: false // Needs manual approval\r\n })\r\n .select()\r\n .single();\r\n \r\n if (responseError) throw responseError;\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `πŸ’Ύ **Customer Response Saved!**\\n\\n` +\r\n `πŸ†” Session: ${aiSessionId}\\n` +\r\n `πŸ“ Response Type: ${responseType}\\n` +\r\n `πŸ“„ Length: ${customerResponse.length} characters\\n\\n` +\r\n `βœ… **Response ready for provider approval**\\n` +\r\n `πŸ” Provider can review in AI tab before sending to customer\\n\\n` +\r\n `**Preview:**\\n\\`\\`\\`\\n${customerResponse.substring(0, 200)}${customerResponse.length > 200 ? '...' : ''}\\`\\`\\``\r\n }]\r\n };\r\n }\r\n \r\n case 'complete-ai-session': {\r\n const { aiSessionId, workCompleted, technicalSummary, invoiceDescription, efficiencyNotes } = args as any;\r\n \r\n // Extract actual session UUID from readable ID\r\n const sessionUuid = aiSessionId.replace('ai-sess-', '');\r\n \r\n // Get session to calculate actual time spent - find by JavaScript filtering\r\n const teamIds = await getAccessibleTeamIds(authContext.teamId);\r\n const { data: allSessions, error: getSessionError } = await supabase\r\n .from('ai_sessions')\r\n .select('id, ticket_id, ai_time_estimate_minutes, created_at')\r\n .in('team_id', teamIds);\r\n \r\n if (getSessionError) {\r\n throw new Error(`Database error: ${getSessionError.message}`);\r\n }\r\n \r\n // Find session with matching prefix\r\n const existingSession = allSessions?.find(s => s.id.toString().startsWith(sessionUuid));\r\n if (!existingSession) {\r\n throw new Error(`Session not found: ${aiSessionId} (searched ${allSessions?.length || 0} sessions)`);\r\n }\r\n\r\n // Calculate actual time spent automatically\r\n const completionTime = new Date();\r\n const sessionStartTime = new Date(existingSession.created_at);\r\n const timeSpentMinutes = Math.round((completionTime.getTime() - sessionStartTime.getTime()) / 60000);\r\n \r\n // Update session to completed with calculated time\r\n const { data: session, error: sessionError } = await supabase\r\n .from('ai_sessions')\r\n .update({\r\n status: 'completed',\r\n actual_time_minutes: timeSpentMinutes,\r\n completed_at: completionTime.toISOString(),\r\n efficiency_score: null // Will be calculated\r\n })\r\n .eq('id', existingSession.id)\r\n .select('id, ticket_id, ai_time_estimate_minutes, created_at')\r\n .single();\r\n \r\n if (sessionError || !session) {\r\n throw new Error(`Failed to update session: ${aiSessionId}`);\r\n }\r\n \r\n // Calculate efficiency score\r\n const efficiencyScore = session.ai_time_estimate_minutes \r\n ? (timeSpentMinutes / session.ai_time_estimate_minutes)\r\n : 1.0;\r\n \r\n // Update efficiency score\r\n await supabase\r\n .from('ai_sessions')\r\n .update({ efficiency_score: efficiencyScore.toFixed(2) })\r\n .eq('id', session.id);\r\n \r\n // Get session duration\r\n const sessionDuration = Math.round(\r\n (completionTime.getTime() - new Date(session.created_at).getTime()) / 60000\r\n );\r\n \r\n // Create work summary\r\n const workSummary = `Completed ${workCompleted.length} tasks including: ${workCompleted.slice(0, 3).join(', ')}${workCompleted.length > 3 ? ' and more' : ''}.`;\r\n \r\n // Get ticket details for agenda event creation\r\n const { data: ticketInfo } = await supabase\r\n .from('tickets')\r\n .select('ticket_number, title, project_id')\r\n .eq('id', session.ticket_id)\r\n .single();\r\n \r\n // CREATE AGENDA EVENT: Now that work is complete, create the timetrack entry\r\n // Use invoice description if provided, otherwise create a technical one\r\n let completionDescription: string;\r\n \r\n if (invoiceDescription) {\r\n // Use the invoice-friendly description provided by Cursor (in ticket language)\r\n completionDescription = `${ticketInfo?.ticket_number || 'Ticket'}: ${invoiceDescription}`;\r\n } else {\r\n // Fallback: Create a technical work description\r\n const workDescription = workCompleted.map((task: string, index: number) => `${index + 1}. ${task}`).join('\\n');\r\n completionDescription = `${ticketInfo?.ticket_number || 'Ticket'}: ${technicalSummary || workSummary}\\n\\nCompleted work:\\n${workDescription}`;\r\n }\r\n \r\n // Create the agenda event with AI estimated time\r\n // Use the original AI estimate for billing, not the actual AI completion time\r\n const estimatedMinutes = session.ai_time_estimate_minutes || timeSpentMinutes;\r\n const sessionStart = new Date(session.created_at);\r\n const estimatedEnd = new Date(sessionStart.getTime() + estimatedMinutes * 60000);\r\n \r\n const { data: agendaEvent, error: agendaError } = await supabase\r\n .from('agenda_events')\r\n .insert({\r\n team_id: authContext.teamId,\r\n user_id: authContext.userId,\r\n title: ticketInfo?.title || 'Development Work',\r\n description: completionDescription,\r\n start_time: sessionStart.toISOString(),\r\n end_time: estimatedEnd.toISOString(),\r\n project_id: ticketInfo?.project_id || null,\r\n ticket_id: session.ticket_id,\r\n ai_session_id: session.id, // Use the actual UUID, not the readable ID\r\n type: 'work',\r\n status: 'draft', // Mark as draft for manual approval\r\n all_day: false,\r\n is_tracked: true,\r\n tracked_duration: estimatedMinutes * 60 // Use AI estimate in seconds for billing\r\n })\r\n .select('id')\r\n .single();\r\n \r\n if (agendaError) {\r\n console.error('⚠️ Failed to create agenda event:', agendaError);\r\n // Don't fail the completion if agenda event creation fails\r\n }\r\n \r\n let responseText = `πŸŽ‰ **AI Session Completed Successfully!**\\n\\n`;\r\n responseText += `πŸ†” Session: ${aiSessionId}\\n`;\r\n responseText += `πŸ“Š **Performance Summary:**\\n`;\r\n responseText += ` β€’ Tasks Completed: ${workCompleted.length}\\n`;\r\n responseText += ` β€’ Time Spent: ${timeSpentMinutes} minutes\\n`;\r\n responseText += ` β€’ Estimated Time: ${session.ai_time_estimate_minutes || 'N/A'} minutes\\n`;\r\n responseText += ` β€’ Efficiency: ${efficiencyScore < 1 ? 'πŸš€' : efficiencyScore > 1.5 ? '⚠️' : '⏱️'} ${(efficiencyScore * 100).toFixed(0)}%\\n`;\r\n responseText += ` β€’ Session Duration: ${sessionDuration} minutes\\n\\n`;\r\n \r\n responseText += `βœ… **Work Completed:**\\n`;\r\n workCompleted.forEach((task: string, index: number) => {\r\n responseText += `${index + 1}. ${task}\\n`;\r\n });\r\n responseText += `\\n`;\r\n \r\n if (technicalSummary) {\r\n responseText += `πŸ”§ **Technical Summary:**\\n${technicalSummary}\\n\\n`;\r\n }\r\n \r\n if (efficiencyNotes) {\r\n responseText += `πŸ“ˆ **Efficiency Notes:**\\n${efficiencyNotes}\\n\\n`;\r\n }\r\n \r\n if (agendaEvent) {\r\n responseText += `πŸ“… **Timetrack Entry Created:**\\n`;\r\n responseText += ` β€’ Agenda event created with work summary\\n`;\r\n responseText += ` β€’ Status: DRAFT (requires approval in agenda)\\n`;\r\n responseText += ` β€’ Duration: ${timeSpentMinutes} minutes\\n`;\r\n responseText += ` β€’ Period: ${sessionStart.toLocaleString()} - ${completionTime.toLocaleString()}\\n\\n`;\r\n }\r\n \r\n // Provide context for Cursor AI to generate customer response\r\n responseText += `πŸ“‹ **Context for Customer Response:**\\n`;\r\n responseText += ` β€’ Use \"get-completion-context\" to retrieve full context\\n`;\r\n responseText += ` β€’ Generate customer-friendly response based on completed work\\n`;\r\n responseText += ` β€’ Focus on business value and customer benefits\\n\\n`;\r\n \r\n responseText += `🎯 **Session archived successfully!**`;\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: responseText\r\n }]\r\n };\r\n }\r\n \r\n case 'log-hours': {\r\n const { projectId, workDescription, estimatedHours, chatContextSummary } = args as any;\r\n \r\n let project = null;\r\n \r\n // Verify project access if projectId is provided\r\n if (projectId) {\r\n const teamIds = await getAccessibleTeamIds(authContext.teamId);\r\n const { data: projectData, error: projectError } = await supabase\r\n .from('projects')\r\n .select('id, name, team_id')\r\n .eq('id', projectId)\r\n .in('team_id', teamIds)\r\n .single();\r\n \r\n if (projectError || !projectData) {\r\n throw new Error(`Project not found or no access: ${projectId}. Please call get-projects first to find the correct project.`);\r\n }\r\n \r\n project = projectData;\r\n }\r\n \r\n // Convert hours to seconds for tracker_entries.duration\r\n const durationSeconds = Math.round(estimatedHours * 3600);\r\n \r\n // Create tracker entry as draft\r\n const now = new Date();\r\n const startTime = new Date(now.getTime() - durationSeconds * 1000); // Back-calculate start time\r\n \r\n const { data: trackerEntry, error: trackerError } = await supabase\r\n .from('tracker_entries')\r\n .insert({\r\n team_id: authContext.teamId,\r\n assigned_id: authContext.userId,\r\n project_id: project?.id || null,\r\n description: workDescription,\r\n duration: durationSeconds,\r\n start: startTime.toISOString(),\r\n stop: now.toISOString(),\r\n date: now.toISOString().split('T')[0],\r\n billed: false\r\n })\r\n .select('id, duration, project_id')\r\n .single();\r\n \r\n if (trackerError || !trackerEntry) {\r\n throw new Error(`Failed to create tracker entry: ${trackerError?.message || 'Unknown error'}`);\r\n }\r\n \r\n let responseText = `⏱️ **Hours Logged Successfully!**\\n\\n`;\r\n responseText += `πŸ“‹ **Entry Details:**\\n`;\r\n if (project) {\r\n responseText += ` β€’ Project: ${project.name}\\n`;\r\n } else {\r\n responseText += ` β€’ Project: (No project assigned)\\n`;\r\n }\r\n responseText += ` β€’ Description: ${workDescription}\\n`;\r\n responseText += ` β€’ Estimated Hours: ${estimatedHours}h (${Math.floor(estimatedHours)}h ${Math.round((estimatedHours % 1) * 60)}m)\\n`;\r\n responseText += ` β€’ Status: DRAFT (not billed yet)\\n`;\r\n responseText += ` β€’ Entry ID: ${trackerEntry.id}\\n\\n`;\r\n \r\n if (chatContextSummary) {\r\n responseText += `πŸ“Š **Work Context:**\\n`;\r\n responseText += `${chatContextSummary.substring(0, 200)}${chatContextSummary.length > 200 ? '...' : ''}\\n\\n`;\r\n }\r\n \r\n responseText += `βœ… Time entry created and ready for review in the tracker!`;\r\n \r\n return {\r\n content: [{\r\n type: 'text',\r\n text: responseText\r\n }]\r\n };\r\n }\r\n \r\n default:\r\n throw new Error(`Unknown tool: ${name}`);\r\n }\r\n \r\n } catch (error) {\r\n console.error(`❌ Tool execution error:`, error);\r\n return {\r\n content: [{\r\n type: 'text',\r\n text: `Error executing ${name}: ${error instanceof Error ? error.message : 'Unknown error'}`\r\n }]\r\n };\r\n }\r\n});\r\n\r\n// Resource read handler\r\nserver.setRequestHandler(ReadResourceRequestSchema, async (request) => {\r\n if (!authContext) {\r\n return {\r\n contents: [{\r\n uri: request.params.uri,\r\n mimeType: 'text/plain',\r\n text: 'Error: Not authenticated. API key validation failed.'\r\n }]\r\n };\r\n }\r\n\r\n const { uri } = request.params;\r\n console.error(`πŸ“š Reading resource: ${uri}`);\r\n \r\n try {\r\n switch (uri) {\r\n case 'tickets://recent': {\r\n const teamIds = await getAccessibleTeamIds(authContext!.teamId);\r\n const { data, error } = await supabase\r\n .from('tickets')\r\n .select(`\r\n id,\r\n ticket_number,\r\n title,\r\n status,\r\n priority,\r\n created_at\r\n `)\r\n .in('team_id', teamIds)\r\n .order('created_at', { ascending: false })\r\n .limit(20);\r\n \r\n if (error) throw error;\r\n \r\n return {\r\n contents: [{\r\n uri,\r\n mimeType: 'application/json',\r\n text: JSON.stringify(data, null, 2)\r\n }]\r\n };\r\n }\r\n \r\n case 'customers://all': {\r\n const teamIds = await getAccessibleTeamIds(authContext!.teamId);\r\n const { data, error } = await supabase\r\n .from('customers')\r\n .select('id, name, email, website, created_at')\r\n .in('team_id', teamIds)\r\n .order('name')\r\n .limit(50);\r\n \r\n if (error) throw error;\r\n \r\n return {\r\n contents: [{\r\n uri,\r\n mimeType: 'application/json',\r\n text: JSON.stringify(data, null, 2)\r\n }]\r\n };\r\n }\r\n \r\n case 'projects://active': {\r\n const teamIds = await getAccessibleTeamIds(authContext!.teamId);\r\n const { data, error } = await supabase\r\n .from('projects')\r\n .select(`\r\n id,\r\n name,\r\n description,\r\n status,\r\n created_at,\r\n customers:customer_id(id, name)\r\n `)\r\n .in('team_id', teamIds)\r\n .eq('status', 'active')\r\n .order('name')\r\n .limit(50);\r\n \r\n if (error) throw error;\r\n \r\n return {\r\n contents: [{\r\n uri,\r\n mimeType: 'application/json',\r\n text: JSON.stringify(data, null, 2)\r\n }]\r\n };\r\n }\r\n \r\n default:\r\n throw new Error(`Unknown resource: ${uri}`);\r\n }\r\n \r\n } catch (error) {\r\n console.error(`❌ Resource read error:`, error);\r\n return {\r\n contents: [{\r\n uri,\r\n mimeType: 'text/plain',\r\n text: `Error reading ${uri}: ${error instanceof Error ? error.message : 'Unknown error'}`\r\n }]\r\n };\r\n }\r\n});\r\n\r\n// Main function\r\nasync function main() {\r\n console.error('πŸš€ Starting MG Tickets MCP Bridge Server...');\r\n console.error(`πŸ”‘ API Key: ${apiKey?.substring(0, 10)}...`);\r\n \r\n // Validate API key\r\n authContext = await validateApiKey(apiKey!);\r\n if (!authContext) {\r\n console.error('❌ API key validation failed. Please check your key and try again.');\r\n process.exit(1);\r\n }\r\n \r\n console.error(`βœ… Authenticated as user ${authContext.userId} in team ${authContext.teamId}`);\r\n console.error(`πŸ“‹ Available scopes: ${authContext.scopes.join(', ')}`);\r\n console.error('πŸ“‘ MCP Bridge Server ready for connections');\r\n \r\n const transport = new StdioServerTransport();\r\n await server.connect(transport);\r\n}\r\n\r\n// Handle graceful shutdown\r\nprocess.on('SIGINT', async () => {\r\n console.error('πŸ‘‹ Shutting down MCP Bridge Server...');\r\n process.exit(0);\r\n});\r\n\r\nprocess.on('SIGTERM', async () => {\r\n console.error('πŸ‘‹ Shutting down MCP Bridge Server...');\r\n process.exit(0);\r\n});\r\n\r\n// Start the server\r\nmain().catch((error) => {\r\n console.error('πŸ’₯ Fatal error:', error);\r\n process.exit(1);\r\n});"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@mgsoftwarebv/mcp-server-bridge",
3
- "version": "2.10.1",
3
+ "version": "2.12.0",
4
4
  "description": "MCP Server bridge for MG Tickets - connects Cursor to HTTP MCP server with image support",
5
5
  "type": "module",
6
6
  "main": "dist/index.js",