@app-connect/core 1.7.24 → 1.7.26

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (137) hide show
  1. package/.env.test +5 -5
  2. package/README.md +441 -441
  3. package/connector/developerPortal.js +31 -42
  4. package/connector/mock.js +84 -77
  5. package/connector/proxy/engine.js +164 -163
  6. package/connector/proxy/index.js +500 -500
  7. package/connector/registry.js +252 -252
  8. package/docs/README.md +50 -50
  9. package/docs/architecture.md +93 -93
  10. package/docs/connectors.md +116 -117
  11. package/docs/handlers.md +125 -125
  12. package/docs/libraries.md +101 -101
  13. package/docs/models.md +144 -144
  14. package/docs/routes.md +115 -115
  15. package/docs/tests.md +73 -73
  16. package/handlers/admin.js +523 -523
  17. package/handlers/appointment.js +193 -0
  18. package/handlers/auth.js +296 -296
  19. package/handlers/calldown.js +99 -99
  20. package/handlers/contact.js +280 -280
  21. package/handlers/disposition.js +82 -80
  22. package/handlers/log.js +984 -973
  23. package/handlers/managedAuth.js +446 -446
  24. package/handlers/plugin.js +208 -208
  25. package/handlers/user.js +142 -142
  26. package/index.js +3140 -2652
  27. package/jest.config.js +56 -56
  28. package/lib/analytics.js +54 -54
  29. package/lib/authSession.js +109 -109
  30. package/lib/cacheCleanup.js +21 -0
  31. package/lib/callLogComposer.js +898 -898
  32. package/lib/callLogLookup.js +34 -0
  33. package/lib/constants.js +8 -8
  34. package/lib/debugTracer.js +177 -177
  35. package/lib/encode.js +30 -30
  36. package/lib/errorHandler.js +218 -206
  37. package/lib/generalErrorMessage.js +41 -41
  38. package/lib/jwt.js +18 -18
  39. package/lib/logger.js +190 -190
  40. package/lib/migrateCallLogsSchema.js +116 -0
  41. package/lib/ringcentral.js +266 -266
  42. package/lib/s3ErrorLogReport.js +65 -65
  43. package/lib/sharedSMSComposer.js +471 -471
  44. package/lib/util.js +67 -67
  45. package/mcp/README.md +412 -395
  46. package/mcp/lib/validator.js +91 -91
  47. package/mcp/mcpHandler.js +425 -425
  48. package/mcp/tools/cancelAppointment.js +101 -0
  49. package/mcp/tools/checkAuthStatus.js +105 -105
  50. package/mcp/tools/confirmAppointment.js +101 -0
  51. package/mcp/tools/createAppointment.js +157 -0
  52. package/mcp/tools/createCallLog.js +327 -316
  53. package/mcp/tools/createContact.js +117 -117
  54. package/mcp/tools/createMessageLog.js +287 -287
  55. package/mcp/tools/doAuth.js +60 -60
  56. package/mcp/tools/findContactByName.js +93 -93
  57. package/mcp/tools/findContactByPhone.js +101 -101
  58. package/mcp/tools/getCallLog.js +111 -102
  59. package/mcp/tools/getGoogleFilePicker.js +99 -99
  60. package/mcp/tools/getHelp.js +43 -43
  61. package/mcp/tools/getPublicConnectors.js +94 -94
  62. package/mcp/tools/getSessionInfo.js +90 -90
  63. package/mcp/tools/index.js +51 -41
  64. package/mcp/tools/listAppointments.js +163 -0
  65. package/mcp/tools/logout.js +96 -96
  66. package/mcp/tools/rcGetCallLogs.js +65 -65
  67. package/mcp/tools/updateAppointment.js +154 -0
  68. package/mcp/tools/updateCallLog.js +130 -126
  69. package/mcp/ui/App/App.tsx +358 -358
  70. package/mcp/ui/App/components/AuthInfoForm.tsx +113 -113
  71. package/mcp/ui/App/components/AuthSuccess.tsx +22 -22
  72. package/mcp/ui/App/components/ConnectorList.tsx +82 -82
  73. package/mcp/ui/App/components/DebugPanel.tsx +43 -43
  74. package/mcp/ui/App/components/OAuthConnect.tsx +270 -270
  75. package/mcp/ui/App/lib/callTool.ts +130 -130
  76. package/mcp/ui/App/lib/debugLog.ts +41 -41
  77. package/mcp/ui/App/lib/developerPortal.ts +111 -111
  78. package/mcp/ui/App/main.css +5 -5
  79. package/mcp/ui/App/root.tsx +13 -13
  80. package/mcp/ui/index.html +13 -13
  81. package/mcp/ui/package-lock.json +6356 -6356
  82. package/mcp/ui/package.json +25 -25
  83. package/mcp/ui/tsconfig.json +26 -26
  84. package/mcp/ui/vite.config.ts +16 -16
  85. package/models/accountDataModel.js +33 -33
  86. package/models/adminConfigModel.js +35 -35
  87. package/models/cacheModel.js +30 -26
  88. package/models/callDownListModel.js +34 -34
  89. package/models/callLogModel.js +33 -27
  90. package/models/dynamo/connectorSchema.js +146 -146
  91. package/models/dynamo/lockSchema.js +24 -24
  92. package/models/dynamo/noteCacheSchema.js +29 -29
  93. package/models/llmSessionModel.js +17 -17
  94. package/models/messageLogModel.js +25 -25
  95. package/models/sequelize.js +16 -16
  96. package/models/userModel.js +45 -45
  97. package/package.json +72 -72
  98. package/releaseNotes.json +1093 -1073
  99. package/test/connector/proxy/engine.test.js +126 -93
  100. package/test/connector/proxy/index.test.js +279 -279
  101. package/test/connector/proxy/sample.json +161 -161
  102. package/test/connector/registry.test.js +415 -415
  103. package/test/handlers/admin.test.js +616 -616
  104. package/test/handlers/auth.test.js +1018 -1015
  105. package/test/handlers/contact.test.js +1014 -1014
  106. package/test/handlers/log.test.js +1298 -1160
  107. package/test/handlers/managedAuth.test.js +458 -458
  108. package/test/handlers/plugin.test.js +380 -380
  109. package/test/index.test.js +105 -105
  110. package/test/lib/cacheCleanup.test.js +42 -0
  111. package/test/lib/callLogComposer.test.js +1231 -1231
  112. package/test/lib/debugTracer.test.js +328 -328
  113. package/test/lib/jwt.test.js +176 -176
  114. package/test/lib/logger.test.js +206 -206
  115. package/test/lib/oauth.test.js +359 -359
  116. package/test/lib/ringcentral.test.js +467 -467
  117. package/test/lib/sharedSMSComposer.test.js +1084 -1084
  118. package/test/lib/util.test.js +329 -329
  119. package/test/mcp/tools/checkAuthStatus.test.js +83 -82
  120. package/test/mcp/tools/createCallLog.test.js +436 -436
  121. package/test/mcp/tools/createContact.test.js +58 -58
  122. package/test/mcp/tools/createMessageLog.test.js +595 -595
  123. package/test/mcp/tools/doAuth.test.js +113 -113
  124. package/test/mcp/tools/findContactByName.test.js +275 -275
  125. package/test/mcp/tools/findContactByPhone.test.js +296 -296
  126. package/test/mcp/tools/getCallLog.test.js +298 -298
  127. package/test/mcp/tools/getGoogleFilePicker.test.js +281 -281
  128. package/test/mcp/tools/getPublicConnectors.test.js +107 -107
  129. package/test/mcp/tools/getSessionInfo.test.js +127 -127
  130. package/test/mcp/tools/logout.test.js +233 -233
  131. package/test/mcp/tools/rcGetCallLogs.test.js +56 -56
  132. package/test/mcp/tools/updateCallLog.test.js +360 -360
  133. package/test/models/accountDataModel.test.js +98 -98
  134. package/test/models/dynamo/connectorSchema.test.js +189 -189
  135. package/test/models/models.test.js +568 -539
  136. package/test/routes/managedAuthRoutes.test.js +104 -129
  137. package/test/setup.js +178 -178
@@ -1,360 +1,360 @@
1
- const updateCallLog = require('../../../mcp/tools/updateCallLog');
2
- const jwt = require('../../../lib/jwt');
3
- const connectorRegistry = require('../../../connector/registry');
4
- const logCore = require('../../../handlers/log');
5
- const util = require('../../../lib/util');
6
-
7
- // Mock dependencies
8
- jest.mock('../../../lib/jwt');
9
- jest.mock('../../../connector/registry');
10
- jest.mock('../../../handlers/log');
11
- jest.mock('../../../lib/util');
12
-
13
- describe('MCP Tool: updateCallLog', () => {
14
- beforeEach(() => {
15
- jest.clearAllMocks();
16
- process.env.HASH_KEY = 'test-hash-key';
17
- });
18
-
19
- describe('tool definition', () => {
20
- test('should have correct tool definition', () => {
21
- expect(updateCallLog.definition).toBeDefined();
22
- expect(updateCallLog.definition.name).toBe('updateCallLog');
23
- expect(updateCallLog.definition.description).toContain('REQUIRES AUTHENTICATION');
24
- expect(updateCallLog.definition.inputSchema).toBeDefined();
25
- });
26
-
27
- test('should require jwtToken and incomingData parameters', () => {
28
- expect(updateCallLog.definition.inputSchema.required).toContain('jwtToken');
29
- expect(updateCallLog.definition.inputSchema.required).toContain('incomingData');
30
- });
31
-
32
- test('should require sessionId in incomingData', () => {
33
- const incomingDataSchema = updateCallLog.definition.inputSchema.properties.incomingData;
34
- expect(incomingDataSchema.required).toContain('sessionId');
35
- });
36
- });
37
-
38
- describe('execute', () => {
39
- test('should update call log successfully', async () => {
40
- // Arrange
41
- const mockIncomingData = {
42
- sessionId: 'session-123',
43
- note: 'Updated call note',
44
- accountId: 'rc-account-123'
45
- };
46
-
47
- jwt.decodeJwt.mockReturnValue({
48
- id: 'user-123',
49
- platform: 'testCRM'
50
- });
51
-
52
- const mockConnector = {
53
- updateCallLog: jest.fn()
54
- };
55
- connectorRegistry.getConnector.mockReturnValue(mockConnector);
56
-
57
- util.getHashValue.mockReturnValue('hashed-account-id');
58
-
59
- logCore.updateCallLog.mockResolvedValue({
60
- successful: true,
61
- logId: 'crm-log-123',
62
- updatedNote: 'Updated call note',
63
- returnMessage: { message: 'Call log updated successfully' }
64
- });
65
-
66
- // Act
67
- const result = await updateCallLog.execute({
68
- jwtToken: 'mock-jwt-token',
69
- incomingData: mockIncomingData
70
- });
71
-
72
- // Assert
73
- expect(result).toEqual({
74
- success: true,
75
- data: {
76
- logId: 'crm-log-123',
77
- updatedNote: 'Updated call note',
78
- message: 'Call log updated successfully'
79
- }
80
- });
81
- expect(jwt.decodeJwt).toHaveBeenCalledWith('mock-jwt-token');
82
- expect(util.getHashValue).toHaveBeenCalledWith('rc-account-123', 'test-hash-key');
83
- expect(logCore.updateCallLog).toHaveBeenCalledWith({
84
- platform: 'testCRM',
85
- userId: 'user-123',
86
- incomingData: mockIncomingData,
87
- hashedAccountId: 'hashed-account-id',
88
- isFromSSCL: false
89
- });
90
- });
91
-
92
- test('should update call log with additional submission', async () => {
93
- // Arrange
94
- const mockIncomingData = {
95
- sessionId: 'session-456',
96
- note: 'Updated note with additional data',
97
- additionalSubmission: {
98
- dealId: 'deal-123',
99
- customField: 'custom-value'
100
- }
101
- };
102
-
103
- jwt.decodeJwt.mockReturnValue({
104
- id: 'user-123',
105
- platform: 'testCRM'
106
- });
107
-
108
- const mockConnector = {
109
- updateCallLog: jest.fn()
110
- };
111
- connectorRegistry.getConnector.mockReturnValue(mockConnector);
112
-
113
- logCore.updateCallLog.mockResolvedValue({
114
- successful: true,
115
- logId: 'crm-log-456',
116
- updatedNote: 'Updated note with additional data'
117
- });
118
-
119
- // Act
120
- const result = await updateCallLog.execute({
121
- jwtToken: 'mock-jwt-token',
122
- incomingData: mockIncomingData
123
- });
124
-
125
- // Assert
126
- expect(result.success).toBe(true);
127
- expect(result.data.logId).toBe('crm-log-456');
128
- });
129
-
130
- test('should update call log without accountId', async () => {
131
- // Arrange
132
- const mockIncomingData = {
133
- sessionId: 'session-789',
134
- note: 'Simple update'
135
- };
136
-
137
- jwt.decodeJwt.mockReturnValue({
138
- id: 'user-123',
139
- platform: 'testCRM'
140
- });
141
-
142
- const mockConnector = {
143
- updateCallLog: jest.fn()
144
- };
145
- connectorRegistry.getConnector.mockReturnValue(mockConnector);
146
-
147
- logCore.updateCallLog.mockResolvedValue({
148
- successful: true,
149
- logId: 'crm-log-789',
150
- updatedNote: 'Simple update'
151
- });
152
-
153
- // Act
154
- const result = await updateCallLog.execute({
155
- jwtToken: 'mock-jwt-token',
156
- incomingData: mockIncomingData
157
- });
158
-
159
- // Assert
160
- expect(result.success).toBe(true);
161
- expect(util.getHashValue).not.toHaveBeenCalled();
162
- expect(logCore.updateCallLog).toHaveBeenCalledWith({
163
- platform: 'testCRM',
164
- userId: 'user-123',
165
- incomingData: mockIncomingData,
166
- hashedAccountId: undefined,
167
- isFromSSCL: false
168
- });
169
- });
170
-
171
- test('should return error when JWT is invalid', async () => {
172
- // Arrange
173
- const mockIncomingData = {
174
- sessionId: 'session-123',
175
- note: 'Test note'
176
- };
177
-
178
- jwt.decodeJwt.mockReturnValue({
179
- platform: 'testCRM'
180
- // id is missing
181
- });
182
-
183
- // Act
184
- const result = await updateCallLog.execute({
185
- jwtToken: 'invalid-token',
186
- incomingData: mockIncomingData
187
- });
188
-
189
- // Assert
190
- expect(result.success).toBe(false);
191
- expect(result.error).toContain('Invalid JWT token');
192
- });
193
-
194
- test('should return error when decodeJwt returns null', async () => {
195
- jwt.decodeJwt.mockReturnValue(null);
196
-
197
- const result = await updateCallLog.execute({
198
- jwtToken: 'invalid-token',
199
- incomingData: {
200
- logData: { sessionId: 'session-123' }
201
- }
202
- });
203
-
204
- expect(result.success).toBe(false);
205
- expect(result.error).toContain('Invalid JWT token');
206
- });
207
-
208
- test('should return error when platform connector not found', async () => {
209
- // Arrange
210
- const mockIncomingData = {
211
- sessionId: 'session-123',
212
- note: 'Test note'
213
- };
214
-
215
- jwt.decodeJwt.mockReturnValue({
216
- id: 'user-123',
217
- platform: 'unknownCRM'
218
- });
219
-
220
- connectorRegistry.getConnector.mockReturnValue(null);
221
-
222
- // Act
223
- const result = await updateCallLog.execute({
224
- jwtToken: 'mock-jwt-token',
225
- incomingData: mockIncomingData
226
- });
227
-
228
- // Assert
229
- expect(result.success).toBe(false);
230
- expect(result.error).toContain('Platform connector not found');
231
- });
232
-
233
- test('should return error when updateCallLog is not implemented', async () => {
234
- // Arrange
235
- const mockIncomingData = {
236
- sessionId: 'session-123',
237
- note: 'Test note'
238
- };
239
-
240
- jwt.decodeJwt.mockReturnValue({
241
- id: 'user-123',
242
- platform: 'testCRM'
243
- });
244
-
245
- const mockConnector = {}; // No updateCallLog method
246
- connectorRegistry.getConnector.mockReturnValue(mockConnector);
247
-
248
- // Act
249
- const result = await updateCallLog.execute({
250
- jwtToken: 'mock-jwt-token',
251
- incomingData: mockIncomingData
252
- });
253
-
254
- // Assert
255
- expect(result.success).toBe(false);
256
- expect(result.error).toContain('not implemented');
257
- });
258
-
259
- test('should return error when update fails', async () => {
260
- // Arrange
261
- const mockIncomingData = {
262
- sessionId: 'session-999',
263
- note: 'Test note'
264
- };
265
-
266
- jwt.decodeJwt.mockReturnValue({
267
- id: 'user-123',
268
- platform: 'testCRM'
269
- });
270
-
271
- const mockConnector = {
272
- updateCallLog: jest.fn()
273
- };
274
- connectorRegistry.getConnector.mockReturnValue(mockConnector);
275
-
276
- logCore.updateCallLog.mockResolvedValue({
277
- successful: false,
278
- returnMessage: { message: 'Log not found in CRM' }
279
- });
280
-
281
- // Act
282
- const result = await updateCallLog.execute({
283
- jwtToken: 'mock-jwt-token',
284
- incomingData: mockIncomingData
285
- });
286
-
287
- // Assert
288
- expect(result.success).toBe(false);
289
- expect(result.error).toBe('Log not found in CRM');
290
- });
291
-
292
- test('should handle unexpected errors gracefully', async () => {
293
- // Arrange
294
- const mockIncomingData = {
295
- sessionId: 'session-error',
296
- note: 'Test note'
297
- };
298
-
299
- jwt.decodeJwt.mockReturnValue({
300
- id: 'user-123',
301
- platform: 'testCRM'
302
- });
303
-
304
- const mockConnector = {
305
- updateCallLog: jest.fn()
306
- };
307
- connectorRegistry.getConnector.mockReturnValue(mockConnector);
308
-
309
- logCore.updateCallLog.mockRejectedValue(
310
- new Error('API timeout')
311
- );
312
-
313
- // Act
314
- const result = await updateCallLog.execute({
315
- jwtToken: 'mock-jwt-token',
316
- incomingData: mockIncomingData
317
- });
318
-
319
- // Assert
320
- expect(result.success).toBe(false);
321
- expect(result.error).toBe('API timeout');
322
- expect(result.errorDetails).toBeDefined();
323
- });
324
-
325
- test('should update call log with empty note', async () => {
326
- // Arrange
327
- const mockIncomingData = {
328
- sessionId: 'session-empty-note',
329
- note: ''
330
- };
331
-
332
- jwt.decodeJwt.mockReturnValue({
333
- id: 'user-123',
334
- platform: 'testCRM'
335
- });
336
-
337
- const mockConnector = {
338
- updateCallLog: jest.fn()
339
- };
340
- connectorRegistry.getConnector.mockReturnValue(mockConnector);
341
-
342
- logCore.updateCallLog.mockResolvedValue({
343
- successful: true,
344
- logId: 'crm-log-empty',
345
- updatedNote: ''
346
- });
347
-
348
- // Act
349
- const result = await updateCallLog.execute({
350
- jwtToken: 'mock-jwt-token',
351
- incomingData: mockIncomingData
352
- });
353
-
354
- // Assert
355
- expect(result.success).toBe(true);
356
- expect(result.data.updatedNote).toBe('');
357
- });
358
- });
359
- });
360
-
1
+ const updateCallLog = require('../../../mcp/tools/updateCallLog');
2
+ const jwt = require('../../../lib/jwt');
3
+ const connectorRegistry = require('../../../connector/registry');
4
+ const logCore = require('../../../handlers/log');
5
+ const util = require('../../../lib/util');
6
+
7
+ // Mock dependencies
8
+ jest.mock('../../../lib/jwt');
9
+ jest.mock('../../../connector/registry');
10
+ jest.mock('../../../handlers/log');
11
+ jest.mock('../../../lib/util');
12
+
13
+ describe('MCP Tool: updateCallLog', () => {
14
+ beforeEach(() => {
15
+ jest.clearAllMocks();
16
+ process.env.HASH_KEY = 'test-hash-key';
17
+ });
18
+
19
+ describe('tool definition', () => {
20
+ test('should have correct tool definition', () => {
21
+ expect(updateCallLog.definition).toBeDefined();
22
+ expect(updateCallLog.definition.name).toBe('updateCallLog');
23
+ expect(updateCallLog.definition.description).toContain('REQUIRES AUTHENTICATION');
24
+ expect(updateCallLog.definition.inputSchema).toBeDefined();
25
+ });
26
+
27
+ test('should require jwtToken and incomingData parameters', () => {
28
+ expect(updateCallLog.definition.inputSchema.required).toContain('jwtToken');
29
+ expect(updateCallLog.definition.inputSchema.required).toContain('incomingData');
30
+ });
31
+
32
+ test('should require sessionId in incomingData', () => {
33
+ const incomingDataSchema = updateCallLog.definition.inputSchema.properties.incomingData;
34
+ expect(incomingDataSchema.required).toContain('sessionId');
35
+ });
36
+ });
37
+
38
+ describe('execute', () => {
39
+ test('should update call log successfully', async () => {
40
+ // Arrange
41
+ const mockIncomingData = {
42
+ sessionId: 'session-123',
43
+ note: 'Updated call note',
44
+ accountId: 'rc-account-123'
45
+ };
46
+
47
+ jwt.decodeJwt.mockReturnValue({
48
+ id: 'user-123',
49
+ platform: 'testCRM'
50
+ });
51
+
52
+ const mockConnector = {
53
+ updateCallLog: jest.fn()
54
+ };
55
+ connectorRegistry.getConnector.mockReturnValue(mockConnector);
56
+
57
+ util.getHashValue.mockReturnValue('hashed-account-id');
58
+
59
+ logCore.updateCallLog.mockResolvedValue({
60
+ successful: true,
61
+ logId: 'crm-log-123',
62
+ updatedNote: 'Updated call note',
63
+ returnMessage: { message: 'Call log updated successfully' }
64
+ });
65
+
66
+ // Act
67
+ const result = await updateCallLog.execute({
68
+ jwtToken: 'mock-jwt-token',
69
+ incomingData: mockIncomingData
70
+ });
71
+
72
+ // Assert
73
+ expect(result).toEqual({
74
+ success: true,
75
+ data: {
76
+ logId: 'crm-log-123',
77
+ updatedNote: 'Updated call note',
78
+ message: 'Call log updated successfully'
79
+ }
80
+ });
81
+ expect(jwt.decodeJwt).toHaveBeenCalledWith('mock-jwt-token');
82
+ expect(util.getHashValue).toHaveBeenCalledWith('rc-account-123', 'test-hash-key');
83
+ expect(logCore.updateCallLog).toHaveBeenCalledWith({
84
+ platform: 'testCRM',
85
+ userId: 'user-123',
86
+ incomingData: mockIncomingData,
87
+ hashedAccountId: 'hashed-account-id',
88
+ isFromSSCL: false
89
+ });
90
+ });
91
+
92
+ test('should update call log with additional submission', async () => {
93
+ // Arrange
94
+ const mockIncomingData = {
95
+ sessionId: 'session-456',
96
+ note: 'Updated note with additional data',
97
+ additionalSubmission: {
98
+ dealId: 'deal-123',
99
+ customField: 'custom-value'
100
+ }
101
+ };
102
+
103
+ jwt.decodeJwt.mockReturnValue({
104
+ id: 'user-123',
105
+ platform: 'testCRM'
106
+ });
107
+
108
+ const mockConnector = {
109
+ updateCallLog: jest.fn()
110
+ };
111
+ connectorRegistry.getConnector.mockReturnValue(mockConnector);
112
+
113
+ logCore.updateCallLog.mockResolvedValue({
114
+ successful: true,
115
+ logId: 'crm-log-456',
116
+ updatedNote: 'Updated note with additional data'
117
+ });
118
+
119
+ // Act
120
+ const result = await updateCallLog.execute({
121
+ jwtToken: 'mock-jwt-token',
122
+ incomingData: mockIncomingData
123
+ });
124
+
125
+ // Assert
126
+ expect(result.success).toBe(true);
127
+ expect(result.data.logId).toBe('crm-log-456');
128
+ });
129
+
130
+ test('should update call log without accountId', async () => {
131
+ // Arrange
132
+ const mockIncomingData = {
133
+ sessionId: 'session-789',
134
+ note: 'Simple update'
135
+ };
136
+
137
+ jwt.decodeJwt.mockReturnValue({
138
+ id: 'user-123',
139
+ platform: 'testCRM'
140
+ });
141
+
142
+ const mockConnector = {
143
+ updateCallLog: jest.fn()
144
+ };
145
+ connectorRegistry.getConnector.mockReturnValue(mockConnector);
146
+
147
+ logCore.updateCallLog.mockResolvedValue({
148
+ successful: true,
149
+ logId: 'crm-log-789',
150
+ updatedNote: 'Simple update'
151
+ });
152
+
153
+ // Act
154
+ const result = await updateCallLog.execute({
155
+ jwtToken: 'mock-jwt-token',
156
+ incomingData: mockIncomingData
157
+ });
158
+
159
+ // Assert
160
+ expect(result.success).toBe(true);
161
+ expect(util.getHashValue).not.toHaveBeenCalled();
162
+ expect(logCore.updateCallLog).toHaveBeenCalledWith({
163
+ platform: 'testCRM',
164
+ userId: 'user-123',
165
+ incomingData: mockIncomingData,
166
+ hashedAccountId: undefined,
167
+ isFromSSCL: false
168
+ });
169
+ });
170
+
171
+ test('should return error when JWT is invalid', async () => {
172
+ // Arrange
173
+ const mockIncomingData = {
174
+ sessionId: 'session-123',
175
+ note: 'Test note'
176
+ };
177
+
178
+ jwt.decodeJwt.mockReturnValue({
179
+ platform: 'testCRM'
180
+ // id is missing
181
+ });
182
+
183
+ // Act
184
+ const result = await updateCallLog.execute({
185
+ jwtToken: 'invalid-token',
186
+ incomingData: mockIncomingData
187
+ });
188
+
189
+ // Assert
190
+ expect(result.success).toBe(false);
191
+ expect(result.error).toContain('Invalid JWT token');
192
+ });
193
+
194
+ test('should return error when decodeJwt returns null', async () => {
195
+ jwt.decodeJwt.mockReturnValue(null);
196
+
197
+ const result = await updateCallLog.execute({
198
+ jwtToken: 'invalid-token',
199
+ incomingData: {
200
+ logData: { sessionId: 'session-123' }
201
+ }
202
+ });
203
+
204
+ expect(result.success).toBe(false);
205
+ expect(result.error).toContain('Invalid JWT token');
206
+ });
207
+
208
+ test('should return error when platform connector not found', async () => {
209
+ // Arrange
210
+ const mockIncomingData = {
211
+ sessionId: 'session-123',
212
+ note: 'Test note'
213
+ };
214
+
215
+ jwt.decodeJwt.mockReturnValue({
216
+ id: 'user-123',
217
+ platform: 'unknownCRM'
218
+ });
219
+
220
+ connectorRegistry.getConnector.mockReturnValue(null);
221
+
222
+ // Act
223
+ const result = await updateCallLog.execute({
224
+ jwtToken: 'mock-jwt-token',
225
+ incomingData: mockIncomingData
226
+ });
227
+
228
+ // Assert
229
+ expect(result.success).toBe(false);
230
+ expect(result.error).toContain('Platform connector not found');
231
+ });
232
+
233
+ test('should return error when updateCallLog is not implemented', async () => {
234
+ // Arrange
235
+ const mockIncomingData = {
236
+ sessionId: 'session-123',
237
+ note: 'Test note'
238
+ };
239
+
240
+ jwt.decodeJwt.mockReturnValue({
241
+ id: 'user-123',
242
+ platform: 'testCRM'
243
+ });
244
+
245
+ const mockConnector = {}; // No updateCallLog method
246
+ connectorRegistry.getConnector.mockReturnValue(mockConnector);
247
+
248
+ // Act
249
+ const result = await updateCallLog.execute({
250
+ jwtToken: 'mock-jwt-token',
251
+ incomingData: mockIncomingData
252
+ });
253
+
254
+ // Assert
255
+ expect(result.success).toBe(false);
256
+ expect(result.error).toContain('not implemented');
257
+ });
258
+
259
+ test('should return error when update fails', async () => {
260
+ // Arrange
261
+ const mockIncomingData = {
262
+ sessionId: 'session-999',
263
+ note: 'Test note'
264
+ };
265
+
266
+ jwt.decodeJwt.mockReturnValue({
267
+ id: 'user-123',
268
+ platform: 'testCRM'
269
+ });
270
+
271
+ const mockConnector = {
272
+ updateCallLog: jest.fn()
273
+ };
274
+ connectorRegistry.getConnector.mockReturnValue(mockConnector);
275
+
276
+ logCore.updateCallLog.mockResolvedValue({
277
+ successful: false,
278
+ returnMessage: { message: 'Log not found in CRM' }
279
+ });
280
+
281
+ // Act
282
+ const result = await updateCallLog.execute({
283
+ jwtToken: 'mock-jwt-token',
284
+ incomingData: mockIncomingData
285
+ });
286
+
287
+ // Assert
288
+ expect(result.success).toBe(false);
289
+ expect(result.error).toBe('Log not found in CRM');
290
+ });
291
+
292
+ test('should handle unexpected errors gracefully', async () => {
293
+ // Arrange
294
+ const mockIncomingData = {
295
+ sessionId: 'session-error',
296
+ note: 'Test note'
297
+ };
298
+
299
+ jwt.decodeJwt.mockReturnValue({
300
+ id: 'user-123',
301
+ platform: 'testCRM'
302
+ });
303
+
304
+ const mockConnector = {
305
+ updateCallLog: jest.fn()
306
+ };
307
+ connectorRegistry.getConnector.mockReturnValue(mockConnector);
308
+
309
+ logCore.updateCallLog.mockRejectedValue(
310
+ new Error('API timeout')
311
+ );
312
+
313
+ // Act
314
+ const result = await updateCallLog.execute({
315
+ jwtToken: 'mock-jwt-token',
316
+ incomingData: mockIncomingData
317
+ });
318
+
319
+ // Assert
320
+ expect(result.success).toBe(false);
321
+ expect(result.error).toBe('API timeout');
322
+ expect(result.errorDetails).toBeDefined();
323
+ });
324
+
325
+ test('should update call log with empty note', async () => {
326
+ // Arrange
327
+ const mockIncomingData = {
328
+ sessionId: 'session-empty-note',
329
+ note: ''
330
+ };
331
+
332
+ jwt.decodeJwt.mockReturnValue({
333
+ id: 'user-123',
334
+ platform: 'testCRM'
335
+ });
336
+
337
+ const mockConnector = {
338
+ updateCallLog: jest.fn()
339
+ };
340
+ connectorRegistry.getConnector.mockReturnValue(mockConnector);
341
+
342
+ logCore.updateCallLog.mockResolvedValue({
343
+ successful: true,
344
+ logId: 'crm-log-empty',
345
+ updatedNote: ''
346
+ });
347
+
348
+ // Act
349
+ const result = await updateCallLog.execute({
350
+ jwtToken: 'mock-jwt-token',
351
+ incomingData: mockIncomingData
352
+ });
353
+
354
+ // Assert
355
+ expect(result.success).toBe(true);
356
+ expect(result.data.updatedNote).toBe('');
357
+ });
358
+ });
359
+ });
360
+